CFIPrograms' most common uses are within debug frames, but it is not their only use. For example, some assembly writers encode them by hand into .cfi_escape directives. This PR extracts printing code for them into its own files, which avoids the need for the main class to depend on DWARFUnit, sections, and similar. One in a series of NFC DebugInfo/DWARF refactoring changes to layer it more cleanly, so that binary CFI parsing can be used from low-level code, (such as byte strings created via .cfi_escape) without circular dependencies. The final goal is to make a more limited dwarf library usable from lower-level code. More information can be found at https://discourse.llvm.org/t/rfc-debuginfo-dwarf-refactor-into-to-lower-and-higher-level-libraries/86665
122 lines
4.2 KiB
C++
122 lines
4.2 KiB
C++
//===- DWARFCFIPrinter.cpp - Print the cfi-portions of .debug_frame -------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/DebugInfo/DWARF/DWARFCFIPrinter.h"
|
|
#include "llvm/DebugInfo/DIContext.h"
|
|
#include "llvm/DebugInfo/DWARF/DWARFCFIProgram.h"
|
|
#include "llvm/DebugInfo/DWARF/DWARFDataExtractor.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
#include "llvm/Support/DataExtractor.h"
|
|
#include "llvm/Support/Errc.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include "llvm/Support/Format.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include <cassert>
|
|
#include <cinttypes>
|
|
#include <cstdint>
|
|
#include <optional>
|
|
|
|
using namespace llvm;
|
|
using namespace dwarf;
|
|
|
|
static void printRegister(raw_ostream &OS, const DIDumpOptions &DumpOpts,
|
|
unsigned RegNum) {
|
|
if (DumpOpts.GetNameForDWARFReg) {
|
|
auto RegName = DumpOpts.GetNameForDWARFReg(RegNum, DumpOpts.IsEH);
|
|
if (!RegName.empty()) {
|
|
OS << RegName;
|
|
return;
|
|
}
|
|
}
|
|
OS << "reg" << RegNum;
|
|
}
|
|
|
|
/// Print \p Opcode's operand number \p OperandIdx which has value \p Operand.
|
|
static void printOperand(raw_ostream &OS, const DIDumpOptions &DumpOpts,
|
|
const CFIProgram &P,
|
|
const CFIProgram::Instruction &Instr,
|
|
unsigned OperandIdx, uint64_t Operand,
|
|
std::optional<uint64_t> &Address) {
|
|
assert(OperandIdx < CFIProgram::MaxOperands);
|
|
uint8_t Opcode = Instr.Opcode;
|
|
CFIProgram::OperandType Type = P.getOperandTypes()[Opcode][OperandIdx];
|
|
|
|
switch (Type) {
|
|
case CFIProgram::OT_Unset: {
|
|
OS << " Unsupported " << (OperandIdx ? "second" : "first") << " operand to";
|
|
auto OpcodeName = P.callFrameString(Opcode);
|
|
if (!OpcodeName.empty())
|
|
OS << " " << OpcodeName;
|
|
else
|
|
OS << format(" Opcode %x", Opcode);
|
|
break;
|
|
}
|
|
case CFIProgram::OT_None:
|
|
break;
|
|
case CFIProgram::OT_Address:
|
|
OS << format(" %" PRIx64, Operand);
|
|
Address = Operand;
|
|
break;
|
|
case CFIProgram::OT_Offset:
|
|
// The offsets are all encoded in a unsigned form, but in practice
|
|
// consumers use them signed. It's most certainly legacy due to
|
|
// the lack of signed variants in the first Dwarf standards.
|
|
OS << format(" %+" PRId64, int64_t(Operand));
|
|
break;
|
|
case CFIProgram::OT_FactoredCodeOffset: // Always Unsigned
|
|
if (P.codeAlign())
|
|
OS << format(" %" PRId64, Operand * P.codeAlign());
|
|
else
|
|
OS << format(" %" PRId64 "*code_alignment_factor", Operand);
|
|
if (Address && P.codeAlign()) {
|
|
*Address += Operand * P.codeAlign();
|
|
OS << format(" to 0x%" PRIx64, *Address);
|
|
}
|
|
break;
|
|
case CFIProgram::OT_SignedFactDataOffset:
|
|
if (P.dataAlign())
|
|
OS << format(" %" PRId64, int64_t(Operand) * P.dataAlign());
|
|
else
|
|
OS << format(" %" PRId64 "*data_alignment_factor", int64_t(Operand));
|
|
break;
|
|
case CFIProgram::OT_UnsignedFactDataOffset:
|
|
if (P.dataAlign())
|
|
OS << format(" %" PRId64, Operand * P.dataAlign());
|
|
else
|
|
OS << format(" %" PRId64 "*data_alignment_factor", Operand);
|
|
break;
|
|
case CFIProgram::OT_Register:
|
|
OS << ' ';
|
|
printRegister(OS, DumpOpts, Operand);
|
|
break;
|
|
case CFIProgram::OT_AddressSpace:
|
|
OS << format(" in addrspace%" PRId64, Operand);
|
|
break;
|
|
case CFIProgram::OT_Expression:
|
|
assert(Instr.Expression && "missing DWARFExpression object");
|
|
OS << " ";
|
|
DWARFExpressionPrinter::print(&Instr.Expression.value(), OS, DumpOpts,
|
|
nullptr);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void llvm::dwarf::printCFIProgram(const CFIProgram &P, raw_ostream &OS,
|
|
const DIDumpOptions &DumpOpts,
|
|
unsigned IndentLevel,
|
|
std::optional<uint64_t> Address) {
|
|
for (const auto &Instr : P) {
|
|
uint8_t Opcode = Instr.Opcode;
|
|
OS.indent(2 * IndentLevel);
|
|
OS << P.callFrameString(Opcode) << ":";
|
|
for (size_t i = 0; i < Instr.Ops.size(); ++i)
|
|
printOperand(OS, DumpOpts, P, Instr, i, Instr.Ops[i], Address);
|
|
OS << '\n';
|
|
}
|
|
}
|