//===- llvm/unittest/MC/DwarfLineTables.cpp ------------------------------===// // // 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/ADT/STLExtras.h" #include "llvm/BinaryFormat/Dwarf.h" #include "llvm/MC/MCAsmInfo.h" #include "llvm/MC/MCContext.h" #include "llvm/MC/MCDwarf.h" #include "llvm/MC/MCRegisterInfo.h" #include "llvm/MC/MCTargetOptions.h" #include "llvm/Support/TargetRegistry.h" #include "llvm/Support/TargetSelect.h" #include "gtest/gtest.h" using namespace llvm; namespace { struct Context { const char *Triple = "x86_64-pc-linux"; std::unique_ptr MRI; std::unique_ptr MAI; std::unique_ptr Ctx; Context() { llvm::InitializeAllTargetInfos(); llvm::InitializeAllTargetMCs(); llvm::InitializeAllDisassemblers(); // If we didn't build x86, do not run the test. std::string Error; const Target *TheTarget = TargetRegistry::lookupTarget(Triple, Error); if (!TheTarget) return; MRI.reset(TheTarget->createMCRegInfo(Triple)); MCTargetOptions MCOptions; MAI.reset(TheTarget->createMCAsmInfo(*MRI, Triple, MCOptions)); Ctx = std::make_unique(MAI.get(), MRI.get(), nullptr); } operator bool() { return Ctx.get(); } operator MCContext &() { return *Ctx; }; }; Context &getContext() { static Context Ctxt; return Ctxt; } } void verifyEncoding(MCDwarfLineTableParams Params, int LineDelta, int AddrDelta, ArrayRef ExpectedEncoding) { SmallString<16> Buffer; raw_svector_ostream EncodingOS(Buffer); MCDwarfLineAddr::Encode(getContext(), Params, LineDelta, AddrDelta, EncodingOS); EXPECT_EQ(ExpectedEncoding, arrayRefFromStringRef(Buffer)); } TEST(DwarfLineTables, TestDefaultParams) { if (!getContext()) return; MCDwarfLineTableParams Params; // Minimal line offset expressible through extended opcode, 0 addr delta const uint8_t Encoding0[] = {13}; // Special opcode Addr += 0, Line += -5 verifyEncoding(Params, -5, 0, Encoding0); // Maximal line offset expressible through extended opcode, const uint8_t Encoding1[] = {26}; // Special opcode Addr += 0, Line += +8 verifyEncoding(Params, 8, 0, Encoding1); // Random value in the middle of the special ocode range const uint8_t Encoding2[] = {146}; // Special opcode Addr += 9, Line += 2 verifyEncoding(Params, 2, 9, Encoding2); // Minimal line offset expressible through extended opcode, max addr delta const uint8_t Encoding3[] = {251}; // Special opcode Addr += 17, Line += -5 verifyEncoding(Params, -5, 17, Encoding3); // Biggest special opcode const uint8_t Encoding4[] = {255}; // Special opcode Addr += 17, Line += -1 verifyEncoding(Params, -1, 17, Encoding4); // Line delta outside of the special opcode range, address delta in range const uint8_t Encoding5[] = {dwarf::DW_LNS_advance_line, 9, 158}; // Special opcode Addr += 10, Line += 0 verifyEncoding(Params, 9, 10, Encoding5); // Address delta outside of the special opcode range, but small // enough to do DW_LNS_const_add_pc + special opcode. const uint8_t Encoding6[] = {dwarf::DW_LNS_const_add_pc, // pc += 17 62}; // Special opcode Addr += 3, Line += 2 verifyEncoding(Params, 2, 20, Encoding6); // Address delta big enough to require the use of DW_LNS_advance_pc // Line delta in special opcode range const uint8_t Encoding7[] = {dwarf::DW_LNS_advance_pc, 100, 20}; // Special opcode Addr += 0, Line += 2 verifyEncoding(Params, 2, 100, Encoding7); // No special opcode possible. const uint8_t Encoding8[] = {dwarf::DW_LNS_advance_line, 20, dwarf::DW_LNS_advance_pc, 100, dwarf::DW_LNS_copy}; verifyEncoding(Params, 20, 100, Encoding8); } TEST(DwarfLineTables, TestCustomParams) { if (!getContext()) return; // Some tests against the example values given in the standard. MCDwarfLineTableParams Params; Params.DWARF2LineOpcodeBase = 13; Params.DWARF2LineBase = -3; Params.DWARF2LineRange = 12; // Minimal line offset expressible through extended opcode, 0 addr delta const uint8_t Encoding0[] = {13}; // Special opcode Addr += 0, Line += -5 verifyEncoding(Params, -3, 0, Encoding0); // Maximal line offset expressible through extended opcode, const uint8_t Encoding1[] = {24}; // Special opcode Addr += 0, Line += +8 verifyEncoding(Params, 8, 0, Encoding1); // Random value in the middle of the special ocode range const uint8_t Encoding2[] = {126}; // Special opcode Addr += 9, Line += 2 verifyEncoding(Params, 2, 9, Encoding2); // Minimal line offset expressible through extended opcode, max addr delta const uint8_t Encoding3[] = {253}; // Special opcode Addr += 20, Line += -3 verifyEncoding(Params, -3, 20, Encoding3); // Biggest special opcode const uint8_t Encoding4[] = {255}; // Special opcode Addr += 17, Line += -1 verifyEncoding(Params, -1, 20, Encoding4); // Line delta outside of the special opcode range, address delta in range const uint8_t Encoding5[] = {dwarf::DW_LNS_advance_line, 9, 136}; // Special opcode Addr += 10, Line += 0 verifyEncoding(Params, 9, 10, Encoding5); // Address delta outside of the special opcode range, but small // enough to do DW_LNS_const_add_pc + special opcode. const uint8_t Encoding6[] = {dwarf::DW_LNS_const_add_pc, // pc += 20 138}; // Special opcode Addr += 10, Line += 2 verifyEncoding(Params, 2, 30, Encoding6); // Address delta big enough to require the use of DW_LNS_advance_pc // Line delta in special opcode range const uint8_t Encoding7[] = {dwarf::DW_LNS_advance_pc, 100, 18}; // Special opcode Addr += 0, Line += 2 verifyEncoding(Params, 2, 100, Encoding7); // No special opcode possible. const uint8_t Encoding8[] = {dwarf::DW_LNS_advance_line, 20, dwarf::DW_LNS_advance_pc, 100, dwarf::DW_LNS_copy}; verifyEncoding(Params, 20, 100, Encoding8); } TEST(DwarfLineTables, TestCustomParams2) { if (!getContext()) return; // Corner case param values. MCDwarfLineTableParams Params; Params.DWARF2LineOpcodeBase = 13; Params.DWARF2LineBase = 1; Params.DWARF2LineRange = 255; const uint8_t Encoding0[] = {dwarf::DW_LNS_advance_line, 248, 1, dwarf::DW_LNS_copy}; verifyEncoding(Params, 248, 0, Encoding0); }