1 //===-- ValueObjectMemory.cpp ---------------------------------*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 10 11 #include "lldb/Core/ValueObjectMemory.h" 12 13 // C Includes 14 // C++ Includes 15 // Other libraries and framework includes 16 // Project includes 17 #include "lldb/Core/Module.h" 18 #include "lldb/Core/ValueObjectList.h" 19 #include "lldb/Core/Value.h" 20 #include "lldb/Core/ValueObject.h" 21 22 #include "lldb/Symbol/ObjectFile.h" 23 #include "lldb/Symbol/SymbolContext.h" 24 #include "lldb/Symbol/Type.h" 25 #include "lldb/Symbol/Variable.h" 26 27 #include "lldb/Target/ExecutionContext.h" 28 #include "lldb/Target/Process.h" 29 #include "lldb/Target/RegisterContext.h" 30 #include "lldb/Target/Target.h" 31 #include "lldb/Target/Thread.h" 32 33 using namespace lldb; 34 using namespace lldb_private; 35 36 ValueObjectSP 37 ValueObjectMemory::Create (ExecutionContextScope *exe_scope, 38 const char *name, 39 const Address &address, 40 lldb::TypeSP &type_sp) 41 { 42 return (new ValueObjectMemory (exe_scope, name, address, type_sp))->GetSP(); 43 } 44 45 ValueObjectSP 46 ValueObjectMemory::Create (ExecutionContextScope *exe_scope, 47 const char *name, 48 const Address &address, 49 const ClangASTType &ast_type) 50 { 51 return (new ValueObjectMemory (exe_scope, name, address, ast_type))->GetSP(); 52 } 53 54 ValueObjectMemory::ValueObjectMemory (ExecutionContextScope *exe_scope, 55 const char *name, 56 const Address &address, 57 lldb::TypeSP &type_sp) : 58 ValueObject(exe_scope), 59 m_address (address), 60 m_type_sp(type_sp), 61 m_clang_type() 62 { 63 // Do not attempt to construct one of these objects with no variable! 64 assert (m_type_sp.get() != NULL); 65 SetName (ConstString(name)); 66 m_value.SetContext(Value::eContextTypeLLDBType, m_type_sp.get()); 67 TargetSP target_sp (GetTargetSP()); 68 lldb::addr_t load_address = m_address.GetLoadAddress(target_sp.get()); 69 if (load_address != LLDB_INVALID_ADDRESS) 70 { 71 m_value.SetValueType(Value::eValueTypeLoadAddress); 72 m_value.GetScalar() = load_address; 73 } 74 else 75 { 76 lldb::addr_t file_address = m_address.GetFileAddress(); 77 if (file_address != LLDB_INVALID_ADDRESS) 78 { 79 m_value.SetValueType(Value::eValueTypeFileAddress); 80 m_value.GetScalar() = file_address; 81 } 82 else 83 { 84 m_value.GetScalar() = m_address.GetOffset(); 85 m_value.SetValueType (Value::eValueTypeScalar); 86 } 87 } 88 } 89 90 ValueObjectMemory::ValueObjectMemory (ExecutionContextScope *exe_scope, 91 const char *name, 92 const Address &address, 93 const ClangASTType &ast_type) : 94 ValueObject(exe_scope), 95 m_address (address), 96 m_type_sp(), 97 m_clang_type(ast_type) 98 { 99 // Do not attempt to construct one of these objects with no variable! 100 assert (m_clang_type.GetASTContext()); 101 assert (m_clang_type.GetOpaqueQualType()); 102 103 TargetSP target_sp (GetTargetSP()); 104 105 SetName (ConstString(name)); 106 m_value.SetContext(Value::eContextTypeClangType, m_clang_type.GetOpaqueQualType()); 107 lldb::addr_t load_address = m_address.GetLoadAddress (target_sp.get()); 108 if (load_address != LLDB_INVALID_ADDRESS) 109 { 110 m_value.SetValueType(Value::eValueTypeLoadAddress); 111 m_value.GetScalar() = load_address; 112 } 113 else 114 { 115 lldb::addr_t file_address = m_address.GetFileAddress(); 116 if (file_address != LLDB_INVALID_ADDRESS) 117 { 118 m_value.SetValueType(Value::eValueTypeFileAddress); 119 m_value.GetScalar() = file_address; 120 } 121 else 122 { 123 m_value.GetScalar() = m_address.GetOffset(); 124 m_value.SetValueType (Value::eValueTypeScalar); 125 } 126 } 127 } 128 129 ValueObjectMemory::~ValueObjectMemory() 130 { 131 } 132 133 lldb::clang_type_t 134 ValueObjectMemory::GetClangTypeImpl () 135 { 136 if (m_type_sp) 137 return m_type_sp->GetClangForwardType(); 138 return m_clang_type.GetOpaqueQualType(); 139 } 140 141 ConstString 142 ValueObjectMemory::GetTypeName() 143 { 144 if (m_type_sp) 145 return m_type_sp->GetName(); 146 return ClangASTType::GetConstTypeName (GetClangAST(), m_clang_type.GetOpaqueQualType()); 147 } 148 149 size_t 150 ValueObjectMemory::CalculateNumChildren() 151 { 152 if (m_type_sp) 153 return m_type_sp->GetNumChildren(true); 154 const bool omit_empty_base_classes = true; 155 return ClangASTContext::GetNumChildren (m_clang_type.GetASTContext(), 156 m_clang_type.GetOpaqueQualType(), 157 omit_empty_base_classes); 158 } 159 160 clang::ASTContext * 161 ValueObjectMemory::GetClangASTImpl () 162 { 163 if (m_type_sp) 164 return m_type_sp->GetClangAST(); 165 return m_clang_type.GetASTContext(); 166 } 167 168 uint64_t 169 ValueObjectMemory::GetByteSize() 170 { 171 if (m_type_sp) 172 return m_type_sp->GetByteSize(); 173 return m_clang_type.GetClangTypeByteSize (); 174 } 175 176 lldb::ValueType 177 ValueObjectMemory::GetValueType() const 178 { 179 // RETHINK: Should this be inherited from somewhere? 180 return lldb::eValueTypeVariableGlobal; 181 } 182 183 bool 184 ValueObjectMemory::UpdateValue () 185 { 186 SetValueIsValid (false); 187 m_error.Clear(); 188 189 ExecutionContext exe_ctx (GetExecutionContextRef()); 190 191 Target *target = exe_ctx.GetTargetPtr(); 192 if (target) 193 { 194 m_data.SetByteOrder(target->GetArchitecture().GetByteOrder()); 195 m_data.SetAddressByteSize(target->GetArchitecture().GetAddressByteSize()); 196 } 197 198 Value old_value(m_value); 199 if (m_address.IsValid()) 200 { 201 Value::ValueType value_type = m_value.GetValueType(); 202 203 switch (value_type) 204 { 205 default: 206 assert(!"Unhandled expression result value kind..."); 207 break; 208 209 case Value::eValueTypeScalar: 210 // The variable value is in the Scalar value inside the m_value. 211 // We can point our m_data right to it. 212 m_error = m_value.GetValueAsData (&exe_ctx, GetClangAST(), m_data, 0, GetModule().get()); 213 break; 214 215 case Value::eValueTypeFileAddress: 216 case Value::eValueTypeLoadAddress: 217 case Value::eValueTypeHostAddress: 218 // The DWARF expression result was an address in the inferior 219 // process. If this variable is an aggregate type, we just need 220 // the address as the main value as all child variable objects 221 // will rely upon this location and add an offset and then read 222 // their own values as needed. If this variable is a simple 223 // type, we read all data for it into m_data. 224 // Make sure this type has a value before we try and read it 225 226 // If we have a file address, convert it to a load address if we can. 227 if (value_type == Value::eValueTypeFileAddress && exe_ctx.GetProcessPtr()) 228 { 229 lldb::addr_t load_addr = m_address.GetLoadAddress(target); 230 if (load_addr != LLDB_INVALID_ADDRESS) 231 { 232 m_value.SetValueType(Value::eValueTypeLoadAddress); 233 m_value.GetScalar() = load_addr; 234 } 235 } 236 237 if (ClangASTContext::IsAggregateType (GetClangType())) 238 { 239 // this value object represents an aggregate type whose 240 // children have values, but this object does not. So we 241 // say we are changed if our location has changed. 242 SetValueDidChange (value_type != old_value.GetValueType() || m_value.GetScalar() != old_value.GetScalar()); 243 } 244 else 245 { 246 // Copy the Value and set the context to use our Variable 247 // so it can extract read its value into m_data appropriately 248 Value value(m_value); 249 if (m_type_sp) 250 value.SetContext(Value::eContextTypeLLDBType, m_type_sp.get()); 251 else 252 value.SetContext(Value::eContextTypeClangType, m_clang_type.GetOpaqueQualType()); 253 254 m_error = value.GetValueAsData(&exe_ctx, GetClangAST(), m_data, 0, GetModule().get()); 255 } 256 break; 257 } 258 259 SetValueIsValid (m_error.Success()); 260 } 261 return m_error.Success(); 262 } 263 264 265 266 bool 267 ValueObjectMemory::IsInScope () 268 { 269 // FIXME: Maybe try to read the memory address, and if that works, then 270 // we are in scope? 271 return true; 272 } 273 274 275 lldb::ModuleSP 276 ValueObjectMemory::GetModule() 277 { 278 return m_address.GetModule(); 279 } 280 281 282