16#define DEBUG_TYPE "jitlink"
30 std::shared_ptr<orc::SymbolStringPool> SSP,
Triple TT,
61bool COFFLinkGraphBuilder::isComdatSection(
62 const object::coff_section *Section) {
66Section &COFFLinkGraphBuilder::getCommonSection() {
70 return *CommonSection;
75 return make_error<JITLinkError>(
"Object is not a relocatable COFF file");
78 return std::move(Err);
81 return std::move(Err);
84 return std::move(Err);
90COFFLinkGraphBuilder::getCOFFSectionName(COFFSectionIndex SectionIndex,
93 switch (SectionIndex) {
109 return *SecNameOrErr;
142 <<
"Creating section for \"" <<
SectionName <<
"\"\n";
155 auto *GraphSec = G->findSectionByName(
SectionName);
161 if (GraphSec->getMemProt() != Prot)
162 return make_error<JITLinkError>(
"MemProt should match");
166 B = &G->createZeroFillBlock(
167 *GraphSec, getSectionSize(Obj, *Sec),
169 (*Sec)->getAlignment(), 0);
176 reinterpret_cast<const char *
>(
Data.data()),
Data.size());
179 if (
auto Err = handleDirectiveSection(
180 StringRef(CharData.data(), CharData.size())))
183 B = &G->createContentBlock(
185 (*Sec)->getAlignment(), 0);
206 return Sym.takeError();
210 SymbolName = *SymNameOrErr;
218 return make_error<JITLinkError>(
219 "Invalid COFF section number:" +
formatv(
"{0:d}: ", SectionIndex) +
220 " (" +
toString(SecOrErr.takeError()) +
")");
223 auto InternedSymbolName = G->intern(std::move(SymbolName));
227 if (
Sym->isFileRecord())
229 dbgs() <<
" " << SymIndex <<
": Skipping FileRecord symbol \""
230 << InternedSymbolName <<
"\" in "
231 << getCOFFSectionName(SectionIndex, Sec, *
Sym)
232 <<
" (index: " << SectionIndex <<
") \n";
234 else if (
Sym->isUndefined()) {
235 GSym = createExternalSymbol(SymIndex, InternedSymbolName, *
Sym, Sec);
236 }
else if (
Sym->isWeakExternal()) {
240 WeakExternalRequests.push_back(
244 createDefinedSymbol(SymIndex, InternedSymbolName, *
Sym, Sec);
250 dbgs() <<
" " << SymIndex
251 <<
": Creating defined graph symbol for COFF symbol \""
252 << InternedSymbolName <<
"\" in "
253 << getCOFFSectionName(SectionIndex, Sec, *
Sym)
254 <<
" (index: " << SectionIndex <<
") \n";
255 dbgs() <<
" " << *GSym <<
"\n";
263 SymIndex +=
Sym->getNumberOfAuxSymbols();
266 if (
auto Err = flushWeakAliasRequests())
269 if (
auto Err = handleAlternateNames())
272 if (
auto Err = calculateImplicitSizeOfSymbols())
278Error COFFLinkGraphBuilder::handleDirectiveSection(
StringRef Str) {
279 auto Parsed = DirectiveParser.
parse(Str);
281 return Parsed.takeError();
282 for (
auto *Arg : *Parsed) {
284 switch (Arg->getOption().getID()) {
285 case COFF_OPT_alternatename: {
289 return make_error<JITLinkError>(
290 "Invalid COFF /alternatename directive");
291 AlternateNames[G->intern(
From)] = G->intern(To);
294 case COFF_OPT_incl: {
295 auto Symbol = &G->addExternalSymbol(S, 0,
false);
296 Symbol->setLive(
true);
297 ExternalSymbols[Symbol->getName()] = Symbol;
300 case COFF_OPT_export:
304 dbgs() <<
"Unknown coff directive: " << Arg->getSpelling() <<
"\n";
313Error COFFLinkGraphBuilder::flushWeakAliasRequests() {
315 for (
auto &WeakExternal : WeakExternalRequests) {
317 Expected<object::COFFSymbolRef> AliasSymbol =
320 return AliasSymbol.takeError();
329 auto NewSymbol = createAliasSymbol(G->intern(WeakExternal.SymbolName),
332 return NewSymbol.takeError();
333 setGraphSymbol(AliasSymbol->getSectionNumber(), WeakExternal.Alias,
336 dbgs() <<
" " << WeakExternal.Alias
337 <<
": Creating weak external symbol for COFF symbol \""
338 << WeakExternal.SymbolName <<
"\" in section "
339 << AliasSymbol->getSectionNumber() <<
"\n";
340 dbgs() <<
" " << **NewSymbol <<
"\n";
343 return make_error<JITLinkError>(
"Weak symbol alias requested but actual "
344 "symbol not found for symbol " +
345 formatv(
"{0:d}", WeakExternal.Alias));
350Error COFFLinkGraphBuilder::handleAlternateNames() {
351 for (
auto &KeyValue : AlternateNames) {
352 auto DefinedSymbolName = KeyValue.second;
353 auto ExternalSymbolsName = KeyValue.first;
354 if (DefinedSymbols.count(DefinedSymbolName) &&
355 ExternalSymbols.count(ExternalSymbolsName)) {
356 auto *
Target = DefinedSymbols[DefinedSymbolName];
357 auto *Alias = ExternalSymbols[ExternalSymbolsName];
358 G->makeDefined(*Alias,
Target->getBlock(),
Target->getOffset(),
365Symbol *COFFLinkGraphBuilder::createExternalSymbol(
366 COFFSymbolIndex SymIndex, orc::SymbolStringPtr SymbolName,
367 object::COFFSymbolRef Symbol,
const object::coff_section *Section) {
369 if (!ExternalSymbols.count(SymbolName)) {
370 Sym = &G->addExternalSymbol(*SymbolName,
Symbol.getValue(),
false);
371 ExternalSymbols[
Sym->getName()] =
Sym;
377 dbgs() <<
" " << SymIndex
378 <<
": Creating external graph symbol for COFF symbol \""
379 <<
Sym->getName() <<
"\" in "
380 << getCOFFSectionName(
Symbol.getSectionNumber(), Section, Symbol)
381 <<
" (index: " <<
Symbol.getSectionNumber() <<
") \n";
387COFFLinkGraphBuilder::createAliasSymbol(orc::SymbolStringPtr SymbolName,
389 if (!
Target.isDefined()) {
391 return make_error<JITLinkError>(
"Weak external symbol with external "
392 "symbol as alternative not supported.");
394 return &G->addDefinedSymbol(
Target.getBlock(),
Target.getOffset(), SymbolName,
406Error COFFLinkGraphBuilder::calculateImplicitSizeOfSymbols() {
410 auto &SymbolSet = SymbolSets[SecIndex];
411 if (SymbolSet.empty())
417 for (
auto It = SymbolSet.rbegin(); It != SymbolSet.rend(); It++) {
419 jitlink::Symbol *
Symbol = It->second;
422 if (
Symbol->getOffset() == LastOffset)
425 CandSize = LastOffset -
Offset;
429 dbgs() <<
" Overlapping symbol range generated for the following "
432 <<
" " << *Symbol <<
"\n";
434 (void)LastDifferentOffset;
436 LastDifferentOffset =
Offset;
451 dbgs() <<
" Empty implicit symbol size generated for the following "
454 <<
" " << *Symbol <<
"\n";
457 Symbol->setSize(CandSize);
463Expected<Symbol *> COFFLinkGraphBuilder::createDefinedSymbol(
464 COFFSymbolIndex SymIndex, orc::SymbolStringPtr SymbolName,
465 object::COFFSymbolRef Symbol,
const object::coff_section *Section) {
469 return &G->addDefinedSymbol(
470 G->createZeroFillBlock(getCommonSection(),
Symbol.getValue(),
471 orc::ExecutorAddr(),
Symbol.getValue(), 0),
477 return &G->addAbsoluteSymbol(SymbolName,
478 orc::ExecutorAddr(
Symbol.getValue()), 0,
482 return make_error<JITLinkError>(
483 "Reserved section number used in regular symbol " +
489 dbgs() <<
" " << SymIndex
490 <<
": Skipping graph symbol since section was not created for "
498 if (
Symbol.isExternal()) {
500 if (!isComdatSection(Section)) {
501 auto GSym = &G->addDefinedSymbol(
507 if (!PendingComdatExports[
Symbol.getSectionNumber()])
508 return make_error<JITLinkError>(
"No pending COMDAT export for symbol " +
511 return exportCOMDATSymbol(SymIndex, SymbolName, Symbol);
517 const object::coff_aux_section_definition *Definition =
518 Symbol.getSectionDefinition();
519 if (!Definition || !isComdatSection(Section)) {
521 return &G->addDefinedSymbol(
527 auto GSym = &G->addDefinedSymbol(
533 if (PendingComdatExports[
Symbol.getSectionNumber()])
534 return make_error<JITLinkError>(
535 "COMDAT export request already exists before symbol " +
537 return createCOMDATExportRequest(SymIndex, Symbol, Definition);
539 return make_error<JITLinkError>(
"Unsupported storage class " +
541 " in symbol " +
formatv(
"{0:d}", SymIndex));
558Expected<Symbol *> COFFLinkGraphBuilder::createCOMDATExportRequest(
559 COFFSymbolIndex SymIndex, object::COFFSymbolRef Symbol,
560 const object::coff_aux_section_definition *Definition) {
562 switch (Definition->Selection) {
582 dbgs() <<
" " << SymIndex
583 <<
": Partially supported IMAGE_COMDAT_SELECT_LARGEST was used"
585 <<
Symbol.getSectionNumber() <<
" (size: " << Definition->Length
593 return make_error<JITLinkError>(
594 "IMAGE_COMDAT_SELECT_NEWEST is not supported.");
597 return make_error<JITLinkError>(
"Invalid comdat selection type: " +
598 formatv(
"{0:d}", Definition->Selection));
601 PendingComdatExports[
Symbol.getSectionNumber()] = {SymIndex,
L,
608COFFLinkGraphBuilder::exportCOMDATSymbol(COFFSymbolIndex SymIndex,
609 orc::SymbolStringPtr SymbolName,
610 object::COFFSymbolRef Symbol) {
612 auto &PendingComdatExport = PendingComdatExports[
Symbol.getSectionNumber()];
616 auto GSym = &G->addDefinedSymbol(
617 *
B,
Symbol.getValue(), SymbolName, 0, PendingComdatExport->Linkage,
621 dbgs() <<
" " << SymIndex
622 <<
": Exporting COMDAT graph symbol for COFF symbol \"" <<
SymbolName
623 <<
"\" in section " <<
Symbol.getSectionNumber() <<
"\n";
624 dbgs() <<
" " << *GSym <<
"\n";
629 PendingComdatExport = std::nullopt;
637 auto IBN =
G.intern(ImageBaseName);
640 for (
auto *
Sym :
G.external_symbols()) {
641 if (
Sym->getName() == IBN) {
648 for (
auto *
Sym :
G.absolute_symbols()) {
649 if (
Sym->getName() == IBN) {
656 for (
auto *
Sym :
G.defined_symbols()) {
657 if (
Sym->hasName() &&
Sym->getName() == IBN) {
BlockVerifier::State From
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
static const char * CommonSectionName
COFFYAML::WeakExternalCharacteristics Characteristics
static Expected< StringRef > getFileName(const DebugStringTableSubsectionRef &Strings, const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID)
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
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.
StringRef - Represent a constant reference to a string, i.e.
std::pair< StringRef, StringRef > split(char Separator) const
Split into two substrings around the first occurrence of a separator character.
constexpr bool empty() const
empty - Check if the string is empty.
Manages the enabling and disabling of subtarget specific features.
Triple - Helper class for working with autoconf configuration names.
An Addressable with content and edges.
void addEdge(Edge::Kind K, Edge::OffsetT Offset, Symbol &Target, Edge::AddendT Addend)
Add an edge to this block.
Expected< opt::InputArgList > parse(StringRef Str)
Expected< std::unique_ptr< LinkGraph > > buildGraph()
virtual Error addRelocations()=0
Symbol * getGraphSymbol(COFFSymbolIndex SymIndex) const
COFFLinkGraphBuilder(const object::COFFObjectFile &Obj, std::shared_ptr< orc::SymbolStringPool > SSP, Triple TT, SubtargetFeatures Features, LinkGraph::GetEdgeKindNameFunction GetEdgeKindName)
virtual ~COFFLinkGraphBuilder()
Block * getGraphBlock(COFFSectionIndex SecIndex) const
void setGraphSymbol(COFFSectionIndex SecIndex, COFFSymbolIndex SymIndex, Symbol &Sym)
void setGraphBlock(COFFSectionIndex SecIndex, Block *B)
Symbol * operator()(LinkGraph &G)
const char *(*)(Edge::Kind) GetEdgeKindNameFunction
StringRef getFileName() const
const dos_header * getDOSHeader() const
Expected< StringRef > getSectionName(DataRefImpl Sec) const override
Expected< COFFSymbolRef > getSymbol(uint32_t index) const
Expected< StringRef > getSymbolName(DataRefImpl Symb) const override
uint32_t getNumberOfSymbols() const
uint32_t getNumberOfSections() const
bool isRelocatableObject() const override
True if this is a relocatable object (.o/.obj).
Expected< const coff_section * > getSection(int32_t index) const
Expected< ArrayRef< uint8_t > > getSectionContents(DataRefImpl Sec) const override
uint64_t getImageBase() const
Represents an address in the executor process.
constexpr char SymbolName[]
Key for Kernel::Metadata::mSymbolName.
@ IMAGE_SCN_CNT_UNINITIALIZED_DATA
@ IMAGE_SYM_CLASS_LABEL
Label.
@ IMAGE_SYM_CLASS_STATIC
Static.
@ IMAGE_COMDAT_SELECT_NODUPLICATES
@ IMAGE_COMDAT_SELECT_LARGEST
@ IMAGE_COMDAT_SELECT_NEWEST
@ IMAGE_COMDAT_SELECT_SAME_SIZE
@ IMAGE_COMDAT_SELECT_ASSOCIATIVE
@ IMAGE_COMDAT_SELECT_EXACT_MATCH
@ IMAGE_COMDAT_SELECT_ANY
@ IMAGE_WEAK_EXTERN_SEARCH_ALIAS
bool isReservedSectionNumber(int32_t SectionNumber)
@ IMAGE_SYM_DTYPE_FUNCTION
A function that returns a base type.
static Triple createTripleWithCOFFFormat(Triple T)
Linkage
Describes symbol linkage. This can be used to resolve definition clashes.
Scope
Defines the scope in which this symbol should be visible: Default – Visible in the public interface o...
MemProt
Describes Read/Write/Exec permissions for memory.
uint64_t ExecutorAddrDiff
@ NoAlloc
NoAlloc memory should not be allocated by the JITLinkMemoryManager at all.
NodeAddr< BlockNode * > Block
This is an optimization pass for GlobalISel generic memory operations.
auto formatv(bool Validate, const char *Fmt, Ts &&...Vals)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
const char * toString(DWARFSectionKind Kind)
Implement std::hash so that hash_code can be used in STL containers.
support::ulittle32_t TagIndex
support::ulittle32_t VirtualSize
support::ulittle32_t SizeOfRawData