| 1 | //===-- ABISysV_arc.cpp ---------------------------------------------------===// |
| 2 | // |
| 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 |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #include "ABISysV_arc.h" |
| 10 | |
| 11 | // C Includes |
| 12 | // C++ Includes |
| 13 | #include <array> |
| 14 | #include <limits> |
| 15 | #include <type_traits> |
| 16 | |
| 17 | // Other libraries and framework includes |
| 18 | #include "llvm/IR/DerivedTypes.h" |
| 19 | #include "llvm/Support/MathExtras.h" |
| 20 | #include "llvm/TargetParser/Triple.h" |
| 21 | |
| 22 | #include "lldb/Core/Module.h" |
| 23 | #include "lldb/Core/PluginManager.h" |
| 24 | #include "lldb/Core/Value.h" |
| 25 | #include "lldb/Symbol/UnwindPlan.h" |
| 26 | #include "lldb/Target/Process.h" |
| 27 | #include "lldb/Target/RegisterContext.h" |
| 28 | #include "lldb/Target/StackFrame.h" |
| 29 | #include "lldb/Target/Target.h" |
| 30 | #include "lldb/Target/Thread.h" |
| 31 | #include "lldb/Utility/ConstString.h" |
| 32 | #include "lldb/Utility/RegisterValue.h" |
| 33 | #include "lldb/Utility/Status.h" |
| 34 | #include "lldb/ValueObject/ValueObjectConstResult.h" |
| 35 | #include "lldb/ValueObject/ValueObjectMemory.h" |
| 36 | #include "lldb/ValueObject/ValueObjectRegister.h" |
| 37 | |
| 38 | #define DEFINE_REG_NAME(reg_num) ConstString(#reg_num).GetCString() |
| 39 | #define DEFINE_REG_NAME_STR(reg_name) ConstString(reg_name).GetCString() |
| 40 | |
| 41 | // The ABI is not a source of such information as size, offset, encoding, etc. |
| 42 | // of a register. Just provides correct dwarf and eh_frame numbers. |
| 43 | |
| 44 | #define DEFINE_GENERIC_REGISTER_STUB(dwarf_num, str_name, generic_num) \ |
| 45 | { \ |
| 46 | DEFINE_REG_NAME(dwarf_num), DEFINE_REG_NAME_STR(str_name), \ |
| 47 | 0, 0, eEncodingInvalid, eFormatDefault, \ |
| 48 | { dwarf_num, dwarf_num, generic_num, LLDB_INVALID_REGNUM, dwarf_num }, \ |
| 49 | nullptr, nullptr, nullptr, \ |
| 50 | } |
| 51 | |
| 52 | #define DEFINE_REGISTER_STUB(dwarf_num, str_name) \ |
| 53 | DEFINE_GENERIC_REGISTER_STUB(dwarf_num, str_name, LLDB_INVALID_REGNUM) |
| 54 | |
| 55 | using namespace lldb; |
| 56 | using namespace lldb_private; |
| 57 | |
| 58 | LLDB_PLUGIN_DEFINE_ADV(ABISysV_arc, ABIARC) |
| 59 | |
| 60 | namespace { |
| 61 | namespace dwarf { |
| 62 | enum regnums { |
| 63 | r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, r14, r15, r16, |
| 64 | r17, r18, r19, r20, r21, r22, r23, r24, r25, r26, |
| 65 | r27, fp = r27, r28, sp = r28, r29, r30, r31, blink = r31, |
| 66 | r32, r33, r34, r35, r36, r37, r38, r39, r40, r41, r42, r43, r44, r45, r46, |
| 67 | r47, r48, r49, r50, r51, r52, r53, r54, r55, r56, r57, r58, r59, r60, |
| 68 | /*reserved,*/ /*limm indicator,*/ r63 = 63, pc = 70, status32 = 74 |
| 69 | }; |
| 70 | |
| 71 | static const std::array<RegisterInfo, 64> g_register_infos = { ._M_elems: { |
| 72 | DEFINE_GENERIC_REGISTER_STUB(r0, nullptr, LLDB_REGNUM_GENERIC_ARG1), |
| 73 | DEFINE_GENERIC_REGISTER_STUB(r1, nullptr, LLDB_REGNUM_GENERIC_ARG2), |
| 74 | DEFINE_GENERIC_REGISTER_STUB(r2, nullptr, LLDB_REGNUM_GENERIC_ARG3), |
| 75 | DEFINE_GENERIC_REGISTER_STUB(r3, nullptr, LLDB_REGNUM_GENERIC_ARG4), |
| 76 | DEFINE_GENERIC_REGISTER_STUB(r4, nullptr, LLDB_REGNUM_GENERIC_ARG5), |
| 77 | DEFINE_GENERIC_REGISTER_STUB(r5, nullptr, LLDB_REGNUM_GENERIC_ARG6), |
| 78 | DEFINE_GENERIC_REGISTER_STUB(r6, nullptr, LLDB_REGNUM_GENERIC_ARG7), |
| 79 | DEFINE_GENERIC_REGISTER_STUB(r7, nullptr, LLDB_REGNUM_GENERIC_ARG8), |
| 80 | DEFINE_REGISTER_STUB(r8, nullptr), |
| 81 | DEFINE_REGISTER_STUB(r9, nullptr), |
| 82 | DEFINE_REGISTER_STUB(r10, nullptr), |
| 83 | DEFINE_REGISTER_STUB(r11, nullptr), |
| 84 | DEFINE_REGISTER_STUB(r12, nullptr), |
| 85 | DEFINE_REGISTER_STUB(r13, nullptr), |
| 86 | DEFINE_REGISTER_STUB(r14, nullptr), |
| 87 | DEFINE_REGISTER_STUB(r15, nullptr), |
| 88 | DEFINE_REGISTER_STUB(r16, nullptr), |
| 89 | DEFINE_REGISTER_STUB(r17, nullptr), |
| 90 | DEFINE_REGISTER_STUB(r18, nullptr), |
| 91 | DEFINE_REGISTER_STUB(r19, nullptr), |
| 92 | DEFINE_REGISTER_STUB(r20, nullptr), |
| 93 | DEFINE_REGISTER_STUB(r21, nullptr), |
| 94 | DEFINE_REGISTER_STUB(r22, nullptr), |
| 95 | DEFINE_REGISTER_STUB(r23, nullptr), |
| 96 | DEFINE_REGISTER_STUB(r24, nullptr), |
| 97 | DEFINE_REGISTER_STUB(r25, nullptr), |
| 98 | DEFINE_REGISTER_STUB(r26, "gp" ), |
| 99 | DEFINE_GENERIC_REGISTER_STUB(r27, "fp" , LLDB_REGNUM_GENERIC_FP), |
| 100 | DEFINE_GENERIC_REGISTER_STUB(r28, "sp" , LLDB_REGNUM_GENERIC_SP), |
| 101 | DEFINE_REGISTER_STUB(r29, "ilink" ), |
| 102 | DEFINE_REGISTER_STUB(r30, nullptr), |
| 103 | DEFINE_GENERIC_REGISTER_STUB(r31, "blink" , LLDB_REGNUM_GENERIC_RA), |
| 104 | DEFINE_REGISTER_STUB(r32, nullptr), |
| 105 | DEFINE_REGISTER_STUB(r33, nullptr), |
| 106 | DEFINE_REGISTER_STUB(r34, nullptr), |
| 107 | DEFINE_REGISTER_STUB(r35, nullptr), |
| 108 | DEFINE_REGISTER_STUB(r36, nullptr), |
| 109 | DEFINE_REGISTER_STUB(r37, nullptr), |
| 110 | DEFINE_REGISTER_STUB(r38, nullptr), |
| 111 | DEFINE_REGISTER_STUB(r39, nullptr), |
| 112 | DEFINE_REGISTER_STUB(r40, nullptr), |
| 113 | DEFINE_REGISTER_STUB(r41, nullptr), |
| 114 | DEFINE_REGISTER_STUB(r42, nullptr), |
| 115 | DEFINE_REGISTER_STUB(r43, nullptr), |
| 116 | DEFINE_REGISTER_STUB(r44, nullptr), |
| 117 | DEFINE_REGISTER_STUB(r45, nullptr), |
| 118 | DEFINE_REGISTER_STUB(r46, nullptr), |
| 119 | DEFINE_REGISTER_STUB(r47, nullptr), |
| 120 | DEFINE_REGISTER_STUB(r48, nullptr), |
| 121 | DEFINE_REGISTER_STUB(r49, nullptr), |
| 122 | DEFINE_REGISTER_STUB(r50, nullptr), |
| 123 | DEFINE_REGISTER_STUB(r51, nullptr), |
| 124 | DEFINE_REGISTER_STUB(r52, nullptr), |
| 125 | DEFINE_REGISTER_STUB(r53, nullptr), |
| 126 | DEFINE_REGISTER_STUB(r54, nullptr), |
| 127 | DEFINE_REGISTER_STUB(r55, nullptr), |
| 128 | DEFINE_REGISTER_STUB(r56, nullptr), |
| 129 | DEFINE_REGISTER_STUB(r57, nullptr), |
| 130 | DEFINE_REGISTER_STUB(r58, "accl" ), |
| 131 | DEFINE_REGISTER_STUB(r59, "acch" ), |
| 132 | DEFINE_REGISTER_STUB(r60, "lp_count" ), |
| 133 | DEFINE_REGISTER_STUB(r63, "pcl" ), |
| 134 | DEFINE_GENERIC_REGISTER_STUB(pc, nullptr, LLDB_REGNUM_GENERIC_PC), |
| 135 | DEFINE_GENERIC_REGISTER_STUB(status32, nullptr, LLDB_REGNUM_GENERIC_FLAGS)} }; |
| 136 | } // namespace dwarf |
| 137 | } // namespace |
| 138 | |
| 139 | const RegisterInfo *ABISysV_arc::GetRegisterInfoArray(uint32_t &count) { |
| 140 | count = dwarf::g_register_infos.size(); |
| 141 | return dwarf::g_register_infos.data(); |
| 142 | } |
| 143 | |
| 144 | size_t ABISysV_arc::GetRedZoneSize() const { return 0; } |
| 145 | |
| 146 | bool ABISysV_arc::IsRegisterFileReduced(RegisterContext ®_ctx) const { |
| 147 | if (!m_is_reg_file_reduced) { |
| 148 | const auto *const rf_build_reg = reg_ctx.GetRegisterInfoByName(reg_name: "rf_build" ); |
| 149 | |
| 150 | const auto reg_value = reg_ctx.ReadRegisterAsUnsigned(reg_info: rf_build_reg, |
| 151 | /*fail_value*/ 0); |
| 152 | // RF_BUILD "Number of Entries" bit. |
| 153 | const uint32_t rf_entries_bit = 1U << 9U; |
| 154 | m_is_reg_file_reduced = (reg_value & rf_entries_bit) != 0; |
| 155 | } |
| 156 | |
| 157 | return m_is_reg_file_reduced.value_or(u: false); |
| 158 | } |
| 159 | |
| 160 | //------------------------------------------------------------------ |
| 161 | // Static Functions |
| 162 | //------------------------------------------------------------------ |
| 163 | |
| 164 | ABISP ABISysV_arc::CreateInstance(ProcessSP process_sp, const ArchSpec &arch) { |
| 165 | return llvm::Triple::arc == arch.GetTriple().getArch() ? |
| 166 | ABISP(new ABISysV_arc(std::move(process_sp), MakeMCRegisterInfo(arch))) : |
| 167 | ABISP(); |
| 168 | } |
| 169 | |
| 170 | static const size_t word_size = 4U; |
| 171 | static const size_t reg_size = word_size; |
| 172 | |
| 173 | static inline size_t AugmentArgSize(size_t size_in_bytes) { |
| 174 | return llvm::alignTo(Value: size_in_bytes, Align: word_size); |
| 175 | } |
| 176 | |
| 177 | static size_t |
| 178 | TotalArgsSizeInWords(const llvm::ArrayRef<ABI::CallArgument> &args) { |
| 179 | size_t total_size = 0; |
| 180 | for (const auto &arg : args) |
| 181 | total_size += |
| 182 | (ABI::CallArgument::TargetValue == arg.type ? AugmentArgSize(size_in_bytes: arg.size) |
| 183 | : reg_size) / |
| 184 | word_size; |
| 185 | |
| 186 | return total_size; |
| 187 | } |
| 188 | |
| 189 | bool ABISysV_arc::PrepareTrivialCall(Thread &thread, addr_t sp, |
| 190 | addr_t func_addr, addr_t return_addr, |
| 191 | llvm::ArrayRef<addr_t> args) const { |
| 192 | // We don't use the traditional trivial call specialized for jit. |
| 193 | return false; |
| 194 | } |
| 195 | |
| 196 | bool ABISysV_arc::PrepareTrivialCall(Thread &thread, addr_t sp, addr_t pc, |
| 197 | addr_t ra, llvm::Type &prototype, |
| 198 | llvm::ArrayRef<ABI::CallArgument> args) const { |
| 199 | auto reg_ctx = thread.GetRegisterContext(); |
| 200 | if (!reg_ctx) |
| 201 | return false; |
| 202 | |
| 203 | uint32_t pc_reg = reg_ctx->ConvertRegisterKindToRegisterNumber( |
| 204 | kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_PC); |
| 205 | if (pc_reg == LLDB_INVALID_REGNUM) |
| 206 | return false; |
| 207 | |
| 208 | uint32_t ra_reg = reg_ctx->ConvertRegisterKindToRegisterNumber( |
| 209 | kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_RA); |
| 210 | if (ra_reg == LLDB_INVALID_REGNUM) |
| 211 | return false; |
| 212 | |
| 213 | uint32_t sp_reg = reg_ctx->ConvertRegisterKindToRegisterNumber( |
| 214 | kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_SP); |
| 215 | if (sp_reg == LLDB_INVALID_REGNUM) |
| 216 | return false; |
| 217 | |
| 218 | Status error; |
| 219 | ProcessSP process = thread.GetProcess(); |
| 220 | if (!process) |
| 221 | return false; |
| 222 | |
| 223 | // Push host data onto target. |
| 224 | for (const auto &arg : args) { |
| 225 | // Skip over target values. |
| 226 | if (arg.type == ABI::CallArgument::TargetValue) |
| 227 | continue; |
| 228 | |
| 229 | // Create space on the stack for this data 4-byte aligned. |
| 230 | sp -= AugmentArgSize(size_in_bytes: arg.size); |
| 231 | |
| 232 | if (process->WriteMemory(vm_addr: sp, buf: arg.data_up.get(), size: arg.size, error) < arg.size |
| 233 | || error.Fail()) |
| 234 | return false; |
| 235 | |
| 236 | // Update the argument with the target pointer. |
| 237 | *const_cast<addr_t *>(&arg.value) = sp; |
| 238 | } |
| 239 | |
| 240 | // Make sure number of parameters matches prototype. |
| 241 | assert(!prototype.isFunctionVarArg()); |
| 242 | assert(prototype.getFunctionNumParams() == args.size()); |
| 243 | |
| 244 | const size_t regs_for_args_count = IsRegisterFileReduced(reg_ctx&: *reg_ctx) ? 4U : 8U; |
| 245 | |
| 246 | // Number of arguments passed on stack. |
| 247 | auto args_size = TotalArgsSizeInWords(args); |
| 248 | auto on_stack = |
| 249 | args_size <= regs_for_args_count ? 0 : args_size - regs_for_args_count; |
| 250 | auto offset = on_stack * word_size; |
| 251 | |
| 252 | uint8_t reg_value[reg_size]; |
| 253 | size_t reg_index = LLDB_REGNUM_GENERIC_ARG1; |
| 254 | |
| 255 | for (const auto &arg : args) { |
| 256 | auto value = reinterpret_cast<const uint8_t *>(&arg.value); |
| 257 | auto size = |
| 258 | ABI::CallArgument::TargetValue == arg.type ? arg.size : reg_size; |
| 259 | |
| 260 | // Pass arguments via registers. |
| 261 | while (size > 0 && reg_index < regs_for_args_count) { |
| 262 | size_t byte_index = 0; |
| 263 | auto end = size < reg_size ? size : reg_size; |
| 264 | |
| 265 | while (byte_index < end) { |
| 266 | reg_value[byte_index++] = *(value++); |
| 267 | --size; |
| 268 | } |
| 269 | |
| 270 | while (byte_index < reg_size) { |
| 271 | reg_value[byte_index++] = 0; |
| 272 | } |
| 273 | |
| 274 | RegisterValue reg_val_obj(llvm::ArrayRef(reg_value, reg_size), |
| 275 | eByteOrderLittle); |
| 276 | if (!reg_ctx->WriteRegister( |
| 277 | reg_info: reg_ctx->GetRegisterInfo(reg_kind: eRegisterKindGeneric, reg_num: reg_index), |
| 278 | reg_value: reg_val_obj)) |
| 279 | return false; |
| 280 | |
| 281 | // NOTE: It's unsafe to iterate through LLDB_REGNUM_GENERICs. |
| 282 | ++reg_index; |
| 283 | } |
| 284 | |
| 285 | if (reg_index < regs_for_args_count || size == 0) |
| 286 | continue; |
| 287 | |
| 288 | // Remaining arguments are passed on the stack. |
| 289 | if (process->WriteMemory(vm_addr: sp - offset, buf: value, size, error) < size || |
| 290 | !error.Success()) |
| 291 | return false; |
| 292 | |
| 293 | offset -= AugmentArgSize(size_in_bytes: size); |
| 294 | } |
| 295 | |
| 296 | // Set stack pointer immediately below arguments. |
| 297 | sp -= on_stack * word_size; |
| 298 | |
| 299 | // Update registers with current function call state. |
| 300 | reg_ctx->WriteRegisterFromUnsigned(reg: pc_reg, uval: pc); |
| 301 | reg_ctx->WriteRegisterFromUnsigned(reg: ra_reg, uval: ra); |
| 302 | reg_ctx->WriteRegisterFromUnsigned(reg: sp_reg, uval: sp); |
| 303 | |
| 304 | return true; |
| 305 | } |
| 306 | |
| 307 | bool ABISysV_arc::GetArgumentValues(Thread &thread, ValueList &values) const { |
| 308 | return false; |
| 309 | } |
| 310 | |
| 311 | Status ABISysV_arc::SetReturnValueObject(StackFrameSP &frame_sp, |
| 312 | ValueObjectSP &new_value_sp) { |
| 313 | Status result; |
| 314 | if (!new_value_sp) { |
| 315 | result = Status::FromErrorString(str: "Empty value object for return value." ); |
| 316 | return result; |
| 317 | } |
| 318 | |
| 319 | CompilerType compiler_type = new_value_sp->GetCompilerType(); |
| 320 | if (!compiler_type) { |
| 321 | result = Status::FromErrorString(str: "Null clang type for return value." ); |
| 322 | return result; |
| 323 | } |
| 324 | |
| 325 | auto ®_ctx = *frame_sp->GetThread()->GetRegisterContext(); |
| 326 | |
| 327 | bool is_signed = false; |
| 328 | if (!compiler_type.IsIntegerOrEnumerationType(is_signed) && |
| 329 | !compiler_type.IsPointerType()) { |
| 330 | result = Status::FromErrorString( |
| 331 | str: "We don't support returning other types at present" ); |
| 332 | return result; |
| 333 | } |
| 334 | |
| 335 | DataExtractor data; |
| 336 | size_t num_bytes = new_value_sp->GetData(data, error&: result); |
| 337 | |
| 338 | if (result.Fail()) { |
| 339 | result = Status::FromErrorStringWithFormat( |
| 340 | format: "Couldn't convert return value to raw data: %s" , result.AsCString()); |
| 341 | return result; |
| 342 | } |
| 343 | |
| 344 | if (num_bytes <= 2 * reg_size) { |
| 345 | offset_t offset = 0; |
| 346 | uint64_t raw_value = data.GetMaxU64(offset_ptr: &offset, byte_size: num_bytes); |
| 347 | |
| 348 | auto reg_info = |
| 349 | reg_ctx.GetRegisterInfo(reg_kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); |
| 350 | if (!reg_ctx.WriteRegisterFromUnsigned(reg_info, uval: raw_value)) { |
| 351 | result = Status::FromErrorStringWithFormat( |
| 352 | format: "Couldn't write value to register %s" , reg_info->name); |
| 353 | return result; |
| 354 | } |
| 355 | |
| 356 | if (num_bytes <= reg_size) |
| 357 | return result; // Successfully written. |
| 358 | |
| 359 | raw_value >>= 32; |
| 360 | reg_info = |
| 361 | reg_ctx.GetRegisterInfo(reg_kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG2); |
| 362 | if (!reg_ctx.WriteRegisterFromUnsigned(reg_info, uval: raw_value)) { |
| 363 | result = Status::FromErrorStringWithFormat( |
| 364 | format: "Couldn't write value to register %s" , reg_info->name); |
| 365 | } |
| 366 | |
| 367 | return result; |
| 368 | } |
| 369 | |
| 370 | result = Status::FromErrorString( |
| 371 | str: "We don't support returning large integer values at present." ); |
| 372 | return result; |
| 373 | } |
| 374 | |
| 375 | template <typename T> |
| 376 | static void SetInteger(Scalar &scalar, uint64_t raw_value, bool is_signed) { |
| 377 | raw_value &= std::numeric_limits<T>::max(); |
| 378 | if (is_signed) |
| 379 | scalar = static_cast<typename std::make_signed<T>::type>(raw_value); |
| 380 | else |
| 381 | scalar = static_cast<T>(raw_value); |
| 382 | } |
| 383 | |
| 384 | static bool SetSizedInteger(Scalar &scalar, uint64_t raw_value, |
| 385 | uint8_t size_in_bytes, bool is_signed) { |
| 386 | switch (size_in_bytes) { |
| 387 | default: |
| 388 | return false; |
| 389 | |
| 390 | case sizeof(uint64_t): |
| 391 | SetInteger<uint64_t>(scalar, raw_value, is_signed); |
| 392 | break; |
| 393 | |
| 394 | case sizeof(uint32_t): |
| 395 | SetInteger<uint32_t>(scalar, raw_value, is_signed); |
| 396 | break; |
| 397 | |
| 398 | case sizeof(uint16_t): |
| 399 | SetInteger<uint16_t>(scalar, raw_value, is_signed); |
| 400 | break; |
| 401 | |
| 402 | case sizeof(uint8_t): |
| 403 | SetInteger<uint8_t>(scalar, raw_value, is_signed); |
| 404 | break; |
| 405 | } |
| 406 | |
| 407 | return true; |
| 408 | } |
| 409 | |
| 410 | static bool SetSizedFloat(Scalar &scalar, uint64_t raw_value, |
| 411 | uint8_t size_in_bytes) { |
| 412 | switch (size_in_bytes) { |
| 413 | default: |
| 414 | return false; |
| 415 | |
| 416 | case sizeof(uint64_t): |
| 417 | scalar = *reinterpret_cast<double *>(&raw_value); |
| 418 | break; |
| 419 | |
| 420 | case sizeof(uint32_t): |
| 421 | scalar = *reinterpret_cast<float *>(&raw_value); |
| 422 | break; |
| 423 | } |
| 424 | |
| 425 | return true; |
| 426 | } |
| 427 | |
| 428 | static uint64_t ReadRawValue(const RegisterContextSP ®_ctx, |
| 429 | uint8_t size_in_bytes) { |
| 430 | auto reg_info_r0 = |
| 431 | reg_ctx->GetRegisterInfo(reg_kind: eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); |
| 432 | |
| 433 | // Extract the register context so we can read arguments from registers. |
| 434 | uint64_t raw_value = |
| 435 | reg_ctx->ReadRegisterAsUnsigned(reg_info: reg_info_r0, fail_value: 0) & UINT32_MAX; |
| 436 | |
| 437 | if (sizeof(uint64_t) == size_in_bytes) |
| 438 | raw_value |= (reg_ctx->ReadRegisterAsUnsigned( |
| 439 | reg_info: reg_ctx->GetRegisterInfo(reg_kind: eRegisterKindGeneric, |
| 440 | LLDB_REGNUM_GENERIC_ARG2), fail_value: 0) & |
| 441 | UINT64_MAX) << 32U; |
| 442 | |
| 443 | return raw_value; |
| 444 | } |
| 445 | |
| 446 | ValueObjectSP |
| 447 | ABISysV_arc::GetReturnValueObjectSimple(Thread &thread, |
| 448 | CompilerType &compiler_type) const { |
| 449 | if (!compiler_type) |
| 450 | return ValueObjectSP(); |
| 451 | |
| 452 | auto reg_ctx = thread.GetRegisterContext(); |
| 453 | if (!reg_ctx) |
| 454 | return ValueObjectSP(); |
| 455 | |
| 456 | Value value; |
| 457 | value.SetCompilerType(compiler_type); |
| 458 | |
| 459 | const uint32_t type_flags = compiler_type.GetTypeInfo(); |
| 460 | // Integer return type. |
| 461 | if (type_flags & eTypeIsInteger) { |
| 462 | const size_t byte_size = |
| 463 | llvm::expectedToOptional(E: compiler_type.GetByteSize(exe_scope: &thread)) |
| 464 | .value_or(u: 0); |
| 465 | auto raw_value = ReadRawValue(reg_ctx, size_in_bytes: byte_size); |
| 466 | |
| 467 | const bool is_signed = (type_flags & eTypeIsSigned) != 0; |
| 468 | if (!SetSizedInteger(scalar&: value.GetScalar(), raw_value, size_in_bytes: byte_size, is_signed)) |
| 469 | return ValueObjectSP(); |
| 470 | |
| 471 | value.SetValueType(Value::ValueType::Scalar); |
| 472 | } |
| 473 | // Pointer return type. |
| 474 | else if (type_flags & eTypeIsPointer) { |
| 475 | auto reg_info_r0 = reg_ctx->GetRegisterInfo(reg_kind: eRegisterKindGeneric, |
| 476 | LLDB_REGNUM_GENERIC_ARG1); |
| 477 | value.GetScalar() = reg_ctx->ReadRegisterAsUnsigned(reg_info: reg_info_r0, fail_value: 0); |
| 478 | |
| 479 | value.SetValueType(Value::ValueType::Scalar); |
| 480 | } |
| 481 | // Floating point return type. |
| 482 | else if (type_flags & eTypeIsFloat) { |
| 483 | uint32_t float_count = 0; |
| 484 | bool is_complex = false; |
| 485 | |
| 486 | if (compiler_type.IsFloatingPointType(count&: float_count, is_complex) && |
| 487 | 1 == float_count && !is_complex) { |
| 488 | const size_t byte_size = |
| 489 | llvm::expectedToOptional(E: compiler_type.GetByteSize(exe_scope: &thread)) |
| 490 | .value_or(u: 0); |
| 491 | auto raw_value = ReadRawValue(reg_ctx, size_in_bytes: byte_size); |
| 492 | |
| 493 | if (!SetSizedFloat(scalar&: value.GetScalar(), raw_value, size_in_bytes: byte_size)) |
| 494 | return ValueObjectSP(); |
| 495 | } |
| 496 | } |
| 497 | // Unsupported return type. |
| 498 | else |
| 499 | return ValueObjectSP(); |
| 500 | |
| 501 | return ValueObjectConstResult::Create(exe_scope: thread.GetStackFrameAtIndex(idx: 0).get(), |
| 502 | value, name: ConstString("" )); |
| 503 | } |
| 504 | |
| 505 | ValueObjectSP ABISysV_arc::GetReturnValueObjectImpl( |
| 506 | Thread &thread, CompilerType &return_compiler_type) const { |
| 507 | ValueObjectSP return_valobj_sp; |
| 508 | |
| 509 | if (!return_compiler_type) |
| 510 | return return_valobj_sp; |
| 511 | |
| 512 | ExecutionContext exe_ctx(thread.shared_from_this()); |
| 513 | return GetReturnValueObjectSimple(thread, compiler_type&: return_compiler_type); |
| 514 | } |
| 515 | |
| 516 | ValueObjectSP ABISysV_arc::GetReturnValueObjectImpl(Thread &thread, |
| 517 | llvm::Type &retType) const { |
| 518 | auto reg_ctx = thread.GetRegisterContext(); |
| 519 | if (!reg_ctx) |
| 520 | return ValueObjectSP(); |
| 521 | |
| 522 | Value value; |
| 523 | // Void return type. |
| 524 | if (retType.isVoidTy()) { |
| 525 | value.GetScalar() = 0; |
| 526 | } |
| 527 | // Integer return type. |
| 528 | else if (retType.isIntegerTy()) { |
| 529 | size_t byte_size = retType.getPrimitiveSizeInBits(); |
| 530 | if (1 != byte_size) // For boolean type. |
| 531 | byte_size /= CHAR_BIT; |
| 532 | |
| 533 | auto raw_value = ReadRawValue(reg_ctx, size_in_bytes: byte_size); |
| 534 | |
| 535 | const bool is_signed = false; // IR Type doesn't provide this info. |
| 536 | if (!SetSizedInteger(scalar&: value.GetScalar(), raw_value, size_in_bytes: byte_size, is_signed)) |
| 537 | return ValueObjectSP(); |
| 538 | } |
| 539 | // Pointer return type. |
| 540 | else if (retType.isPointerTy()) { |
| 541 | auto reg_info_r0 = reg_ctx->GetRegisterInfo(reg_kind: eRegisterKindGeneric, |
| 542 | LLDB_REGNUM_GENERIC_ARG1); |
| 543 | value.GetScalar() = reg_ctx->ReadRegisterAsUnsigned(reg_info: reg_info_r0, fail_value: 0); |
| 544 | value.SetValueType(Value::ValueType::Scalar); |
| 545 | } |
| 546 | // Floating point return type. |
| 547 | else if (retType.isFloatingPointTy()) { |
| 548 | const size_t byte_size = retType.getPrimitiveSizeInBits() / CHAR_BIT; |
| 549 | auto raw_value = ReadRawValue(reg_ctx, size_in_bytes: byte_size); |
| 550 | |
| 551 | if (!SetSizedFloat(scalar&: value.GetScalar(), raw_value, size_in_bytes: byte_size)) |
| 552 | return ValueObjectSP(); |
| 553 | } |
| 554 | // Unsupported return type. |
| 555 | else |
| 556 | return ValueObjectSP(); |
| 557 | |
| 558 | return ValueObjectConstResult::Create(exe_scope: thread.GetStackFrameAtIndex(idx: 0).get(), |
| 559 | value, name: ConstString("" )); |
| 560 | } |
| 561 | |
| 562 | UnwindPlanSP ABISysV_arc::CreateFunctionEntryUnwindPlan() { |
| 563 | UnwindPlan::Row row; |
| 564 | |
| 565 | // Our Call Frame Address is the stack pointer value. |
| 566 | row.GetCFAValue().SetIsRegisterPlusOffset(reg_num: dwarf::sp, offset: 0); |
| 567 | |
| 568 | // The previous PC is in the BLINK, all other registers are the same. |
| 569 | row.SetRegisterLocationToRegister(reg_num: dwarf::pc, other_reg_num: dwarf::blink, can_replace: true); |
| 570 | |
| 571 | auto plan_sp = std::make_shared<UnwindPlan>(args: eRegisterKindDWARF); |
| 572 | plan_sp->AppendRow(row: std::move(row)); |
| 573 | plan_sp->SetSourceName("arc at-func-entry default" ); |
| 574 | plan_sp->SetSourcedFromCompiler(eLazyBoolNo); |
| 575 | return plan_sp; |
| 576 | } |
| 577 | |
| 578 | UnwindPlanSP ABISysV_arc::CreateDefaultUnwindPlan() { return nullptr; } |
| 579 | |
| 580 | bool ABISysV_arc::RegisterIsVolatile(const RegisterInfo *reg_info) { |
| 581 | if (nullptr == reg_info) |
| 582 | return false; |
| 583 | |
| 584 | // Volatile registers are: r0..r12. |
| 585 | uint32_t regnum = reg_info->kinds[eRegisterKindDWARF]; |
| 586 | if (regnum <= 12) |
| 587 | return true; |
| 588 | |
| 589 | static const std::string ra_reg_name = "blink" ; |
| 590 | return ra_reg_name == reg_info->name; |
| 591 | } |
| 592 | |
| 593 | void ABISysV_arc::Initialize() { |
| 594 | PluginManager::RegisterPlugin(name: GetPluginNameStatic(), |
| 595 | description: "System V ABI for ARC targets" , create_callback: CreateInstance); |
| 596 | } |
| 597 | |
| 598 | void ABISysV_arc::Terminate() { |
| 599 | PluginManager::UnregisterPlugin(create_callback: CreateInstance); |
| 600 | } |
| 601 | |