180814287SRaphael Isemann //===-- OptionValuePathMappings.cpp ---------------------------------------===//
267cc0636SGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
667cc0636SGreg Clayton //
767cc0636SGreg Clayton //===----------------------------------------------------------------------===//
867cc0636SGreg Clayton
967cc0636SGreg Clayton #include "lldb/Interpreter/OptionValuePathMappings.h"
1067cc0636SGreg Clayton
11dbd7fabaSJonas Devlieghere #include "lldb/Host/FileSystem.h"
12145d95c9SPavel Labath #include "lldb/Utility/Args.h"
135713a05bSZachary Turner #include "lldb/Utility/FileSpec.h"
14bf9a7730SZachary Turner #include "lldb/Utility/Stream.h"
1567cc0636SGreg Clayton
1667cc0636SGreg Clayton using namespace lldb;
1767cc0636SGreg Clayton using namespace lldb_private;
1893c1b3caSPavel Labath
VerifyPathExists(const char * path)19b9c1b51eSKate Stone static bool VerifyPathExists(const char *path) {
206a311641SGreg Clayton if (path && path[0])
21dbd7fabaSJonas Devlieghere return FileSystem::Instance().Exists(path);
226a311641SGreg Clayton else
236a311641SGreg Clayton return false;
246a311641SGreg Clayton }
256a311641SGreg Clayton
DumpValue(const ExecutionContext * exe_ctx,Stream & strm,uint32_t dump_mask)26b9c1b51eSKate Stone void OptionValuePathMappings::DumpValue(const ExecutionContext *exe_ctx,
27b9c1b51eSKate Stone Stream &strm, uint32_t dump_mask) {
2867cc0636SGreg Clayton if (dump_mask & eDumpOptionType)
2967cc0636SGreg Clayton strm.Printf("(%s)", GetTypeAsCString());
30b9c1b51eSKate Stone if (dump_mask & eDumpOptionValue) {
3167cc0636SGreg Clayton if (dump_mask & eDumpOptionType)
3267cc0636SGreg Clayton strm.Printf(" =%s", (m_path_mappings.GetSize() > 0) ? "\n" : "");
3367cc0636SGreg Clayton m_path_mappings.Dump(&strm);
3467cc0636SGreg Clayton }
3567cc0636SGreg Clayton }
3667cc0636SGreg Clayton
SetValueFromString(llvm::StringRef value,VarSetOperationType op)3797206d57SZachary Turner Status OptionValuePathMappings::SetValueFromString(llvm::StringRef value,
38b9c1b51eSKate Stone VarSetOperationType op) {
3997206d57SZachary Turner Status error;
40771ef6d4SMalcolm Parsons Args args(value.str());
4167cc0636SGreg Clayton const size_t argc = args.GetArgumentCount();
4267cc0636SGreg Clayton
43b9c1b51eSKate Stone switch (op) {
4467cc0636SGreg Clayton case eVarSetOperationClear:
4567cc0636SGreg Clayton Clear();
46332e8b1cSGreg Clayton NotifyValueChanged();
4767cc0636SGreg Clayton break;
4867cc0636SGreg Clayton
4967cc0636SGreg Clayton case eVarSetOperationReplace:
50b9c1b51eSKate Stone // Must be at least one index + 1 pair of paths, and the pair count must be
51b9c1b51eSKate Stone // even
52b9c1b51eSKate Stone if (argc >= 3 && (((argc - 1) & 1) == 0)) {
533a6ba367SMichał Górny uint32_t idx;
5467cc0636SGreg Clayton const uint32_t count = m_path_mappings.GetSize();
553a6ba367SMichał Górny if (!llvm::to_integer(args.GetArgumentAtIndex(0), idx) || idx > count) {
56b9c1b51eSKate Stone error.SetErrorStringWithFormat(
573a6ba367SMichał Górny "invalid file list index %s, index must be 0 through %u",
583a6ba367SMichał Górny args.GetArgumentAtIndex(0), count);
59b9c1b51eSKate Stone } else {
606a311641SGreg Clayton bool changed = false;
61ca47ac3dSWalter Erquinigo for (size_t i = 1; i < argc; idx++, i += 2) {
626a311641SGreg Clayton const char *orginal_path = args.GetArgumentAtIndex(i);
636a311641SGreg Clayton const char *replace_path = args.GetArgumentAtIndex(i + 1);
64b9c1b51eSKate Stone if (VerifyPathExists(replace_path)) {
65dfd499a6SXu Jun if (!m_path_mappings.Replace(orginal_path, replace_path, idx,
66dfd499a6SXu Jun m_notify_changes))
67dfd499a6SXu Jun m_path_mappings.Append(orginal_path, replace_path,
68dfd499a6SXu Jun m_notify_changes);
696a311641SGreg Clayton changed = true;
70b9c1b51eSKate Stone } else {
7130350c25SWalter Erquinigo std::string previousError =
7230350c25SWalter Erquinigo error.Fail() ? std::string(error.AsCString()) + "\n" : "";
73b9c1b51eSKate Stone error.SetErrorStringWithFormat(
7430350c25SWalter Erquinigo "%sthe replacement path doesn't exist: \"%s\"",
7530350c25SWalter Erquinigo previousError.c_str(), replace_path);
766a311641SGreg Clayton }
776a311641SGreg Clayton }
786a311641SGreg Clayton if (changed)
79332e8b1cSGreg Clayton NotifyValueChanged();
8067cc0636SGreg Clayton }
81b9c1b51eSKate Stone } else {
82b9c1b51eSKate Stone error.SetErrorString("replace operation takes an array index followed by "
83b9c1b51eSKate Stone "one or more path pairs");
8467cc0636SGreg Clayton }
8567cc0636SGreg Clayton break;
8667cc0636SGreg Clayton
8767cc0636SGreg Clayton case eVarSetOperationAssign:
88b9c1b51eSKate Stone if (argc < 2 || (argc & 1)) {
8967cc0636SGreg Clayton error.SetErrorString("assign operation takes one or more path pairs");
9067cc0636SGreg Clayton break;
9167cc0636SGreg Clayton }
9267cc0636SGreg Clayton m_path_mappings.Clear(m_notify_changes);
9367cc0636SGreg Clayton // Fall through to append case
9462e0681aSJason Molenda LLVM_FALLTHROUGH;
9567cc0636SGreg Clayton case eVarSetOperationAppend:
96b9c1b51eSKate Stone if (argc < 2 || (argc & 1)) {
9767cc0636SGreg Clayton error.SetErrorString("append operation takes one or more path pairs");
9867cc0636SGreg Clayton break;
99b9c1b51eSKate Stone } else {
1006a311641SGreg Clayton bool changed = false;
101b9c1b51eSKate Stone for (size_t i = 0; i < argc; i += 2) {
1026a311641SGreg Clayton const char *orginal_path = args.GetArgumentAtIndex(i);
1036a311641SGreg Clayton const char *replace_path = args.GetArgumentAtIndex(i + 1);
104b9c1b51eSKate Stone if (VerifyPathExists(replace_path)) {
105dfd499a6SXu Jun m_path_mappings.Append(orginal_path, replace_path, m_notify_changes);
10667cc0636SGreg Clayton m_value_was_set = true;
1076a311641SGreg Clayton changed = true;
108b9c1b51eSKate Stone } else {
10930350c25SWalter Erquinigo std::string previousError =
11030350c25SWalter Erquinigo error.Fail() ? std::string(error.AsCString()) + "\n" : "";
111b9c1b51eSKate Stone error.SetErrorStringWithFormat(
11230350c25SWalter Erquinigo "%sthe replacement path doesn't exist: \"%s\"",
11330350c25SWalter Erquinigo previousError.c_str(), replace_path);
1146a311641SGreg Clayton }
1156a311641SGreg Clayton }
1166a311641SGreg Clayton if (changed)
117332e8b1cSGreg Clayton NotifyValueChanged();
11867cc0636SGreg Clayton }
11967cc0636SGreg Clayton break;
12067cc0636SGreg Clayton
12167cc0636SGreg Clayton case eVarSetOperationInsertBefore:
12267cc0636SGreg Clayton case eVarSetOperationInsertAfter:
123b9c1b51eSKate Stone // Must be at least one index + 1 pair of paths, and the pair count must be
124b9c1b51eSKate Stone // even
125b9c1b51eSKate Stone if (argc >= 3 && (((argc - 1) & 1) == 0)) {
1263a6ba367SMichał Górny uint32_t idx;
12767cc0636SGreg Clayton const uint32_t count = m_path_mappings.GetSize();
1283a6ba367SMichał Górny if (!llvm::to_integer(args.GetArgumentAtIndex(0), idx) || idx > count) {
129b9c1b51eSKate Stone error.SetErrorStringWithFormat(
1303a6ba367SMichał Górny "invalid file list index %s, index must be 0 through %u",
1313a6ba367SMichał Górny args.GetArgumentAtIndex(0), count);
132b9c1b51eSKate Stone } else {
1336a311641SGreg Clayton bool changed = false;
13467cc0636SGreg Clayton if (op == eVarSetOperationInsertAfter)
13567cc0636SGreg Clayton ++idx;
13630350c25SWalter Erquinigo for (size_t i = 1; i < argc; i += 2) {
1376a311641SGreg Clayton const char *orginal_path = args.GetArgumentAtIndex(i);
1386a311641SGreg Clayton const char *replace_path = args.GetArgumentAtIndex(i + 1);
139b9c1b51eSKate Stone if (VerifyPathExists(replace_path)) {
140dfd499a6SXu Jun m_path_mappings.Insert(orginal_path, replace_path, idx,
141dfd499a6SXu Jun m_notify_changes);
1426a311641SGreg Clayton changed = true;
14330350c25SWalter Erquinigo idx++;
144b9c1b51eSKate Stone } else {
14530350c25SWalter Erquinigo std::string previousError =
14630350c25SWalter Erquinigo error.Fail() ? std::string(error.AsCString()) + "\n" : "";
147b9c1b51eSKate Stone error.SetErrorStringWithFormat(
14830350c25SWalter Erquinigo "%sthe replacement path doesn't exist: \"%s\"",
14930350c25SWalter Erquinigo previousError.c_str(), replace_path);
1506a311641SGreg Clayton }
1516a311641SGreg Clayton }
1526a311641SGreg Clayton if (changed)
153332e8b1cSGreg Clayton NotifyValueChanged();
15467cc0636SGreg Clayton }
155b9c1b51eSKate Stone } else {
156b9c1b51eSKate Stone error.SetErrorString("insert operation takes an array index followed by "
157b9c1b51eSKate Stone "one or more path pairs");
15867cc0636SGreg Clayton }
15967cc0636SGreg Clayton break;
16067cc0636SGreg Clayton
16167cc0636SGreg Clayton case eVarSetOperationRemove:
162b9c1b51eSKate Stone if (argc > 0) {
16367cc0636SGreg Clayton std::vector<int> remove_indexes;
164ca47ac3dSWalter Erquinigo for (size_t i = 0; i < argc; ++i) {
1653a6ba367SMichał Górny int idx;
1663a6ba367SMichał Górny if (!llvm::to_integer(args.GetArgumentAtIndex(i), idx) || idx < 0 ||
1673a6ba367SMichał Górny idx >= (int)m_path_mappings.GetSize()) {
168b9c1b51eSKate Stone error.SetErrorStringWithFormat(
169b9c1b51eSKate Stone "invalid array index '%s', aborting remove operation",
170b9c1b51eSKate Stone args.GetArgumentAtIndex(i));
171ca47ac3dSWalter Erquinigo break;
172ca47ac3dSWalter Erquinigo } else
173ca47ac3dSWalter Erquinigo remove_indexes.push_back(idx);
17467cc0636SGreg Clayton }
175ca47ac3dSWalter Erquinigo
176ca47ac3dSWalter Erquinigo // Sort and then erase in reverse so indexes are always valid
177*cd9a5cfdSDmitri Gribenko llvm::sort(remove_indexes);
178ca47ac3dSWalter Erquinigo for (auto index : llvm::reverse(remove_indexes))
179ca47ac3dSWalter Erquinigo m_path_mappings.Remove(index, m_notify_changes);
180ca47ac3dSWalter Erquinigo NotifyValueChanged();
181b9c1b51eSKate Stone } else {
18267cc0636SGreg Clayton error.SetErrorString("remove operation takes one or more array index");
18367cc0636SGreg Clayton }
18467cc0636SGreg Clayton break;
18567cc0636SGreg Clayton
18667cc0636SGreg Clayton case eVarSetOperationInvalid:
187c95f7e2aSPavel Labath error = OptionValue::SetValueFromString(value, op);
18867cc0636SGreg Clayton break;
18967cc0636SGreg Clayton }
19067cc0636SGreg Clayton return error;
19167cc0636SGreg Clayton }
192