| 1 | //===-- CXXFunctionPointer.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 "lldb/DataFormatters/CXXFunctionPointer.h" |
| 10 | |
| 11 | #include "lldb/Target/ABI.h" |
| 12 | #include "lldb/Target/SectionLoadList.h" |
| 13 | #include "lldb/Target/Target.h" |
| 14 | #include "lldb/Utility/Stream.h" |
| 15 | #include "lldb/ValueObject/ValueObject.h" |
| 16 | #include "lldb/lldb-enumerations.h" |
| 17 | |
| 18 | #include <string> |
| 19 | |
| 20 | using namespace lldb; |
| 21 | using namespace lldb_private; |
| 22 | using namespace lldb_private::formatters; |
| 23 | |
| 24 | bool lldb_private::formatters::CXXFunctionPointerSummaryProvider( |
| 25 | ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) { |
| 26 | StreamString sstr; |
| 27 | auto [func_ptr_address, func_ptr_address_type] = valobj.GetPointerValue(); |
| 28 | if (func_ptr_address != 0 && func_ptr_address != LLDB_INVALID_ADDRESS) { |
| 29 | switch (func_ptr_address_type) { |
| 30 | case eAddressTypeInvalid: |
| 31 | case eAddressTypeFile: |
| 32 | case eAddressTypeHost: |
| 33 | break; |
| 34 | |
| 35 | case eAddressTypeLoad: { |
| 36 | ExecutionContext exe_ctx(valobj.GetExecutionContextRef()); |
| 37 | |
| 38 | Address so_addr; |
| 39 | Target *target = exe_ctx.GetTargetPtr(); |
| 40 | if (target && target->HasLoadedSections()) { |
| 41 | target->ResolveLoadAddress(load_addr: func_ptr_address, so_addr); |
| 42 | if (so_addr.GetSection() == nullptr) { |
| 43 | // If we have an address that doesn't correspond to any symbol, |
| 44 | // it might have authentication bits. Strip them & see if it |
| 45 | // now points to a symbol -- if so, do the SymbolContext lookup |
| 46 | // based on the stripped address. |
| 47 | // If we find a symbol with the ptrauth bits stripped, print the |
| 48 | // raw value into the stream, and replace the Address with the |
| 49 | // one that points to a symbol for a fuller description. |
| 50 | if (Process *process = exe_ctx.GetProcessPtr()) { |
| 51 | if (ABISP abi_sp = process->GetABI()) { |
| 52 | addr_t fixed_addr = abi_sp->FixCodeAddress(pc: func_ptr_address); |
| 53 | if (fixed_addr != func_ptr_address) { |
| 54 | Address test_address; |
| 55 | test_address.SetLoadAddress(load_addr: fixed_addr, target); |
| 56 | if (test_address.GetSection() != nullptr) { |
| 57 | int addrsize = target->GetArchitecture().GetAddressByteSize(); |
| 58 | sstr.Printf(format: "actual=0x%*.*" PRIx64 " " , addrsize * 2, |
| 59 | addrsize * 2, fixed_addr); |
| 60 | so_addr = test_address; |
| 61 | } |
| 62 | } |
| 63 | } |
| 64 | } |
| 65 | } |
| 66 | |
| 67 | if (so_addr.IsValid()) { |
| 68 | so_addr.Dump(s: &sstr, exe_scope: exe_ctx.GetBestExecutionContextScope(), |
| 69 | style: Address::DumpStyleResolvedDescription, |
| 70 | fallback_style: Address::DumpStyleSectionNameOffset); |
| 71 | } |
| 72 | } |
| 73 | } break; |
| 74 | } |
| 75 | } |
| 76 | if (sstr.GetSize() > 0) { |
| 77 | if (valobj.GetValueType() == lldb::eValueTypeVTableEntry) |
| 78 | stream.PutCString(cstr: sstr.GetData()); |
| 79 | else |
| 80 | stream.Printf(format: "(%s)" , sstr.GetData()); |
| 81 | return true; |
| 82 | } else |
| 83 | return false; |
| 84 | } |
| 85 | |