xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/SymbolFile/PDB/PDBLocationToDWARFExpression.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1dda28197Spatrick //===-- PDBLocationToDWARFExpression.cpp ----------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "PDBLocationToDWARFExpression.h"
10061da546Spatrick 
11061da546Spatrick #include "lldb/Core/Section.h"
12061da546Spatrick #include "lldb/Core/StreamBuffer.h"
13061da546Spatrick #include "lldb/Core/dwarf.h"
14061da546Spatrick #include "lldb/Expression/DWARFExpression.h"
15061da546Spatrick #include "lldb/Symbol/Variable.h"
16061da546Spatrick #include "lldb/Utility/DataBufferHeap.h"
17061da546Spatrick 
18061da546Spatrick #include "llvm/DebugInfo/CodeView/CodeView.h"
19061da546Spatrick #include "llvm/DebugInfo/PDB/IPDBSession.h"
20061da546Spatrick #include "llvm/DebugInfo/PDB/PDBSymbolData.h"
21061da546Spatrick 
22061da546Spatrick #include "Plugins/SymbolFile/NativePDB/CodeViewRegisterMapping.h"
23061da546Spatrick #include "Plugins/SymbolFile/NativePDB/PdbFPOProgramToDWARFExpression.h"
24061da546Spatrick 
25061da546Spatrick using namespace lldb;
26061da546Spatrick using namespace lldb_private;
27061da546Spatrick using namespace lldb_private::npdb;
28*f6aab3d8Srobert using namespace lldb_private::dwarf;
29061da546Spatrick using namespace llvm::pdb;
30061da546Spatrick 
31061da546Spatrick static std::unique_ptr<IPDBFrameData>
GetCorrespondingFrameData(const IPDBSession & session,const Variable::RangeList & ranges)32061da546Spatrick GetCorrespondingFrameData(const IPDBSession &session,
33061da546Spatrick                           const Variable::RangeList &ranges) {
34061da546Spatrick   auto enumFrameData = session.getFrameData();
35061da546Spatrick   if (!enumFrameData)
36061da546Spatrick     return nullptr;
37061da546Spatrick 
38061da546Spatrick   std::unique_ptr<IPDBFrameData> found;
39061da546Spatrick   while (auto fd = enumFrameData->getNext()) {
40061da546Spatrick     Range<lldb::addr_t, lldb::addr_t> fdRange(fd->getVirtualAddress(),
41061da546Spatrick                                               fd->getLengthBlock());
42061da546Spatrick 
43061da546Spatrick     for (size_t i = 0; i < ranges.GetSize(); i++) {
44061da546Spatrick       auto range = ranges.GetEntryAtIndex(i);
45061da546Spatrick       if (!range)
46061da546Spatrick         continue;
47061da546Spatrick 
48061da546Spatrick       if (!range->DoesIntersect(fdRange))
49061da546Spatrick         continue;
50061da546Spatrick 
51061da546Spatrick       found = std::move(fd);
52061da546Spatrick 
53061da546Spatrick       break;
54061da546Spatrick     }
55061da546Spatrick   }
56061da546Spatrick 
57061da546Spatrick   return found;
58061da546Spatrick }
59061da546Spatrick 
EmitVFrameEvaluationDWARFExpression(llvm::StringRef program,llvm::Triple::ArchType arch_type,Stream & stream)60061da546Spatrick static bool EmitVFrameEvaluationDWARFExpression(
61061da546Spatrick     llvm::StringRef program, llvm::Triple::ArchType arch_type, Stream &stream) {
62061da546Spatrick   // VFrame value always stored in $TO pseudo-register
63061da546Spatrick   return TranslateFPOProgramToDWARFExpression(program, "$T0", arch_type,
64061da546Spatrick                                               stream);
65061da546Spatrick }
66061da546Spatrick 
ConvertPDBLocationToDWARFExpression(ModuleSP module,const PDBSymbolData & symbol,const Variable::RangeList & ranges,bool & is_constant)67061da546Spatrick DWARFExpression ConvertPDBLocationToDWARFExpression(
68061da546Spatrick     ModuleSP module, const PDBSymbolData &symbol,
69061da546Spatrick     const Variable::RangeList &ranges, bool &is_constant) {
70061da546Spatrick   is_constant = true;
71061da546Spatrick 
72061da546Spatrick   if (!module)
73061da546Spatrick     return DWARFExpression();
74061da546Spatrick 
75061da546Spatrick   const ArchSpec &architecture = module->GetArchitecture();
76061da546Spatrick   llvm::Triple::ArchType arch_type = architecture.GetMachine();
77061da546Spatrick   ByteOrder byte_order = architecture.GetByteOrder();
78061da546Spatrick   uint32_t address_size = architecture.GetAddressByteSize();
79061da546Spatrick   uint32_t byte_size = architecture.GetDataByteSize();
80061da546Spatrick   if (byte_order == eByteOrderInvalid || address_size == 0)
81061da546Spatrick     return DWARFExpression();
82061da546Spatrick 
83061da546Spatrick   RegisterKind register_kind = eRegisterKindDWARF;
84061da546Spatrick   StreamBuffer<32> stream(Stream::eBinary, address_size, byte_order);
85061da546Spatrick   switch (symbol.getLocationType()) {
86061da546Spatrick   case PDB_LocType::Static:
87061da546Spatrick   case PDB_LocType::TLS: {
88061da546Spatrick     stream.PutHex8(DW_OP_addr);
89061da546Spatrick 
90061da546Spatrick     SectionList *section_list = module->GetSectionList();
91061da546Spatrick     if (!section_list)
92061da546Spatrick       return DWARFExpression();
93061da546Spatrick 
94061da546Spatrick     uint32_t section_id = symbol.getAddressSection();
95061da546Spatrick 
96061da546Spatrick     auto section = section_list->FindSectionByID(section_id);
97061da546Spatrick     if (!section)
98061da546Spatrick       return DWARFExpression();
99061da546Spatrick 
100061da546Spatrick     uint32_t offset = symbol.getAddressOffset();
101061da546Spatrick     stream.PutMaxHex64(section->GetFileAddress() + offset, address_size,
102061da546Spatrick                        byte_order);
103061da546Spatrick 
104061da546Spatrick     is_constant = false;
105061da546Spatrick 
106061da546Spatrick     break;
107061da546Spatrick   }
108061da546Spatrick   case PDB_LocType::RegRel: {
109061da546Spatrick     uint32_t reg_num;
110061da546Spatrick     auto reg_id = symbol.getRegisterId();
111061da546Spatrick     if (reg_id == llvm::codeview::RegisterId::VFRAME) {
112061da546Spatrick       if (auto fd = GetCorrespondingFrameData(symbol.getSession(), ranges)) {
113061da546Spatrick         if (EmitVFrameEvaluationDWARFExpression(fd->getProgram(), arch_type,
114061da546Spatrick                                                 stream)) {
115061da546Spatrick           int32_t offset = symbol.getOffset();
116061da546Spatrick           stream.PutHex8(DW_OP_consts);
117061da546Spatrick           stream.PutSLEB128(offset);
118061da546Spatrick           stream.PutHex8(DW_OP_plus);
119061da546Spatrick 
120061da546Spatrick           register_kind = eRegisterKindLLDB;
121061da546Spatrick 
122061da546Spatrick           is_constant = false;
123061da546Spatrick           break;
124061da546Spatrick         }
125061da546Spatrick       }
126061da546Spatrick 
127061da546Spatrick       register_kind = eRegisterKindGeneric;
128061da546Spatrick       reg_num = LLDB_REGNUM_GENERIC_FP;
129061da546Spatrick     } else {
130061da546Spatrick       register_kind = eRegisterKindLLDB;
131061da546Spatrick       reg_num = GetLLDBRegisterNumber(arch_type, reg_id);
132061da546Spatrick       if (reg_num == LLDB_INVALID_REGNUM)
133061da546Spatrick         return DWARFExpression();
134061da546Spatrick     }
135061da546Spatrick 
136061da546Spatrick     if (reg_num > 31) {
137061da546Spatrick       stream.PutHex8(DW_OP_bregx);
138061da546Spatrick       stream.PutULEB128(reg_num);
139061da546Spatrick     } else
140061da546Spatrick       stream.PutHex8(DW_OP_breg0 + reg_num);
141061da546Spatrick 
142061da546Spatrick     int32_t offset = symbol.getOffset();
143061da546Spatrick     stream.PutSLEB128(offset);
144061da546Spatrick 
145061da546Spatrick     is_constant = false;
146061da546Spatrick 
147061da546Spatrick     break;
148061da546Spatrick   }
149061da546Spatrick   case PDB_LocType::Enregistered: {
150061da546Spatrick     register_kind = eRegisterKindLLDB;
151061da546Spatrick     uint32_t reg_num = GetLLDBRegisterNumber(arch_type, symbol.getRegisterId());
152061da546Spatrick     if (reg_num == LLDB_INVALID_REGNUM)
153061da546Spatrick       return DWARFExpression();
154061da546Spatrick 
155061da546Spatrick     if (reg_num > 31) {
156061da546Spatrick       stream.PutHex8(DW_OP_regx);
157061da546Spatrick       stream.PutULEB128(reg_num);
158061da546Spatrick     } else
159061da546Spatrick       stream.PutHex8(DW_OP_reg0 + reg_num);
160061da546Spatrick 
161061da546Spatrick     is_constant = false;
162061da546Spatrick 
163061da546Spatrick     break;
164061da546Spatrick   }
165061da546Spatrick   case PDB_LocType::Constant: {
166061da546Spatrick     Variant value = symbol.getValue();
167061da546Spatrick     stream.PutRawBytes(&value.Value, sizeof(value.Value),
168061da546Spatrick                        endian::InlHostByteOrder());
169061da546Spatrick     break;
170061da546Spatrick   }
171061da546Spatrick   default:
172061da546Spatrick     return DWARFExpression();
173061da546Spatrick   }
174061da546Spatrick 
175061da546Spatrick   DataBufferSP buffer =
176061da546Spatrick       std::make_shared<DataBufferHeap>(stream.GetData(), stream.GetSize());
177061da546Spatrick   DataExtractor extractor(buffer, byte_order, address_size, byte_size);
178*f6aab3d8Srobert   DWARFExpression result(extractor);
179061da546Spatrick   result.SetRegisterKind(register_kind);
180061da546Spatrick 
181061da546Spatrick   return result;
182061da546Spatrick }
183