167cc0636SGreg Clayton //===-- OptionValueArray.cpp ------------------------------------*- C++ -*-===// 267cc0636SGreg Clayton // 367cc0636SGreg Clayton // The LLVM Compiler Infrastructure 467cc0636SGreg Clayton // 567cc0636SGreg Clayton // This file is distributed under the University of Illinois Open Source 667cc0636SGreg Clayton // License. See LICENSE.TXT for details. 767cc0636SGreg Clayton // 867cc0636SGreg Clayton //===----------------------------------------------------------------------===// 967cc0636SGreg Clayton 1067cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueArray.h" 1167cc0636SGreg Clayton 1267cc0636SGreg Clayton // C Includes 1367cc0636SGreg Clayton // C++ Includes 1467cc0636SGreg Clayton // Other libraries and framework includes 1567cc0636SGreg Clayton // Project includes 1667cc0636SGreg Clayton #include "lldb/Core/Stream.h" 1767cc0636SGreg Clayton #include "lldb/Interpreter/Args.h" 1867cc0636SGreg Clayton 1967cc0636SGreg Clayton using namespace lldb; 2067cc0636SGreg Clayton using namespace lldb_private; 2167cc0636SGreg Clayton 2267cc0636SGreg Clayton void 2367cc0636SGreg Clayton OptionValueArray::DumpValue (const ExecutionContext *exe_ctx, Stream &strm, uint32_t dump_mask) 2467cc0636SGreg Clayton { 2567cc0636SGreg Clayton const Type array_element_type = ConvertTypeMaskToType (m_type_mask); 2667cc0636SGreg Clayton if (dump_mask & eDumpOptionType) 2767cc0636SGreg Clayton { 2867cc0636SGreg Clayton if ((GetType() == eTypeArray) && (m_type_mask != eTypeInvalid)) 2967cc0636SGreg Clayton strm.Printf ("(%s of %ss)", GetTypeAsCString(), GetBuiltinTypeAsCString(array_element_type)); 3067cc0636SGreg Clayton else 3167cc0636SGreg Clayton strm.Printf ("(%s)", GetTypeAsCString()); 3267cc0636SGreg Clayton } 3367cc0636SGreg Clayton if (dump_mask & eDumpOptionValue) 3467cc0636SGreg Clayton { 3567cc0636SGreg Clayton if (dump_mask & eDumpOptionType) 3667cc0636SGreg Clayton strm.Printf (" =%s", (m_values.size() > 0) ? "\n" : ""); 3767cc0636SGreg Clayton strm.IndentMore(); 3867cc0636SGreg Clayton const uint32_t size = m_values.size(); 3967cc0636SGreg Clayton for (uint32_t i = 0; i<size; ++i) 4067cc0636SGreg Clayton { 4167cc0636SGreg Clayton strm.Indent(); 4267cc0636SGreg Clayton strm.Printf("[%u]: ", i); 4367cc0636SGreg Clayton const uint32_t extra_dump_options = m_raw_value_dump ? eDumpOptionRaw : 0; 4467cc0636SGreg Clayton switch (array_element_type) 4567cc0636SGreg Clayton { 4667cc0636SGreg Clayton default: 4767cc0636SGreg Clayton case eTypeArray: 4867cc0636SGreg Clayton case eTypeDictionary: 4967cc0636SGreg Clayton case eTypeProperties: 5067cc0636SGreg Clayton case eTypeFileSpecList: 5167cc0636SGreg Clayton case eTypePathMap: 5267cc0636SGreg Clayton m_values[i]->DumpValue(exe_ctx, strm, dump_mask | extra_dump_options); 5367cc0636SGreg Clayton break; 5467cc0636SGreg Clayton 5567cc0636SGreg Clayton case eTypeBoolean: 5667cc0636SGreg Clayton case eTypeEnum: 5767cc0636SGreg Clayton case eTypeFileSpec: 5867cc0636SGreg Clayton case eTypeFormat: 5967cc0636SGreg Clayton case eTypeSInt64: 6067cc0636SGreg Clayton case eTypeString: 6167cc0636SGreg Clayton case eTypeUInt64: 6267cc0636SGreg Clayton case eTypeUUID: 6367cc0636SGreg Clayton // No need to show the type for dictionaries of simple items 6467cc0636SGreg Clayton m_values[i]->DumpValue(exe_ctx, strm, (dump_mask & (~eDumpOptionType)) | extra_dump_options); 6567cc0636SGreg Clayton break; 6667cc0636SGreg Clayton } 6767cc0636SGreg Clayton if (i < (size - 1)) 6867cc0636SGreg Clayton strm.EOL(); 6967cc0636SGreg Clayton } 7067cc0636SGreg Clayton strm.IndentLess(); 7167cc0636SGreg Clayton } 7267cc0636SGreg Clayton } 7367cc0636SGreg Clayton 7467cc0636SGreg Clayton Error 7567cc0636SGreg Clayton OptionValueArray::SetValueFromCString (const char *value, VarSetOperationType op) 7667cc0636SGreg Clayton { 7767cc0636SGreg Clayton Args args(value); 7867cc0636SGreg Clayton return SetArgs (args, op); 7967cc0636SGreg Clayton } 8067cc0636SGreg Clayton 8167cc0636SGreg Clayton 8267cc0636SGreg Clayton lldb::OptionValueSP 8367cc0636SGreg Clayton OptionValueArray::GetSubValue (const ExecutionContext *exe_ctx, 8467cc0636SGreg Clayton const char *name, 8567cc0636SGreg Clayton bool will_modify, 8667cc0636SGreg Clayton Error &error) const 8767cc0636SGreg Clayton { 8867cc0636SGreg Clayton if (name && name[0] == '[') 8967cc0636SGreg Clayton { 9067cc0636SGreg Clayton const char *end_bracket = strchr (name+1, ']'); 9167cc0636SGreg Clayton if (end_bracket) 9267cc0636SGreg Clayton { 9367cc0636SGreg Clayton const char *sub_value = NULL; 9467cc0636SGreg Clayton if (end_bracket[1]) 9567cc0636SGreg Clayton sub_value = end_bracket + 1; 9667cc0636SGreg Clayton std::string index_str (name+1, end_bracket); 9767cc0636SGreg Clayton const size_t array_count = m_values.size(); 9867cc0636SGreg Clayton int32_t idx = Args::StringToSInt32(index_str.c_str(), INT32_MAX, 0, NULL); 9967cc0636SGreg Clayton if (idx != INT32_MAX) 10067cc0636SGreg Clayton { 10167cc0636SGreg Clayton ; 10267cc0636SGreg Clayton uint32_t new_idx = UINT32_MAX; 10367cc0636SGreg Clayton if (idx < 0) 10467cc0636SGreg Clayton { 10567cc0636SGreg Clayton // Access from the end of the array if the index is negative 10667cc0636SGreg Clayton new_idx = array_count - idx; 10767cc0636SGreg Clayton } 10867cc0636SGreg Clayton else 10967cc0636SGreg Clayton { 11067cc0636SGreg Clayton // Just a standard index 11167cc0636SGreg Clayton new_idx = idx; 11267cc0636SGreg Clayton } 11367cc0636SGreg Clayton 11467cc0636SGreg Clayton if (new_idx < array_count) 11567cc0636SGreg Clayton { 11667cc0636SGreg Clayton if (m_values[new_idx]) 11767cc0636SGreg Clayton { 11867cc0636SGreg Clayton if (sub_value) 11967cc0636SGreg Clayton return m_values[new_idx]->GetSubValue (exe_ctx, sub_value, will_modify, error); 12067cc0636SGreg Clayton else 12167cc0636SGreg Clayton return m_values[new_idx]; 12267cc0636SGreg Clayton } 12367cc0636SGreg Clayton } 12467cc0636SGreg Clayton else 12567cc0636SGreg Clayton { 12667cc0636SGreg Clayton if (array_count == 0) 12767cc0636SGreg Clayton error.SetErrorStringWithFormat("index %i is not valid for an empty array", idx); 12867cc0636SGreg Clayton else if (idx > 0) 129*d01b2953SDaniel Malea error.SetErrorStringWithFormat("index %i out of range, valid values are 0 through %" PRIu64, idx, (uint64_t)(array_count - 1)); 13067cc0636SGreg Clayton else 131*d01b2953SDaniel Malea error.SetErrorStringWithFormat("negative index %i out of range, valid values are -1 through -%" PRIu64, idx, (uint64_t)array_count); 13267cc0636SGreg Clayton } 13367cc0636SGreg Clayton } 13467cc0636SGreg Clayton } 13567cc0636SGreg Clayton } 13667cc0636SGreg Clayton else 13767cc0636SGreg Clayton { 13867cc0636SGreg Clayton error.SetErrorStringWithFormat("invalid value path '%s', %s values only support '[<index>]' subvalues where <index> is a positive or negative array index", name, GetTypeAsCString()); 13967cc0636SGreg Clayton } 14067cc0636SGreg Clayton return OptionValueSP(); 14167cc0636SGreg Clayton } 14267cc0636SGreg Clayton 14367cc0636SGreg Clayton 14467cc0636SGreg Clayton size_t 14567cc0636SGreg Clayton OptionValueArray::GetArgs (Args &args) const 14667cc0636SGreg Clayton { 14767cc0636SGreg Clayton const uint32_t size = m_values.size(); 14867cc0636SGreg Clayton std::vector<const char *> argv; 14967cc0636SGreg Clayton for (uint32_t i = 0; i<size; ++i) 15067cc0636SGreg Clayton { 15167cc0636SGreg Clayton const char *string_value = m_values[i]->GetStringValue (); 15267cc0636SGreg Clayton if (string_value) 15367cc0636SGreg Clayton argv.push_back(string_value); 15467cc0636SGreg Clayton } 15567cc0636SGreg Clayton 15667cc0636SGreg Clayton if (argv.empty()) 15767cc0636SGreg Clayton args.Clear(); 15867cc0636SGreg Clayton else 15967cc0636SGreg Clayton args.SetArguments(argv.size(), &argv[0]); 16067cc0636SGreg Clayton return args.GetArgumentCount(); 16167cc0636SGreg Clayton } 16267cc0636SGreg Clayton 16367cc0636SGreg Clayton Error 16467cc0636SGreg Clayton OptionValueArray::SetArgs (const Args &args, VarSetOperationType op) 16567cc0636SGreg Clayton { 16667cc0636SGreg Clayton Error error; 16767cc0636SGreg Clayton const size_t argc = args.GetArgumentCount(); 16867cc0636SGreg Clayton switch (op) 16967cc0636SGreg Clayton { 17067cc0636SGreg Clayton case eVarSetOperationInvalid: 17167cc0636SGreg Clayton error.SetErrorString("unsupported operation"); 17267cc0636SGreg Clayton break; 17367cc0636SGreg Clayton 17467cc0636SGreg Clayton case eVarSetOperationInsertBefore: 17567cc0636SGreg Clayton case eVarSetOperationInsertAfter: 17667cc0636SGreg Clayton if (argc > 1) 17767cc0636SGreg Clayton { 17867cc0636SGreg Clayton uint32_t idx = Args::StringToUInt32(args.GetArgumentAtIndex(0), UINT32_MAX); 17967cc0636SGreg Clayton const uint32_t count = GetSize(); 18067cc0636SGreg Clayton if (idx > count) 18167cc0636SGreg Clayton { 18267cc0636SGreg Clayton error.SetErrorStringWithFormat("invalid insert array index %u, index must be 0 through %u", idx, count); 18367cc0636SGreg Clayton } 18467cc0636SGreg Clayton else 18567cc0636SGreg Clayton { 18667cc0636SGreg Clayton if (op == eVarSetOperationInsertAfter) 18767cc0636SGreg Clayton ++idx; 18867cc0636SGreg Clayton for (size_t i=1; i<argc; ++i, ++idx) 18967cc0636SGreg Clayton { 19067cc0636SGreg Clayton lldb::OptionValueSP value_sp (CreateValueFromCStringForTypeMask (args.GetArgumentAtIndex(i), 19167cc0636SGreg Clayton m_type_mask, 19267cc0636SGreg Clayton error)); 19367cc0636SGreg Clayton if (value_sp) 19467cc0636SGreg Clayton { 19567cc0636SGreg Clayton if (error.Fail()) 19667cc0636SGreg Clayton return error; 19767cc0636SGreg Clayton if (idx >= m_values.size()) 19867cc0636SGreg Clayton m_values.push_back(value_sp); 19967cc0636SGreg Clayton else 20067cc0636SGreg Clayton m_values.insert(m_values.begin() + idx, value_sp); 20167cc0636SGreg Clayton } 20267cc0636SGreg Clayton else 20367cc0636SGreg Clayton { 20467cc0636SGreg Clayton error.SetErrorString("array of complex types must subclass OptionValueArray"); 20567cc0636SGreg Clayton return error; 20667cc0636SGreg Clayton } 20767cc0636SGreg Clayton } 20867cc0636SGreg Clayton } 20967cc0636SGreg Clayton } 21067cc0636SGreg Clayton else 21167cc0636SGreg Clayton { 21267cc0636SGreg Clayton error.SetErrorString("insert operation takes an array index followed by one or more values"); 21367cc0636SGreg Clayton } 21467cc0636SGreg Clayton break; 21567cc0636SGreg Clayton 21667cc0636SGreg Clayton case eVarSetOperationRemove: 21767cc0636SGreg Clayton if (argc > 0) 21867cc0636SGreg Clayton { 21967cc0636SGreg Clayton const uint32_t size = m_values.size(); 22067cc0636SGreg Clayton std::vector<int> remove_indexes; 22167cc0636SGreg Clayton bool all_indexes_valid = true; 22267cc0636SGreg Clayton size_t i; 22367cc0636SGreg Clayton for (i=0; i<argc; ++i) 22467cc0636SGreg Clayton { 22567cc0636SGreg Clayton const int idx = Args::StringToSInt32(args.GetArgumentAtIndex(i), INT32_MAX); 22667cc0636SGreg Clayton if (idx >= size) 22767cc0636SGreg Clayton { 22867cc0636SGreg Clayton all_indexes_valid = false; 22967cc0636SGreg Clayton break; 23067cc0636SGreg Clayton } 23167cc0636SGreg Clayton else 23267cc0636SGreg Clayton remove_indexes.push_back(idx); 23367cc0636SGreg Clayton } 23467cc0636SGreg Clayton 23567cc0636SGreg Clayton if (all_indexes_valid) 23667cc0636SGreg Clayton { 23767cc0636SGreg Clayton size_t num_remove_indexes = remove_indexes.size(); 23867cc0636SGreg Clayton if (num_remove_indexes) 23967cc0636SGreg Clayton { 24067cc0636SGreg Clayton // Sort and then erase in reverse so indexes are always valid 24167cc0636SGreg Clayton if (num_remove_indexes > 1) 24267cc0636SGreg Clayton { 24367cc0636SGreg Clayton std::sort(remove_indexes.begin(), remove_indexes.end()); 24467cc0636SGreg Clayton for (std::vector<int>::const_reverse_iterator pos = remove_indexes.rbegin(), end = remove_indexes.rend(); pos != end; ++pos) 24567cc0636SGreg Clayton { 24667cc0636SGreg Clayton m_values.erase(m_values.begin() + *pos); 24767cc0636SGreg Clayton } 24867cc0636SGreg Clayton } 24967cc0636SGreg Clayton else 25067cc0636SGreg Clayton { 25167cc0636SGreg Clayton // Only one index 25267cc0636SGreg Clayton m_values.erase(m_values.begin() + remove_indexes.front()); 25367cc0636SGreg Clayton } 25467cc0636SGreg Clayton } 25567cc0636SGreg Clayton } 25667cc0636SGreg Clayton else 25767cc0636SGreg Clayton { 25867cc0636SGreg Clayton error.SetErrorStringWithFormat("invalid array index '%s', aborting remove operation", args.GetArgumentAtIndex(i)); 25967cc0636SGreg Clayton } 26067cc0636SGreg Clayton } 26167cc0636SGreg Clayton else 26267cc0636SGreg Clayton { 26367cc0636SGreg Clayton error.SetErrorString("remove operation takes one or more array indices"); 26467cc0636SGreg Clayton } 26567cc0636SGreg Clayton break; 26667cc0636SGreg Clayton 26767cc0636SGreg Clayton case eVarSetOperationClear: 26867cc0636SGreg Clayton Clear (); 26967cc0636SGreg Clayton break; 27067cc0636SGreg Clayton 27167cc0636SGreg Clayton case eVarSetOperationReplace: 27267cc0636SGreg Clayton if (argc > 1) 27367cc0636SGreg Clayton { 27467cc0636SGreg Clayton uint32_t idx = Args::StringToUInt32(args.GetArgumentAtIndex(0), UINT32_MAX); 27567cc0636SGreg Clayton const uint32_t count = GetSize(); 27667cc0636SGreg Clayton if (idx > count) 27767cc0636SGreg Clayton { 27867cc0636SGreg Clayton error.SetErrorStringWithFormat("invalid replace array index %u, index must be 0 through %u", idx, count); 27967cc0636SGreg Clayton } 28067cc0636SGreg Clayton else 28167cc0636SGreg Clayton { 28267cc0636SGreg Clayton for (size_t i=1; i<argc; ++i, ++idx) 28367cc0636SGreg Clayton { 28467cc0636SGreg Clayton lldb::OptionValueSP value_sp (CreateValueFromCStringForTypeMask (args.GetArgumentAtIndex(i), 28567cc0636SGreg Clayton m_type_mask, 28667cc0636SGreg Clayton error)); 28767cc0636SGreg Clayton if (value_sp) 28867cc0636SGreg Clayton { 28967cc0636SGreg Clayton if (error.Fail()) 29067cc0636SGreg Clayton return error; 29167cc0636SGreg Clayton if (idx < count) 29267cc0636SGreg Clayton m_values[idx] = value_sp; 29367cc0636SGreg Clayton else 29467cc0636SGreg Clayton m_values.push_back(value_sp); 29567cc0636SGreg Clayton } 29667cc0636SGreg Clayton else 29767cc0636SGreg Clayton { 29867cc0636SGreg Clayton error.SetErrorString("array of complex types must subclass OptionValueArray"); 29967cc0636SGreg Clayton return error; 30067cc0636SGreg Clayton } 30167cc0636SGreg Clayton } 30267cc0636SGreg Clayton } 30367cc0636SGreg Clayton } 30467cc0636SGreg Clayton else 30567cc0636SGreg Clayton { 30667cc0636SGreg Clayton error.SetErrorString("replace operation takes an array index followed by one or more values"); 30767cc0636SGreg Clayton } 30867cc0636SGreg Clayton break; 30967cc0636SGreg Clayton 31067cc0636SGreg Clayton case eVarSetOperationAssign: 31167cc0636SGreg Clayton m_values.clear(); 31267cc0636SGreg Clayton // Fall through to append case 31367cc0636SGreg Clayton case eVarSetOperationAppend: 31467cc0636SGreg Clayton for (size_t i=0; i<argc; ++i) 31567cc0636SGreg Clayton { 31667cc0636SGreg Clayton lldb::OptionValueSP value_sp (CreateValueFromCStringForTypeMask (args.GetArgumentAtIndex(i), 31767cc0636SGreg Clayton m_type_mask, 31867cc0636SGreg Clayton error)); 31967cc0636SGreg Clayton if (value_sp) 32067cc0636SGreg Clayton { 32167cc0636SGreg Clayton if (error.Fail()) 32267cc0636SGreg Clayton return error; 32367cc0636SGreg Clayton m_value_was_set = true; 32467cc0636SGreg Clayton AppendValue(value_sp); 32567cc0636SGreg Clayton } 32667cc0636SGreg Clayton else 32767cc0636SGreg Clayton { 32867cc0636SGreg Clayton error.SetErrorString("array of complex types must subclass OptionValueArray"); 32967cc0636SGreg Clayton } 33067cc0636SGreg Clayton } 33167cc0636SGreg Clayton break; 33267cc0636SGreg Clayton } 33367cc0636SGreg Clayton return error; 33467cc0636SGreg Clayton } 33567cc0636SGreg Clayton 33667cc0636SGreg Clayton lldb::OptionValueSP 33767cc0636SGreg Clayton OptionValueArray::DeepCopy () const 33867cc0636SGreg Clayton { 33967cc0636SGreg Clayton OptionValueArray *copied_array = new OptionValueArray (m_type_mask, m_raw_value_dump); 34067cc0636SGreg Clayton lldb::OptionValueSP copied_value_sp(copied_array); 34167cc0636SGreg Clayton const uint32_t size = m_values.size(); 34267cc0636SGreg Clayton for (uint32_t i = 0; i<size; ++i) 34367cc0636SGreg Clayton { 34467cc0636SGreg Clayton copied_array->AppendValue (m_values[i]->DeepCopy()); 34567cc0636SGreg Clayton } 34667cc0636SGreg Clayton return copied_value_sp; 34767cc0636SGreg Clayton } 34867cc0636SGreg Clayton 34967cc0636SGreg Clayton 35067cc0636SGreg Clayton 351