178a685aaSJim Ingham //===-- ValueObjectDynamicValue.cpp ---------------------------------*- C++ -*-===// 278a685aaSJim Ingham // 378a685aaSJim Ingham // The LLVM Compiler Infrastructure 478a685aaSJim Ingham // 578a685aaSJim Ingham // This file is distributed under the University of Illinois Open Source 678a685aaSJim Ingham // License. See LICENSE.TXT for details. 778a685aaSJim Ingham // 878a685aaSJim Ingham //===----------------------------------------------------------------------===// 978a685aaSJim Ingham 1078a685aaSJim Ingham 1178a685aaSJim Ingham #include "lldb/Core/ValueObjectDynamicValue.h" 1278a685aaSJim Ingham 1378a685aaSJim Ingham // C Includes 1478a685aaSJim Ingham // C++ Includes 1578a685aaSJim Ingham // Other libraries and framework includes 1678a685aaSJim Ingham // Project includes 17d228483dSEnrico Granata #include "lldb/Core/Log.h" 1878a685aaSJim Ingham #include "lldb/Core/Module.h" 1978a685aaSJim Ingham #include "lldb/Core/ValueObjectList.h" 2078a685aaSJim Ingham #include "lldb/Core/Value.h" 2178a685aaSJim Ingham #include "lldb/Core/ValueObject.h" 2278a685aaSJim Ingham 2321fd13f9SEnrico Granata #include "lldb/Symbol/ClangASTType.h" 2478a685aaSJim Ingham #include "lldb/Symbol/ObjectFile.h" 2578a685aaSJim Ingham #include "lldb/Symbol/SymbolContext.h" 2678a685aaSJim Ingham #include "lldb/Symbol/Type.h" 2778a685aaSJim Ingham #include "lldb/Symbol/Variable.h" 2878a685aaSJim Ingham 2978a685aaSJim Ingham #include "lldb/Target/ExecutionContext.h" 3078a685aaSJim Ingham #include "lldb/Target/LanguageRuntime.h" 3178a685aaSJim Ingham #include "lldb/Target/Process.h" 3278a685aaSJim Ingham #include "lldb/Target/RegisterContext.h" 3378a685aaSJim Ingham #include "lldb/Target/Target.h" 3478a685aaSJim Ingham #include "lldb/Target/Thread.h" 3578a685aaSJim Ingham 3678a685aaSJim Ingham using namespace lldb_private; 3778a685aaSJim Ingham 382837b766SJim Ingham ValueObjectDynamicValue::ValueObjectDynamicValue (ValueObject &parent, lldb::DynamicValueType use_dynamic) : 3978a685aaSJim Ingham ValueObject(parent), 4078a685aaSJim Ingham m_address (), 41f7b1a34eSEnrico Granata m_dynamic_type_info(), 422837b766SJim Ingham m_use_dynamic (use_dynamic) 4378a685aaSJim Ingham { 446f3533fbSEnrico Granata SetName (parent.GetName()); 4578a685aaSJim Ingham } 4678a685aaSJim Ingham 4778a685aaSJim Ingham ValueObjectDynamicValue::~ValueObjectDynamicValue() 4878a685aaSJim Ingham { 4978a685aaSJim Ingham m_owning_valobj_sp.reset(); 5078a685aaSJim Ingham } 5178a685aaSJim Ingham 5257ee3067SGreg Clayton ClangASTType 537277284fSSean Callanan ValueObjectDynamicValue::GetClangTypeImpl () 5478a685aaSJim Ingham { 55dc4db5a6SEnrico Granata const bool success = UpdateValueIfNeeded(false); 56dc4db5a6SEnrico Granata if (success) 57dc4db5a6SEnrico Granata { 58dc4db5a6SEnrico Granata if (m_dynamic_type_info.HasType()) 5978a685aaSJim Ingham return m_value.GetClangType(); 6078a685aaSJim Ingham else 6178a685aaSJim Ingham return m_parent->GetClangType(); 6278a685aaSJim Ingham } 63dc4db5a6SEnrico Granata return m_parent->GetClangType(); 64dc4db5a6SEnrico Granata } 6578a685aaSJim Ingham 6678a685aaSJim Ingham ConstString 6778a685aaSJim Ingham ValueObjectDynamicValue::GetTypeName() 6878a685aaSJim Ingham { 69c3e320a7SEnrico Granata const bool success = UpdateValueIfNeeded(false); 70f7b1a34eSEnrico Granata if (success) 71f7b1a34eSEnrico Granata { 72f7b1a34eSEnrico Granata if (m_dynamic_type_info.HasName()) 73f7b1a34eSEnrico Granata return m_dynamic_type_info.GetName(); 74f7b1a34eSEnrico Granata } 75f7b1a34eSEnrico Granata return m_parent->GetTypeName(); 76f7b1a34eSEnrico Granata } 77f7b1a34eSEnrico Granata 78dc4db5a6SEnrico Granata TypeImpl 79dc4db5a6SEnrico Granata ValueObjectDynamicValue::GetTypeImpl () 80dc4db5a6SEnrico Granata { 81dc4db5a6SEnrico Granata const bool success = UpdateValueIfNeeded(false); 82df7c7f99SEnrico Granata if (success && m_type_impl.IsValid()) 83dc4db5a6SEnrico Granata { 84dc4db5a6SEnrico Granata return m_type_impl; 85dc4db5a6SEnrico Granata } 86dc4db5a6SEnrico Granata return m_parent->GetTypeImpl(); 87dc4db5a6SEnrico Granata } 88dc4db5a6SEnrico Granata 89f7b1a34eSEnrico Granata ConstString 90f7b1a34eSEnrico Granata ValueObjectDynamicValue::GetQualifiedTypeName() 91f7b1a34eSEnrico Granata { 92f7b1a34eSEnrico Granata const bool success = UpdateValueIfNeeded(false); 93f7b1a34eSEnrico Granata if (success) 94f7b1a34eSEnrico Granata { 95f7b1a34eSEnrico Granata if (m_dynamic_type_info.HasName()) 96f7b1a34eSEnrico Granata return m_dynamic_type_info.GetName(); 97f7b1a34eSEnrico Granata } 98e8daa2f8SEnrico Granata return m_parent->GetQualifiedTypeName(); 99e8daa2f8SEnrico Granata } 100e8daa2f8SEnrico Granata 101e8daa2f8SEnrico Granata ConstString 102e8daa2f8SEnrico Granata ValueObjectDynamicValue::GetDisplayTypeName() 103e8daa2f8SEnrico Granata { 104e8daa2f8SEnrico Granata const bool success = UpdateValueIfNeeded(false); 105e8daa2f8SEnrico Granata if (success) 106e8daa2f8SEnrico Granata { 107e8daa2f8SEnrico Granata if (m_dynamic_type_info.HasType()) 108e8daa2f8SEnrico Granata return GetClangType().GetDisplayTypeName(); 109e8daa2f8SEnrico Granata if (m_dynamic_type_info.HasName()) 110e8daa2f8SEnrico Granata return m_dynamic_type_info.GetName(); 111e8daa2f8SEnrico Granata } 112e8daa2f8SEnrico Granata return m_parent->GetDisplayTypeName(); 11378a685aaSJim Ingham } 11478a685aaSJim Ingham 115c7bece56SGreg Clayton size_t 11678a685aaSJim Ingham ValueObjectDynamicValue::CalculateNumChildren() 11778a685aaSJim Ingham { 118c3e320a7SEnrico Granata const bool success = UpdateValueIfNeeded(false); 119dc4db5a6SEnrico Granata if (success && m_dynamic_type_info.HasType()) 12057ee3067SGreg Clayton return GetClangType().GetNumChildren (true); 12178a685aaSJim Ingham else 12278a685aaSJim Ingham return m_parent->GetNumChildren(); 12378a685aaSJim Ingham } 12478a685aaSJim Ingham 125faac1118SGreg Clayton uint64_t 12678a685aaSJim Ingham ValueObjectDynamicValue::GetByteSize() 12778a685aaSJim Ingham { 128c3e320a7SEnrico Granata const bool success = UpdateValueIfNeeded(false); 129dc4db5a6SEnrico Granata if (success && m_dynamic_type_info.HasType()) 13057ee3067SGreg Clayton return m_value.GetValueByteSize(NULL); 13178a685aaSJim Ingham else 13278a685aaSJim Ingham return m_parent->GetByteSize(); 13378a685aaSJim Ingham } 13478a685aaSJim Ingham 13578a685aaSJim Ingham lldb::ValueType 13678a685aaSJim Ingham ValueObjectDynamicValue::GetValueType() const 13778a685aaSJim Ingham { 13878a685aaSJim Ingham return m_parent->GetValueType(); 13978a685aaSJim Ingham } 14078a685aaSJim Ingham 141dc4db5a6SEnrico Granata 142dc4db5a6SEnrico Granata static TypeAndOrName 143dc4db5a6SEnrico Granata FixupTypeAndOrName (const TypeAndOrName& type_andor_name, 144dc4db5a6SEnrico Granata ValueObject& parent) 145dc4db5a6SEnrico Granata { 146dc4db5a6SEnrico Granata TypeAndOrName ret(type_andor_name); 147dc4db5a6SEnrico Granata if (type_andor_name.HasType()) 148dc4db5a6SEnrico Granata { 149dc4db5a6SEnrico Granata // The type will always be the type of the dynamic object. If our parent's type was a pointer, 150dc4db5a6SEnrico Granata // then our type should be a pointer to the type of the dynamic object. If a reference, then the original type 151dc4db5a6SEnrico Granata // should be okay... 152dc4db5a6SEnrico Granata ClangASTType orig_type = type_andor_name.GetClangASTType(); 153dc4db5a6SEnrico Granata ClangASTType corrected_type = orig_type; 154dc4db5a6SEnrico Granata if (parent.IsPointerType()) 155dc4db5a6SEnrico Granata corrected_type = orig_type.GetPointerType (); 156dc4db5a6SEnrico Granata else if (parent.IsPointerOrReferenceType()) 157dc4db5a6SEnrico Granata corrected_type = orig_type.GetLValueReferenceType (); 158dc4db5a6SEnrico Granata ret.SetClangASTType(corrected_type); 159dc4db5a6SEnrico Granata } 160dc4db5a6SEnrico Granata else /*if (m_dynamic_type_info.HasName())*/ 161dc4db5a6SEnrico Granata { 162dc4db5a6SEnrico Granata // If we are here we need to adjust our dynamic type name to include the correct & or * symbol 163dc4db5a6SEnrico Granata std::string corrected_name (type_andor_name.GetName().GetCString()); 164dc4db5a6SEnrico Granata if (parent.IsPointerType()) 165dc4db5a6SEnrico Granata corrected_name.append(" *"); 166dc4db5a6SEnrico Granata else if (parent.IsPointerOrReferenceType()) 167dc4db5a6SEnrico Granata corrected_name.append(" &"); 168fcf0c4e3SEnrico Granata // the parent type should be a correctly pointer'ed or referenc'ed type 169fcf0c4e3SEnrico Granata ret.SetClangASTType(parent.GetClangType()); 170dc4db5a6SEnrico Granata ret.SetName(corrected_name.c_str()); 171dc4db5a6SEnrico Granata } 172dc4db5a6SEnrico Granata return ret; 173dc4db5a6SEnrico Granata } 174dc4db5a6SEnrico Granata 17578a685aaSJim Ingham bool 17678a685aaSJim Ingham ValueObjectDynamicValue::UpdateValue () 17778a685aaSJim Ingham { 17878a685aaSJim Ingham SetValueIsValid (false); 17978a685aaSJim Ingham m_error.Clear(); 18078a685aaSJim Ingham 181c3e320a7SEnrico Granata if (!m_parent->UpdateValueIfNeeded(false)) 18278a685aaSJim Ingham { 183007d5be6SGreg Clayton // The dynamic value failed to get an error, pass the error along 184007d5be6SGreg Clayton if (m_error.Success() && m_parent->GetError().Fail()) 185007d5be6SGreg Clayton m_error = m_parent->GetError(); 18678a685aaSJim Ingham return false; 18778a685aaSJim Ingham } 18878a685aaSJim Ingham 1892837b766SJim Ingham // Setting our type_sp to NULL will route everything back through our 1902837b766SJim Ingham // parent which is equivalent to not using dynamic values. 1912837b766SJim Ingham if (m_use_dynamic == lldb::eNoDynamicValues) 1922837b766SJim Ingham { 193f7b1a34eSEnrico Granata m_dynamic_type_info.Clear(); 1942837b766SJim Ingham return true; 1952837b766SJim Ingham } 1962837b766SJim Ingham 197cc4d0146SGreg Clayton ExecutionContext exe_ctx (GetExecutionContextRef()); 198c14ee32dSGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 199c14ee32dSGreg Clayton if (target) 20078a685aaSJim Ingham { 201c14ee32dSGreg Clayton m_data.SetByteOrder(target->GetArchitecture().GetByteOrder()); 202c14ee32dSGreg Clayton m_data.SetAddressByteSize(target->GetArchitecture().GetAddressByteSize()); 20378a685aaSJim Ingham } 20478a685aaSJim Ingham 20578a685aaSJim Ingham // First make sure our Type and/or Address haven't changed: 206cc4d0146SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 20778a685aaSJim Ingham if (!process) 20878a685aaSJim Ingham return false; 20978a685aaSJim Ingham 21061be0903SJim Ingham TypeAndOrName class_type_or_name; 21178a685aaSJim Ingham Address dynamic_address; 21278a685aaSJim Ingham bool found_dynamic_type = false; 21378a685aaSJim Ingham 21478a685aaSJim Ingham lldb::LanguageType known_type = m_parent->GetObjectRuntimeLanguage(); 21578a685aaSJim Ingham if (known_type != lldb::eLanguageTypeUnknown && known_type != lldb::eLanguageTypeC) 21678a685aaSJim Ingham { 21778a685aaSJim Ingham LanguageRuntime *runtime = process->GetLanguageRuntime (known_type); 21878a685aaSJim Ingham if (runtime) 2192837b766SJim Ingham found_dynamic_type = runtime->GetDynamicTypeAndAddress (*m_parent, m_use_dynamic, class_type_or_name, dynamic_address); 22078a685aaSJim Ingham } 22178a685aaSJim Ingham else 22278a685aaSJim Ingham { 22378a685aaSJim Ingham LanguageRuntime *cpp_runtime = process->GetLanguageRuntime (lldb::eLanguageTypeC_plus_plus); 22478a685aaSJim Ingham if (cpp_runtime) 2252837b766SJim Ingham found_dynamic_type = cpp_runtime->GetDynamicTypeAndAddress (*m_parent, m_use_dynamic, class_type_or_name, dynamic_address); 22678a685aaSJim Ingham 22778a685aaSJim Ingham if (!found_dynamic_type) 22878a685aaSJim Ingham { 22978a685aaSJim Ingham LanguageRuntime *objc_runtime = process->GetLanguageRuntime (lldb::eLanguageTypeObjC); 23078a685aaSJim Ingham if (objc_runtime) 2319910bc85SEnrico Granata found_dynamic_type = objc_runtime->GetDynamicTypeAndAddress (*m_parent, m_use_dynamic, class_type_or_name, dynamic_address); 23278a685aaSJim Ingham } 23378a685aaSJim Ingham } 23478a685aaSJim Ingham 23561be0903SJim Ingham // Getting the dynamic value may have run the program a bit, and so marked us as needing updating, but we really 23661be0903SJim Ingham // don't... 23761be0903SJim Ingham 23861be0903SJim Ingham m_update_point.SetUpdated(); 23961be0903SJim Ingham 240df7c7f99SEnrico Granata if (found_dynamic_type) 241df7c7f99SEnrico Granata { 242df7c7f99SEnrico Granata if (class_type_or_name.HasType()) 243df7c7f99SEnrico Granata { 244df7c7f99SEnrico Granata // TypeSP are always generated from debug info 245df7c7f99SEnrico Granata if (!class_type_or_name.HasTypeSP() && class_type_or_name.GetClangASTType().IsRuntimeGeneratedType()) 246dc4db5a6SEnrico Granata { 247dc4db5a6SEnrico Granata m_type_impl = TypeImpl(m_parent->GetClangType(),FixupTypeAndOrName(class_type_or_name, *m_parent).GetClangASTType()); 248dc4db5a6SEnrico Granata class_type_or_name.SetClangASTType(ClangASTType()); 249dc4db5a6SEnrico Granata } 250df7c7f99SEnrico Granata else 251df7c7f99SEnrico Granata { 2529aceaa1bSEnrico Granata m_type_impl = TypeImpl(m_parent->GetClangType(),FixupTypeAndOrName(class_type_or_name, *m_parent).GetClangASTType()); 253df7c7f99SEnrico Granata } 254df7c7f99SEnrico Granata } 255df7c7f99SEnrico Granata else 256df7c7f99SEnrico Granata { 257df7c7f99SEnrico Granata m_type_impl.Clear(); 258df7c7f99SEnrico Granata } 259df7c7f99SEnrico Granata } 260df7c7f99SEnrico Granata else 261df7c7f99SEnrico Granata { 262df7c7f99SEnrico Granata m_type_impl.Clear(); 263df7c7f99SEnrico Granata } 264dc4db5a6SEnrico Granata 26578a685aaSJim Ingham // If we don't have a dynamic type, then make ourselves just a echo of our parent. 26678a685aaSJim Ingham // Or we could return false, and make ourselves an echo of our parent? 26778a685aaSJim Ingham if (!found_dynamic_type) 26878a685aaSJim Ingham { 269f7b1a34eSEnrico Granata if (m_dynamic_type_info) 27075badc46SEnrico Granata SetValueDidChange(true); 271bd83b87dSEnrico Granata ClearDynamicTypeInformation(); 272f7b1a34eSEnrico Granata m_dynamic_type_info.Clear(); 27378a685aaSJim Ingham m_value = m_parent->GetValue(); 27457ee3067SGreg Clayton m_error = m_value.GetValueAsData (&exe_ctx, m_data, 0, GetModule().get()); 27578a685aaSJim Ingham return m_error.Success(); 27678a685aaSJim Ingham } 27778a685aaSJim Ingham 27878a685aaSJim Ingham Value old_value(m_value); 27978a685aaSJim Ingham 2805160ce5cSGreg Clayton Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 281d228483dSEnrico Granata 282e3e91517SEnrico Granata bool has_changed_type = false; 283e3e91517SEnrico Granata 284f7b1a34eSEnrico Granata if (!m_dynamic_type_info) 28578a685aaSJim Ingham { 286f7b1a34eSEnrico Granata m_dynamic_type_info = class_type_or_name; 287e3e91517SEnrico Granata has_changed_type = true; 28878a685aaSJim Ingham } 289f7b1a34eSEnrico Granata else if (class_type_or_name != m_dynamic_type_info) 29078a685aaSJim Ingham { 29178a685aaSJim Ingham // We are another type, we need to tear down our children... 292f7b1a34eSEnrico Granata m_dynamic_type_info = class_type_or_name; 29378a685aaSJim Ingham SetValueDidChange (true); 294e3e91517SEnrico Granata has_changed_type = true; 29578a685aaSJim Ingham } 29678a685aaSJim Ingham 297e3e91517SEnrico Granata if (has_changed_type) 298e3e91517SEnrico Granata ClearDynamicTypeInformation (); 299e3e91517SEnrico Granata 30078a685aaSJim Ingham if (!m_address.IsValid() || m_address != dynamic_address) 30178a685aaSJim Ingham { 30278a685aaSJim Ingham if (m_address.IsValid()) 30378a685aaSJim Ingham SetValueDidChange (true); 30478a685aaSJim Ingham 30578a685aaSJim Ingham // We've moved, so we should be fine... 30678a685aaSJim Ingham m_address = dynamic_address; 307cc4d0146SGreg Clayton lldb::TargetSP target_sp (GetTargetSP()); 308cc4d0146SGreg Clayton lldb::addr_t load_address = m_address.GetLoadAddress(target_sp.get()); 30978a685aaSJim Ingham m_value.GetScalar() = load_address; 31078a685aaSJim Ingham } 31178a685aaSJim Ingham 312dc4db5a6SEnrico Granata m_dynamic_type_info = FixupTypeAndOrName(m_dynamic_type_info, *m_parent); 31378a685aaSJim Ingham 31457ee3067SGreg Clayton //m_value.SetContext (Value::eContextTypeClangType, corrected_type); 315dc4db5a6SEnrico Granata m_value.SetClangType (m_dynamic_type_info.GetClangASTType()); 31678a685aaSJim Ingham 31778a685aaSJim Ingham // Our address is the location of the dynamic type stored in memory. It isn't a load address, 31878a685aaSJim Ingham // because we aren't pointing to the LOCATION that stores the pointer to us, we're pointing to us... 31978a685aaSJim Ingham m_value.SetValueType(Value::eValueTypeScalar); 32078a685aaSJim Ingham 321e3e91517SEnrico Granata if (has_changed_type && log) 322324a1036SSaleem Abdulrasool log->Printf("[%s %p] has a new dynamic type %s", GetName().GetCString(), 323324a1036SSaleem Abdulrasool static_cast<void*>(this), GetTypeName().GetCString()); 324e3e91517SEnrico Granata 325f7b1a34eSEnrico Granata if (m_address.IsValid() && m_dynamic_type_info) 32678a685aaSJim Ingham { 32778a685aaSJim Ingham // The variable value is in the Scalar value inside the m_value. 32878a685aaSJim Ingham // We can point our m_data right to it. 32957ee3067SGreg Clayton m_error = m_value.GetValueAsData (&exe_ctx, m_data, 0, GetModule().get()); 33078a685aaSJim Ingham if (m_error.Success()) 33178a685aaSJim Ingham { 332*d07cfd3aSEnrico Granata if (!CanProvideValue()) 33378a685aaSJim Ingham { 33478a685aaSJim Ingham // this value object represents an aggregate type whose 33578a685aaSJim Ingham // children have values, but this object does not. So we 33678a685aaSJim Ingham // say we are changed if our location has changed. 33778a685aaSJim Ingham SetValueDidChange (m_value.GetValueType() != old_value.GetValueType() || m_value.GetScalar() != old_value.GetScalar()); 33878a685aaSJim Ingham } 33978a685aaSJim Ingham 34078a685aaSJim Ingham SetValueIsValid (true); 34178a685aaSJim Ingham return true; 34278a685aaSJim Ingham } 34378a685aaSJim Ingham } 34478a685aaSJim Ingham 34578a685aaSJim Ingham // We get here if we've failed above... 34678a685aaSJim Ingham SetValueIsValid (false); 34778a685aaSJim Ingham return false; 34878a685aaSJim Ingham } 34978a685aaSJim Ingham 35078a685aaSJim Ingham 35178a685aaSJim Ingham 35278a685aaSJim Ingham bool 35378a685aaSJim Ingham ValueObjectDynamicValue::IsInScope () 35478a685aaSJim Ingham { 35578a685aaSJim Ingham return m_parent->IsInScope(); 35678a685aaSJim Ingham } 35778a685aaSJim Ingham 35807a4ac22SEnrico Granata bool 35907a4ac22SEnrico Granata ValueObjectDynamicValue::SetValueFromCString (const char *value_str, Error& error) 36007a4ac22SEnrico Granata { 36107a4ac22SEnrico Granata if (!UpdateValueIfNeeded(false)) 36207a4ac22SEnrico Granata { 36307a4ac22SEnrico Granata error.SetErrorString("unable to read value"); 36407a4ac22SEnrico Granata return false; 36507a4ac22SEnrico Granata } 36607a4ac22SEnrico Granata 36707a4ac22SEnrico Granata uint64_t my_value = GetValueAsUnsigned(UINT64_MAX); 36807a4ac22SEnrico Granata uint64_t parent_value = m_parent->GetValueAsUnsigned(UINT64_MAX); 36907a4ac22SEnrico Granata 37007a4ac22SEnrico Granata if (my_value == UINT64_MAX || parent_value == UINT64_MAX) 37107a4ac22SEnrico Granata { 37207a4ac22SEnrico Granata error.SetErrorString("unable to read value"); 37307a4ac22SEnrico Granata return false; 37407a4ac22SEnrico Granata } 37507a4ac22SEnrico Granata 37607a4ac22SEnrico Granata // if we are at an offset from our parent, in order to set ourselves correctly we would need 37707a4ac22SEnrico Granata // to change the new value so that it refers to the correct dynamic type. we choose not to deal 37807a4ac22SEnrico Granata // with that - if anything more than a value overwrite is required, you should be using the 37907a4ac22SEnrico Granata // expression parser instead of the value editing facility 38007a4ac22SEnrico Granata if (my_value != parent_value) 38107a4ac22SEnrico Granata { 38207a4ac22SEnrico Granata // but NULL'ing out a value should always be allowed 38307a4ac22SEnrico Granata if (strcmp(value_str,"0")) 38407a4ac22SEnrico Granata { 38507a4ac22SEnrico Granata error.SetErrorString("unable to modify dynamic value, use 'expression' command"); 38607a4ac22SEnrico Granata return false; 38707a4ac22SEnrico Granata } 38807a4ac22SEnrico Granata } 38907a4ac22SEnrico Granata 39007a4ac22SEnrico Granata bool ret_val = m_parent->SetValueFromCString(value_str,error); 39107a4ac22SEnrico Granata SetNeedsUpdate(); 39207a4ac22SEnrico Granata return ret_val; 39307a4ac22SEnrico Granata } 394389823e9SSean Callanan 395389823e9SSean Callanan bool 396389823e9SSean Callanan ValueObjectDynamicValue::SetData (DataExtractor &data, Error &error) 397389823e9SSean Callanan { 398389823e9SSean Callanan if (!UpdateValueIfNeeded(false)) 399389823e9SSean Callanan { 400389823e9SSean Callanan error.SetErrorString("unable to read value"); 401389823e9SSean Callanan return false; 402389823e9SSean Callanan } 403389823e9SSean Callanan 404389823e9SSean Callanan uint64_t my_value = GetValueAsUnsigned(UINT64_MAX); 405389823e9SSean Callanan uint64_t parent_value = m_parent->GetValueAsUnsigned(UINT64_MAX); 406389823e9SSean Callanan 407389823e9SSean Callanan if (my_value == UINT64_MAX || parent_value == UINT64_MAX) 408389823e9SSean Callanan { 409389823e9SSean Callanan error.SetErrorString("unable to read value"); 410389823e9SSean Callanan return false; 411389823e9SSean Callanan } 412389823e9SSean Callanan 413389823e9SSean Callanan // if we are at an offset from our parent, in order to set ourselves correctly we would need 414389823e9SSean Callanan // to change the new value so that it refers to the correct dynamic type. we choose not to deal 415389823e9SSean Callanan // with that - if anything more than a value overwrite is required, you should be using the 416389823e9SSean Callanan // expression parser instead of the value editing facility 417389823e9SSean Callanan if (my_value != parent_value) 418389823e9SSean Callanan { 419389823e9SSean Callanan // but NULL'ing out a value should always be allowed 420389823e9SSean Callanan lldb::offset_t offset = 0; 421389823e9SSean Callanan 422389823e9SSean Callanan if (data.GetPointer(&offset) != 0) 423389823e9SSean Callanan { 424389823e9SSean Callanan error.SetErrorString("unable to modify dynamic value, use 'expression' command"); 425389823e9SSean Callanan return false; 426389823e9SSean Callanan } 427389823e9SSean Callanan } 428389823e9SSean Callanan 429389823e9SSean Callanan bool ret_val = m_parent->SetData(data, error); 430389823e9SSean Callanan SetNeedsUpdate(); 431389823e9SSean Callanan return ret_val; 432389823e9SSean Callanan } 433