
These are identified by misc-include-cleaner. I've filtered out those that break builds. Also, I'm staying away from llvm-config.h, config.h, and Compiler.h, which likely cause platform- or compiler-specific build failures.
119 lines
4.1 KiB
C++
119 lines
4.1 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/DWARFExpressionPrinter.h"
|
|
#include "llvm/DebugInfo/DWARF/LowLevel/DWARFCFIProgram.h"
|
|
#include "llvm/Support/Compiler.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 << " ";
|
|
printDwarfExpression(&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';
|
|
}
|
|
}
|