Under unusual circumstances (jitting a function that causes the creation of

another stub, but then never calling the jitted function) can cause the JIT to
leave a stub in place. Judging by the comments this is a known deficiency, so
we're just not going to use AssertingVH for the StubToFunctionTy map.

Also shorten some lines longer than 80 columns.

This fixes the "make check" failure with ocaml on x86-64 linux.


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@70185 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Nick Lewycky 2009-04-27 05:09:44 +00:00
parent 23c70f44e4
commit e2bcf13be5

View File

@ -55,32 +55,35 @@ static JIT *TheJIT = 0;
// //
namespace { namespace {
class JITResolverState { class JITResolverState {
public:
typedef std::map<AssertingVH<Function>, void*> FunctionToStubMapTy;
typedef std::map<void*, Function*> StubToFunctionMapTy;
typedef std::map<AssertingVH<GlobalValue>, void*> GlobalToIndirectSymMapTy;
private: private:
/// FunctionToStubMap - Keep track of the stub created for a particular /// FunctionToStubMap - Keep track of the stub created for a particular
/// function so that we can reuse them if necessary. /// function so that we can reuse them if necessary.
std::map<AssertingVH<Function>, void*> FunctionToStubMap; FunctionToStubMapTy FunctionToStubMap;
/// StubToFunctionMap - Keep track of the function that each stub /// StubToFunctionMap - Keep track of the function that each stub
/// corresponds to. /// corresponds to.
std::map<void*, AssertingVH<Function> > StubToFunctionMap; StubToFunctionMapTy StubToFunctionMap;
/// GlobalToIndirectSymMap - Keep track of the indirect symbol created for a /// GlobalToIndirectSymMap - Keep track of the indirect symbol created for a
/// particular GlobalVariable so that we can reuse them if necessary. /// particular GlobalVariable so that we can reuse them if necessary.
std::map<GlobalValue*, void*> GlobalToIndirectSymMap; GlobalToIndirectSymMapTy GlobalToIndirectSymMap;
public: public:
std::map<AssertingVH<Function>, void*>& getFunctionToStubMap(const MutexGuard& locked) { FunctionToStubMapTy& getFunctionToStubMap(const MutexGuard& locked) {
assert(locked.holds(TheJIT->lock)); assert(locked.holds(TheJIT->lock));
return FunctionToStubMap; return FunctionToStubMap;
} }
std::map<void*, AssertingVH<Function> >& getStubToFunctionMap(const MutexGuard& locked) { StubToFunctionMapTy& getStubToFunctionMap(const MutexGuard& locked) {
assert(locked.holds(TheJIT->lock)); assert(locked.holds(TheJIT->lock));
return StubToFunctionMap; return StubToFunctionMap;
} }
std::map<GlobalValue*, void*>& GlobalToIndirectSymMapTy& getGlobalToIndirectSymMap(const MutexGuard& locked) {
getGlobalToIndirectSymMap(const MutexGuard& locked) {
assert(locked.holds(TheJIT->lock)); assert(locked.holds(TheJIT->lock));
return GlobalToIndirectSymMap; return GlobalToIndirectSymMap;
} }
@ -89,6 +92,10 @@ namespace {
/// JITResolver - Keep track of, and resolve, call sites for functions that /// JITResolver - Keep track of, and resolve, call sites for functions that
/// have not yet been compiled. /// have not yet been compiled.
class JITResolver { class JITResolver {
typedef JITResolverState::FunctionToStubMapTy FunctionToStubMapTy;
typedef JITResolverState::StubToFunctionMapTy StubToFunctionMapTy;
typedef JITResolverState::GlobalToIndirectSymMapTy GlobalToIndirectSymMapTy;
/// LazyResolverFn - The target lazy resolver function that we actually /// LazyResolverFn - The target lazy resolver function that we actually
/// rewrite instructions to use. /// rewrite instructions to use.
TargetJITInfo::LazyResolverFn LazyResolverFn; TargetJITInfo::LazyResolverFn LazyResolverFn;
@ -276,18 +283,17 @@ void JITResolver::getRelocatableGVs(SmallVectorImpl<GlobalValue*> &GVs,
SmallVectorImpl<void*> &Ptrs) { SmallVectorImpl<void*> &Ptrs) {
MutexGuard locked(TheJIT->lock); MutexGuard locked(TheJIT->lock);
std::map<AssertingVH<Function>,void*> &FM =state.getFunctionToStubMap(locked); FunctionToStubMapTy &FM = state.getFunctionToStubMap(locked);
std::map<GlobalValue*,void*> &GM = state.getGlobalToIndirectSymMap(locked); GlobalToIndirectSymMapTy &GM = state.getGlobalToIndirectSymMap(locked);
for (std::map<AssertingVH<Function>,void*>::iterator i = FM.begin(), for (FunctionToStubMapTy::iterator i = FM.begin(), e = FM.end(); i != e; ++i){
e = FM.end(); i != e; ++i) {
Function *F = i->first; Function *F = i->first;
if (F->isDeclaration() && F->hasExternalLinkage()) { if (F->isDeclaration() && F->hasExternalLinkage()) {
GVs.push_back(i->first); GVs.push_back(i->first);
Ptrs.push_back(i->second); Ptrs.push_back(i->second);
} }
} }
for (std::map<GlobalValue*,void*>::iterator i = GM.begin(), e = GM.end(); for (GlobalToIndirectSymMapTy::iterator i = GM.begin(), e = GM.end();
i != e; ++i) { i != e; ++i) {
GVs.push_back(i->first); GVs.push_back(i->first);
Ptrs.push_back(i->second); Ptrs.push_back(i->second);
@ -297,9 +303,9 @@ void JITResolver::getRelocatableGVs(SmallVectorImpl<GlobalValue*> &GVs,
GlobalValue *JITResolver::invalidateStub(void *Stub) { GlobalValue *JITResolver::invalidateStub(void *Stub) {
MutexGuard locked(TheJIT->lock); MutexGuard locked(TheJIT->lock);
std::map<AssertingVH<Function>,void*> &FM =state.getFunctionToStubMap(locked); FunctionToStubMapTy &FM = state.getFunctionToStubMap(locked);
std::map<void*,AssertingVH<Function> > &SM=state.getStubToFunctionMap(locked); StubToFunctionMapTy &SM = state.getStubToFunctionMap(locked);
std::map<GlobalValue*,void*> &GM = state.getGlobalToIndirectSymMap(locked); GlobalToIndirectSymMapTy &GM = state.getGlobalToIndirectSymMap(locked);
// Look up the cheap way first, to see if it's a function stub we are // Look up the cheap way first, to see if it's a function stub we are
// invalidating. If so, remove it from both the forward and reverse maps. // invalidating. If so, remove it from both the forward and reverse maps.
@ -311,7 +317,7 @@ GlobalValue *JITResolver::invalidateStub(void *Stub) {
} }
// Otherwise, it might be an indirect symbol stub. Find it and remove it. // Otherwise, it might be an indirect symbol stub. Find it and remove it.
for (std::map<GlobalValue*,void*>::iterator i = GM.begin(), e = GM.end(); for (GlobalToIndirectSymMapTy::iterator i = GM.begin(), e = GM.end();
i != e; ++i) { i != e; ++i) {
if (i->second != Stub) if (i->second != Stub)
continue; continue;
@ -349,7 +355,7 @@ void *JITResolver::JITCompilerFn(void *Stub) {
// The address given to us for the stub may not be exactly right, it might be // The address given to us for the stub may not be exactly right, it might be
// a little bit after the stub. As such, use upper_bound to find it. // a little bit after the stub. As such, use upper_bound to find it.
std::map<void*, AssertingVH<Function> >::iterator I = StubToFunctionMapTy::iterator I =
JR.state.getStubToFunctionMap(locked).upper_bound(Stub); JR.state.getStubToFunctionMap(locked).upper_bound(Stub);
assert(I != JR.state.getStubToFunctionMap(locked).begin() && assert(I != JR.state.getStubToFunctionMap(locked).begin() &&
"This is not a known stub!"); "This is not a known stub!");