180814287SRaphael Isemann //===-- CXXFunctionPointer.cpp---------------------------------------------===//
2419d7918SEnrico Granata //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6419d7918SEnrico Granata //
7419d7918SEnrico Granata //===----------------------------------------------------------------------===//
8419d7918SEnrico Granata 
9c6bbb8b6SEnrico Granata #include "lldb/DataFormatters/CXXFunctionPointer.h"
10c6bbb8b6SEnrico Granata 
11d717cc9fSEnrico Granata #include "lldb/Core/ValueObject.h"
12*223e8ca0SJason Molenda #include "lldb/Target/ABI.h"
13419d7918SEnrico Granata #include "lldb/Target/SectionLoadList.h"
14419d7918SEnrico Granata #include "lldb/Target/Target.h"
15bf9a7730SZachary Turner #include "lldb/Utility/Stream.h"
16419d7918SEnrico Granata 
17419d7918SEnrico Granata #include <string>
18419d7918SEnrico Granata 
19419d7918SEnrico Granata using namespace lldb;
20419d7918SEnrico Granata using namespace lldb_private;
21419d7918SEnrico Granata using namespace lldb_private::formatters;
22419d7918SEnrico Granata 
CXXFunctionPointerSummaryProvider(ValueObject & valobj,Stream & stream,const TypeSummaryOptions & options)23b9c1b51eSKate Stone bool lldb_private::formatters::CXXFunctionPointerSummaryProvider(
24b9c1b51eSKate Stone     ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
25419d7918SEnrico Granata   std::string destination;
26419d7918SEnrico Granata   StreamString sstr;
27419d7918SEnrico Granata   AddressType func_ptr_address_type = eAddressTypeInvalid;
28419d7918SEnrico Granata   addr_t func_ptr_address = valobj.GetPointerValue(&func_ptr_address_type);
29b9c1b51eSKate Stone   if (func_ptr_address != 0 && func_ptr_address != LLDB_INVALID_ADDRESS) {
30b9c1b51eSKate Stone     switch (func_ptr_address_type) {
31419d7918SEnrico Granata     case eAddressTypeInvalid:
32419d7918SEnrico Granata     case eAddressTypeFile:
33419d7918SEnrico Granata     case eAddressTypeHost:
34419d7918SEnrico Granata       break;
35419d7918SEnrico Granata 
36b9c1b51eSKate Stone     case eAddressTypeLoad: {
37419d7918SEnrico Granata       ExecutionContext exe_ctx(valobj.GetExecutionContextRef());
38419d7918SEnrico Granata 
39419d7918SEnrico Granata       Address so_addr;
40419d7918SEnrico Granata       Target *target = exe_ctx.GetTargetPtr();
41a6682a41SJonas Devlieghere       if (target && !target->GetSectionLoadList().IsEmpty()) {
42*223e8ca0SJason Molenda         target->GetSectionLoadList().ResolveLoadAddress(func_ptr_address,
43*223e8ca0SJason Molenda                                                         so_addr);
44*223e8ca0SJason Molenda         if (so_addr.GetSection() == nullptr) {
45*223e8ca0SJason Molenda           // If we have an address that doesn't correspond to any symbol,
46*223e8ca0SJason Molenda           // it might have authentication bits.  Strip them & see if it
47*223e8ca0SJason Molenda           // now points to a symbol -- if so, do the SymbolContext lookup
48*223e8ca0SJason Molenda           // based on the stripped address.
49*223e8ca0SJason Molenda           // If we find a symbol with the ptrauth bits stripped, print the
50*223e8ca0SJason Molenda           // raw value into the stream, and replace the Address with the
51*223e8ca0SJason Molenda           // one that points to a symbol for a fuller description.
52*223e8ca0SJason Molenda           if (Process *process = exe_ctx.GetProcessPtr()) {
53*223e8ca0SJason Molenda             if (ABISP abi_sp = process->GetABI()) {
54*223e8ca0SJason Molenda               addr_t fixed_addr = abi_sp->FixCodeAddress(func_ptr_address);
55*223e8ca0SJason Molenda               if (fixed_addr != func_ptr_address) {
56*223e8ca0SJason Molenda                 Address test_address;
57*223e8ca0SJason Molenda                 test_address.SetLoadAddress(fixed_addr, target);
58*223e8ca0SJason Molenda                 if (test_address.GetSection() != nullptr) {
59*223e8ca0SJason Molenda                   int addrsize = target->GetArchitecture().GetAddressByteSize();
60*223e8ca0SJason Molenda                   sstr.Printf("actual=0x%*.*" PRIx64 " ", addrsize * 2,
61*223e8ca0SJason Molenda                               addrsize * 2, fixed_addr);
62*223e8ca0SJason Molenda                   so_addr = test_address;
63*223e8ca0SJason Molenda                 }
64*223e8ca0SJason Molenda               }
65*223e8ca0SJason Molenda             }
66*223e8ca0SJason Molenda           }
67*223e8ca0SJason Molenda         }
68*223e8ca0SJason Molenda 
69*223e8ca0SJason Molenda         if (so_addr.IsValid()) {
70b9c1b51eSKate Stone           so_addr.Dump(&sstr, exe_ctx.GetBestExecutionContextScope(),
71419d7918SEnrico Granata                        Address::DumpStyleResolvedDescription,
72419d7918SEnrico Granata                        Address::DumpStyleSectionNameOffset);
73419d7918SEnrico Granata         }
74419d7918SEnrico Granata       }
75b9c1b51eSKate Stone     } break;
76419d7918SEnrico Granata     }
77419d7918SEnrico Granata   }
78b9c1b51eSKate Stone   if (sstr.GetSize() > 0) {
79419d7918SEnrico Granata     stream.Printf("(%s)", sstr.GetData());
80419d7918SEnrico Granata     return true;
81b9c1b51eSKate Stone   } else
82419d7918SEnrico Granata     return false;
83419d7918SEnrico Granata }
84