33 AcceleratorRecords(&
GlobalData.getAllocator()) {
44 OrigUnit(&OrigUnit), getUnitFromOffset(UnitFromOffset),
46 AcceleratorRecords(&
GlobalData.getAllocator()) {
50 DWARFDie CUDie = OrigUnit.getUnitDIE();
54 if (std::optional<DWARFFormValue> Val = CUDie.
find(dwarf::DW_AT_language))
57 if (!
GlobalData.getOptions().NoODR && Language.has_value() &&
69 LineTablePtr = File.Dwarf->getLineTableForUnit(&
getOrigUnit());
82 for (
DIEInfo &Info : DieInfoArray)
83 Info.unsetFlagsWhichSetDuringLiveAnalysis();
89 Dependencies.reset(
nullptr);
96 AcceleratorRecords.erase();
100 DebugAddrIndexMap.clear();
101 StmtSeqListAttributes.clear();
117 OutDieOffsetArray.resize(
getOrigUnit().getNumDIEs(), 0);
124 bool IsODRUnavailableFunctionScope) {
131 bool ChildIsODRUnavailableFunctionScope = IsODRUnavailableFunctionScope;
133 if (DieInfo.getIsInMouduleScope())
134 ChildInfo.setIsInMouduleScope();
136 if (DieInfo.getIsInFunctionScope())
137 ChildInfo.setIsInFunctionScope();
139 if (DieInfo.getIsInAnonNamespaceScope())
140 ChildInfo.setIsInAnonNamespaceScope();
142 switch (CurChild->getTag()) {
143 case dwarf::DW_TAG_module:
144 ChildInfo.setIsInMouduleScope();
145 if (DieEntry->
getTag() == dwarf::DW_TAG_compile_unit &&
150 case dwarf::DW_TAG_subprogram:
151 ChildInfo.setIsInFunctionScope();
152 if (!ChildIsODRUnavailableFunctionScope &&
153 !ChildInfo.getIsInMouduleScope()) {
155 {dwarf::DW_AT_abstract_origin, dwarf::DW_AT_specification}))
156 ChildIsODRUnavailableFunctionScope =
true;
159 case dwarf::DW_TAG_namespace: {
162 if (
find(CurChild, dwarf::DW_AT_extension))
165 if (!NamespaceEntry.
CU->
find(NamespaceEntry.
DieEntry, dwarf::DW_AT_name))
166 ChildInfo.setIsInAnonNamespaceScope();
173 ChildInfo.setTrackLiveness();
175 if ((!ChildInfo.getIsInAnonNamespaceScope() &&
176 !ChildIsODRUnavailableFunctionScope && !NoODR))
177 ChildInfo.setODRAvailable();
179 if (CurChild->hasChildren())
180 analyzeDWARFStructureRec(CurChild, ChildIsODRUnavailableFunctionScope);
187 if (LineTablePtr->hasFileAtIndex(FileIdx)) {
191 if (It == ResolvedFullPaths.end()) {
192 std::string OrigFileName;
193 bool FoundFileName = LineTablePtr->getFileNameByIndex(
198 assert(FoundFileName &&
"Must get file name from line table");
208 ResolvedParentPaths.find(ParentPath);
209 if (ParentIt == ResolvedParentPaths.end()) {
214 .insert({ParentPath, GlobalStrings.
insert(RealPath).first})
222 It = ResolvedFullPaths
223 .insert(std::make_pair(
224 FileIdx, GlobalStrings.
insert(ResolvedPath).first))
236 if (ObjFileIdx > std::numeric_limits<uint32_t>::max())
238 "object files exceeds UINT32_MAX");
239 if (LocalIdx > std::numeric_limits<uint32_t>::max())
241 "local index exceeds UINT32_MAX");
243 Priority = (ObjFileIdx << 32) | LocalIdx;
249 ResolvedFullPaths.shrink_and_clear();
250 ResolvedParentPaths.clear();
255 Dependencies.reset(
nullptr);
256 StmtSeqListAttributes.clear();
263 if (!Language || Language != dwarf::DW_LANG_Swift)
266 if (!
GlobalData.getOptions().ParseableSwiftInterfaces)
271 if (!Path.ends_with(
".swiftinterface"))
283 if (!DeveloperDir.
empty() && Path.starts_with(DeveloperDir))
287 if (std::optional<DWARFFormValue> Val =
find(DieEntry, dwarf::DW_AT_name)) {
305 PendingSwiftInterfaces.emplace_back(*
Name, ResolvedPath);
311 for (
auto &Pending : PendingSwiftInterfaces) {
312 auto &Entry = Map[Pending.ModuleName];
313 if (!Entry.empty() && Entry != Pending.ResolvedPath)
314 warn(
Twine(
"conflicting parseable interfaces for Swift Module ") +
315 Pending.ModuleName +
": " + Entry +
" and " + Pending.ResolvedPath +
317 Entry = Pending.ResolvedPath;
319 PendingSwiftInterfaces.clear();
340 assert(ChildInfo.getODRAvailable());
343 ChildrenIndexAssigner.getChildIndex(*
this, CurChild)))
346 if (
Error Err = assignTypeNamesRec(CurChild, NameBuilder))
354 if (std::optional<SectionDescriptor *> DebugInfoSection =
361 Patch.
RefCU.getPointer()->getDieOutOffset(
366 ->ListDebugULEB128DieRefPatch.forEach(
370 Patch.
RefCU.getPointer()->getDieOutOffset(
375 if (std::optional<SectionDescriptor *> DebugLocSection =
378 ->ListDebugULEB128DieRefPatch.forEach(
382 Patch.
RefCU.getPointer()->getDieOutOffset(
387 if (std::optional<SectionDescriptor *> DebugLocListsSection =
389 (*DebugLocListsSection)
390 ->ListDebugULEB128DieRefPatch.forEach(
394 Patch.
RefCU.getPointer()->getDieOutOffset(
409 RefCU = getUnitFromOffset(*
Offset);
417 if (std::optional<uint32_t> RefDieIdx =
425 }
else if (RefCU && CanResolveInterCUReferences) {
430 if (ReferredCUStage < Stage::Loaded || ReferredCUStage >
Stage::Cloned)
433 if (std::optional<uint32_t> RefDieIdx =
449 if (std::optional<DWARFFormValue> AttrVal =
find(DieEntry, Attr))
457 std::lock_guard<std::mutex> Guard(RangesMutex);
459 Ranges.
insert({FuncLowPc, FuncHighPc}, PcOffset);
461 LowPc = std::min(*LowPc, FuncLowPc + PcOffset);
463 LowPc = FuncLowPc + PcOffset;
464 this->HighPc = std::max(HighPc, FuncHighPc + PcOffset);
468 std::lock_guard<std::mutex> Guard(LabelsMutex);
469 Labels.
insert({LabelLowPc, PcOffset});
489 if (!DebugInfoSection.ListDebugLocPatch.empty()) {
494 uint64_t OffsetAfterUnitLength = emitLocListHeader(OutLocationSection);
496 DebugInfoSection.ListDebugLocPatch.forEach([&](
DebugLocPatch &Patch) {
505 if (!OriginalLocations) {
510 LinkedLocationExpressionsVector LinkedLocationExpressions;
512 LinkedLocationExpressionsWithOffsetPatches LinkedExpression;
514 if (CurExpression.Range) {
516 LinkedExpression.Expression.Range = {
526 LinkedExpression.Expression.Expr,
528 LinkedExpression.Patches);
530 LinkedLocationExpressions.push_back({LinkedExpression});
535 OutLocationSection.
OS.
tell());
536 emitLocListFragment(LinkedLocationExpressions, OutLocationSection);
539 if (OffsetAfterUnitLength > 0) {
540 assert(OffsetAfterUnitLength -
542 OffsetAfterUnitLength);
543 OutLocationSection.
apply(
544 OffsetAfterUnitLength -
546 dwarf::DW_FORM_sec_offset,
547 OutLocationSection.
OS.
tell() - OffsetAfterUnitLength);
559 uint64_t OffsetAfterUnitLength = OutLocationSection.
OS.
tell();
573 return OffsetAfterUnitLength;
577uint64_t CompileUnit::emitLocListFragment(
578 const LinkedLocationExpressionsVector &LinkedLocationExpression,
580 uint64_t OffsetBeforeLocationExpression = 0;
583 uint64_t BaseAddress = 0;
584 if (std::optional<uint64_t> LowPC =
getLowPc())
585 BaseAddress = *LowPC;
587 for (
const LinkedLocationExpressionsWithOffsetPatches &LocExpression :
588 LinkedLocationExpression) {
589 if (LocExpression.Expression.Range) {
591 LocExpression.Expression.Range->LowPC - BaseAddress,
594 LocExpression.Expression.Range->HighPC - BaseAddress,
598 OutLocationSection.
emitIntVal(LocExpression.Expression.Expr.size(), 2);
599 OffsetBeforeLocationExpression = OutLocationSection.
OS.
tell();
600 for (uint64_t *OffsetPtr : LocExpression.Patches)
601 *OffsetPtr += OffsetBeforeLocationExpression;
603 OutLocationSection.
OS
604 << StringRef((
const char *)LocExpression.Expression.Expr.data(),
605 LocExpression.Expression.Expr.size());
613 return OffsetBeforeLocationExpression;
616 std::optional<uint64_t> BaseAddress;
617 for (
const LinkedLocationExpressionsWithOffsetPatches &LocExpression :
618 LinkedLocationExpression) {
619 if (LocExpression.Expression.Range) {
623 BaseAddress = LocExpression.Expression.Range->LowPC;
626 OutLocationSection.
emitIntVal(dwarf::DW_LLE_base_addressx, 1);
628 OutLocationSection.
OS);
632 OutLocationSection.
emitIntVal(dwarf::DW_LLE_offset_pair, 1);
635 encodeULEB128(LocExpression.Expression.Range->LowPC - *BaseAddress,
636 OutLocationSection.
OS);
639 encodeULEB128(LocExpression.Expression.Range->HighPC - *BaseAddress,
640 OutLocationSection.
OS);
643 OutLocationSection.
emitIntVal(dwarf::DW_LLE_default_location, 1);
645 encodeULEB128(LocExpression.Expression.Expr.size(), OutLocationSection.
OS);
646 OffsetBeforeLocationExpression = OutLocationSection.
OS.
tell();
647 for (uint64_t *OffsetPtr : LocExpression.Patches)
648 *OffsetPtr += OffsetBeforeLocationExpression;
650 OutLocationSection.
OS << StringRef(
651 (
const char *)LocExpression.Expression.Expr.data(),
652 LocExpression.Expression.Expr.size());
656 OutLocationSection.
emitIntVal(dwarf::DW_LLE_end_of_list, 1);
657 return OffsetBeforeLocationExpression;
660Error CompileUnit::emitDebugAddrSection() {
661 if (
GlobalData.getOptions().UpdateIndexTablesOnly)
667 if (DebugAddrIndexMap.empty())
670 SectionDescriptor &OutAddrSection =
677 uint64_t OffsetAfterSectionLength = OutAddrSection.
OS.
tell();
689 for (uint64_t AddrValue : DebugAddrIndexMap.getValues())
693 OutAddrSection.
apply(
694 OffsetAfterSectionLength -
696 dwarf::DW_FORM_sec_offset,
697 OutAddrSection.
OS.
tell() - OffsetAfterSectionLength);
709 LinkedFunctionRanges.
insert(
712 emitAranges(LinkedFunctionRanges);
730 if (!DebugInfoSection.ListDebugRangePatch.empty()) {
731 std::optional<AddressRangeValuePair> CachedRange;
732 uint64_t OffsetAfterUnitLength = emitRangeListHeader(OutRangeSection);
735 DebugInfoSection.ListDebugRangePatch.forEach([&](
DebugRangePatch &Patch) {
737 CompileUnitRangePtr = &Patch;
747 InputDebugRangesSectionOffset)) {
749 for (
const auto &
Range : *InputRanges) {
750 if (!CachedRange || !CachedRange->Range.contains(
Range.LowPC))
756 warn(
"inconsistent range data.");
761 LinkedRanges.
insert({
Range.LowPC + CachedRange->Value,
762 Range.HighPC + CachedRange->Value});
766 warn(
"invalid range list ignored.");
771 OutRangeSection.
OS.
tell());
772 emitRangeListFragment(LinkedRanges, OutRangeSection);
776 if (CompileUnitRangePtr !=
nullptr) {
780 dwarf::DW_FORM_sec_offset,
781 OutRangeSection.
OS.
tell());
782 emitRangeListFragment(LinkedFunctionRanges, OutRangeSection);
785 if (OffsetAfterUnitLength > 0) {
786 assert(OffsetAfterUnitLength -
788 OffsetAfterUnitLength);
789 OutRangeSection.
apply(
790 OffsetAfterUnitLength -
792 dwarf::DW_FORM_sec_offset,
793 OutRangeSection.
OS.
tell() - OffsetAfterUnitLength);
804 uint64_t OffsetAfterUnitLength = OutRangeSection.
OS.
tell();
818 return OffsetAfterUnitLength;
821void CompileUnit::emitRangeListFragment(
const AddressRanges &LinkedRanges,
825 uint64_t BaseAddress = 0;
826 if (std::optional<uint64_t> LowPC =
getLowPc())
827 BaseAddress = *LowPC;
829 for (
const AddressRange &
Range : LinkedRanges) {
842 std::optional<uint64_t> BaseAddress;
843 for (
const AddressRange &
Range : LinkedRanges) {
845 BaseAddress =
Range.start();
848 OutRangeSection.
emitIntVal(dwarf::DW_RLE_base_addressx, 1);
853 OutRangeSection.
emitIntVal(dwarf::DW_RLE_offset_pair, 1);
863 OutRangeSection.
emitIntVal(dwarf::DW_RLE_end_of_list, 1);
866void CompileUnit::emitAranges(AddressRanges &LinkedFunctionRanges) {
867 if (LinkedFunctionRanges.
empty())
870 SectionDescriptor &DebugInfoSection =
872 SectionDescriptor &OutArangesSection =
887 uint64_t OffsetAfterArangesLengthField = OutArangesSection.
OS.
tell();
890 OutArangesSection.notePatch(
891 DebugOffsetPatch{OutArangesSection.
OS.
tell(), &DebugInfoSection});
897 for (
size_t Idx = 0; Idx <
Padding; Idx++)
901 for (
const AddressRange &
Range : LinkedFunctionRanges) {
912 uint64_t OffsetAfterArangesEnd = OutArangesSection.
OS.
tell();
915 OutArangesSection.
apply(
916 OffsetAfterArangesLengthField -
918 dwarf::DW_FORM_sec_offset,
919 OffsetAfterArangesEnd - OffsetAfterArangesLengthField);
927 DWARFDie OrigUnitDie = OrigUnit.getUnitDIE();
930 if (std::optional<uint64_t> MacroAttr =
934 emitMacroTableImpl(Table, *MacroAttr,
true);
939 if (std::optional<uint64_t> MacroAttr =
943 emitMacroTableImpl(Table, *MacroAttr,
false);
952 bool hasDWARFv5Header) {
958 bool DefAttributeIsReported =
false;
959 bool UndefAttributeIsReported =
false;
960 bool ImportAttributeIsReported =
false;
962 for (
const DWARFDebugMacro::MacroList &
List : MacroTable->MacroLists) {
963 if (OffsetToMacroTable == List.Offset) {
965 if (hasDWARFv5Header) {
967 OutSection.emitIntVal(List.Header.Version, sizeof(List.Header.Version));
969 uint8_t Flags = List.Header.Flags;
973 DWARFDebugMacro::HeaderFlagMask::MACRO_OPCODE_OPERANDS_TABLE) {
975 ~DWARFDebugMacro::HeaderFlagMask::MACRO_OPCODE_OPERANDS_TABLE;
976 warn(
"opcode_operands_table is not supported yet.");
980 std::optional<uint64_t> StmtListOffset;
981 if (Flags & DWARFDebugMacro::HeaderFlagMask::MACRO_DEBUG_LINE_OFFSET) {
983 for (auto &V : getOutUnitDIE()->values()) {
984 if (V.getAttribute() == dwarf::DW_AT_stmt_list) {
985 StmtListOffset = V.getDIEInteger().getValue();
990 if (!StmtListOffset) {
991 Flags &= ~DWARFDebugMacro::HeaderFlagMask::MACRO_DEBUG_LINE_OFFSET;
992 warn(
"couldn`t find line table for macro table.");
997 OutSection.emitIntVal(Flags, sizeof(Flags));
1000 if (StmtListOffset) {
1001 OutSection.notePatch(DebugOffsetPatch{
1002 OutSection.OS.tell(),
1003 &getOrCreateSectionDescriptor(DebugSectionKind::DebugLine)});
1006 OutSection.emitIntVal(0xBADDEF, List.Header.getOffsetByteSize());
1011 for (const DWARFDebugMacro::Entry &MacroEntry : List.Macros) {
1012 if (MacroEntry.Type == 0) {
1013 encodeULEB128(MacroEntry.Type, OutSection.OS);
1017 uint8_t MacroType = MacroEntry.Type;
1018 switch (MacroType) {
1020 bool HasVendorSpecificExtension =
1021 (!hasDWARFv5Header &&
1022 MacroType == dwarf::DW_MACINFO_vendor_ext) ||
1023 (hasDWARFv5Header && (MacroType >= dwarf::DW_MACRO_lo_user &&
1024 MacroType <= dwarf::DW_MACRO_hi_user));
1026 if (HasVendorSpecificExtension) {
1028 OutSection.emitIntVal(MacroType, 1);
1031 encodeULEB128(MacroEntry.ExtConstant, OutSection.OS);
1034 OutSection.emitString(dwarf::DW_FORM_string, MacroEntry.ExtStr);
1036 warn(
"unknown macro type. skip.");
1044 case dwarf::DW_MACRO_define:
1045 case dwarf::DW_MACRO_undef: {
1047 OutSection.emitIntVal(MacroType, 1);
1050 encodeULEB128(MacroEntry.Line, OutSection.OS);
1053 OutSection.emitString(dwarf::DW_FORM_string, MacroEntry.MacroStr);
1055 case dwarf::DW_MACRO_define_strp:
1056 case dwarf::DW_MACRO_undef_strp:
1057 case dwarf::DW_MACRO_define_strx:
1058 case dwarf::DW_MACRO_undef_strx: {
1061 switch (MacroType) {
1062 case dwarf::DW_MACRO_define_strx: {
1063 MacroType = dwarf::DW_MACRO_define_strp;
1064 if (!DefAttributeIsReported) {
1065 warn(
"DW_MACRO_define_strx unsupported yet. Convert to "
1066 "DW_MACRO_define_strp.");
1067 DefAttributeIsReported = true;
1070 case dwarf::DW_MACRO_undef_strx: {
1071 MacroType = dwarf::DW_MACRO_undef_strp;
1072 if (!UndefAttributeIsReported) {
1073 warn(
"DW_MACRO_undef_strx unsupported yet. Convert to "
1074 "DW_MACRO_undef_strp.");
1075 UndefAttributeIsReported = true;
1084 OutSection.emitIntVal(MacroType, 1);
1087 encodeULEB128(MacroEntry.Line, OutSection.OS);
1090 OutSection.emitString(dwarf::DW_FORM_strp, MacroEntry.MacroStr);
1093 case dwarf::DW_MACRO_start_file: {
1095 OutSection.emitIntVal(MacroType, 1);
1097 encodeULEB128(MacroEntry.Line, OutSection.OS);
1099 encodeULEB128(MacroEntry.File, OutSection.OS);
1101 case dwarf::DW_MACRO_end_file: {
1103 OutSection.emitIntVal(MacroType, 1);
1105 case dwarf::DW_MACRO_import:
1106 case dwarf::DW_MACRO_import_sup: {
1107 if (!ImportAttributeIsReported) {
1108 warn(
"DW_MACRO_import and DW_MACRO_import_sup are unsupported "
1110 ImportAttributeIsReported = true;
1124 std::optional<int64_t> VarAddressAdjustment,
1129 uint8_t OrigAddressByteSize = OrigUnit.getAddressByteSize();
1132 for (
auto &
Op : InputExpression) {
1133 auto Desc =
Op.getDescription();
1136 if ((
Desc.Op.size() == 2 &&
Desc.Op[0] == Encoding::BaseTypeRef) ||
1137 (
Desc.Op.size() == 2 &&
Desc.Op[1] == Encoding::BaseTypeRef &&
1138 Desc.Op[0] != Encoding::Size1))
1139 warn(
"unsupported DW_OP encoding.");
1141 if ((
Desc.Op.size() == 1 &&
Desc.Op[0] == Encoding::BaseTypeRef) ||
1142 (
Desc.Op.size() == 2 &&
Desc.Op[1] == Encoding::BaseTypeRef &&
1143 Desc.Op[0] == Encoding::Size1)) {
1145 assert(OpOffset <
Op.getEndOffset());
1146 uint32_t ULEBsize =
Op.getEndOffset() - OpOffset - 1;
1150 assert(!
Op.getSubCode() &&
"SubOps not yet supported");
1153 if (
Desc.Op.size() == 1) {
1154 RefOffset =
Op.getRawOperand(0);
1157 RefOffset =
Op.getRawOperand(1);
1161 unsigned RealSize = 0;
1165 if (RefOffset > 0 ||
Op.getCode() != dwarf::DW_OP_convert) {
1166 RefOffset += OrigUnit.getOffset();
1168 if (std::optional<uint32_t> Idx =
1169 OrigUnit.getDIEIndexForOffset(RefOffset))
1178 Section.notePatchWithOffsetUpdate(
1185 if (RealSize > ULEBsize) {
1188 warn(
"base type ref doesn't fit.");
1190 assert(RealSize == ULEBsize &&
"padding failed");
1193 }
else if (!
getGlobalData().getOptions().UpdateIndexTablesOnly &&
1194 Op.getCode() == dwarf::DW_OP_addrx) {
1195 if (std::optional<object::SectionedAddress> SA =
1196 OrigUnit.getAddrOffsetSectionItem(
Op.getRawOperand(0))) {
1201 OutputExpression.
push_back(dwarf::DW_OP_addr);
1202 uint64_t LinkedAddress = SA->Address + VarAddressAdjustment.value_or(0);
1206 reinterpret_cast<const uint8_t *
>(&LinkedAddress),
1207 OrigAddressByteSize);
1208 OutputExpression.
append(AddressBytes.
begin(), AddressBytes.
end());
1210 warn(
"cann't read DW_OP_addrx operand.");
1211 }
else if (!
getGlobalData().getOptions().UpdateIndexTablesOnly &&
1212 Op.getCode() == dwarf::DW_OP_constx) {
1213 if (std::optional<object::SectionedAddress> SA =
1214 OrigUnit.getAddrOffsetSectionItem(
Op.getRawOperand(0))) {
1219 std::optional<uint8_t> OutOperandKind;
1220 switch (OrigAddressByteSize) {
1222 OutOperandKind = dwarf::DW_OP_const2u;
1225 OutOperandKind = dwarf::DW_OP_const4u;
1228 OutOperandKind = dwarf::DW_OP_const8u;
1232 formatv((
"unsupported address size: {0}."), OrigAddressByteSize));
1236 if (OutOperandKind) {
1237 OutputExpression.
push_back(*OutOperandKind);
1239 SA->Address + VarAddressAdjustment.value_or(0);
1243 reinterpret_cast<const uint8_t *
>(&LinkedAddress),
1244 OrigAddressByteSize);
1245 OutputExpression.
append(AddressBytes.
begin(), AddressBytes.
end());
1248 warn(
"cann't read DW_OP_constx operand.");
1255 OpOffset =
Op.getEndOffset();
1260 std::optional<std::reference_wrapper<const Triple>> TargetTriple,
1269 if (ArtificialTypeUnit)
1273 std::pair<DIE *, TypeEntry *> OutCUDie =
cloneDIE(
1275 std::nullopt, std::nullopt, Allocator, ArtificialTypeUnit);
1278 if (!TargetTriple.has_value() || (OutCUDie.first ==
nullptr))
1301 if (
Error Err = emitDebugAddrSection())
1317 uint64_t OutOffset, std::optional<int64_t> FuncAddressAdjustment,
1323 bool NeedToClonePlainDIE = Info.needToKeepInPlainDwarf();
1324 bool NeedToCloneTypeDIE =
1325 (InputDieEntry->
getTag() != dwarf::DW_TAG_compile_unit) &&
1326 Info.needToPlaceInTypeTable();
1327 std::pair<DIE *, TypeEntry *> ClonedDIE;
1331 if (NeedToClonePlainDIE)
1334 ClonedDIE.first = createPlainDIEandCloneAttributes(
1335 InputDieEntry, PlainDIEGenerator, OutOffset, FuncAddressAdjustment,
1336 VarAddressAdjustment);
1337 if (NeedToCloneTypeDIE) {
1340 assert(ArtificialTypeUnit !=
nullptr);
1344 ClonedDIE.second = createTypeDIEandCloneAttributes(
1345 InputDieEntry, TypeDIEGenerator, ClonedParentTypeDIE,
1346 ArtificialTypeUnit, SiblingOrdinal);
1349 ClonedDIE.second ? ClonedDIE.second : ClonedParentTypeDIE;
1351 bool HasPlainChildrenToClone =
1352 (ClonedDIE.first && Info.getKeepPlainChildren());
1354 bool HasTypeChildrenToClone =
1355 ((ClonedDIE.second ||
1356 InputDieEntry->
getTag() == dwarf::DW_TAG_compile_unit) &&
1357 Info.getKeepTypeChildren());
1360 if (HasPlainChildrenToClone || HasTypeChildrenToClone) {
1366 std::pair<DIE *, TypeEntry *> ClonedChild =
cloneDIE(
1367 CurChild, TypeParentForChild, OutOffset, FuncAddressAdjustment,
1368 VarAddressAdjustment, Allocator, ArtificialTypeUnit, ChildOrdinal);
1370 if (ClonedChild.first) {
1372 ClonedChild.first->getOffset() + ClonedChild.first->getSize();
1373 PlainDIEGenerator.
addChild(ClonedChild.first);
1376 assert(ClonedDIE.first ==
nullptr ||
1377 HasPlainChildrenToClone == ClonedDIE.first->hasChildren());
1380 if (HasPlainChildrenToClone)
1381 OutOffset +=
sizeof(int8_t);
1385 if (ClonedDIE.first !=
nullptr)
1386 ClonedDIE.first->setSize(OutOffset - ClonedDIE.first->getOffset());
1391DIE *CompileUnit::createPlainDIEandCloneAttributes(
1393 uint64_t &OutOffset, std::optional<int64_t> &FuncAddressAdjustment,
1394 std::optional<int64_t> &VarAddressAdjustment) {
1397 DIE *ClonedDIE =
nullptr;
1398 bool HasLocationExpressionAddress =
false;
1399 if (InputDieEntry->
getTag() == dwarf::DW_TAG_subprogram) {
1401 FuncAddressAdjustment =
1403 getDIE(InputDieEntry),
false);
1404 }
else if (InputDieEntry->
getTag() == dwarf::DW_TAG_label) {
1406 std::optional<uint64_t> lowPC =
1410 if (It != Labels.
end())
1411 FuncAddressAdjustment = It->second;
1413 }
else if (InputDieEntry->
getTag() == dwarf::DW_TAG_variable) {
1415 std::pair<bool, std::optional<int64_t>> LocExprAddrAndRelocAdjustment =
1417 getDIE(InputDieEntry),
false);
1419 HasLocationExpressionAddress = LocExprAddrAndRelocAdjustment.first;
1420 if (LocExprAddrAndRelocAdjustment.first &&
1421 LocExprAddrAndRelocAdjustment.second)
1422 VarAddressAdjustment = *LocExprAddrAndRelocAdjustment.second;
1425 ClonedDIE = PlainDIEGenerator.
createDIE(InputDieEntry->
getTag(), OutOffset);
1432 DIEAttributeCloner AttributesCloner(ClonedDIE, *
this,
this, InputDieEntry,
1433 PlainDIEGenerator, FuncAddressAdjustment,
1434 VarAddressAdjustment,
1435 HasLocationExpressionAddress);
1436 AttributesCloner.clone();
1439 AcceleratorRecordsSaver AccelRecordsSaver(
getGlobalData(), *
this,
this);
1440 AccelRecordsSaver.save(InputDieEntry, ClonedDIE, AttributesCloner.AttrInfo,
1444 AttributesCloner.finalizeAbbreviations(
Info.getKeepPlainChildren());
1450DIE *CompileUnit::allocateTypeDie(
TypeEntryBody *TypeDescriptor,
1453 bool IsParentDeclaration) {
1458 if (!IsDeclaration && !IsParentDeclaration) {
1461 if (Priority >= TypeDescriptor->
DiePriority.load(std::memory_order_relaxed))
1465 if (TypeDescriptor->
Die.load(std::memory_order_relaxed))
1469 while (TypeDescriptor->
Lock.test_and_set(std::memory_order_acquire))
1474 if (!IsDeclaration && !IsParentDeclaration) {
1477 TypeDescriptor->
DiePriority.load(std::memory_order_relaxed)) {
1478 TypeDescriptor->
DiePriority.store(Priority, std::memory_order_relaxed);
1480 TypeDescriptor->
Die.store(Result, std::memory_order_relaxed);
1482 }
else if (!TypeDescriptor->
Die.load(std::memory_order_relaxed)) {
1491 (BetterParent || Priority < TypeDescriptor->DeclarationDiePriority)) {
1495 TypeDescriptor->
DeclarationDie.store(Result, std::memory_order_relaxed);
1499 TypeDescriptor->
Lock.clear(std::memory_order_release);
1503TypeEntry *CompileUnit::createTypeDIEandCloneAttributes(
1504 const DWARFDebugInfoEntry *InputDieEntry,
DIEGenerator &TypeDIEGenerator,
1506 uint32_t SiblingOrdinal) {
1507 assert(ArtificialTypeUnit !=
nullptr);
1508 uint32_t InputDieIdx =
getDIEIndex(InputDieEntry);
1511 assert(Entry !=
nullptr);
1512 assert(ClonedParentTypeDIE !=
nullptr);
1513 TypeEntryBody *EntryBody =
1515 Entry, ClonedParentTypeDIE);
1522 if (std::optional<uint32_t> ParentIdx = InputDieEntry->
getParentIdx()) {
1524 if (ParentTag == dwarf::DW_TAG_structure_type ||
1525 ParentTag == dwarf::DW_TAG_class_type ||
1526 ParentTag == dwarf::DW_TAG_union_type ||
1527 ParentTag == dwarf::DW_TAG_interface_type) {
1528 uint32_t Prev = EntryBody->
SortKey.load(std::memory_order_relaxed);
1529 while (SiblingOrdinal < Prev &&
1530 !EntryBody->
SortKey.compare_exchange_weak(
1531 Prev, SiblingOrdinal, std::memory_order_relaxed,
1532 std::memory_order_relaxed))
1537 bool IsDeclaration =
1540 bool ParentIsDeclaration =
false;
1541 if (std::optional<uint32_t> ParentIdx = InputDieEntry->
getParentIdx())
1542 ParentIsDeclaration =
1546 allocateTypeDie(EntryBody, TypeDIEGenerator, InputDieEntry->
getTag(),
1547 IsDeclaration, ParentIsDeclaration);
1549 if (OutDIE !=
nullptr) {
1550 assert(ArtificialTypeUnit !=
nullptr);
1553 DIEAttributeCloner AttributesCloner(OutDIE, *
this, ArtificialTypeUnit,
1554 InputDieEntry, TypeDIEGenerator,
1555 std::nullopt, std::nullopt,
false);
1556 AttributesCloner.clone();
1559 AcceleratorRecordsSaver AccelRecordsSaver(
getGlobalData(), *
this,
1560 ArtificialTypeUnit);
1561 AccelRecordsSaver.save(InputDieEntry, OutDIE, AttributesCloner.AttrInfo,
1566 OutDIE->
setSize(AttributesCloner.getOutOffset() + 1);
1575 if (InputLineTable ==
nullptr) {
1577 warn(
"cann't load line table.");
1589 OutLineTable.
Rows = InputLineTable->
Rows;
1592 if (OutLineTable.
Rows.size() == 1 && OutLineTable.
Rows[0].EndSequence)
1593 OutLineTable.
Rows.clear();
1600 filterLineTableRows(*InputLineTable, OutLineTable.
Rows, OrigRowIndices);
1602 if (StmtSeqListAttributes.empty())
1622 &RowIndexToSeqStartOffset))
1626 buildStmtSeqOffsetToFirstRowIndex(*InputLineTable);
1627 patchStmtSeqAttributes(SeqOffsetToFirstRowIndex, RowIndexToSeqStartOffset);
1631void CompileUnit::filterLineTableRows(
1633 std::vector<DWARFDebugLine::Row> &NewRows,
1635 NewRows.reserve(InputLineTable.
Rows.size());
1636 NewRowIndices.
reserve(InputLineTable.
Rows.size());
1642 std::vector<DWARFDebugLine::Row> Seq;
1644 constexpr uint64_t InvalidRowIndex = std::numeric_limits<uint64_t>::max();
1647 std::optional<AddressRangeValuePair> CurrRange;
1667 if (!CurrRange || !CurrRange->Range.contains(Row.Address.Address)) {
1671 CurrRange ? CurrRange->Range.end() + CurrRange->Value : -1ULL;
1672 CurrRange = FunctionRanges.getRangeThatContains(Row.Address.Address);
1673 if (StopAddress != -1ULL && !Seq.empty()) {
1678 auto NextLine = Seq.back();
1679 NextLine.Address.Address = StopAddress;
1680 NextLine.EndSequence = 1;
1681 NextLine.PrologueEnd = 0;
1682 NextLine.BasicBlock = 0;
1683 NextLine.EpilogueBegin = 0;
1684 Seq.push_back(NextLine);
1694 if (Row.EndSequence && Seq.empty())
1699 Seq.emplace_back(Row);
1702 if (Row.EndSequence)
1707void CompileUnit::patchStmtSeqAttributes(
1708 const DenseMap<uint64_t, uint64_t> &SeqOffsetToFirstRowIndex,
1709 const DenseMap<uint64_t, uint64_t> &RowIndexToSeqStartOffset) {
1712 for (
const CompileUnit::StmtSeqPatch &Patch : StmtSeqListAttributes) {
1713 uint64_t NewStmtSeq = InvalidOffset;
1714 auto RowIt = SeqOffsetToFirstRowIndex.
find(Patch.InputStmtSeqOffset);
1715 if (RowIt != SeqOffsetToFirstRowIndex.
end()) {
1716 auto OffIt = RowIndexToSeqStartOffset.
find(RowIt->second);
1717 if (OffIt != RowIndexToSeqStartOffset.
end())
1718 NewStmtSeq = OffIt->second;
1724 *Patch.Value = DIEValue(Patch.Value->getAttribute(), Patch.Value->getForm(),
1725 DIEInteger(NewStmtSeq));
1729DenseMap<uint64_t, uint64_t> CompileUnit::buildStmtSeqOffsetToFirstRowIndex(
1730 const DWARFDebugLine::LineTable &InputLineTable)
const {
1733 SmallVector<uint64_t> StmtAttrs;
1734 StmtAttrs.
reserve(StmtSeqListAttributes.size());
1735 for (
const StmtSeqPatch &Patch : StmtSeqListAttributes)
1736 StmtAttrs.
push_back(Patch.InputStmtSeqOffset);
1740 DenseMap<uint64_t, uint64_t>
Result;
1746void CompileUnit::insertLineSequence(std::vector<DWARFDebugLine::Row> &Seq,
1747 SmallVectorImpl<uint64_t> &SeqIndices,
1748 std::vector<DWARFDebugLine::Row> &Rows,
1749 SmallVectorImpl<uint64_t> &RowIndices) {
1751 "Seq and SeqIndices must be kept in lockstep");
1753 "Rows and RowIndices must be kept in lockstep");
1757 auto ClearSeq = [&] {
1762 if (!Rows.empty() && Rows.back().Address < Seq.front().Address) {
1769 object::SectionedAddress Front = Seq.front().Address;
1771 Rows, [=](
const DWARFDebugLine::Row &O) {
return O.Address < Front; });
1772 size_t InsertIdx = std::distance(Rows.begin(), InsertPoint);
1778 if (InsertPoint != Rows.end() && InsertPoint->Address == Front &&
1779 InsertPoint->EndSequence) {
1780 *InsertPoint = Seq.front();
1781 RowIndices[InsertIdx] = SeqIndices.
front();
1782 Rows.insert(InsertPoint + 1, Seq.begin() + 1, Seq.end());
1783 RowIndices.
insert(RowIndices.
begin() + InsertIdx + 1,
1784 SeqIndices.
begin() + 1, SeqIndices.
end());
1786 Rows.insert(InsertPoint, Seq.begin(), Seq.end());
1794#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
1814 llvm::errs() <<
" KeepPlainChildren: " << getKeepPlainChildren();
1815 llvm::errs() <<
" KeepTypeChildren: " << getKeepTypeChildren();
1816 llvm::errs() <<
" IsInMouduleScope: " << getIsInMouduleScope();
1817 llvm::errs() <<
" IsInFunctionScope: " << getIsInFunctionScope();
1818 llvm::errs() <<
" IsInAnonNamespaceScope: " << getIsInAnonNamespaceScope();
1819 llvm::errs() <<
" ODRAvailable: " << getODRAvailable();
1820 llvm::errs() <<
" TrackLiveness: " << getTrackLiveness();
1825std::optional<std::pair<StringRef, StringRef>>
1836 return std::nullopt;
1841std::optional<std::pair<StringRef, StringRef>>
1843 FileNamesCache::iterator FileData =
FileNames.find(FileIdx);
1845 return std::make_pair(
StringRef(FileData->second.first),
1850 if (LineTable->hasFileAtIndex(FileIdx)) {
1853 LineTable->Prologue.getFileNameEntry(FileIdx);
1858 return std::nullopt;
1861 std::string FileName = *
Name;
1863 FileNamesCache::iterator FileData =
1865 .insert(std::make_pair(
1867 std::make_pair(std::string(
""), std::move(FileName))))
1869 return std::make_pair(
StringRef(FileData->second.first),
1879 if ((Entry.DirIdx != 0) &&
1880 Entry.DirIdx < LineTable->Prologue.IncludeDirectories.size()) {
1882 LineTable->Prologue.IncludeDirectories[Entry.DirIdx]
1885 IncludeDir = *DirName;
1888 return std::nullopt;
1892 if (0 < Entry.DirIdx &&
1893 Entry.DirIdx <= LineTable->Prologue.IncludeDirectories.size()) {
1895 LineTable->Prologue.IncludeDirectories[Entry.DirIdx - 1]
1898 IncludeDir = *DirName;
1901 return std::nullopt;
1914 FileNamesCache::iterator FileData =
1917 std::make_pair(FileIdx, std::make_pair(std::string(FilePath),
1918 std::move(FileName))))
1920 return std::make_pair(
StringRef(FileData->second.first),
1925 return std::nullopt;
1928#define MAX_REFERENCIES_DEPTH 1000
1931 std::optional<UnitEntryPairTy> RefDiePair;
1935 CUDiePair.
DieEntry, dwarf::DW_AT_extension,
1937 if (!RefDiePair || !RefDiePair->DieEntry)
1940 CUDiePair = *RefDiePair;
1947 if (std::optional<uint32_t> ParentIdx =
DieEntry->getParentIdx())
1950 return std::nullopt;
1986 bool InterCUProcessingStarted, std::atomic<bool> &HasNewInterconnectedCUs) {
1990 return Dependencies->resolveDependenciesAndMarkLiveness(
1991 InterCUProcessingStarted, HasNewInterconnectedCUs);
1995 assert(Dependencies.get());
1997 return Dependencies->updateDependenciesCompleteness();
2001 assert(Dependencies.get());
2003 Dependencies->verifyKeepChain();
2008 dwarf::DW_AT_type, dwarf::DW_AT_specification,
2009 dwarf::DW_AT_abstract_origin, dwarf::DW_AT_import};
2011 return ODRAttributes;
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
#define LLVM_DUMP_METHOD
Mark debug helper function definitions like dump() that should not be stripped from debug builds.
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
#define MAX_REFERENCIES_DEPTH
static bool isValid(const char C)
Returns true if C is a valid mangled character: <0-9a-zA-Z_>.
std::optional< T > getRangeThatContains(uint64_t Addr) const
void insert(AddressRange Range, int64_t Value)
The AddressRanges class helps normalize address range collections.
Collection::const_iterator insert(AddressRange Range)
Represent a constant reference to an array (0 or more elements consecutively in memory),...
std::pair< KeyDataTy *, bool > insert(const KeyTy &NewValue)
Insert new value NewValue or return already existing entry.
A structured debug information entry.
DWARFDebugInfoEntry - A DIE with only the minimum required data.
dwarf::Tag getTag() const
std::optional< uint32_t > getParentIdx() const
Returns index of the parent die.
const DWARFAbbreviationDeclaration * getAbbreviationDeclarationPtr() const
Utility class that carries the DWARF compile/type unit and the debug info entry in an object.
LLVM_ABI std::optional< DWARFFormValue > find(dwarf::Attribute Attr) const
Extract the specified attribute from this DIE.
const DWARFDebugInfoEntry * getDebugInfoEntry() const
LLVM_ABI const char * getName(DINameKind Kind) const
Return the DIE name resolving DW_AT_specification or DW_AT_abstract_origin references if necessary.
Encoding
Size and signedness of expression operations' operands.
StringRef getData() const
const dwarf::FormParams & getFormParams() const
DWARFDie getUnitDIE(bool ExtractUnitDIEOnly=true)
uint8_t getAddressByteSize() const
const char * getCompilationDir()
Expected< DWARFLocationExpressionsVector > findLoclistFromOffset(uint64_t Offset)
bool isLittleEndian() const
uint64_t getOffset() const
iterator find(const_arg_type_t< KeyT > Val)
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT > iterator
DenseMapIterator< KeyT, ValueT, KeyInfoT, BucketT, true > const_iterator
std::pair< iterator, bool > insert(const std::pair< KeyT, ValueT > &KV)
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
Error takeError()
Take ownership of the stored error.
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
void reserve(size_type N)
iterator erase(const_iterator CI)
void append(ItTy in_start, ItTy in_end)
Add the specified range to the end of the SmallVector.
iterator insert(iterator I, T &&Elt)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringMapIterBase< ValueTy, false > iterator
Represent a constant reference to a string, i.e.
std::string str() const
Get the contents as an std::string.
constexpr bool empty() const
Check if the string is empty.
StringRef slice(size_t Start, size_t End) const
Return a reference to the substring from [Start, End).
Triple - Helper class for working with autoconf configuration names.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
This class represents DWARF information for source file and it's address map.
std::unique_ptr< AddressesMap > Addresses
Helpful address information(list of valid address ranges, relocations).
std::unique_ptr< DWARFContext > Dwarf
Source DWARF information.
@ Pub
.debug_pubnames, .debug_pubtypes
std::map< std::string, std::string > SwiftInterfacesMapTy
CompileUnit(DWARFUnit &OrigUnit, unsigned ID, bool CanUseODR, StringRef ClangModuleName)
TypeUnit * getAsTypeUnit()
Returns TypeUnit if applicable.
DwarfUnit * operator->()
Accessor for common functionality.
PointerUnion< CompileUnit *, TypeUnit * > Ptr
CompileUnit * getAsCompileUnit()
Returns CompileUnit if applicable.
OutputUnitVariantPtr(CompileUnit *U)
void addLabelLowPc(uint64_t LabelLowPc, int64_t PcOffset)
Add the low_pc of a label that is relocated by applying offset PCOffset.
Error cloneAndEmitDebugLocations()
Clone and emit debug locations(.debug_loc/.debug_loclists).
void cloneDieAttrExpression(const DWARFExpression &InputExpression, SmallVectorImpl< uint8_t > &OutputExpression, SectionDescriptor &Section, std::optional< int64_t > VarAddressAdjustment, OffsetsPtrVector &PatchesOffsets)
Clone attribute location axpression.
void maybeResetToLoadedStage()
Reset compile units data(results of liveness analysis, clonning) if current stage greater than Stage:...
void addFunctionRange(uint64_t LowPC, uint64_t HighPC, int64_t PCOffset)
Add a function range [LowPC, HighPC) that is relocated by applying offset PCOffset.
void analyzeImportedModule(const DWARFDebugInfoEntry *DieEntry)
Collect references to parseable Swift interfaces in imported DW_TAG_module blocks.
std::pair< DIE *, TypeEntry * > cloneDIE(const DWARFDebugInfoEntry *InputDieEntry, TypeEntry *ClonedParentTypeDIE, uint64_t OutOffset, std::optional< int64_t > FuncAddressAdjustment, std::optional< int64_t > VarAddressAdjustment, BumpPtrAllocator &Allocator, TypeUnit *ArtificialTypeUnit, uint32_t SiblingOrdinal=std::numeric_limits< uint32_t >::max())
void cleanupDataAfterClonning()
Cleanup unneeded resources after compile unit is cloned.
Error assignTypeNames(TypePool &TypePoolRef)
Search for type entries and assign names.
llvm::Error setPriority(uint64_t ObjFileIdx, uint64_t LocalIdx)
Set deterministic priority for type DIE allocation ordering.
@ TypeTable
Corresponding DIE goes to the type table only.
@ PlainDwarf
Corresponding DIE goes to the plain dwarf only.
Error cloneAndEmitLineTable(const Triple &TargetTriple)
Error cloneAndEmitRanges()
Clone and emit ranges.
void mergeSwiftInterfaces(DWARFLinkerBase::SwiftInterfacesMapTy &Map)
Merge the Swift interface entries collected by analyzeImportedModule into Map, emitting a warning for...
void updateDieRefPatchesWithClonedOffsets()
After cloning stage the output DIEs offsets are deallocated.
uint64_t getDebugAddrIndex(uint64_t Addr)
Returns index(inside .debug_addr) of an address.
const DWARFFile & getContaingFile() const
Returns DWARFFile containing this compile unit.
bool resolveDependenciesAndMarkLiveness(bool InterCUProcessingStarted, std::atomic< bool > &HasNewInterconnectedCUs)
Search for subprograms and variables referencing live code and discover dependend DIEs.
bool updateDependenciesCompleteness()
Check dependend DIEs for incompatible placement.
bool loadInputDIEs()
Load DIEs of input compilation unit.
const RangesTy & getFunctionRanges() const
Returns function ranges of this unit.
Error cloneAndEmitDebugMacro()
Clone and emit debug macros(.debug_macinfo/.debug_macro).
Error cloneAndEmit(std::optional< std::reference_wrapper< const Triple > > TargetTriple, TypeUnit *ArtificialTypeUnit)
Clone and emit this compilation unit.
void setStage(Stage Stage)
Set stage of overall processing.
Stage getStage() const
Returns stage of overall processing.
CompileUnit(LinkingGlobalData &GlobalData, unsigned ID, StringRef ClangModuleName, DWARFFile &File, OffsetToUnitTy UnitFromOffset, dwarf::FormParams Format, llvm::endianness Endianess)
void verifyDependencies()
Check DIEs to have a consistent marking(keep marking, placement marking).
Stage
The stages of new compile unit processing.
@ Cloned
Output DWARF is generated.
@ CreatedNotLoaded
Created, linked with input DWARF file.
@ Loaded
Input DWARF is loaded.
std::optional< uint64_t > getLowPc() const
Returns value of DW_AT_low_pc attribute.
std::optional< std::pair< StringRef, StringRef > > getDirAndFilenameFromLineTable(const DWARFFormValue &FileIdxValue)
Returns directory and file from the line table by index.
std::optional< UnitEntryPairTy > resolveDIEReference(const DWARFFormValue &RefValue, ResolveInterCUReferencesMode CanResolveInterCUReferences)
Resolve the DIE attribute reference that has been extracted in RefValue.
void loadLineTable()
Loads unit line table.
uint64_t getPriority() const
StringEntry * getFileName(unsigned FileIdx, StringPool &GlobalStrings)
Returns name of the file for the FileIdx from the unit`s line table.
This class is a helper to create output DIE tree.
void addChild(DIE *Child)
Adds a specified Child to the current DIE.
DIE * createDIE(dwarf::Tag DieTag, uint32_t OutOffset)
Creates a DIE of specified tag DieTag and OutOffset.
This class discovers DIEs dependencies: marks "live" DIEs, marks DIE locations (whether DIE should be...
std::string UnitName
The name of this unit.
LinkingGlobalData & getGlobalData()
Return global data.
std::vector< std::unique_ptr< DIEAbbrev > > Abbreviations
Storage for the unique Abbreviations.
DIE * OutUnitDIE
Output unit DIE.
std::string SysRoot
The DW_AT_LLVM_sysroot of this unit.
bool isClangModule() const
Return true if this compile unit is from Clang module.
unsigned ID
Unique ID for the unit.
const std::string & getClangModuleName() const
Return Clang module name;.
void setOutUnitDIE(DIE *UnitDie)
Set output unit DIE.
DwarfUnit(LinkingGlobalData &GlobalData, unsigned ID, StringRef ClangModuleName)
std::string ClangModuleName
If this is a Clang module, this holds the module's name.
FoldingSet< DIEAbbrev > AbbreviationsSet
FoldingSet that uniques the abbreviations.
StringRef getSysRoot()
Return the DW_AT_LLVM_sysroot of the compile unit or an empty StringRef.
DIE * getOutUnitDIE()
Returns output unit DIE.
This class keeps data and services common for the whole linking process.
This class helps to assign indexes for DIE children.
LinkingGlobalData & GlobalData
dwarf::FormParams Format
Format for sections.
const dwarf::FormParams & getFormParams() const
Return size of address.
void eraseSections()
Erases data of all sections.
std::optional< const SectionDescriptor * > tryGetSectionDescriptor(DebugSectionKind SectionKind) const
Returns descriptor for the specified section of SectionKind.
void setOutputFormat(dwarf::FormParams Format, llvm::endianness Endianness)
Sets output format for all keeping sections.
uint16_t getVersion() const
Return DWARF version.
uint16_t getDebugInfoHeaderSize() const
Return size of header of debug_info table.
llvm::endianness getEndianness() const
Endiannes for the sections.
SectionDescriptor & getOrCreateSectionDescriptor(DebugSectionKind SectionKind)
Returns descriptor for the specified section of SectionKind.
const SectionDescriptor & getSectionDescriptor(DebugSectionKind SectionKind) const
Returns descriptor for the specified section of SectionKind.
The helper class to build type name based on DIE properties.
Error assignName(UnitEntryPairTy InputUnitEntryPair, std::optional< std::pair< size_t, size_t > > ChildIndex)
Create synthetic name for the specified DIE InputUnitEntryPair and assign created name to the DIE typ...
Keeps cloned data for the type DIE.
bool DeclarationParentIsDeclaration
uint64_t DeclarationDiePriority
std::atomic< uint32_t > SortKey
std::atomic< DIE * > DeclarationDie
std::atomic< DIE * > Die
TypeEntryBody keeps partially cloned DIEs corresponding to this type.
std::atomic< uint64_t > DiePriority
TypePool keeps type descriptors which contain partially cloned DIE correspinding to each type.
BumpPtrAllocator & getThreadLocalAllocator()
Return thread local allocator used by pool.
TypeEntryBody * getOrCreateTypeEntryBody(TypeEntry *Entry, TypeEntry *ParentEntry)
Create or return existing type entry body for the specified Entry.
TypeEntry * getRoot() const
Return root for all type entries.
Type Unit is used to represent an artificial compilation unit which keeps all type information.
TypePool & getTypePool()
Returns global type pool.
uint64_t tell() const
tell - Return the current offset with the file.
void rememberDieOutOffset(uint32_t Idx, uint64_t Offset)
Idx index of the DIE.
TypeEntry * getDieTypeEntry(uint32_t Idx)
Idx index of the DIE.
DIEInfo & getDIEInfo(unsigned Idx)
Idx index of the DIE.
const DWARFDebugInfoEntry * getSiblingEntry(const DWARFDebugInfoEntry *Die) const
const DWARFDebugInfoEntry * getFirstChildEntry(const DWARFDebugInfoEntry *Die) const
std::optional< uint32_t > getDIEIndexForOffset(uint64_t Offset)
DWARFDie getDIE(const DWARFDebugInfoEntry *Die)
const DWARFDebugInfoEntry * getDebugInfoEntry(unsigned Index) const
DWARFUnit & getOrigUnit() const
Returns paired compile unit from input DWARF.
DWARFDie getUnitDIE(bool ExtractUnitDIEOnly=true)
uint32_t getDIEIndex(const DWARFDebugInfoEntry *Die) const
std::optional< DWARFFormValue > find(uint32_t DieIdx, ArrayRef< dwarf::Attribute > Attrs) const
Error emitDebugInfo(const Triple &TargetTriple)
Emit .debug_info section for unit DIEs.
Error emitDebugLine(const Triple &TargetTriple, const DWARFDebugLine::LineTable &OutLineTable, ArrayRef< uint64_t > OrigRowIndices={}, DenseMap< uint64_t, uint64_t > *RowIndexToSeqStartOffset=nullptr)
Emit .debug_line section.
Error emitDebugStringOffsetSection()
Emit the .debug_str_offsets section for current unit.
void emitPubAccelerators()
Emit .debug_pubnames and .debug_pubtypes for Unit.
void warn(const Twine &Warning, const DWARFDie *DIE=nullptr)
Error emitAbbreviations()
constexpr char Align[]
Key for Kernel::Arg::Metadata::mAlign.
bool isODRLanguage(uint16_t Language)
function_ref< CompileUnit *(uint64_t Offset)> OffsetToUnitTy
SmallVector< uint64_t * > OffsetsPtrVector
Type for list of pointers to patches offsets.
StringMapEntry< std::atomic< TypeEntryBody * > > TypeEntry
ArrayRef< dwarf::Attribute > getODRAttributes()
ResolveInterCUReferencesMode
StringRef guessDeveloperDir(StringRef SysRoot)
Make a best effort to guess the Xcode.app/Contents/Developer path from an SDK path.
DebugSectionKind
List of tracked debug tables.
void buildStmtSeqOffsetToFirstRowIndex(const DWARFDebugLine::LineTable <, ArrayRef< uint64_t > SortedStmtSeqOffsets, DenseMap< uint64_t, uint64_t > &SeqOffToFirstRow)
Build a map from an input DW_AT_LLVM_stmt_sequence byte offset to the first-row index (in LT....
StringMapEntry< EmptyStringSetTag > StringEntry
StringEntry keeps data of the string: the length, external offset and a string body which is placed r...
bool isInToolchainDir(StringRef Path)
Make a best effort to determine whether Path is inside a toolchain.
bool isPathAbsoluteOnWindowsOrPosix(const Twine &Path)
std::optional< uint64_t > toAddress(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract an address.
std::optional< const char * > toString(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract a string value from it.
std::optional< uint64_t > toSectionOffset(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract an section offset.
StringRef toStringRef(const std::optional< DWARFFormValue > &V, StringRef Default={})
Take an optional DWARFFormValue and try to extract a string value from it.
std::optional< uint64_t > toUnsigned(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract an unsigned constant.
@ DW_ARANGES_VERSION
Section version number for .debug_aranges.
LLVM_ABI std::error_code real_path(const Twine &path, SmallVectorImpl< char > &output, bool expand_tilde=false)
Collapse all .
LLVM_ABI StringRef parent_path(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get parent path.
LLVM_ABI bool is_relative(const Twine &path, Style style=Style::native)
Is path relative?
LLVM_ABI StringRef filename(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get filename.
LLVM_ABI void append(SmallVectorImpl< char > &path, const Twine &a, const Twine &b="", const Twine &c="", const Twine &d="")
Append to path.
void swapByteOrder(T &Value)
This is an optimization pass for GlobalISel generic memory operations.
auto find(R &&Range, const T &Val)
Provide wrappers to std::find which take ranges instead of having to pass begin/end explicitly.
void fill(R &&Range, T &&Value)
Provide wrappers to std::fill which take ranges instead of having to pass begin/end explicitly.
auto enumerate(FirstRange &&First, RestRanges &&...Rest)
Given two or more input ranges, returns a new range whose values are tuples (A, B,...
auto partition_point(R &&Range, Predicate P)
Binary search for the first iterator in a range where a predicate is false.
void append_range(Container &C, Range &&R)
Wrapper function to append range R to container C.
auto unique(Range &&R, Predicate P)
Error createStringError(std::error_code EC, char const *Fmt, const Ts &... Vals)
Create formatted StringError object.
static void insertLineSequence(std::vector< TrackedRow > &Seq, std::vector< TrackedRow > &Rows)
Insert the new line info sequence Seq into the current set of already linked line info Rows.
auto formatv(bool Validate, const char *Fmt, Ts &&...Vals)
void sort(IteratorTy Start, IteratorTy End)
@ Dwarf
DWARF v5 .debug_names.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
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...
uint64_t offsetToAlignment(uint64_t Value, Align Alignment)
Returns the offset to the next integer (mod 2**64) that is greater than or equal to Value and is a mu...
LLVM_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
FunctionAddr VTableAddr uintptr_t uintptr_t Data
DWARFExpression::Operation Op
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
unsigned encodeULEB128(uint64_t Value, raw_ostream &OS, unsigned PadTo=0)
Utility function to encode a ULEB128 value to an output stream.
bool isCompileUnit(const std::unique_ptr< DWARFUnit > &U)
BumpPtrAllocatorImpl<> BumpPtrAllocator
The standard BumpPtrAllocator which just uses the default template parameters.
void consumeError(Error Err)
Consume a Error without doing anything.
dwarf::FormParams FormParams
Version, address size (starting in v5), and DWARF32/64 format; these parameters affect interpretation...
Standard .debug_line state machine structure.
object::SectionedAddress Address
The program-counter value corresponding to a machine instruction generated by the compiler and sectio...
Represents a single DWARF expression, whose value is location-dependent.
Information gathered about source DIEs.
LLVM_DUMP_METHOD void dump()
bool needToPlaceInTypeTable() const
DieOutputPlacement getPlacement() const
This structure is used to update reference to the DIE.
uint64_t RefDieIdxOrClonedOffset
PointerIntPair< CompileUnit *, 1 > RefCU
This structure is used to update location list offset into .debug_loc/.debug_loclists.
int64_t AddrAdjustmentValue
This structure is used to update range list offset into .debug_ranges/.debug_rnglists.
bool IsCompileUnitRanges
Indicates patch which points to immediate compile unit's attribute.
This structure is used to update reference to the DIE of ULEB128 form.
uint64_t RefDieIdxOrClonedOffset
PointerIntPair< CompileUnit *, 1 > RefCU
dwarf::FormParams getFormParams() const
Returns FormParams used by section.
This structure is used to keep data of the concrete section.
raw_svector_ostream OS
Stream which stores data to the Contents.
void emitUnitLength(uint64_t Length)
Emit unit length into the current section contents.
void emitOffset(uint64_t Val)
Emit specified offset value into the current section contents.
void emitIntVal(uint64_t Val, unsigned Size)
Emit specified integer value into the current section contents.
void apply(uint64_t PatchOffset, dwarf::Form AttrForm, uint64_t Val)
Write specified Value of AttrForm to the PatchOffset.
uint64_t getIntVal(uint64_t PatchOffset, unsigned Size)
Returns integer value of Size located by specified PatchOffset.
This is a helper structure which keeps a debug info entry with it's containing compilation unit.
UnitEntryPairTy()=default
std::optional< UnitEntryPairTy > getParent()
UnitEntryPairTy getNamespaceOrigin()
const DWARFDebugInfoEntry * DieEntry