@@ -145,6 +145,7 @@ bool ValueObjectDynamicValue::UpdateValue() {
145
145
Address dynamic_address;
146
146
bool found_dynamic_type = false ;
147
147
Value::ValueType value_type;
148
+ llvm::ArrayRef<uint8_t > local_buffer;
148
149
149
150
LanguageRuntime *runtime = nullptr ;
150
151
@@ -157,7 +158,7 @@ bool ValueObjectDynamicValue::UpdateValue() {
157
158
// Try the preferred runtime first.
158
159
found_dynamic_type = preferred_runtime->GetDynamicTypeAndAddress (
159
160
*m_parent, m_use_dynamic, class_type_or_name, dynamic_address,
160
- value_type);
161
+ value_type, local_buffer );
161
162
if (found_dynamic_type)
162
163
// Set the operative `runtime` for later use in this function.
163
164
runtime = preferred_runtime;
@@ -166,20 +167,20 @@ bool ValueObjectDynamicValue::UpdateValue() {
166
167
// Fallback to the runtime for `known_type`.
167
168
found_dynamic_type = runtime->GetDynamicTypeAndAddress (
168
169
*m_parent, m_use_dynamic, class_type_or_name, dynamic_address,
169
- value_type);
170
+ value_type, local_buffer );
170
171
} else {
171
172
runtime = process->GetLanguageRuntime (lldb::eLanguageTypeC_plus_plus);
172
173
if (runtime)
173
174
found_dynamic_type = runtime->GetDynamicTypeAndAddress (
174
175
*m_parent, m_use_dynamic, class_type_or_name, dynamic_address,
175
- value_type);
176
+ value_type, local_buffer );
176
177
177
178
if (!found_dynamic_type) {
178
179
runtime = process->GetLanguageRuntime (lldb::eLanguageTypeObjC);
179
180
if (runtime)
180
181
found_dynamic_type = runtime->GetDynamicTypeAndAddress (
181
182
*m_parent, m_use_dynamic, class_type_or_name, dynamic_address,
182
- value_type);
183
+ value_type, local_buffer );
183
184
}
184
185
}
185
186
@@ -239,11 +240,30 @@ bool ValueObjectDynamicValue::UpdateValue() {
239
240
if (m_address.IsValid ())
240
241
SetValueDidChange (true );
241
242
242
- // We've moved, so we should be fine...
243
- m_address = dynamic_address;
244
- lldb::TargetSP target_sp (GetTargetSP ());
245
- lldb::addr_t load_address = m_address.GetLoadAddress (target_sp.get ());
246
- m_value.GetScalar () = load_address;
243
+ auto *exe_scope = exe_ctx.GetBestExecutionContextScope ();
244
+ // If we found a host address, and the dynamic type fits in the local buffer
245
+ // that was found, point to thar buffer. Later on this function will copy
246
+ // the buffer over.
247
+ if (value_type == Value::ValueType::HostAddress) {
248
+ // If we found a host address but it doesn't fit in the buffer, there's
249
+ // nothing we can do.
250
+ if (local_buffer.empty () ||
251
+ local_buffer.size () <
252
+ m_dynamic_type_info.GetCompilerType ().GetByteSize (exe_scope)) {
253
+ SetValueIsValid (false );
254
+ return false ;
255
+ }
256
+
257
+ m_value.GetScalar () = (uint64_t )local_buffer.data ();
258
+ m_address = LLDB_INVALID_ADDRESS;
259
+ } else {
260
+ // Otherwise we have a legitimate address on the target. Point to the load
261
+ // address.
262
+ m_address = dynamic_address;
263
+ lldb::TargetSP target_sp (GetTargetSP ());
264
+ lldb::addr_t load_address = m_address.GetLoadAddress (target_sp.get ());
265
+ m_value.GetScalar () = load_address;
266
+ }
247
267
}
248
268
249
269
if (runtime)
@@ -258,7 +278,11 @@ bool ValueObjectDynamicValue::UpdateValue() {
258
278
LLDB_LOGF (log , " [%s %p] has a new dynamic type %s" , GetName ().GetCString (),
259
279
static_cast <void *>(this ), GetTypeName ().GetCString ());
260
280
261
- if (m_address.IsValid () && m_dynamic_type_info) {
281
+ // m_address could be invalid but we could still have a local buffer
282
+ // containing the dynamic value.
283
+ if ((m_address.IsValid () ||
284
+ m_value.GetValueType () == Value::ValueType::HostAddress) &&
285
+ m_dynamic_type_info) {
262
286
// The variable value is in the Scalar value inside the m_value. We can
263
287
// point our m_data right to it.
264
288
m_error = m_value.GetValueAsData (&exe_ctx, m_data, GetModule ().get ());
0 commit comments