17#include "llvm/Config/config.h"
66 unsigned MinInsnLength = Context.getAsmInfo().getMinInstAlignment();
67 if (MinInsnLength == 1)
69 if (AddrDelta % MinInsnLength != 0) {
73 return AddrDelta / MinInsnLength;
77 UseRelocs = Ctx.getAsmInfo().doesDwarfUseRelocationsAcrossSections();
80 Ctx.getObjectFileInfo()->getDwarfLineStrSection();
81 assert(DwarfLineStrSection &&
"DwarfLineStrSection must not be NULL");
134static inline const MCExpr *
152 auto I = MCLineDivisions.find(Sec);
153 if (
I == MCLineDivisions.end())
156 auto &Entries =
I->second;
170 Entries.push_back(EndEntry);
181 unsigned FileNum, LastLine, Column, Flags, Isa, Discriminator;
197 bool EndEntryEmitted =
false;
198 for (
auto It = LineEntries.begin(); It != LineEntries.end(); ++It) {
199 auto LineEntry = *It;
200 MCSymbol *CurrLabel = LineEntry.getLabel();
203 if (LineEntry.LineStreamLabel) {
205 auto *Label = CurrLabel;
206 auto NextIt = It + 1;
211 if (!Label && NextIt != LineEntries.end()) {
212 Label = NextIt->getLabel();
218 MCOS->
emitLabel(LineEntry.LineStreamLabel, LineEntry.StreamLabelDefLoc);
222 if (LineEntry.IsEndEntry) {
226 EndEntryEmitted =
true;
230 int64_t LineDelta =
static_cast<int64_t
>(LineEntry.getLine()) - LastLine;
232 if (FileNum != LineEntry.getFileNum()) {
233 FileNum = LineEntry.getFileNum();
234 MCOS->
emitInt8(dwarf::DW_LNS_set_file);
237 if (Column != LineEntry.getColumn()) {
238 Column = LineEntry.getColumn();
239 MCOS->
emitInt8(dwarf::DW_LNS_set_column);
242 if (Discriminator != LineEntry.getDiscriminator() &&
244 Discriminator = LineEntry.getDiscriminator();
246 MCOS->
emitInt8(dwarf::DW_LNS_extended_op);
248 MCOS->
emitInt8(dwarf::DW_LNE_set_discriminator);
251 if (Isa != LineEntry.getIsa()) {
252 Isa = LineEntry.getIsa();
253 MCOS->
emitInt8(dwarf::DW_LNS_set_isa);
257 Flags = LineEntry.getFlags();
258 MCOS->
emitInt8(dwarf::DW_LNS_negate_stmt);
261 MCOS->
emitInt8(dwarf::DW_LNS_set_basic_block);
263 MCOS->
emitInt8(dwarf::DW_LNS_set_prologue_end);
265 MCOS->
emitInt8(dwarf::DW_LNS_set_epilogue_begin);
274 LastLine = LineEntry.getLine();
275 PrevLabel = CurrLabel;
276 IsAtStartSeq =
false;
284 if (!EndEntryEmitted && !IsAtStartSeq)
293 auto *LineSym = ctx.createTempSymbol();
295 const MCDwarfLoc &DwarfLoc = ctx.getCurrentDwarfLoc();
314 if (LineTables.empty())
318 std::optional<MCDwarfLineStr> LineStr;
320 LineStr.emplace(context);
326 for (
const auto &CUIDTablePair : LineTables) {
327 CUIDTablePair.second.emitCU(MCOS, Params, LineStr);
331 LineStr->emitSection(MCOS);
336 if (!HasSplitLineTable)
338 std::optional<MCDwarfLineStr> NoLineStr(std::nullopt);
340 MCOS.
emitLabel(Header.Emit(&MCOS, Params, {}, NoLineStr).second);
343std::pair<MCSymbol *, MCSymbol *>
345 std::optional<MCDwarfLineStr> &LineStr)
const {
346 static const char StandardOpcodeLengths[] = {
360 assert(std::size(StandardOpcodeLengths) >=
362 return Emit(MCOS, Params,
370 if (!Context.getAsmInfo().doesSetDirectiveSuppressReloc())
374 MCSymbol *ABS = Context.createTempSymbol();
394 if (!LineStrings.isFinalized())
395 LineStrings.finalizeInOrder();
397 Data.resize(LineStrings.getSize());
403 return LineStrings.add(Path);
412 if (Ctx.getAsmInfo().needsDwarfSectionOffsetDirective()) {
422void MCDwarfLineTableHeader::emitV2FileDirTables(
MCStreamer *MCOS)
const {
443 bool EmitMD5,
bool HasAnySource,
444 std::optional<MCDwarfLineStr> &LineStr) {
456 StringRef(
reinterpret_cast<const char *
>(Cksum.data()), Cksum.size()));
467 LineStr->emitRef(MCOS, Source);
475void MCDwarfLineTableHeader::emitV5FileDirTables(
476 MCStreamer *MCOS, std::optional<MCDwarfLineStr> &LineStr)
const {
483 : dwarf::DW_FORM_string);
486 SmallString<256> Dir;
493 CompDir = LineStr->getSaver().save(CompDir);
497 LineStr->emitRef(MCOS, CompDir);
499 LineStr->emitRef(MCOS, Dir);
513 uint64_t Entries = 2;
521 : dwarf::DW_FORM_string);
531 : dwarf::DW_FORM_string);
541 "No root file and no .file directives");
548std::pair<MCSymbol *, MCSymbol *>
551 std::optional<MCDwarfLineStr> &LineStr)
const {
571 if (LineTableVersion >= 5) {
591 if (LineTableVersion >= 4)
599 for (
char Length : StandardOpcodeLengths)
604 if (LineTableVersion >= 5)
605 emitV5FileDirTables(MCOS, LineStr);
607 emitV2FileDirTables(MCOS);
613 return std::make_pair(LineStartSym, LineEndSym);
617 std::optional<MCDwarfLineStr> &LineStr)
const {
618 MCSymbol *LineEndSym = Header.Emit(MCOS, Params, LineStr).second;
621 for (
const auto &LineSec : MCLineSections.getMCLineEntries())
622 emitOne(MCOS, LineSec.first, LineSec.second);
631 std::optional<MD5::MD5Result> Checksum,
632 std::optional<StringRef> Source,
633 uint16_t DwarfVersion,
unsigned FileNumber) {
634 return Header.tryGetFile(Directory, FileName, Checksum, Source, DwarfVersion,
640 std::optional<MD5::MD5Result> Checksum) {
643 return RootFile.
Checksum == Checksum;
648 std::optional<MD5::MD5Result> Checksum,
649 std::optional<StringRef> Source,
650 uint16_t DwarfVersion,
unsigned FileNumber) {
653 if (FileName.
empty()) {
654 FileName =
"<stdin>";
666 if (FileNumber == 0) {
674 if (!IterBool.second)
675 return IterBool.first->second;
685 if (!File.Name.empty())
689 if (Directory.
empty()) {
692 if (!tFileName.
empty()) {
694 if (!Directory.
empty())
695 FileName = tFileName;
702 if (Directory.
empty()) {
716 File.Name = std::string(FileName);
717 File.DirIndex = DirIndex;
718 File.Checksum = Checksum;
720 File.Source = Source;
721 if (Source.has_value())
730 int64_t LineDelta,
uint64_t AddrDelta) {
745 int64_t LineDelta,
uint64_t AddrDelta,
748 bool NeedCopy =
false;
760 if (AddrDelta == MaxSpecialAddrDelta)
761 Out.
push_back(dwarf::DW_LNS_const_add_pc);
762 else if (AddrDelta) {
766 Out.
push_back(dwarf::DW_LNS_extended_op);
768 Out.
push_back(dwarf::DW_LNE_end_sequence);
779 Out.
push_back(dwarf::DW_LNS_advance_line);
788 if (LineDelta == 0 && AddrDelta == 0) {
797 if (AddrDelta < 256 + MaxSpecialAddrDelta) {
806 Opcode = Temp + (AddrDelta - MaxSpecialAddrDelta) * Params.
DWARF2LineRange;
808 Out.
push_back(dwarf::DW_LNS_const_add_pc);
821 assert(Temp <= 255 &&
"Buggy special opcode encoding.");
844 ? dwarf::DW_FORM_sec_offset
846 : dwarf::DW_FORM_data4);
847 EmitAbbrev(MCOS, dwarf::DW_AT_stmt_list, SecOffsetForm);
850 EmitAbbrev(MCOS, dwarf::DW_AT_ranges, SecOffsetForm);
852 EmitAbbrev(MCOS, dwarf::DW_AT_low_pc, dwarf::DW_FORM_addr);
853 EmitAbbrev(MCOS, dwarf::DW_AT_high_pc, dwarf::DW_FORM_addr);
855 EmitAbbrev(MCOS, dwarf::DW_AT_name, dwarf::DW_FORM_string);
857 EmitAbbrev(MCOS, dwarf::DW_AT_comp_dir, dwarf::DW_FORM_string);
859 if (!DwarfDebugFlags.
empty())
860 EmitAbbrev(MCOS, dwarf::DW_AT_APPLE_flags, dwarf::DW_FORM_string);
861 EmitAbbrev(MCOS, dwarf::DW_AT_producer, dwarf::DW_FORM_string);
864 EmitAbbrev(MCOS, dwarf::DW_AT_language_name, dwarf::DW_FORM_data2);
866 EmitAbbrev(MCOS, dwarf::DW_AT_language, dwarf::DW_FORM_data2);
874 EmitAbbrev(MCOS, dwarf::DW_AT_name, dwarf::DW_FORM_string);
875 EmitAbbrev(MCOS, dwarf::DW_AT_decl_file, dwarf::DW_FORM_data4);
876 EmitAbbrev(MCOS, dwarf::DW_AT_decl_line, dwarf::DW_FORM_data4);
877 EmitAbbrev(MCOS, dwarf::DW_AT_low_pc, dwarf::DW_FORM_addr);
889 const MCSymbol *InfoSectionSymbol) {
896 unsigned UnitLengthBytes =
902 int Length = UnitLengthBytes + 2 + OffsetSize + 1 + 1;
908 int Pad = 2 * AddrSize - (
Length & (2 * AddrSize - 1));
909 if (Pad == 2 * AddrSize)
915 Length += 2 * AddrSize * Sections.size();
930 if (InfoSectionSymbol)
940 for(
int i = 0; i < Pad; i++)
946 const MCSymbol *StartSymbol = Sec->getBeginSymbol();
947 MCSymbol *EndSymbol = Sec->getEndSymbol(context);
948 assert(StartSymbol &&
"StartSymbol must not be NULL");
949 assert(EndSymbol &&
"EndSymbol must not be NULL");
967 const MCSymbol *AbbrevSectionSymbol,
982 unsigned UnitLengthBytes =
1004 MCOS->
emitInt8(dwarf::DW_UT_compile);
1009 if (AbbrevSectionSymbol)
1025 if (LineSectionSymbol)
1043 const auto TextSection = Sections.begin();
1044 assert(TextSection != Sections.end() &&
"No text section found");
1046 MCSymbol *StartSymbol = (*TextSection)->getBeginSymbol();
1047 MCSymbol *EndSymbol = (*TextSection)->getEndSymbol(context);
1048 assert(StartSymbol &&
"StartSymbol must not be NULL");
1049 assert(EndSymbol &&
"EndSymbol must not be NULL");
1063 if (MCDwarfDirs.
size() > 0) {
1072 MCDwarfFiles.
empty()
1086 if (!DwarfDebugFlags.
empty()){
1093 if (!DwarfDebugProducer.
empty())
1101 MCOS->
emitInt16(dwarf::DW_LNAME_Assembly);
1106 MCOS->
emitInt16(dwarf::DW_LANG_Mips_Assembler);
1112 const std::vector<MCGenDwarfLabelEntry> &Entries =
1114 for (
const auto &Entry : Entries) {
1159 const MCSymbol *StartSymbol = Sec->getBeginSymbol();
1160 const MCSymbol *EndSymbol = Sec->getEndSymbol(context);
1161 const MCExpr *SectionStartAddr =
1163 const MCExpr *SectionSize =
1165 MCOS->
emitInt8(dwarf::DW_RLE_start_length);
1166 MCOS->
emitValue(SectionStartAddr, AddrSize);
1169 MCOS->
emitInt8(dwarf::DW_RLE_end_of_list);
1176 const MCSymbol *StartSymbol = Sec->getBeginSymbol();
1177 const MCSymbol *EndSymbol = Sec->getEndSymbol(context);
1180 const MCExpr *SectionStartAddr =
1183 MCOS->
emitValue(SectionStartAddr, AddrSize);
1186 const MCExpr *SectionSize =
1197 return RangesSymbol;
1209 bool CreateDwarfSectionSymbols =
1211 MCSymbol *LineSectionSymbol =
nullptr;
1212 if (CreateDwarfSectionSymbols)
1214 MCSymbol *AbbrevSectionSymbol =
nullptr;
1215 MCSymbol *InfoSectionSymbol =
nullptr;
1228 const bool UseRangesSection =
1231 CreateDwarfSectionSymbols |= UseRangesSection;
1234 if (CreateDwarfSectionSymbols) {
1239 if (CreateDwarfSectionSymbols) {
1249 if (UseRangesSection) {
1258 EmitGenDwarfInfo(MCOS, AbbrevSectionSymbol, LineSectionSymbol, RangesSymbol);
1270 if (Symbol->isTemporary())
1281 if (Name.starts_with(
"_"))
1282 Name = Name.
substr(1, Name.size()-1);
1289 unsigned CurBuffer =
SrcMgr.FindBufferContainingLoc(
Loc);
1290 unsigned LineNumber =
SrcMgr.FindLineNumber(
Loc, CurBuffer);
1305 auto ReplaceReg = [=](
unsigned &Reg) {
1311 ReplaceReg(
F.Register);
1312 ReplaceReg(
F.Register2);
1316 ReplaceReg(
F.Register);
1321 ReplaceReg(
F.Register);
1326 ReplaceReg(
F.Register);
1327 ReplaceReg(
F.MaskRegister);
1330 ReplaceReg(
F.Register);
1331 ReplaceReg(
F.SpillRegister);
1332 ReplaceReg(
F.MaskRegister);
1334 std::visit(Visitor, ExtraFields);
1348 unsigned symbolEncoding) {
1350 unsigned format = symbolEncoding & 0x0f;
1369 unsigned symbolEncoding,
bool isEH) {
1382 unsigned symbolEncoding) {
1393class FrameEmitterImpl {
1394 int64_t CFAOffset = 0;
1395 int64_t InitialCFAOffset = 0;
1397 MCObjectStreamer &Streamer;
1400 FrameEmitterImpl(
bool IsEH, MCObjectStreamer &Streamer)
1401 : IsEH(IsEH), Streamer(Streamer) {}
1404 void EmitCompactUnwind(
const MCDwarfFrameInfo &frame);
1406 const MCSymbol &EmitCIE(
const MCDwarfFrameInfo &
F);
1407 void EmitFDE(
const MCSymbol &cieStart,
const MCDwarfFrameInfo &frame,
1408 bool LastInSection,
const MCSymbol &SectionStart);
1410 MCSymbol *BaseLabel);
1411 void emitCFIInstruction(
const MCCFIInstruction &Instr);
1422 if (DwarfReg < 32) {
1423 OS <<
uint8_t(dwarf::DW_OP_reg0 + DwarfReg);
1425 OS <<
uint8_t(dwarf::DW_OP_regx);
1432 auto *MRI = Streamer.getContext().getRegisterInfo();
1434 switch (
Instr.getOperation()) {
1436 unsigned Reg1 =
Instr.getRegister();
1437 unsigned Reg2 =
Instr.getRegister2();
1439 Reg1 = MRI->getDwarfRegNumFromDwarfEHRegNum(Reg1);
1440 Reg2 = MRI->getDwarfRegNumFromDwarfEHRegNum(Reg2);
1442 Streamer.emitInt8(dwarf::DW_CFA_register);
1443 Streamer.emitULEB128IntValue(Reg1);
1444 Streamer.emitULEB128IntValue(Reg2);
1448 Streamer.emitInt8(dwarf::DW_CFA_GNU_window_save);
1452 Streamer.emitInt8(dwarf::DW_CFA_AARCH64_negate_ra_state);
1456 Streamer.emitInt8(dwarf::DW_CFA_AARCH64_negate_ra_state_with_pc);
1460 unsigned Reg =
Instr.getRegister();
1461 Streamer.emitInt8(dwarf::DW_CFA_undefined);
1462 Streamer.emitULEB128IntValue(
Reg);
1467 const bool IsRelative =
1470 Streamer.emitInt8(dwarf::DW_CFA_def_cfa_offset);
1473 CFAOffset +=
Instr.getOffset();
1475 CFAOffset =
Instr.getOffset();
1477 Streamer.emitULEB128IntValue(CFAOffset);
1482 unsigned Reg =
Instr.getRegister();
1484 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1485 Streamer.emitInt8(dwarf::DW_CFA_def_cfa);
1486 Streamer.emitULEB128IntValue(
Reg);
1487 CFAOffset =
Instr.getOffset();
1488 Streamer.emitULEB128IntValue(CFAOffset);
1493 unsigned Reg =
Instr.getRegister();
1495 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1496 Streamer.emitInt8(dwarf::DW_CFA_def_cfa_register);
1497 Streamer.emitULEB128IntValue(
Reg);
1503 unsigned Reg =
Instr.getRegister();
1505 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1506 Streamer.emitIntValue(dwarf::DW_CFA_LLVM_def_aspace_cfa, 1);
1507 Streamer.emitULEB128IntValue(
Reg);
1508 CFAOffset =
Instr.getOffset();
1509 Streamer.emitULEB128IntValue(CFAOffset);
1510 Streamer.emitULEB128IntValue(
Instr.getAddressSpace());
1516 const bool IsRelative =
1519 unsigned Reg =
Instr.getRegister();
1521 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1529 Streamer.emitInt8(dwarf::DW_CFA_offset_extended_sf);
1530 Streamer.emitULEB128IntValue(
Reg);
1531 Streamer.emitSLEB128IntValue(
Offset);
1532 }
else if (
Reg < 64) {
1533 Streamer.emitInt8(dwarf::DW_CFA_offset +
Reg);
1534 Streamer.emitULEB128IntValue(
Offset);
1536 Streamer.emitInt8(dwarf::DW_CFA_offset_extended);
1537 Streamer.emitULEB128IntValue(
Reg);
1538 Streamer.emitULEB128IntValue(
Offset);
1543 Streamer.emitInt8(dwarf::DW_CFA_remember_state);
1546 Streamer.emitInt8(dwarf::DW_CFA_restore_state);
1549 unsigned Reg =
Instr.getRegister();
1550 Streamer.emitInt8(dwarf::DW_CFA_same_value);
1551 Streamer.emitULEB128IntValue(
Reg);
1555 unsigned Reg =
Instr.getRegister();
1557 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1559 Streamer.emitInt8(dwarf::DW_CFA_restore |
Reg);
1561 Streamer.emitInt8(dwarf::DW_CFA_restore_extended);
1562 Streamer.emitULEB128IntValue(
Reg);
1567 Streamer.emitInt8(dwarf::DW_CFA_GNU_args_size);
1568 Streamer.emitULEB128IntValue(
Instr.getOffset());
1572 Streamer.emitBytes(
Instr.getValues());
1576 Streamer.emitLabel(
Instr.getCfiLabel(),
Instr.getLoc());
1579 unsigned Reg =
Instr.getRegister();
1581 Reg = MRI->getDwarfRegNumFromDwarfEHRegNum(
Reg);
1587 Streamer.emitInt8(dwarf::DW_CFA_val_offset_sf);
1588 Streamer.emitULEB128IntValue(
Reg);
1589 Streamer.emitSLEB128IntValue(
Offset);
1591 Streamer.emitInt8(dwarf::DW_CFA_val_offset);
1592 Streamer.emitULEB128IntValue(
Reg);
1593 Streamer.emitULEB128IntValue(
Offset);
1613 const auto &Fields =
1619 if (Fields.Reg1SizeInBits % 8 == 0) {
1620 OSBlock << uint8_t(dwarf::DW_OP_piece);
1623 OSBlock << uint8_t(dwarf::DW_OP_bit_piece);
1628 if (Fields.Reg2SizeInBits % 8 == 0) {
1629 OSBlock << uint8_t(dwarf::DW_OP_piece);
1632 OSBlock << uint8_t(dwarf::DW_OP_bit_piece);
1637 Streamer.emitInt8(dwarf::DW_CFA_expression);
1638 Streamer.emitULEB128IntValue(Fields.Register);
1639 Streamer.emitULEB128IntValue(
Block.size());
1668 const auto &Fields =
1675 if (VRs.size() == 1 && VRs[0].SizeInBits % 8 == 0) {
1677 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1678 << uint8_t(dwarf::DW_OP_LLVM_offset_uconst);
1679 encodeULEB128((VRs[0].SizeInBits / 8) * VRs[0].Lane, OSBlock);
1681 for (
const auto &VR : VRs) {
1686 OSBlock << uint8_t(dwarf::DW_OP_bit_piece);
1692 Streamer.emitInt8(dwarf::DW_CFA_expression);
1693 Streamer.emitULEB128IntValue(Fields.Register);
1694 Streamer.emitULEB128IntValue(
Block.size());
1710 const auto &Fields =
1716 OSBlock << uint8_t(dwarf::DW_OP_swap);
1717 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1718 << uint8_t(dwarf::DW_OP_LLVM_offset_uconst);
1720 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1721 << uint8_t(dwarf::DW_OP_LLVM_call_frame_entry_reg);
1723 OSBlock << uint8_t(dwarf::DW_OP_deref_size);
1724 OSBlock << uint8_t(Fields.MaskRegisterSizeInBits / 8);
1725 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1726 << uint8_t(dwarf::DW_OP_LLVM_select_bit_piece);
1730 Streamer.emitInt8(dwarf::DW_CFA_expression);
1731 Streamer.emitULEB128IntValue(Fields.Register);
1732 Streamer.emitULEB128IntValue(
Block.size());
1755 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1756 << uint8_t(dwarf::DW_OP_LLVM_call_frame_entry_reg);
1758 OSBlock << uint8_t(dwarf::DW_OP_deref_size)
1759 << uint8_t(Fields.MaskRegisterSizeInBits / 8);
1760 OSBlock << uint8_t(dwarf::DW_OP_LLVM_user)
1761 << uint8_t(dwarf::DW_OP_LLVM_select_bit_piece);
1765 Streamer.emitInt8(dwarf::DW_CFA_expression);
1766 Streamer.emitULEB128IntValue(Fields.Register);
1767 Streamer.emitULEB128IntValue(
Block.size());
1782 if (Label && !
Label->isDefined())
continue;
1785 if (BaseLabel && Label) {
1787 if (ThisSym != BaseLabel) {
1788 Streamer.emitDwarfAdvanceFrameAddr(BaseLabel, ThisSym,
Instr.getLoc());
1789 BaseLabel = ThisSym;
1793 emitCFIInstruction(Instr);
1825 if (!Encoding)
return;
1829 if (!DwarfEHFrameOnly && Frame.
Lsda)
1830 Encoding |= 0x40000000;
1835 Streamer.emitSymbolValue(Frame.
Begin,
Size);
1844 Streamer.emitIntValue(Encoding,
Size);
1851 Streamer.emitIntValue(0,
Size);
1855 if (!DwarfEHFrameOnly && Frame.
Lsda)
1856 Streamer.emitSymbolValue(Frame.
Lsda,
Size);
1858 Streamer.emitIntValue(0,
Size);
1864 switch (DwarfVersion) {
1877 MCContext &context = Streamer.getContext();
1882 Streamer.emitLabel(sectionStart);
1897 *sectionEnd, UnitLengthBytes);
1903 Streamer.emitIntValue(CIE_ID, OffsetSize);
1907 Streamer.emitInt8(CIEVersion);
1911 Augmentation +=
"z";
1913 Augmentation +=
"P";
1915 Augmentation +=
"L";
1916 Augmentation +=
"R";
1918 Augmentation +=
"S";
1920 Augmentation +=
"B";
1922 Augmentation +=
"G";
1923 Streamer.emitBytes(Augmentation);
1925 Streamer.emitInt8(0);
1927 if (CIEVersion >= 4) {
1932 Streamer.emitInt8(0);
1943 if (
RAReg ==
static_cast<unsigned>(INT_MAX))
1946 if (CIEVersion == 1) {
1948 "DWARF 2 encodes return_address_register in one byte");
1949 Streamer.emitInt8(
RAReg);
1951 Streamer.emitULEB128IntValue(
RAReg);
1955 unsigned augmentationLength = 0;
1959 augmentationLength += 1;
1961 augmentationLength +=
1965 augmentationLength += 1;
1967 augmentationLength += 1;
1969 Streamer.emitULEB128IntValue(augmentationLength);
1992 emitCFIInstructions(Instructions,
nullptr);
1995 InitialCFAOffset = CFAOffset;
2000 Streamer.emitLabel(sectionEnd);
2001 return *sectionStart;
2004void FrameEmitterImpl::EmitFDE(
const MCSymbol &cieStart,
2008 MCContext &context = Streamer.getContext();
2013 CFAOffset = InitialCFAOffset;
2026 Streamer.emitLabel(fdeStart);
2039 Streamer.emitSymbolValue(&cieStart, OffsetSize,
2044 unsigned PCEncoding =
2056 unsigned augmentationLength = 0;
2061 Streamer.emitULEB128IntValue(augmentationLength);
2076 Streamer.emitValueToAlignment(
Align(Alignment));
2078 Streamer.emitLabel(fdeEnd);
2086 explicit CIEKey(
const MCDwarfFrameInfo &Frame,
bool IsEH)
2087 : Personality(Frame.Personality),
2088 PersonalityEncoding(Frame.PersonalityEncoding),
2089 LsdaEncoding(Frame.LsdaEncoding), IsSignalFrame(Frame.IsSignalFrame),
2090 IsSimple(Frame.IsSimple),
RAReg(Frame.
RAReg),
2091 IsBKeyFrame(Frame.IsBKeyFrame),
2092 IsMTETaggedFrame(Frame.IsMTETaggedFrame), IsEH(IsEH) {}
2094 StringRef PersonalityName()
const {
2097 return Personality->getName();
2103 return std::make_tuple(
RAReg, IsSimple) <
2104 std::make_tuple(
Other.RAReg,
Other.IsSimple);
2106 return std::make_tuple(PersonalityName(), PersonalityEncoding, LsdaEncoding,
2107 IsSignalFrame, IsSimple,
RAReg, IsBKeyFrame,
2109 std::make_tuple(
Other.PersonalityName(),
Other.PersonalityEncoding,
2112 Other.IsMTETaggedFrame);
2120 return Personality ==
Other.Personality &&
2121 PersonalityEncoding ==
Other.PersonalityEncoding &&
2122 LsdaEncoding ==
Other.LsdaEncoding &&
2123 IsSignalFrame ==
Other.IsSignalFrame && IsSimple ==
Other.IsSimple &&
2125 IsMTETaggedFrame ==
Other.IsMTETaggedFrame;
2129 const MCSymbol *Personality =
nullptr;
2130 unsigned PersonalityEncoding = 0;
2131 unsigned LsdaEncoding = -1;
2132 bool IsSignalFrame =
false;
2133 bool IsSimple =
false;
2134 unsigned RAReg = UINT_MAX;
2135 bool IsBKeyFrame =
false;
2136 bool IsMTETaggedFrame =
false;
2145 const MCAsmInfo &AsmInfo = Context.getAsmInfo();
2146 FrameEmitterImpl
Emitter(IsEH, Streamer);
2153 bool SectionEmitted =
false;
2156 if (!SectionEmitted) {
2159 SectionEmitted =
true;
2161 NeedsEHFrameSection |=
2164 Emitter.EmitCompactUnwind(Frame);
2172 if (!NeedsEHFrameSection && IsEH)
return;
2179 MCSymbol *SectionStart = Context.createTempSymbol();
2187 std::vector<MCDwarfFrameInfo> FrameArrayX(FrameArray.
begin(), FrameArray.
end());
2190 return CIEKey(
X, IsEH) < CIEKey(
Y, IsEH);
2193 const MCSymbol *LastCIEStart =
nullptr;
2194 for (
auto I = FrameArrayX.begin(), E = FrameArrayX.end();
I != E;) {
2207 CIEKey
Key(Frame, IsEH);
2208 if (!LastCIEStart ||
Key != LastKey) {
2210 LastCIEStart = &
Emitter.EmitCIE(Frame);
2213 Emitter.EmitFDE(*LastCIEStart, Frame,
I == E, *SectionStart);
2230 uint8_t Opcode = dwarf::DW_CFA_advance_loc | AddrDelta;
2233 Out.
push_back(dwarf::DW_CFA_advance_loc1);
2236 Out.
push_back(dwarf::DW_CFA_advance_loc2);
2240 Out.
push_back(dwarf::DW_CFA_advance_loc4);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
dxil DXContainer Global Emitter
This file contains constants used for implementing Dwarf debug support.
static void emitFDESymbol(MCObjectStreamer &streamer, const MCSymbol &symbol, unsigned symbolEncoding, bool isEH)
static uint64_t SpecialAddr(MCDwarfLineTableParams Params, uint64_t op)
Given a special op, return the address skip amount (in units of DWARF2_LINE_MIN_INSN_LENGTH).
static void EmitGenDwarfAranges(MCStreamer *MCOS, const MCSymbol *InfoSectionSymbol)
static bool isRootFile(const MCDwarfFile &RootFile, StringRef &Directory, StringRef &FileName, std::optional< MD5::MD5Result > Checksum)
static uint64_t ScaleAddrDelta(MCContext &Context, uint64_t AddrDelta)
static const MCExpr * forceExpAbs(MCStreamer &OS, const MCExpr *Expr)
static void emitAbsValue(MCStreamer &OS, const MCExpr *Value, unsigned Size)
static void encodeDwarfRegisterLocation(int DwarfReg, raw_ostream &OS)
static void emitOneV5FileEntry(MCStreamer *MCOS, const MCDwarfFile &DwarfFile, bool EmitMD5, bool HasAnySource, std::optional< MCDwarfLineStr > &LineStr)
static const MCExpr * makeEndMinusStartExpr(MCContext &Ctx, const MCSymbol &Start, const MCSymbol &End, int IntVal)
static unsigned getCIEVersion(bool IsEH, unsigned DwarfVersion)
static void EmitGenDwarfInfo(MCStreamer *MCOS, const MCSymbol *AbbrevSectionSymbol, const MCSymbol *LineSectionSymbol, const MCSymbol *RangesSymbol)
static void EmitAbbrev(MCStreamer *MCOS, uint64_t Name, uint64_t Form)
static void EmitPersonality(MCStreamer &streamer, const MCSymbol &symbol, unsigned symbolEncoding)
static void emitEncodingByte(MCObjectStreamer &Streamer, unsigned Encoding)
static int getDataAlignmentFactor(MCStreamer &streamer)
static MCSymbol * emitGenDwarfRanges(MCStreamer *MCOS)
static const MCExpr * makeStartPlusIntExpr(MCContext &Ctx, const MCSymbol &Start, int IntVal)
static void EmitGenDwarfAbbrev(MCStreamer *MCOS)
static unsigned getSizeForEncoding(MCStreamer &streamer, unsigned symbolEncoding)
#define DWARF2_FLAG_IS_STMT
#define DWARF2_FLAG_BASIC_BLOCK
#define DWARF2_LINE_DEFAULT_IS_STMT
#define DWARF2_FLAG_PROLOGUE_END
#define DWARF2_FLAG_EPILOGUE_BEGIN
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
static constexpr MCPhysReg RAReg
This file defines the SmallString class.
This file defines the SmallVector class.
static TableGen::Emitter::Opt Y("gen-skeleton-entry", EmitSkeleton, "Generate example skeleton entry")
Represent a constant reference to an array (0 or more elements consecutively in memory),...
size_t size() const
Get the array size.
Tagged union holding either a T or a Error.
This class is intended to be used as a base class for asm properties and features specific to the tar...
unsigned getMinInstAlignment() const
const std::vector< MCCFIInstruction > & getInitialFrameState() const
bool needsDwarfSectionOffsetDirective() const
bool doesDwarfUseRelocationsAcrossSections() const
virtual const MCExpr * getExprForFDESymbol(const MCSymbol *Sym, unsigned Encoding, MCStreamer &Streamer) const
bool isStackGrowthDirectionUp() const
True if target stack grow up.
unsigned getCalleeSaveStackSlotSize() const
Get the callee-saved register stack slot size in bytes.
bool doDwarfFDESymbolsUseAbsDiff() const
virtual const MCExpr * getExprForPersonalitySymbol(const MCSymbol *Sym, unsigned Encoding, MCStreamer &Streamer) const
unsigned getCodePointerSize() const
Get the code pointer size in bytes.
static LLVM_ABI const MCBinaryExpr * create(Opcode Op, const MCExpr *LHS, const MCExpr *RHS, MCContext &Ctx, SMLoc Loc=SMLoc())
LLVM_ABI void replaceRegister(unsigned FromReg, unsigned ToReg)
Replaces in place all references to FromReg with ToReg.
@ OpLLVMVectorRegisterMask
static LLVM_ABI const MCConstantExpr * create(int64_t Value, MCContext &Ctx, bool PrintInHex=false, unsigned SizeInBytes=0)
Context object for machine code objects.
const MCObjectFileInfo * getObjectFileInfo() const
LLVM_ABI void remapDebugPath(SmallVectorImpl< char > &Path)
Remap one path in-place as per the debug prefix map.
const SetVector< MCSection * > & getGenDwarfSectionSyms()
const SmallVectorImpl< std::string > & getMCDwarfDirs(unsigned CUID=0)
StringRef getDwarfDebugProducer()
StringRef getDwarfDebugFlags()
LLVM_ABI MCSymbol * createTempSymbol()
Create a temporary symbol with a unique name.
StringRef getCompilationDir() const
Get the compilation directory for DW_AT_comp_dir The compilation directory should be set with setComp...
MCDwarfLineTable & getMCDwarfLineTable(unsigned CUID)
unsigned getDwarfCompileUnitID()
const MCRegisterInfo * getRegisterInfo() const
const SmallVectorImpl< MCDwarfFile > & getMCDwarfFiles(unsigned CUID=0)
const std::map< unsigned, MCDwarfLineTable > & getMCDwarfLineTables() const
unsigned getGenDwarfFileNumber()
uint16_t getDwarfVersion() const
LLVM_ABI void finalizeDwarfSections(MCStreamer &MCOS)
Remove empty sections from SectionsForRanges, to avoid generating useless debug info for them.
void addMCGenDwarfLabelEntry(const MCGenDwarfLabelEntry &E)
LLVM_ABI MCSymbol * getOrCreateSymbol(const Twine &Name)
Lookup the symbol inside with the specified Name.
const MCDwarfLoc & getCurrentDwarfLoc()
dwarf::DwarfFormat getDwarfFormat() const
const MCAsmInfo & getAsmInfo() const
const std::vector< MCGenDwarfLabelEntry > & getMCGenDwarfLabelEntries() const
LLVM_ABI void Emit(MCStreamer &MCOS, MCDwarfLineTableParams Params, MCSection *Section) const
static LLVM_ABI void emit(MCObjectStreamer &streamer, bool isEH)
static LLVM_ABI void encodeAdvanceLoc(MCContext &Context, uint64_t AddrDelta, SmallVectorImpl< char > &OS)
static LLVM_ABI void Emit(MCStreamer *MCOS, MCDwarfLineTableParams Params, int64_t LineDelta, uint64_t AddrDelta)
Utility function to emit the encoding to a streamer.
static LLVM_ABI void encode(MCContext &Context, MCDwarfLineTableParams Params, int64_t LineDelta, uint64_t AddrDelta, SmallVectorImpl< char > &OS)
Utility function to encode a Dwarf pair of LineDelta and AddrDeltas.
Instances of this class represent the line information for the dwarf line table entries.
void setEndLabel(MCSymbol *EndLabel)
MCDwarfLineEntry(MCSymbol *label, const MCDwarfLoc loc, MCSymbol *lineStreamLabel=nullptr, SMLoc streamLabelDefLoc={})
MCSymbol * LineStreamLabel
static LLVM_ABI void make(MCStreamer *MCOS, MCSection *Section)
LLVM_ABI void emitSection(MCStreamer *MCOS)
Emit the .debug_line_str section if appropriate.
LLVM_ABI MCDwarfLineStr(MCContext &Ctx)
Construct an instance that can emit .debug_line_str (for use in a normal v5 line table).
LLVM_ABI SmallString< 0 > getFinalizedData()
Returns finalized section.
LLVM_ABI void emitRef(MCStreamer *MCOS, StringRef Path)
Emit a reference to the string.
LLVM_ABI size_t addString(StringRef Path)
Adds path Path to the line string.
LLVM_ABI void endCurrentSeqAndEmitLineStreamLabel(MCStreamer *MCOS, SMLoc DefLoc, StringRef Name)
MCDwarfFile & getRootFile()
const MCLineSection & getMCLineSections() const
static LLVM_ABI void emit(MCStreamer *MCOS, MCDwarfLineTableParams Params)
static LLVM_ABI void emitOne(MCStreamer *MCOS, MCSection *Section, const MCLineSection::MCDwarfLineEntryCollection &LineEntries)
LLVM_ABI Expected< unsigned > tryGetFile(StringRef &Directory, StringRef &FileName, std::optional< MD5::MD5Result > Checksum, std::optional< StringRef > Source, uint16_t DwarfVersion, unsigned FileNumber=0)
LLVM_ABI void emitCU(MCStreamer *MCOS, MCDwarfLineTableParams Params, std::optional< MCDwarfLineStr > &LineStr) const
Instances of this class represent the information from a dwarf .loc directive.
Base class for the full range of assembler expressions which are needed for parsing.
static LLVM_ABI void Emit(MCStreamer *MCOS)
MCGenDwarfLabelEntry(StringRef name, unsigned fileNumber, unsigned lineNumber, MCSymbol *label)
static LLVM_ABI void Make(MCSymbol *Symbol, MCStreamer *MCOS, SourceMgr &SrcMgr, SMLoc &Loc)
LLVM_ABI void addEndEntry(MCSymbol *EndLabel)
void addLineEntry(const MCDwarfLineEntry &LineEntry, MCSection *Sec)
std::vector< MCDwarfLineEntry > MCDwarfLineEntryCollection
MCSection * getDwarfRangesSection() const
bool getSupportsCompactUnwindWithoutEHFrame() const
MCSection * getDwarfLineStrSection() const
unsigned getCompactUnwindDwarfEHFrameOnly() const
MCSection * getDwarfRnglistsSection() const
MCSection * getDwarfLineSection() const
MCSection * getDwarfInfoSection() const
MCSection * getDwarfFrameSection() const
unsigned getFDEEncoding() const
MCSection * getDwarfAbbrevSection() const
bool getOmitDwarfIfHaveCompactUnwind() const
MCSection * getDwarfARangesSection() const
MCSection * getCompactUnwindSection() const
Streaming object file generation interface.
void generateCompactUnwindEncodings()
void emitValueToAlignment(Align Alignment, int64_t Fill=0, uint8_t FillLen=1, unsigned MaxBytesToEmit=0) override
Emit some number of copies of Value until the byte alignment ByteAlignment is reached.
void emitLabel(MCSymbol *Symbol, SMLoc Loc=SMLoc()) override
Emit a label for Symbol into the current section.
MCRegisterInfo base class - We assume that the target defines a static array of MCRegisterDesc object...
MCRegister getRARegister() const
This method should return the register where the return address can be found.
virtual int64_t getDwarfRegNum(MCRegister Reg, bool isEH) const
Map a target register to an equivalent dwarf register number.
Instances of this class represent a uniqued identifier for a section in the current translation unit.
MCSymbol * getBeginSymbol()
Streaming machine code generation interface.
virtual void emitAssignment(MCSymbol *Symbol, const MCExpr *Value)
Emit an assignment of Value to Symbol.
virtual void emitBinaryData(StringRef Data)
Functionally identical to EmitBytes.
virtual void emitDwarfUnitLength(uint64_t Length, const Twine &Comment)
Emit a unit length field.
MCContext & getContext() const
virtual void AddComment(const Twine &T, bool EOL=true)
Add a textual comment.
virtual void emitCOFFSecRel32(MCSymbol const *Symbol, uint64_t Offset)
Emits a COFF section relative relocation.
void emitValue(const MCExpr *Value, unsigned Size, SMLoc Loc=SMLoc())
void emitSymbolValue(const MCSymbol *Sym, unsigned Size, bool IsSectionRelative=false)
Special case of EmitValue that avoids the client having to pass in a MCExpr for MCSymbols.
virtual void emitDwarfLineStartLabel(MCSymbol *StartSym)
Emit the debug line start label.
virtual void emitAbsoluteSymbolDiff(const MCSymbol *Hi, const MCSymbol *Lo, unsigned Size)
Emit the absolute difference between two symbols.
virtual void emitLabel(MCSymbol *Symbol, SMLoc Loc=SMLoc())
Emit a label for Symbol into the current section.
virtual void emitIntValue(uint64_t Value, unsigned Size)
Special case of EmitValue that avoids the client having to pass in a MCExpr for constant integers.
virtual void emitDwarfAdvanceLineAddr(int64_t LineDelta, const MCSymbol *LastLabel, const MCSymbol *Label, unsigned PointerSize)
If targets does not support representing debug line section by .loc/.file directives in assembly outp...
void emitInt16(uint64_t Value)
virtual MCSymbol * getDwarfLineTableSymbol(unsigned CUID)
unsigned emitULEB128IntValue(uint64_t Value, unsigned PadTo=0)
Special case of EmitULEB128Value that avoids the client having to pass in a MCExpr for constant integ...
virtual void emitULEB128Value(const MCExpr *Value)
ArrayRef< MCDwarfFrameInfo > getDwarfFrameInfos() const
virtual void switchSection(MCSection *Section, uint32_t Subsec=0)
Set the current section where code is being emitted to Section.
void emitInt32(uint64_t Value)
MCSection * getCurrentSectionOnly() const
virtual void emitDwarfLineEndEntry(MCSection *Section, MCSymbol *LastLabel, MCSymbol *EndLabel=nullptr)
Emit the debug line end entry.
void emitInt8(uint64_t Value)
void emitFill(uint64_t NumBytes, uint8_t FillValue)
Emit NumBytes bytes worth of the value specified by FillValue.
virtual void emitBytes(StringRef Data)
Emit the bytes in Data into the output.
static const MCSymbolRefExpr * create(const MCSymbol *Symbol, MCContext &Ctx, SMLoc Loc=SMLoc())
MCSymbol - Instances of this class represent a symbol name in the MC file, and MCSymbols are created ...
MCSection & getSection() const
Get the section associated with a defined, non-absolute symbol.
Wrapper class representing virtual and physical registers.
Represents a location in source code.
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
StringRef str() const
Explicit conversion to StringRef.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void push_back(const T &Elt)
This owns the files read by a parser, handles include stacks, and handles diagnostic wrangling.
Represent a constant reference to a string, i.e.
constexpr StringRef substr(size_t Start, size_t N=npos) const
Return a reference to the substring from [Start, Start + N).
constexpr bool empty() const
Check if the string is empty.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
LLVM Value Representation.
This class implements an extremely fast bulk output stream that can only output to a stream.
A raw_ostream that writes to an SmallVector or SmallString.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
const uint32_t DW_CIE_ID
Special ID values that distinguish a CIE from a FDE in DWARF CFI.
uint8_t getUnitLengthFieldByteSize(DwarfFormat Format)
Get the byte size of the unit length field depending on the DWARF format.
const uint64_t DW64_CIE_ID
DwarfFormat
Constants that define the DWARF format as 32 or 64 bit.
uint8_t getDwarfOffsetByteSize(DwarfFormat Format)
The size of a reference determined by the DWARF 32/64-bit format.
@ DW_LENGTH_DWARF64
Indicator of 64-bit DWARF format.
LLVM_ABI MCSymbol * emitListsTableHeaderStart(MCStreamer &S)
NodeAddr< InstrNode * > Instr
void write(void *memory, value_type value, endianness endian)
Write a value to memory with a particular endianness.
LLVM_ABI StringRef get_separator(Style style=Style::native)
Return the preferred separator for this platform.
LLVM_ABI StringRef parent_path(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get parent path.
LLVM_ABI StringRef filename(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get filename.
This is an optimization pass for GlobalISel generic memory operations.
bool operator<(int64_t V1, const APSInt &V2)
void stable_sort(R &&Range)
auto find(R &&Range, const T &Val)
Provide wrappers to std::find which take ranges instead of having to pass begin/end explicitly.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
LLVM_ABI std::error_code inconvertibleErrorCode()
The value returned by this function can be returned from convertToErrorCode for Error values where no...
bool operator!=(uint64_t V1, const APInt &V2)
constexpr bool isUIntN(unsigned N, uint64_t x)
Checks if an unsigned integer fits into the given (dynamic) bit width.
bool operator==(const AddressRangeValuePair &LHS, const AddressRangeValuePair &RHS)
LLVM_ABI SourceMgr SrcMgr
constexpr bool isUInt(uint64_t x)
Checks if an unsigned integer fits into the given bit width.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
format_object< Ts... > format(const char *Fmt, const Ts &... Vals)
These are helper functions used to produce formatted output.
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
LLVM_ABI unsigned getULEB128Size(uint64_t Value)
Utility function to get the size of the ULEB128-encoded value.
FunctionAddr VTableAddr uintptr_t uintptr_t Data
ArrayRef(const T &OneElt) -> ArrayRef< T >
unsigned encodeULEB128(uint64_t Value, raw_ostream &OS, unsigned PadTo=0)
Utility function to encode a ULEB128 value to an output stream.
void appendLEB128(SmallVectorImpl< U > &Buffer, T Value)
constexpr decltype(auto) makeVisitor(CallableTs &&...Callables)
Returns an opaquely-typed Callable object whose operator() overload set is the sum of the operator() ...
StringRef toStringRef(bool B)
Construct a string ref from a boolean.
This struct is a compact representation of a valid (non-zero power of two) alignment.
Held in ExtraFields when OpLLVMRegisterPair.
Held in ExtraFields when OpLLVMVectorOffset.
Held in ExtraFields when OpLLVMVectorRegisterMask.
Held in ExtraFields when OpLLVMVectorRegisters.
std::vector< VectorRegisterWithLane > VectorRegisters
Instances of this class represent the name of the dwarf .file directive and its associated dwarf file...
std::optional< MD5::MD5Result > Checksum
The MD5 checksum, if there is one.
const MCSymbol * Personality
unsigned PersonalityEncoding
uint64_t CompactUnwindEncoding
std::vector< MCCFIInstruction > Instructions
uint8_t DWARF2LineOpcodeBase
First special line opcode - leave room for the standard opcodes.
uint8_t DWARF2LineRange
Range of line offsets in a special line info. opcode.
int8_t DWARF2LineBase
Minimum line offset in a special line info.