21 AcceleratorRecords(&
GlobalData.getAllocator()) {
29 LineTable.Prologue.MinInstLength = 1;
30 LineTable.Prologue.MaxOpsPerInst = 1;
31 LineTable.Prologue.DefaultIsStmt = 1;
32 LineTable.Prologue.LineBase = -5;
33 LineTable.Prologue.LineRange = 14;
34 LineTable.Prologue.OpcodeBase = 13;
35 LineTable.Prologue.StandardOpcodeLengths = {0, 1, 1, 1, 1, 0,
42 prepareDataForTreeCreation();
57 DIE *UnitDIE = DIETreeGenerator.
createDIE(dwarf::DW_TAG_compile_unit, 0);
62 ProducerString +=
"llvm DWARFLinkerParallel library version ";
66 GlobalData.getStringPool().insert(ProducerString.
str()).first},
68 OutOffset += DIETreeGenerator
74 OutOffset += DIETreeGenerator
76 dwarf::DW_FORM_data2, *Language)
84 OutOffset += DIETreeGenerator
89 if (!LineTable.Prologue.FileNames.empty()) {
93 OutOffset += DIETreeGenerator
95 dwarf::DW_FORM_sec_offset, 0xbaddef)
102 OutOffset += DIETreeGenerator
111 OutOffset += DIETreeGenerator
113 dwarf::DW_FORM_sec_offset,
120 finalizeTypeEntryRec(UnitDIE->
getOffset(), UnitDIE, Types.getRoot());
123 for (
uint64_t *OffsetPtr : PatchesOffsets)
130void TypeUnit::prepareDataForTreeCreation() {
153 return LHS.Directory->first() < RHS.Directory->first() ||
154 (!(RHS.Directory->first() < LHS.Directory->first()) &&
155 LHS.FilePath->first() < RHS.FilePath->first());
158 DebugInfoSection.ListDebugTypeDeclFilePatch.sort(PatchesComparator);
163 getScalarFormForValue(
164 DebugInfoSection.ListDebugTypeDeclFilePatch.size())
167 DebugInfoSection.ListDebugTypeDeclFilePatch.forEach(
168 [&](DebugTypeDeclFilePatch &Patch) {
181 DIEGenerator DIEGen(Patch.
Die, Types.getThreadLocalAllocator(),
185 .addScalarAttribute(dwarf::DW_AT_decl_file,
186 DeclFileForm, FileIdx)
192 if (!
GlobalData.getOptions().AllowNonDeterministicOutput) {
195 forEach([&](SectionDescriptor &OutSection) {
196 std::function<bool(
const DebugStrPatch &
LHS,
const DebugStrPatch &
RHS)>
197 StrPatchesComparator =
198 [&](
const DebugStrPatch &
LHS,
const DebugStrPatch &
RHS) {
199 return LHS.String->getKey() <
RHS.String->getKey();
201 OutSection.ListDebugStrPatch.sort(StrPatchesComparator);
203 std::function<bool(
const DebugTypeStrPatch &
LHS,
204 const DebugTypeStrPatch &
RHS)>
205 TypeStrPatchesComparator = [&](
const DebugTypeStrPatch &
LHS,
206 const DebugTypeStrPatch &
RHS) {
207 return LHS.String->getKey() <
RHS.String->getKey();
209 OutSection.ListDebugTypeStrPatch.sort(TypeStrPatchesComparator);
214 if (!
GlobalData.getOptions().AllowNonDeterministicOutput) {
217 forEach([&](SectionDescriptor &OutSection) {
218 std::function<bool(
const DebugLineStrPatch &
LHS,
219 const DebugLineStrPatch &
RHS)>
220 LineStrPatchesComparator = [&](
const DebugLineStrPatch &
LHS,
221 const DebugLineStrPatch &
RHS) {
222 return LHS.String->getKey() <
RHS.String->getKey();
224 OutSection.ListDebugLineStrPatch.sort(LineStrPatchesComparator);
226 std::function<bool(
const DebugTypeLineStrPatch &
LHS,
227 const DebugTypeLineStrPatch &
RHS)>
228 TypeLineStrPatchesComparator =
229 [&](
const DebugTypeLineStrPatch &
LHS,
230 const DebugTypeLineStrPatch &
RHS) {
231 return LHS.String->getKey() <
RHS.String->getKey();
233 OutSection.ListDebugTypeLineStrPatch.sort(TypeLineStrPatchesComparator);
239uint64_t TypeUnit::finalizeTypeEntryRec(uint64_t OutOffset, DIE *OutDIE,
241 bool HasChildren = !
Entry->getValue().load()->Children.empty();
242 DIEGenerator DIEGen(OutDIE, Types.getThreadLocalAllocator(), *
this);
243 OutOffset += DIEGen.finalizeAbbreviations(HasChildren,
nullptr);
244 OutOffset += OutDIE->
getSize() - 1;
247 Entry->getValue().load()->Children.forEach([&](
TypeEntry *ChildEntry) {
248 DIE *ChildDIE = &ChildEntry->
getValue().load()->getFinalDie();
249 DIEGen.addChild(ChildDIE);
253 OutOffset = finalizeTypeEntryRec(OutOffset, ChildDIE, ChildEntry);
257 OutOffset +=
sizeof(int8_t);
264uint32_t TypeUnit::addFileNameIntoLinetable(
StringEntry *Dir,
268 if (Dir->
first() ==
"") {
271 DirectoriesMapTy::iterator DirEntry = DirectoriesMap.find(Dir);
272 if (DirEntry == DirectoriesMap.end()) {
274 assert(LineTable.Prologue.IncludeDirectories.size() < UINT32_MAX);
275 DirIdx = LineTable.Prologue.IncludeDirectories.size();
276 DirectoriesMap.insert({Dir, DirIdx});
277 LineTable.Prologue.IncludeDirectories.push_back(
281 DirIdx = DirEntry->second;
288 auto [FileEntry,
Inserted] = FileNamesMap.try_emplace(
289 {FileName, DirIdx}, LineTable.Prologue.FileNames.size());
292 assert(LineTable.Prologue.FileNames.size() < UINT32_MAX);
293 LineTable.Prologue.FileNames.push_back(DWARFDebugLine::FileNameEntry());
295 dwarf::DW_FORM_string, FileName->
getKeyData());
296 LineTable.Prologue.FileNames.back().DirIdx = DirIdx;
299 uint32_t FileIdx = FileEntry->second;
300 return getVersion() < 5 ? FileIdx + 1 : FileIdx;
303std::pair<dwarf::Form, uint8_t>
304TypeUnit::getScalarFormForValue(uint64_t
Value)
const {
305 if (
Value > 0xFFFFFFFF)
306 return std::make_pair(dwarf::DW_FORM_data8, 8);
309 return std::make_pair(dwarf::DW_FORM_data4, 4);
312 return std::make_pair(dwarf::DW_FORM_data2, 2);
314 return std::make_pair(dwarf::DW_FORM_data1, 1);
317uint8_t TypeUnit::getSizeByAttrForm(
dwarf::Form Form)
const {
318 if (Form == dwarf::DW_FORM_data1)
321 if (Form == dwarf::DW_FORM_data2)
324 if (Form == dwarf::DW_FORM_data4)
327 if (Form == dwarf::DW_FORM_data8)
330 if (Form == dwarf::DW_FORM_data16)
358 if (!LineTable.Prologue.FileNames.empty()) {
369 Tasks.push_back([&]() ->
Error {
382 Tasks, [&](std::function<
Error(
void)>
F) {
return F(); }))
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
A structured debug information entry.
unsigned getAbbrevNumber() const
unsigned getOffset() const
Get the compile/type unit relative offset of this DIE.
void setOffset(unsigned O)
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
StringRef str() const
Explicit conversion to StringRef.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
const ValueTy & getValue() const
const char * getKeyData() const
getKeyData - Return the start of the string data that is the key for this value.
Triple - Helper class for working with autoconf configuration names.
@ Pub
.debug_pubnames, .debug_pubtypes
This class is a helper to create output DIE tree.
std::pair< DIEValue &, size_t > addStringPlaceholderAttribute(dwarf::Attribute Attr, dwarf::Form AttrForm)
Adds string attribute with dummy offset to the current DIE.
DIE * createDIE(dwarf::Tag DieTag, uint32_t OutOffset)
Creates a DIE of specified tag DieTag and OutOffset.
std::pair< DIEValue &, size_t > addScalarAttribute(dwarf::Attribute Attr, dwarf::Form AttrForm, uint64_t Value)
Adds specified scalar attribute to the current DIE.
std::string UnitName
The name of this unit.
IndexedValuesMap< const StringEntry * > DebugStringIndexMap
Maps a string into the index inside .debug_str_offsets section.
unsigned ID
Unique ID for the unit.
StringRef getUnitName() const
Returns this unit name.
void setOutUnitDIE(DIE *UnitDie)
Set output unit DIE.
DwarfUnit(LinkingGlobalData &GlobalData, unsigned ID, StringRef ClangModuleName)
DIE * getOutUnitDIE()
Returns output unit DIE.
This class keeps data and services common for the whole linking process.
const DWARFLinkerOptions & getOptions() const
Returns linking options.
uint16_t getDebugStrOffsetsHeaderSize() const
Return size of header of debug_str_offsets table.
LinkingGlobalData & GlobalData
dwarf::FormParams Format
Format for sections.
const dwarf::FormParams & getFormParams() const
Return size of address.
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.
void forEach(function_ref< void(SectionDescriptor &)> Handler)
Enumerate all sections and call Handler for each.
SectionDescriptor & getOrCreateSectionDescriptor(DebugSectionKind SectionKind)
Returns descriptor for the specified section of SectionKind.
Error finishCloningAndEmit(const Triple &TargetTriple)
Emits resulting dwarf based on information from DIE tree.
void createDIETree(BumpPtrAllocator &Allocator)
Generates DIE tree based on information from TypesMap.
TypeUnit(LinkingGlobalData &GlobalData, unsigned ID, std::optional< uint16_t > Language, dwarf::FormParams Format, llvm::endianness Endianess)
LLVM_ABI void spawn(std::function< void()> f)
Error emitDebugInfo(const Triple &TargetTriple)
Emit .debug_info section for unit DIEs.
Error emitDebugStringOffsetSection()
Emit the .debug_str_offsets section for current unit.
void emitPubAccelerators()
Emit .debug_pubnames and .debug_pubtypes for Unit.
Error emitAbbreviations()
Error emitDebugLine(const Triple &TargetTriple, const DWARFDebugLine::LineTable &OutLineTable)
Emit .debug_line section.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
SmallVector< uint64_t * > OffsetsPtrVector
Type for list of pointers to patches offsets.
StringMapEntry< std::atomic< TypeEntryBody * > > TypeEntry
StringMapEntry< std::nullopt_t > StringEntry
StringEntry keeps data of the string: the length, external offset and a string body which is placed r...
This is an optimization pass for GlobalISel generic memory operations.
FunctionAddr VTableAddr Value
auto formatv(bool Validate, const char *Fmt, Ts &&...Vals)
BumpPtrAllocatorImpl BumpPtrAllocator
The standard BumpPtrAllocator which just uses the default template parameters.
LLVM_ABI unsigned getULEB128Size(uint64_t Value)
Utility function to get the size of the ULEB128-encoded value.
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
Error parallelForEachError(RangeTy &&R, FuncTy Fn)
bool AllowNonDeterministicOutput
Allow to generate valid, but non deterministic output.
This structure is used to update strings offsets into .debug_str.
This structure is used to keep data of the concrete section.
void notePatchWithOffsetUpdate(const T &Patch, OffsetsPtrVector &PatchesOffsetsList)
While creating patches, offsets to attributes may be partially unknown(because size of abbreviation n...