From: Chris Lattner Date: Wed, 4 Dec 2002 04:47:34 +0000 (+0000) Subject: Implement lazy resolution of function calls X-Git-Url: http://plrg.eecs.uci.edu/git/?a=commitdiff_plain;h=d6c10ea817f8ce2f8fdc24beadaab264767f1dcb;p=oota-llvm.git Implement lazy resolution of function calls git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@4899 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/tools/jello/Emitter.cpp b/tools/jello/Emitter.cpp index 89695ad2b4f..0e54322ef9b 100644 --- a/tools/jello/Emitter.cpp +++ b/tools/jello/Emitter.cpp @@ -68,7 +68,9 @@ void Emitter::emitByte(unsigned char B) { // resolved on demand. Keep track of these markers. // void Emitter::emitPCRelativeDisp(Value *V) { - unsigned ZeroAddr = -(unsigned)CurByte; // Calculate displacement to null + TheVM.addFunctionRef(CurByte, cast(V)); + + unsigned ZeroAddr = -(unsigned)CurByte-4; // Calculate displacement to null *(unsigned*)CurByte = ZeroAddr; // 4 byte offset CurByte += 4; } diff --git a/tools/jello/VM.cpp b/tools/jello/VM.cpp index 497c42ef90e..b5a8ca1c22d 100644 --- a/tools/jello/VM.cpp +++ b/tools/jello/VM.cpp @@ -43,6 +43,21 @@ int VM::run(Function *F) { return PF(); } +void *VM::resolveFunctionReference(void *RefAddr) { + Function *F = FunctionRefs[RefAddr]; + assert(F && "Reference address not known!"); + + void *Addr = getPointerToFunction(F); + assert(Addr && "Pointer to function unknown!"); + + FunctionRefs.erase(RefAddr); + return Addr; +} + +const std::string &VM::getFunctionReferencedName(void *RefAddr) { + return FunctionRefs[RefAddr]->getName(); +} + /// getPointerToFunction - This method is used to get the address of the /// specified function, compiling it if neccesary. diff --git a/tools/jello/VM.h b/tools/jello/VM.h index 5d0cd38941c..b25cd4625e4 100644 --- a/tools/jello/VM.h +++ b/tools/jello/VM.h @@ -10,6 +10,7 @@ #include "llvm/PassManager.h" #include #include +#include class TargetMachine; class Function; @@ -23,12 +24,21 @@ class VM { PassManager PM; // Passes to compile a function MachineCodeEmitter *MCE; // MCE object + // GlobalAddress - A mapping between LLVM values and their native code + // generated versions... std::map GlobalAddress; + + // FunctionRefs - A mapping between addresses that refer to unresolved + // functions and the LLVM function object itself. This is used by the fault + // handler to lazily patch up references... + // + std::map FunctionRefs; public: VM(const std::string &name, Module &m, TargetMachine &tm) : ExeName(name), M(m), TM(tm) { MCE = createEmitter(*this); // Initialize MCE setupPassManager(); + registerCallback(); } ~VM(); @@ -41,10 +51,19 @@ public: CurVal = Addr; } + void addFunctionRef(void *Ref, Function *F) { + FunctionRefs[Ref] = F; + } + + const std::string &getFunctionReferencedName(void *RefAddr); + + void *resolveFunctionReference(void *RefAddr); + private: static MachineCodeEmitter *createEmitter(VM &V); void setupPassManager(); void *getPointerToFunction(Function *F); + void registerCallback(); }; #endif