<div><span><span><div>Hi all,</div>
<div><br></div>
<div>I need to run c++ prgrams using lli.
However, I find lli cannot handle the alias instruction correctly. Following is
an example:</div>
<div><br></div>
<div>------------ example.cc
-------------</div>
<div><br></div>
<div>#include
<iostream></div>
<div>using namespace std;</div>
<div><br></div>
<div>class
cls {</div>
<div>public:</div>
<div>    cls();</div>
<div>   
~cls();</div>
<div>};</div>
<div><br></div>
<div>cls::cls() {</div>
<div> 
  cout << "constructor" <<
endl;</div>
<div>};</div>
<div><br></div>
<div>cls::~cls() {</div>
<div> 
  cout << "destructor" <<
endl;</div>
<div>}</div>
<div><br></div>
<div>int main(int argc, char
*argv[])</div>
<div>{</div>
<div>    cls *A = new
cls();</div>
<div>    delete A;</div>
<div>    return
0;</div>
<div>}</div>
<div><br></div>
<div>----------- end of file
-----------</div>
<div><br></div>
<div>I compile and run the program
by:</div>
<div><br></div>
<div>$ clang++ -emit-llvm -S example.cc -o
example.S</div>
<div>$ llvm-as example.S -o example.bc</div>
<div>$ lli
example.bc</div>
<div><br></div>
<div>and get the "Recursive compilation" error
output:</div>
<div><br></div>
<div>lli:
/home/dongpeng/Program/llvm/lib/ExecutionEngine/JIT/JIT.cpp:467: void
llvm::JIT::runJITOnFunctionUnlocked(llvm::Function*, const
llvm::MutexGuard&): Assertion `!isAlreadyCodeGenerating && "Error:
Recursive compilation detected!"' failed.</div>
<div>0  lli    
        0x0000000001323d37
llvm::sys::PrintStackTrace(_IO_FILE*) + 38</div>
<div>1  lli    
        0x0000000001323fbe</div>
<div>2  lli  
          0x0000000001323a02</div>
<div>3
 libpthread.so.0 0x00007fd4c168b210</div>
<div>4  libc.so.6  
    0x00007fd4c06de1d5 gsignal + 53</div>
<div>5  libc.so.6
      0x00007fd4c06e1388 abort + 328</div>
<div>6  libc.so.6
      0x00007fd4c06d7252</div>
<div>7  libc.so.6  
    0x00007fd4c06d7302</div>
<div>8  lli      
      0x0000000000cf1620
llvm::JIT::runJITOnFunctionUnlocked(llvm::Function*, llvm::MutexGuard
const&) + 66</div>
<div>9  lli          
  0x0000000000cf1a3a llvm::JIT::getPointerToFunction(llvm::Function*) +
694</div>
<div>10 lli            
0x00000000010fb64c llvm::ExecutionEngine::getPointerToGlobal(llvm::GlobalValue
const*) + 70</div>
<div>11 lli            
0x0000000000cfa2f0</div>
<div>12 lli            
0x0000000000cfaca3</div>
<div>13 lli            
0x00000000008c9368</div>
<div>14 lli            
0x0000000000d6a9e1
llvm::MachineFunctionPass::runOnFunction(llvm::Function&) + 95</div>
<div>15
lli             0x000000000129a53f
llvm::FPPassManager::runOnFunction(llvm::Function&) + 385</div>
<div>16 lli
            0x000000000129a286
llvm::legacy::FunctionPassManagerImpl::run(llvm::Function&) +
84</div>
<div>17 lli            
0x0000000001299e5e llvm::legacy::FunctionPassManager::run(llvm::Function&)
+ 180</div>
<div>18 lli            
0x0000000000cf175c llvm::JIT::jitTheFunction(llvm::Function*, llvm::MutexGuard
const&) + 72</div>
<div>19 lli            
0x0000000000cf1637 llvm::JIT::runJITOnFunctionUnlocked(llvm::Function*,
llvm::MutexGuard const&) + 89</div>
<div>20 lli        
    0x0000000000cf1a3a
llvm::JIT::getPointerToFunction(llvm::Function*) + 694</div>
<div>21 lli  
          0x0000000000817815 main + 4732</div>
<div>22
libc.so.6       0x00007fd4c06ca995 __libc_start_main +
245</div>
<div>23 lli            
0x000000000080d479</div>
<div>Stack dump:</div>
<div>0.    Program
arguments: lli example.bc </div>
<div>1.    Running pass 'X86
Machine Code Emitter' on function
'@main'</div>
<div>Aborted</div>
<div><br></div>
<div><br></div>
<div>I look into
the problem and find that it is caused by the alias
instructions:</div>
<div><br></div>
<div>@_ZN3clsC1Ev = alias void (%class.cls*)*
@_ZN3clsC2Ev</div>
<div>@_ZN3clsD1Ev = alias void (%class.cls*)*
@_ZN3clsD2Ev</div>
<div><br></div>
<div>The first alias is for the constructor
and the other one is for the destructor. When finishing JIT the main function,
JITEmitter needs to resolve all the names in main. When it is trying to resolve
the alias name @_ZN3clsC1Ev, the following code in JITEmitter.cpp will go to
resolve and JIT the original name of the alias name. In my case, they are
@_ZN3clsC2Ev and @_ZN3clsD2Ev. This procedure will cause a recursive
compilation error. </div>
<div><br></div>
<div>----------------
lib/ExecutionEngine/JIT/JITEmitter.cpp
------------------------</div>
<div><br></div>
<div>void
*JITEmitter::getPointerToGlobal(GlobalValue *V, void
*Reference,</div>
<div>               
                   
 bool MayNeedFarStub) {</div>
<div>  if (GlobalVariable *GV =
dyn_cast<GlobalVariable>(V))</div>
<div>    return
TheJIT->getOrEmitGlobalVariable(GV);</div>
<div><br></div>
<div>  if
(GlobalAlias *GA = dyn_cast<GlobalAlias>(V))</div>
<div>   
return
TheJIT->getPointerToGlobal(GA->resolveAliasedGlobal(false));</div>
<div><br></div>
<div><br></div>
<div>One possible solution is to let JITEmitter emit a stub for the name like
following:</div>
<div><br></div>
<div>@@ -687,8 +687,19 @@</div>
<div>   if (GlobalVariable *GV =
dyn_cast<GlobalVariable>(V))</div>
<div>     return TheJIT->getOrEmitGlobalVariable(GV);</div>
<div> </div>
<div>-  if (GlobalAlias *GA = dyn_cast<GlobalAlias>(V))</div>
<div>-    return
TheJIT->getPointerToGlobal(GA->resolveAliasedGlobal(false));</div>
<div>+  if (GlobalAlias *GA = dyn_cast<GlobalAlias>(V)) {</div>
<div>+      Function *F =
const_cast<Function*>(dyn_cast<Function>(GA->resolveAliasedGlobal(false)));</div>
<div>+      if (F == NULL) {</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">      </span>  return TheJIT->getPointerToGlobal(GA->resolveAliasedGlobal(false));</div>
<div>+      } else {</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">      </span>    void *FnStub = Resolver.getLazyFunctionStubIfAvailable(F);</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">      </span>    if (FnStub)</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">              </span>return FnStub;</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">      </span>    else</div>
<div>+  <span class="Apple-tab-span" style="white-space:pre">              </span>return Resolver.getLazyFunctionStub(F);</div>
<div>+      }</div>
<div>+  }</div>
<div>+</div>
<div> </div>
<div>   // If we have already compiled the function, return a pointer to its body.</div>
<div>   Function *F = cast<Function>(V);</div>
<div><br></div>
<div>Attached is the patch. I am not sure it is the correct way to solve the problem. Any comments are appreciated. Thanks for your help!</div>
<div><br></div>
<div><br></div>
<div>Sincerely,</div>
<div>Dongpeng</div></span></span></div>