blob: 92532a3e52c98c67ae334adc3e491c42d12c5fcd [file] [log] [blame]
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001//===-- ValueObjectChild.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#include "lldb/Core/ValueObjectChild.h"
11
Chris Lattner30fdc8d2010-06-08 16:52:24 +000012#include "lldb/Core/Module.h"
13#include "lldb/Core/ValueObjectList.h"
14
Greg Claytone1a916a2010-07-21 22:12:05 +000015#include "lldb/Symbol/ClangASTType.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000016#include "lldb/Symbol/ObjectFile.h"
17#include "lldb/Symbol/SymbolContext.h"
18#include "lldb/Symbol/Type.h"
19#include "lldb/Symbol/Variable.h"
20
21#include "lldb/Target/ExecutionContext.h"
22#include "lldb/Target/Process.h"
23#include "lldb/Target/Target.h"
24
25using namespace lldb_private;
26
27ValueObjectChild::ValueObjectChild
28(
Jim Ingham6035b672011-03-31 00:19:25 +000029 ValueObject &parent,
Chris Lattner30fdc8d2010-06-08 16:52:24 +000030 clang::ASTContext *clang_ast,
31 void *clang_type,
32 const ConstString &name,
33 uint32_t byte_size,
34 int32_t byte_offset,
35 uint32_t bitfield_bit_size,
Greg Clayton8f92f0a2010-10-14 22:52:14 +000036 uint32_t bitfield_bit_offset,
Greg Claytone221f822011-01-21 01:59:00 +000037 bool is_base_class,
Enrico Granata9128ee22011-09-06 19:20:51 +000038 bool is_deref_of_parent,
39 AddressType child_ptr_or_ref_addr_type
Chris Lattner30fdc8d2010-06-08 16:52:24 +000040) :
Greg Clayton8f92f0a2010-10-14 22:52:14 +000041 ValueObject (parent),
Chris Lattner30fdc8d2010-06-08 16:52:24 +000042 m_clang_ast (clang_ast),
43 m_clang_type (clang_type),
44 m_byte_size (byte_size),
45 m_byte_offset (byte_offset),
46 m_bitfield_bit_size (bitfield_bit_size),
Greg Clayton8f92f0a2010-10-14 22:52:14 +000047 m_bitfield_bit_offset (bitfield_bit_offset),
Greg Claytone221f822011-01-21 01:59:00 +000048 m_is_base_class (is_base_class),
49 m_is_deref_of_parent (is_deref_of_parent)
Chris Lattner30fdc8d2010-06-08 16:52:24 +000050{
51 m_name = name;
Enrico Granata9128ee22011-09-06 19:20:51 +000052 SetAddressTypeOfChildren(child_ptr_or_ref_addr_type);
Chris Lattner30fdc8d2010-06-08 16:52:24 +000053}
54
55ValueObjectChild::~ValueObjectChild()
56{
57}
58
Chris Lattner30fdc8d2010-06-08 16:52:24 +000059lldb::ValueType
60ValueObjectChild::GetValueType() const
61{
62 return m_parent->GetValueType();
63}
64
65uint32_t
66ValueObjectChild::CalculateNumChildren()
67{
Greg Clayton6beaaa62011-01-17 03:46:26 +000068 return ClangASTContext::GetNumChildren (GetClangAST (), m_clang_type, true);
Chris Lattner30fdc8d2010-06-08 16:52:24 +000069}
70
Chris Lattner30fdc8d2010-06-08 16:52:24 +000071ConstString
72ValueObjectChild::GetTypeName()
73{
74 if (m_type_name.IsEmpty())
75 {
Greg Claytone3055942011-06-30 02:28:26 +000076 m_type_name = ClangASTType::GetConstTypeName (GetClangType());
Chris Lattner30fdc8d2010-06-08 16:52:24 +000077 if (m_type_name)
78 {
79 if (m_bitfield_bit_size > 0)
80 {
81 const char *clang_type_name = m_type_name.AsCString();
82 if (clang_type_name)
83 {
Greg Claytonc982c762010-07-09 20:39:50 +000084 std::vector<char> bitfield_type_name (strlen(clang_type_name) + 32, 0);
Greg Clayton471b31c2010-07-20 22:52:08 +000085 ::snprintf (&bitfield_type_name.front(), bitfield_type_name.size(), "%s:%u", clang_type_name, m_bitfield_bit_size);
86 m_type_name.SetCString(&bitfield_type_name.front());
Chris Lattner30fdc8d2010-06-08 16:52:24 +000087 }
88 }
89 }
90 }
91 return m_type_name;
92}
93
Jim Ingham6035b672011-03-31 00:19:25 +000094bool
95ValueObjectChild::UpdateValue ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +000096{
97 m_error.Clear();
98 SetValueIsValid (false);
99 ValueObject* parent = m_parent;
100 if (parent)
101 {
Enrico Granatac3e320a2011-08-02 17:27:39 +0000102 if (parent->UpdateValueIfNeeded(false))
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000103 {
Greg Clayton526e5af2010-11-13 03:52:47 +0000104 m_value.SetContext(Value::eContextTypeClangType, m_clang_type);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000105
106 // Copy the parent scalar value and the scalar value type
107 m_value.GetScalar() = parent->GetValue().GetScalar();
108 Value::ValueType value_type = parent->GetValue().GetValueType();
109 m_value.SetValueType (value_type);
110
Greg Clayton1be10fc2010-09-29 01:12:09 +0000111 if (ClangASTContext::IsPointerOrReferenceType (parent->GetClangType()))
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000112 {
Enrico Granata9128ee22011-09-06 19:20:51 +0000113 lldb::addr_t addr = parent->GetPointerValue ();
Greg Clayton6fdfc7e2011-08-16 00:44:29 +0000114 m_value.GetScalar() = addr;
Greg Clayton7c8a9662010-11-02 01:50:16 +0000115
116 if (addr == LLDB_INVALID_ADDRESS)
117 {
118 m_error.SetErrorString ("parent address is invalid.");
119 }
120 else if (addr == 0)
121 {
122 m_error.SetErrorString ("parent is NULL");
123 }
124 else
125 {
Greg Claytondf797c12010-11-02 21:21:20 +0000126 m_value.GetScalar() += m_byte_offset;
Enrico Granata9128ee22011-09-06 19:20:51 +0000127 AddressType addr_type = parent->GetAddressTypeOfChildren();
128
129 switch (addr_type)
130 {
131 case eAddressTypeFile:
132 if (m_update_point.GetProcessSP().get() != NULL && m_update_point.GetProcessSP()->IsAlive() == true)
133 m_value.SetValueType (Value::eValueTypeLoadAddress);
134 else
135 m_value.SetValueType(Value::eValueTypeFileAddress);
136 break;
137 case eAddressTypeLoad:
138 m_value.SetValueType (Value::eValueTypeLoadAddress);
139 break;
140 case eAddressTypeHost:
141 m_value.SetValueType(Value::eValueTypeHostAddress);
142 break;
143 case eAddressTypeInvalid:
144 default:
145 // TODO: does this make sense?
146 m_value.SetValueType(Value::eValueTypeScalar);
147 break;
148 }
Greg Clayton7c8a9662010-11-02 01:50:16 +0000149 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000150 }
151 else
152 {
153 switch (value_type)
154 {
155 case Value::eValueTypeLoadAddress:
156 case Value::eValueTypeFileAddress:
157 case Value::eValueTypeHostAddress:
158 {
159 lldb::addr_t addr = m_value.GetScalar().ULongLong(LLDB_INVALID_ADDRESS);
Greg Clayton7c8a9662010-11-02 01:50:16 +0000160 if (addr == LLDB_INVALID_ADDRESS)
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000161 {
Greg Clayton7c8a9662010-11-02 01:50:16 +0000162 m_error.SetErrorString ("parent address is invalid.");
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000163 }
Greg Clayton7c8a9662010-11-02 01:50:16 +0000164 else if (addr == 0)
165 {
166 m_error.SetErrorString ("parent is NULL");
167 }
168 else
169 {
170 // Set this object's scalar value to the address of its
Jim Ingham16e0c682011-08-12 23:34:31 +0000171 // value by adding its byte offset to the parent address
Greg Clayton7c8a9662010-11-02 01:50:16 +0000172 m_value.GetScalar() += GetByteOffset();
173 }
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000174 }
175 break;
176
177 case Value::eValueTypeScalar:
178 // TODO: What if this is a register value? Do we try and
179 // extract the child value from within the parent data?
180 // Probably...
181 default:
Greg Clayton86edbf42011-10-26 00:56:27 +0000182 m_error.SetErrorString ("parent has invalid value.");
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000183 break;
184 }
185 }
186
187 if (m_error.Success())
188 {
Jim Ingham6035b672011-03-31 00:19:25 +0000189 ExecutionContext exe_ctx (GetExecutionContextScope());
Greg Clayton644247c2011-07-07 01:59:51 +0000190 m_error = m_value.GetValueAsData (&exe_ctx, GetClangAST (), m_data, 0, GetModule());
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000191 }
192 }
193 else
194 {
Greg Clayton86edbf42011-10-26 00:56:27 +0000195 m_error.SetErrorStringWithFormat("parent failed to evaluate: %s", parent->GetError().AsCString());
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000196 }
197 }
198 else
199 {
200 m_error.SetErrorString("ValueObjectChild has a NULL parent ValueObject.");
201 }
Jim Ingham6035b672011-03-31 00:19:25 +0000202
203 return m_error.Success();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000204}
205
206
207bool
Jim Ingham6035b672011-03-31 00:19:25 +0000208ValueObjectChild::IsInScope ()
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000209{
Jim Ingham6035b672011-03-31 00:19:25 +0000210 return m_parent->IsInScope ();
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000211}