1 //===----------- JITSymbol.cpp - JITSymbol class implementation -----------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // JITSymbol class implementation plus helper functions.
11 //===----------------------------------------------------------------------===//
13 #include "llvm/ExecutionEngine/JITSymbol.h"
14 #include "llvm/IR/Function.h"
15 #include "llvm/IR/GlobalAlias.h"
16 #include "llvm/IR/GlobalValue.h"
17 #include "llvm/IR/ModuleSummaryIndex.h"
18 #include "llvm/Object/ObjectFile.h"
22 JITSymbolFlags
llvm::JITSymbolFlags::fromGlobalValue(const GlobalValue
&GV
) {
23 assert(GV
.hasName() && "Can't get flags for anonymous symbol");
25 JITSymbolFlags Flags
= JITSymbolFlags::None
;
26 if (GV
.hasWeakLinkage() || GV
.hasLinkOnceLinkage())
27 Flags
|= JITSymbolFlags::Weak
;
28 if (GV
.hasCommonLinkage())
29 Flags
|= JITSymbolFlags::Common
;
30 if (!GV
.hasLocalLinkage() && !GV
.hasHiddenVisibility())
31 Flags
|= JITSymbolFlags::Exported
;
33 if (isa
<Function
>(GV
))
34 Flags
|= JITSymbolFlags::Callable
;
35 else if (isa
<GlobalAlias
>(GV
) &&
36 isa
<Function
>(cast
<GlobalAlias
>(GV
).getAliasee()))
37 Flags
|= JITSymbolFlags::Callable
;
39 // Check for a linker-private-global-prefix on the symbol name, in which
40 // case it must be marked as non-exported.
41 if (auto *M
= GV
.getParent()) {
42 const auto &DL
= M
->getDataLayout();
43 StringRef LPGP
= DL
.getLinkerPrivateGlobalPrefix();
44 if (!LPGP
.empty() && GV
.getName().front() == '\01' &&
45 GV
.getName().substr(1).startswith(LPGP
))
46 Flags
&= ~JITSymbolFlags::Exported
;
52 JITSymbolFlags
llvm::JITSymbolFlags::fromSummary(GlobalValueSummary
*S
) {
53 JITSymbolFlags Flags
= JITSymbolFlags::None
;
54 auto L
= S
->linkage();
55 if (GlobalValue::isWeakLinkage(L
) || GlobalValue::isLinkOnceLinkage(L
))
56 Flags
|= JITSymbolFlags::Weak
;
57 if (GlobalValue::isCommonLinkage(L
))
58 Flags
|= JITSymbolFlags::Common
;
59 if (GlobalValue::isExternalLinkage(L
) || GlobalValue::isExternalWeakLinkage(L
))
60 Flags
|= JITSymbolFlags::Exported
;
62 if (isa
<FunctionSummary
>(S
))
63 Flags
|= JITSymbolFlags::Callable
;
68 Expected
<JITSymbolFlags
>
69 llvm::JITSymbolFlags::fromObjectSymbol(const object::SymbolRef
&Symbol
) {
70 Expected
<uint32_t> SymbolFlagsOrErr
= Symbol
.getFlags();
71 if (!SymbolFlagsOrErr
)
72 // TODO: Test this error.
73 return SymbolFlagsOrErr
.takeError();
75 JITSymbolFlags Flags
= JITSymbolFlags::None
;
76 if (*SymbolFlagsOrErr
& object::BasicSymbolRef::SF_Weak
)
77 Flags
|= JITSymbolFlags::Weak
;
78 if (*SymbolFlagsOrErr
& object::BasicSymbolRef::SF_Common
)
79 Flags
|= JITSymbolFlags::Common
;
80 if (*SymbolFlagsOrErr
& object::BasicSymbolRef::SF_Exported
)
81 Flags
|= JITSymbolFlags::Exported
;
83 auto SymbolType
= Symbol
.getType();
85 return SymbolType
.takeError();
87 if (*SymbolType
== object::SymbolRef::ST_Function
)
88 Flags
|= JITSymbolFlags::Callable
;
94 llvm::ARMJITSymbolFlags::fromObjectSymbol(const object::SymbolRef
&Symbol
) {
95 Expected
<uint32_t> SymbolFlagsOrErr
= Symbol
.getFlags();
96 if (!SymbolFlagsOrErr
)
97 // TODO: Actually report errors helpfully.
98 report_fatal_error(SymbolFlagsOrErr
.takeError());
99 ARMJITSymbolFlags Flags
;
100 if (*SymbolFlagsOrErr
& object::BasicSymbolRef::SF_Thumb
)
101 Flags
|= ARMJITSymbolFlags::Thumb
;
105 /// Performs lookup by, for each symbol, first calling
106 /// findSymbolInLogicalDylib and if that fails calling
108 void LegacyJITSymbolResolver::lookup(const LookupSet
&Symbols
,
109 OnResolvedFunction OnResolved
) {
110 JITSymbolResolver::LookupResult Result
;
111 for (auto &Symbol
: Symbols
) {
112 std::string SymName
= Symbol
.str();
113 if (auto Sym
= findSymbolInLogicalDylib(SymName
)) {
114 if (auto AddrOrErr
= Sym
.getAddress())
115 Result
[Symbol
] = JITEvaluatedSymbol(*AddrOrErr
, Sym
.getFlags());
117 OnResolved(AddrOrErr
.takeError());
120 } else if (auto Err
= Sym
.takeError()) {
121 OnResolved(std::move(Err
));
124 // findSymbolInLogicalDylib failed. Lets try findSymbol.
125 if (auto Sym
= findSymbol(SymName
)) {
126 if (auto AddrOrErr
= Sym
.getAddress())
127 Result
[Symbol
] = JITEvaluatedSymbol(*AddrOrErr
, Sym
.getFlags());
129 OnResolved(AddrOrErr
.takeError());
132 } else if (auto Err
= Sym
.takeError()) {
133 OnResolved(std::move(Err
));
136 OnResolved(make_error
<StringError
>("Symbol not found: " + Symbol
,
137 inconvertibleErrorCode()));
143 OnResolved(std::move(Result
));
146 /// Performs flags lookup by calling findSymbolInLogicalDylib and
147 /// returning the flags value for that symbol.
148 Expected
<JITSymbolResolver::LookupSet
>
149 LegacyJITSymbolResolver::getResponsibilitySet(const LookupSet
&Symbols
) {
150 JITSymbolResolver::LookupSet Result
;
152 for (auto &Symbol
: Symbols
) {
153 std::string SymName
= Symbol
.str();
154 if (auto Sym
= findSymbolInLogicalDylib(SymName
)) {
155 // If there's an existing def but it is not strong, then the caller is
156 // responsible for it.
157 if (!Sym
.getFlags().isStrong())
158 Result
.insert(Symbol
);
159 } else if (auto Err
= Sym
.takeError())
160 return std::move(Err
);
162 // If there is no existing definition then the caller is responsible for
164 Result
.insert(Symbol
);
168 return std::move(Result
);