1 //===-- ABISysV_ppc64.cpp -------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "ABISysV_ppc64.h"
11 #include "llvm/ADT/STLExtras.h"
12 #include "llvm/TargetParser/Triple.h"
14 #include "Plugins/TypeSystem/Clang/TypeSystemClang.h"
15 #include "Utility/PPC64LE_DWARF_Registers.h"
16 #include "Utility/PPC64_DWARF_Registers.h"
17 #include "lldb/Core/Module.h"
18 #include "lldb/Core/PluginManager.h"
19 #include "lldb/Core/Value.h"
20 #include "lldb/Core/ValueObjectConstResult.h"
21 #include "lldb/Core/ValueObjectMemory.h"
22 #include "lldb/Core/ValueObjectRegister.h"
23 #include "lldb/Symbol/UnwindPlan.h"
24 #include "lldb/Target/Process.h"
25 #include "lldb/Target/RegisterContext.h"
26 #include "lldb/Target/StackFrame.h"
27 #include "lldb/Target/Target.h"
28 #include "lldb/Target/Thread.h"
29 #include "lldb/Utility/ConstString.h"
30 #include "lldb/Utility/DataExtractor.h"
31 #include "lldb/Utility/LLDBLog.h"
32 #include "lldb/Utility/Log.h"
33 #include "lldb/Utility/RegisterValue.h"
34 #include "lldb/Utility/Status.h"
36 #include "clang/AST/ASTContext.h"
37 #include "clang/AST/Attr.h"
38 #include "clang/AST/Decl.h"
40 #define DECLARE_REGISTER_INFOS_PPC64_STRUCT
41 #include "Plugins/Process/Utility/RegisterInfos_ppc64.h"
42 #undef DECLARE_REGISTER_INFOS_PPC64_STRUCT
44 #define DECLARE_REGISTER_INFOS_PPC64LE_STRUCT
45 #include "Plugins/Process/Utility/RegisterInfos_ppc64le.h"
46 #undef DECLARE_REGISTER_INFOS_PPC64LE_STRUCT
50 using namespace lldb_private
;
52 LLDB_PLUGIN_DEFINE(ABISysV_ppc64
)
54 const lldb_private::RegisterInfo
*
55 ABISysV_ppc64::GetRegisterInfoArray(uint32_t &count
) {
56 if (GetByteOrder() == lldb::eByteOrderLittle
) {
57 count
= std::size(g_register_infos_ppc64le
);
58 return g_register_infos_ppc64le
;
60 count
= std::size(g_register_infos_ppc64
);
61 return g_register_infos_ppc64
;
65 size_t ABISysV_ppc64::GetRedZoneSize() const { return 224; }
67 lldb::ByteOrder
ABISysV_ppc64::GetByteOrder() const {
68 return GetProcessSP()->GetByteOrder();
74 ABISysV_ppc64::CreateInstance(lldb::ProcessSP process_sp
,
75 const ArchSpec
&arch
) {
76 if (arch
.GetTriple().isPPC64())
78 new ABISysV_ppc64(std::move(process_sp
), MakeMCRegisterInfo(arch
)));
82 bool ABISysV_ppc64::PrepareTrivialCall(Thread
&thread
, addr_t sp
,
83 addr_t func_addr
, addr_t return_addr
,
84 llvm::ArrayRef
<addr_t
> args
) const {
85 Log
*log
= GetLog(LLDBLog::Expressions
);
89 s
.Printf("ABISysV_ppc64::PrepareTrivialCall (tid = 0x%" PRIx64
90 ", sp = 0x%" PRIx64
", func_addr = 0x%" PRIx64
91 ", return_addr = 0x%" PRIx64
,
92 thread
.GetID(), (uint64_t)sp
, (uint64_t)func_addr
,
93 (uint64_t)return_addr
);
95 for (size_t i
= 0; i
< args
.size(); ++i
)
96 s
.Printf(", arg%" PRIu64
" = 0x%" PRIx64
, static_cast<uint64_t>(i
+ 1),
99 log
->PutString(s
.GetString());
102 RegisterContext
*reg_ctx
= thread
.GetRegisterContext().get();
106 const RegisterInfo
*reg_info
= nullptr;
108 if (args
.size() > 8) // TODO handle more than 8 arguments
111 for (size_t i
= 0; i
< args
.size(); ++i
) {
112 reg_info
= reg_ctx
->GetRegisterInfo(eRegisterKindGeneric
,
113 LLDB_REGNUM_GENERIC_ARG1
+ i
);
114 LLDB_LOGF(log
, "About to write arg%" PRIu64
" (0x%" PRIx64
") into %s",
115 static_cast<uint64_t>(i
+ 1), args
[i
], reg_info
->name
);
116 if (!reg_ctx
->WriteRegisterFromUnsigned(reg_info
, args
[i
]))
120 // First, align the SP
122 LLDB_LOGF(log
, "16-byte aligning SP: 0x%" PRIx64
" to 0x%" PRIx64
,
123 (uint64_t)sp
, (uint64_t)(sp
& ~0xfull
));
125 sp
&= ~(0xfull
); // 16-byte alignment
127 sp
-= 544; // allocate frame to save TOC, RA and SP.
131 const RegisterInfo
*pc_reg_info
=
132 reg_ctx
->GetRegisterInfo(eRegisterKindGeneric
, LLDB_REGNUM_GENERIC_PC
);
133 const RegisterInfo
*sp_reg_info
=
134 reg_ctx
->GetRegisterInfo(eRegisterKindGeneric
, LLDB_REGNUM_GENERIC_SP
);
135 ProcessSP
process_sp(thread
.GetProcess());
136 const RegisterInfo
*lr_reg_info
=
137 reg_ctx
->GetRegisterInfo(eRegisterKindGeneric
, LLDB_REGNUM_GENERIC_RA
);
138 const RegisterInfo
*r2_reg_info
= reg_ctx
->GetRegisterInfoAtIndex(2);
139 const RegisterInfo
*r12_reg_info
= reg_ctx
->GetRegisterInfoAtIndex(12);
141 // Save return address onto the stack.
143 "Pushing the return address onto the stack: 0x%" PRIx64
145 (uint64_t)sp
, (uint64_t)return_addr
);
146 if (!process_sp
->WritePointerToMemory(sp
+ 16, return_addr
, error
))
149 // Write the return address to link register.
150 LLDB_LOGF(log
, "Writing LR: 0x%" PRIx64
, (uint64_t)return_addr
);
151 if (!reg_ctx
->WriteRegisterFromUnsigned(lr_reg_info
, return_addr
))
154 // Write target address to %r12 register.
155 LLDB_LOGF(log
, "Writing R12: 0x%" PRIx64
, (uint64_t)func_addr
);
156 if (!reg_ctx
->WriteRegisterFromUnsigned(r12_reg_info
, func_addr
))
159 // Read TOC pointer value.
160 reg_value
= reg_ctx
->ReadRegisterAsUnsigned(r2_reg_info
, 0);
162 // Write TOC pointer onto the stack.
163 uint64_t stack_offset
;
164 if (GetByteOrder() == lldb::eByteOrderLittle
)
169 LLDB_LOGF(log
, "Writing R2 (TOC) at SP(0x%" PRIx64
")+%d: 0x%" PRIx64
,
170 (uint64_t)(sp
+ stack_offset
), (int)stack_offset
,
171 (uint64_t)reg_value
);
172 if (!process_sp
->WritePointerToMemory(sp
+ stack_offset
, reg_value
, error
))
175 // Read the current SP value.
176 reg_value
= reg_ctx
->ReadRegisterAsUnsigned(sp_reg_info
, 0);
178 // Save current SP onto the stack.
179 LLDB_LOGF(log
, "Writing SP at SP(0x%" PRIx64
")+0: 0x%" PRIx64
, (uint64_t)sp
,
180 (uint64_t)reg_value
);
181 if (!process_sp
->WritePointerToMemory(sp
, reg_value
, error
))
184 // %r1 is set to the actual stack value.
185 LLDB_LOGF(log
, "Writing SP: 0x%" PRIx64
, (uint64_t)sp
);
187 if (!reg_ctx
->WriteRegisterFromUnsigned(sp_reg_info
, sp
))
190 // %pc is set to the address of the called function.
192 LLDB_LOGF(log
, "Writing IP: 0x%" PRIx64
, (uint64_t)func_addr
);
194 if (!reg_ctx
->WriteRegisterFromUnsigned(pc_reg_info
, func_addr
))
200 static bool ReadIntegerArgument(Scalar
&scalar
, unsigned int bit_width
,
201 bool is_signed
, Thread
&thread
,
202 uint32_t *argument_register_ids
,
203 unsigned int ¤t_argument_register
,
204 addr_t
¤t_stack_argument
) {
206 return false; // Scalar can't hold large integer arguments
208 if (current_argument_register
< 6) {
209 scalar
= thread
.GetRegisterContext()->ReadRegisterAsUnsigned(
210 argument_register_ids
[current_argument_register
], 0);
211 current_argument_register
++;
213 scalar
.SignExtend(bit_width
);
215 uint32_t byte_size
= (bit_width
+ (8 - 1)) / 8;
217 if (thread
.GetProcess()->ReadScalarIntegerFromMemory(
218 current_stack_argument
, byte_size
, is_signed
, scalar
, error
)) {
219 current_stack_argument
+= byte_size
;
227 bool ABISysV_ppc64::GetArgumentValues(Thread
&thread
, ValueList
&values
) const {
228 unsigned int num_values
= values
.GetSize();
229 unsigned int value_index
;
231 // Extract the register context so we can read arguments from registers
233 RegisterContext
*reg_ctx
= thread
.GetRegisterContext().get();
238 // Get the pointer to the first stack argument so we have a place to start
241 addr_t sp
= reg_ctx
->GetSP(0);
246 uint64_t stack_offset
;
247 if (GetByteOrder() == lldb::eByteOrderLittle
)
252 // jump over return address.
253 addr_t current_stack_argument
= sp
+ stack_offset
;
254 uint32_t argument_register_ids
[8];
256 for (size_t i
= 0; i
< 8; ++i
) {
257 argument_register_ids
[i
] =
259 ->GetRegisterInfo(eRegisterKindGeneric
,
260 LLDB_REGNUM_GENERIC_ARG1
+ i
)
261 ->kinds
[eRegisterKindLLDB
];
264 unsigned int current_argument_register
= 0;
266 for (value_index
= 0; value_index
< num_values
; ++value_index
) {
267 Value
*value
= values
.GetValueAtIndex(value_index
);
272 // We currently only support extracting values with Clang QualTypes. Do we
273 // care about others?
274 CompilerType compiler_type
= value
->GetCompilerType();
275 std::optional
<uint64_t> bit_size
= compiler_type
.GetBitSize(&thread
);
280 if (compiler_type
.IsIntegerOrEnumerationType(is_signed
)) {
281 ReadIntegerArgument(value
->GetScalar(), *bit_size
, is_signed
, thread
,
282 argument_register_ids
, current_argument_register
,
283 current_stack_argument
);
284 } else if (compiler_type
.IsPointerType()) {
285 ReadIntegerArgument(value
->GetScalar(), *bit_size
, false, thread
,
286 argument_register_ids
, current_argument_register
,
287 current_stack_argument
);
294 Status
ABISysV_ppc64::SetReturnValueObject(lldb::StackFrameSP
&frame_sp
,
295 lldb::ValueObjectSP
&new_value_sp
) {
298 error
.SetErrorString("Empty value object for return value.");
302 CompilerType compiler_type
= new_value_sp
->GetCompilerType();
303 if (!compiler_type
) {
304 error
.SetErrorString("Null clang type for return value.");
308 Thread
*thread
= frame_sp
->GetThread().get();
314 RegisterContext
*reg_ctx
= thread
->GetRegisterContext().get();
316 bool set_it_simple
= false;
317 if (compiler_type
.IsIntegerOrEnumerationType(is_signed
) ||
318 compiler_type
.IsPointerType()) {
319 const RegisterInfo
*reg_info
= reg_ctx
->GetRegisterInfoByName("r3", 0);
323 size_t num_bytes
= new_value_sp
->GetData(data
, data_error
);
324 if (data_error
.Fail()) {
325 error
.SetErrorStringWithFormat(
326 "Couldn't convert return value to raw data: %s",
327 data_error
.AsCString());
330 lldb::offset_t offset
= 0;
331 if (num_bytes
<= 8) {
332 uint64_t raw_value
= data
.GetMaxU64(&offset
, num_bytes
);
334 if (reg_ctx
->WriteRegisterFromUnsigned(reg_info
, raw_value
))
335 set_it_simple
= true;
337 error
.SetErrorString("We don't support returning longer than 64 bit "
338 "integer values at present.");
340 } else if (compiler_type
.IsFloatingPointType(count
, is_complex
)) {
342 error
.SetErrorString(
343 "We don't support returning complex values at present");
345 std::optional
<uint64_t> bit_width
=
346 compiler_type
.GetBitSize(frame_sp
.get());
348 error
.SetErrorString("can't get size of type");
351 if (*bit_width
<= 64) {
354 size_t num_bytes
= new_value_sp
->GetData(data
, data_error
);
355 if (data_error
.Fail()) {
356 error
.SetErrorStringWithFormat(
357 "Couldn't convert return value to raw data: %s",
358 data_error
.AsCString());
362 unsigned char buffer
[16];
363 ByteOrder byte_order
= data
.GetByteOrder();
365 data
.CopyByteOrderedData(0, num_bytes
, buffer
, 16, byte_order
);
366 set_it_simple
= true;
368 // FIXME - don't know how to do 80 bit long doubles yet.
369 error
.SetErrorString(
370 "We don't support returning float values > 64 bits at present");
375 if (!set_it_simple
) {
376 // Okay we've got a structure or something that doesn't fit in a simple
377 // register. We should figure out where it really goes, but we don't
379 error
.SetErrorString("We only support setting simple integer and float "
380 "return types at present.");
387 // ReturnValueExtractor
392 #define LOG_PREFIX "ReturnValueExtractor: "
394 class ReturnValueExtractor
{
395 // This class represents a register, from which data may be extracted.
397 // It may be constructed by directly specifying its index (where 0 is the
398 // first register used to return values) or by specifying the offset of a
399 // given struct field, in which case the appropriated register index will be
404 GPR
, // General Purpose Register
405 FPR
// Floating Point Register
410 // offs - field offset in struct
411 Register(Type ty
, uint32_t index
, uint32_t offs
, RegisterContext
*reg_ctx
,
412 ByteOrder byte_order
)
413 : m_index(index
), m_offs(offs
% sizeof(uint64_t)),
414 m_avail(sizeof(uint64_t) - m_offs
), m_type(ty
), m_reg_ctx(reg_ctx
),
415 m_byte_order(byte_order
) {}
417 // explicit index, no offset
418 Register(Type ty
, uint32_t index
, RegisterContext
*reg_ctx
,
419 ByteOrder byte_order
)
420 : Register(ty
, index
, 0, reg_ctx
, byte_order
) {}
422 // GPR, calculate index from offs
423 Register(uint32_t offs
, RegisterContext
*reg_ctx
, ByteOrder byte_order
)
424 : Register(GPR
, offs
/ sizeof(uint64_t), offs
, reg_ctx
, byte_order
) {}
426 uint32_t Index() const { return m_index
; }
428 // register offset where data is located
429 uint32_t Offs() const { return m_offs
; }
431 // available bytes in this register
432 uint32_t Avail() const { return m_avail
; }
434 bool IsValid() const {
436 LLDB_LOG(m_log
, LOG_PREFIX
437 "No more than 8 registers should be used to return values");
443 std::string
GetName() const {
445 return ("r" + llvm::Twine(m_index
+ 3)).str();
447 return ("f" + llvm::Twine(m_index
+ 1)).str();
450 // get raw register data
451 bool GetRawData(uint64_t &raw_data
) {
452 const RegisterInfo
*reg_info
=
453 m_reg_ctx
->GetRegisterInfoByName(GetName());
455 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to get RegisterInfo");
459 RegisterValue reg_val
;
460 if (!m_reg_ctx
->ReadRegister(reg_info
, reg_val
)) {
461 LLDB_LOG(m_log
, LOG_PREFIX
"ReadRegister() failed");
466 uint32_t rc
= reg_val
.GetAsMemoryData(
467 *reg_info
, &raw_data
, sizeof(raw_data
), m_byte_order
, error
);
468 if (rc
!= sizeof(raw_data
)) {
469 LLDB_LOG(m_log
, LOG_PREFIX
"GetAsMemoryData() failed");
481 RegisterContext
*m_reg_ctx
;
482 ByteOrder m_byte_order
;
483 Log
*m_log
= GetLog(LLDBLog::Expressions
);
486 Register
GetGPR(uint32_t index
) const {
487 return Register(Register::GPR
, index
, m_reg_ctx
, m_byte_order
);
490 Register
GetFPR(uint32_t index
) const {
491 return Register(Register::FPR
, index
, m_reg_ctx
, m_byte_order
);
494 Register
GetGPRByOffs(uint32_t offs
) const {
495 return Register(offs
, m_reg_ctx
, m_byte_order
);
500 static llvm::Expected
<ReturnValueExtractor
> Create(Thread
&thread
,
501 CompilerType
&type
) {
502 RegisterContext
*reg_ctx
= thread
.GetRegisterContext().get();
504 return llvm::make_error
<llvm::StringError
>(
505 LOG_PREFIX
"Failed to get RegisterContext",
506 llvm::inconvertibleErrorCode());
508 ProcessSP process_sp
= thread
.GetProcess();
510 return llvm::make_error
<llvm::StringError
>(
511 LOG_PREFIX
"GetProcess() failed", llvm::inconvertibleErrorCode());
513 return ReturnValueExtractor(thread
, type
, reg_ctx
, process_sp
);
516 // main method: get value of the type specified at construction time
517 ValueObjectSP
GetValue() {
518 const uint32_t type_flags
= m_type
.GetTypeInfo();
520 // call the appropriate type handler
522 ValueObjectSP valobj_sp
;
523 if (type_flags
& eTypeIsScalar
) {
524 if (type_flags
& eTypeIsInteger
) {
525 value_sp
= GetIntegerValue(0);
526 } else if (type_flags
& eTypeIsFloat
) {
527 if (type_flags
& eTypeIsComplex
) {
528 LLDB_LOG(m_log
, LOG_PREFIX
"Complex numbers are not supported yet");
529 return ValueObjectSP();
531 value_sp
= GetFloatValue(m_type
, 0);
534 } else if (type_flags
& eTypeIsPointer
) {
535 value_sp
= GetPointerValue(0);
539 valobj_sp
= ValueObjectConstResult::Create(
540 m_thread
.GetStackFrameAtIndex(0).get(), *value_sp
, ConstString(""));
541 } else if (type_flags
& eTypeIsVector
) {
542 valobj_sp
= GetVectorValueObject();
543 } else if (type_flags
& eTypeIsStructUnion
|| type_flags
& eTypeIsClass
) {
544 valobj_sp
= GetStructValueObject();
553 CompilerType
&m_type
;
554 uint64_t m_byte_size
;
555 std::unique_ptr
<DataBufferHeap
> m_data_up
;
556 int32_t m_src_offs
= 0;
557 int32_t m_dst_offs
= 0;
558 bool m_packed
= false;
559 Log
*m_log
= GetLog(LLDBLog::Expressions
);
560 RegisterContext
*m_reg_ctx
;
561 ProcessSP m_process_sp
;
562 ByteOrder m_byte_order
;
563 uint32_t m_addr_size
;
568 ReturnValueExtractor(Thread
&thread
, CompilerType
&type
,
569 RegisterContext
*reg_ctx
, ProcessSP process_sp
)
570 : m_thread(thread
), m_type(type
),
571 m_byte_size(m_type
.GetByteSize(&thread
).value_or(0)),
572 m_data_up(new DataBufferHeap(m_byte_size
, 0)), m_reg_ctx(reg_ctx
),
573 m_process_sp(process_sp
), m_byte_order(process_sp
->GetByteOrder()),
575 process_sp
->GetTarget().GetArchitecture().GetAddressByteSize()) {}
577 // build a new scalar value
578 ValueSP
NewScalarValue(CompilerType
&type
) {
579 ValueSP
value_sp(new Value
);
580 value_sp
->SetCompilerType(type
);
581 value_sp
->SetValueType(Value::ValueType::Scalar
);
585 // get an integer value in the specified register
586 ValueSP
GetIntegerValue(uint32_t reg_index
) {
588 auto reg
= GetGPR(reg_index
);
589 if (!reg
.GetRawData(raw_value
))
592 // build value from data
593 ValueSP
value_sp(NewScalarValue(m_type
));
595 uint32_t type_flags
= m_type
.GetTypeInfo();
596 bool is_signed
= (type_flags
& eTypeIsSigned
) != 0;
598 switch (m_byte_size
) {
599 case sizeof(uint64_t):
601 value_sp
->GetScalar() = (int64_t)(raw_value
);
603 value_sp
->GetScalar() = (uint64_t)(raw_value
);
606 case sizeof(uint32_t):
608 value_sp
->GetScalar() = (int32_t)(raw_value
& UINT32_MAX
);
610 value_sp
->GetScalar() = (uint32_t)(raw_value
& UINT32_MAX
);
613 case sizeof(uint16_t):
615 value_sp
->GetScalar() = (int16_t)(raw_value
& UINT16_MAX
);
617 value_sp
->GetScalar() = (uint16_t)(raw_value
& UINT16_MAX
);
620 case sizeof(uint8_t):
622 value_sp
->GetScalar() = (int8_t)(raw_value
& UINT8_MAX
);
624 value_sp
->GetScalar() = (uint8_t)(raw_value
& UINT8_MAX
);
628 llvm_unreachable("Invalid integer size");
634 // get a floating point value on the specified register
635 ValueSP
GetFloatValue(CompilerType
&type
, uint32_t reg_index
) {
637 auto reg
= GetFPR(reg_index
);
638 if (!reg
.GetRawData(raw_data
))
641 // build value from data
642 ValueSP
value_sp(NewScalarValue(type
));
644 DataExtractor
de(&raw_data
, sizeof(raw_data
), m_byte_order
, m_addr_size
);
647 std::optional
<uint64_t> byte_size
= type
.GetByteSize(m_process_sp
.get());
650 switch (*byte_size
) {
652 value_sp
->GetScalar() = (float)de
.GetDouble(&offset
);
656 value_sp
->GetScalar() = de
.GetDouble(&offset
);
660 llvm_unreachable("Invalid floating point size");
666 // get pointer value from register
667 ValueSP
GetPointerValue(uint32_t reg_index
) {
669 auto reg
= GetGPR(reg_index
);
670 if (!reg
.GetRawData(raw_data
))
673 // build value from raw data
674 ValueSP
value_sp(NewScalarValue(m_type
));
675 value_sp
->GetScalar() = raw_data
;
679 // build the ValueObject from our data buffer
680 ValueObjectSP
BuildValueObject() {
681 DataExtractor
de(DataBufferSP(m_data_up
.release()), m_byte_order
,
683 return ValueObjectConstResult::Create(&m_thread
, m_type
, ConstString(""),
687 // get a vector return value
688 ValueObjectSP
GetVectorValueObject() {
689 const uint32_t MAX_VRS
= 2;
691 // get first V register used to return values
692 const RegisterInfo
*vr
[MAX_VRS
];
693 vr
[0] = m_reg_ctx
->GetRegisterInfoByName("vr2");
695 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to get vr2 RegisterInfo");
696 return ValueObjectSP();
699 const uint32_t vr_size
= vr
[0]->byte_size
;
701 if (m_byte_size
> 2 * vr_size
) {
704 "Returning vectors that don't fit in 2 VR regs is not supported");
705 return ValueObjectSP();
708 // load vr3, if needed
709 if (m_byte_size
> vr_size
) {
711 vr
[1] = m_reg_ctx
->GetRegisterInfoByName("vr3");
713 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to get vr3 RegisterInfo");
714 return ValueObjectSP();
718 // Get the whole contents of vector registers and let the logic here
719 // arrange the data properly.
721 RegisterValue vr_val
[MAX_VRS
];
723 std::unique_ptr
<DataBufferHeap
> vr_data(
724 new DataBufferHeap(vrs
* vr_size
, 0));
726 for (uint32_t i
= 0; i
< vrs
; i
++) {
727 if (!m_reg_ctx
->ReadRegister(vr
[i
], vr_val
[i
])) {
728 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to read vector register contents");
729 return ValueObjectSP();
731 if (!vr_val
[i
].GetAsMemoryData(*vr
[i
], vr_data
->GetBytes() + i
* vr_size
,
732 vr_size
, m_byte_order
, error
)) {
733 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to extract vector register bytes");
734 return ValueObjectSP();
738 // The compiler generated code seems to always put the vector elements at
739 // the end of the vector register, in case they don't occupy all of it.
740 // This offset variable handles this.
742 if (m_byte_size
< vr_size
)
743 offs
= vr_size
- m_byte_size
;
745 // copy extracted data to our buffer
746 memcpy(m_data_up
->GetBytes(), vr_data
->GetBytes() + offs
, m_byte_size
);
747 return BuildValueObject();
750 // get a struct return value
751 ValueObjectSP
GetStructValueObject() {
752 // case 1: get from stack
753 if (m_byte_size
> 2 * sizeof(uint64_t)) {
755 auto reg
= GetGPR(0);
756 if (!reg
.GetRawData(addr
))
760 size_t rc
= m_process_sp
->ReadMemory(addr
, m_data_up
->GetBytes(),
762 if (rc
!= m_byte_size
) {
763 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to read memory pointed by r3");
764 return ValueObjectSP();
766 return BuildValueObject();
769 // get number of children
770 const bool omit_empty_base_classes
= true;
771 uint32_t n
= m_type
.GetNumChildren(omit_empty_base_classes
, nullptr);
773 LLDB_LOG(m_log
, LOG_PREFIX
"No children found in struct");
777 // case 2: homogeneous double or float aggregate
778 CompilerType elem_type
;
779 if (m_type
.IsHomogeneousAggregate(&elem_type
)) {
780 uint32_t type_flags
= elem_type
.GetTypeInfo();
781 std::optional
<uint64_t> elem_size
=
782 elem_type
.GetByteSize(m_process_sp
.get());
785 if (type_flags
& eTypeIsComplex
|| !(type_flags
& eTypeIsFloat
)) {
787 LOG_PREFIX
"Unexpected type found in homogeneous aggregate");
791 for (uint32_t i
= 0; i
< n
; i
++) {
792 ValueSP val_sp
= GetFloatValue(elem_type
, i
);
798 size_t rc
= val_sp
->GetScalar().GetAsMemoryData(
799 m_data_up
->GetBytes() + m_dst_offs
, *elem_size
, m_byte_order
,
801 if (rc
!= *elem_size
) {
802 LLDB_LOG(m_log
, LOG_PREFIX
"Failed to get float data");
805 m_dst_offs
+= *elem_size
;
807 return BuildValueObject();
810 // case 3: get from GPRs
812 // first, check if this is a packed struct or not
813 auto ast
= m_type
.GetTypeSystem().dyn_cast_or_null
<TypeSystemClang
>();
815 clang::RecordDecl
*record_decl
= TypeSystemClang::GetAsRecordDecl(m_type
);
818 auto attrs
= record_decl
->attrs();
819 for (const auto &attr
: attrs
) {
820 if (attr
->getKind() == clang::attr::Packed
) {
828 LLDB_LOG(m_log
, LOG_PREFIX
"{0} struct",
829 m_packed
? "packed" : "not packed");
831 for (uint32_t i
= 0; i
< n
; i
++) {
834 GetChildType(i
, name
, size
);
835 // NOTE: the offset returned by GetChildCompilerTypeAtIndex()
836 // can't be used because it never considers alignment bytes
837 // between struct fields.
838 LLDB_LOG(m_log
, LOG_PREFIX
"field={0}, size={1}", name
, size
);
839 if (!ExtractField(size
))
840 return ValueObjectSP();
843 return BuildValueObject();
846 // extract 'size' bytes at 'offs' from GPRs
847 bool ExtractFromRegs(int32_t offs
, uint32_t size
, void *buf
) {
849 auto reg
= GetGPRByOffs(offs
);
853 uint32_t n
= std::min(reg
.Avail(), size
);
856 if (!reg
.GetRawData(raw_data
))
859 memcpy(buf
, (char *)&raw_data
+ reg
.Offs(), n
);
862 buf
= (char *)buf
+ n
;
867 // extract one field from GPRs and put it in our buffer
868 bool ExtractField(uint32_t size
) {
869 auto reg
= GetGPRByOffs(m_src_offs
);
875 uint32_t n
= m_src_offs
% size
;
877 // not 'size' bytes aligned
880 LOG_PREFIX
"Extracting {0} alignment bytes at offset {1}", n
,
882 // get alignment bytes
883 if (!ExtractFromRegs(m_src_offs
, n
, m_data_up
->GetBytes() + m_dst_offs
))
891 LLDB_LOG(m_log
, LOG_PREFIX
"Extracting {0} field bytes at offset {1}", size
,
893 if (!ExtractFromRegs(m_src_offs
, size
, m_data_up
->GetBytes() + m_dst_offs
))
901 CompilerType
GetChildType(uint32_t i
, std::string
&name
, uint32_t &size
) {
902 // GetChild constant inputs
903 const bool transparent_pointers
= false;
904 const bool omit_empty_base_classes
= true;
905 const bool ignore_array_bounds
= false;
906 // GetChild output params
908 uint32_t child_bitfield_bit_size
;
909 uint32_t child_bitfield_bit_offset
;
910 bool child_is_base_class
;
911 bool child_is_deref_of_parent
;
912 ValueObject
*valobj
= nullptr;
913 uint64_t language_flags
;
914 ExecutionContext exe_ctx
;
915 m_thread
.CalculateExecutionContext(exe_ctx
);
917 return m_type
.GetChildCompilerTypeAtIndex(
918 &exe_ctx
, i
, transparent_pointers
, omit_empty_base_classes
,
919 ignore_array_bounds
, name
, size
, child_offs
, child_bitfield_bit_size
,
920 child_bitfield_bit_offset
, child_is_base_class
,
921 child_is_deref_of_parent
, valobj
, language_flags
);
927 } // anonymous namespace
930 ABISysV_ppc64::GetReturnValueObjectSimple(Thread
&thread
,
931 CompilerType
&type
) const {
933 return ValueObjectSP();
935 auto exp_extractor
= ReturnValueExtractor::Create(thread
, type
);
936 if (!exp_extractor
) {
937 Log
*log
= GetLog(LLDBLog::Expressions
);
938 LLDB_LOG_ERROR(log
, exp_extractor
.takeError(),
939 "Extracting return value failed: {0}");
940 return ValueObjectSP();
943 return exp_extractor
.get().GetValue();
946 ValueObjectSP
ABISysV_ppc64::GetReturnValueObjectImpl(
947 Thread
&thread
, CompilerType
&return_compiler_type
) const {
948 return GetReturnValueObjectSimple(thread
, return_compiler_type
);
951 bool ABISysV_ppc64::CreateFunctionEntryUnwindPlan(UnwindPlan
&unwind_plan
) {
953 unwind_plan
.SetRegisterKind(eRegisterKindDWARF
);
959 if (GetByteOrder() == lldb::eByteOrderLittle
) {
960 lr_reg_num
= ppc64le_dwarf::dwarf_lr_ppc64le
;
961 sp_reg_num
= ppc64le_dwarf::dwarf_r1_ppc64le
;
962 pc_reg_num
= ppc64le_dwarf::dwarf_pc_ppc64le
;
964 lr_reg_num
= ppc64_dwarf::dwarf_lr_ppc64
;
965 sp_reg_num
= ppc64_dwarf::dwarf_r1_ppc64
;
966 pc_reg_num
= ppc64_dwarf::dwarf_pc_ppc64
;
969 UnwindPlan::RowSP
row(new UnwindPlan::Row
);
971 // Our Call Frame Address is the stack pointer value
972 row
->GetCFAValue().SetIsRegisterPlusOffset(sp_reg_num
, 0);
974 // The previous PC is in the LR
975 row
->SetRegisterLocationToRegister(pc_reg_num
, lr_reg_num
, true);
976 unwind_plan
.AppendRow(row
);
978 // All other registers are the same.
980 unwind_plan
.SetSourceName("ppc64 at-func-entry default");
981 unwind_plan
.SetSourcedFromCompiler(eLazyBoolNo
);
986 bool ABISysV_ppc64::CreateDefaultUnwindPlan(UnwindPlan
&unwind_plan
) {
988 unwind_plan
.SetRegisterKind(eRegisterKindDWARF
);
994 if (GetByteOrder() == lldb::eByteOrderLittle
) {
995 sp_reg_num
= ppc64le_dwarf::dwarf_r1_ppc64le
;
996 pc_reg_num
= ppc64le_dwarf::dwarf_lr_ppc64le
;
997 cr_reg_num
= ppc64le_dwarf::dwarf_cr_ppc64le
;
999 sp_reg_num
= ppc64_dwarf::dwarf_r1_ppc64
;
1000 pc_reg_num
= ppc64_dwarf::dwarf_lr_ppc64
;
1001 cr_reg_num
= ppc64_dwarf::dwarf_cr_ppc64
;
1004 UnwindPlan::RowSP
row(new UnwindPlan::Row
);
1005 const int32_t ptr_size
= 8;
1006 row
->SetUnspecifiedRegistersAreUndefined(true);
1007 row
->GetCFAValue().SetIsRegisterDereferenced(sp_reg_num
);
1009 row
->SetRegisterLocationToAtCFAPlusOffset(pc_reg_num
, ptr_size
* 2, true);
1010 row
->SetRegisterLocationToIsCFAPlusOffset(sp_reg_num
, 0, true);
1011 row
->SetRegisterLocationToAtCFAPlusOffset(cr_reg_num
, ptr_size
, true);
1013 unwind_plan
.AppendRow(row
);
1014 unwind_plan
.SetSourceName("ppc64 default unwind plan");
1015 unwind_plan
.SetSourcedFromCompiler(eLazyBoolNo
);
1016 unwind_plan
.SetUnwindPlanValidAtAllInstructions(eLazyBoolNo
);
1017 unwind_plan
.SetUnwindPlanForSignalTrap(eLazyBoolNo
);
1018 unwind_plan
.SetReturnAddressRegister(pc_reg_num
);
1022 bool ABISysV_ppc64::RegisterIsVolatile(const RegisterInfo
*reg_info
) {
1023 return !RegisterIsCalleeSaved(reg_info
);
1026 // See "Register Usage" in the
1027 // "System V Application Binary Interface"
1028 // "64-bit PowerPC ELF Application Binary Interface Supplement" current version
1029 // is 2 released 2015 at
1030 // https://members.openpowerfoundation.org/document/dl/576
1031 bool ABISysV_ppc64::RegisterIsCalleeSaved(const RegisterInfo
*reg_info
) {
1033 // Preserved registers are :
1035 // cr2-cr4 (partially preserved)
1036 // f14-f31 (not yet)
1037 // v20-v31 (not yet)
1040 const char *name
= reg_info
->name
;
1041 if (name
[0] == 'r') {
1042 if ((name
[1] == '1' || name
[1] == '2') && name
[2] == '\0')
1044 if (name
[1] == '1' && name
[2] > '2')
1046 if ((name
[1] == '2' || name
[1] == '3') && name
[2] != '\0')
1050 if (name
[0] == 'f' && name
[1] >= '0' && name
[2] <= '9') {
1051 if (name
[2] == '\0')
1053 if (name
[1] == '1' && name
[2] >= '4')
1055 if ((name
[1] == '2' || name
[1] == '3') && name
[2] != '\0')
1059 if (name
[0] == 's' && name
[1] == 'p' && name
[2] == '\0') // sp
1061 if (name
[0] == 'f' && name
[1] == 'p' && name
[2] == '\0') // fp
1063 if (name
[0] == 'p' && name
[1] == 'c' && name
[2] == '\0') // pc
1069 void ABISysV_ppc64::Initialize() {
1070 PluginManager::RegisterPlugin(
1071 GetPluginNameStatic(), "System V ABI for ppc64 targets", CreateInstance
);
1074 void ABISysV_ppc64::Terminate() {
1075 PluginManager::UnregisterPlugin(CreateInstance
);