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