<html><head><meta http-equiv="Content-Type" content="text/html charset=us-ascii"></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space;" class="">Thanks so much! This seems to do the trick. I would have spun my wheels for a long time before discovering all of this, wow.<div class=""><br class=""></div><div class="">Do I even want to know what additional chickens need to be sacrificed to get this to work on Windows?</div><div class=""><br class=""></div><div class=""><span class="Apple-tab-span" style="white-space:pre"> </span>-- lg</div><div class=""><br class=""><div><blockquote type="cite" class=""><div class="">On May 18, 2016, at 1:52 PM, Lang Hames <<a href="mailto:lhames@gmail.com" class="">lhames@gmail.com</a>> wrote:</div><br class="Apple-interchange-newline"><div class=""><div dir="ltr" class="">Hi Larry,<div class=""><br class=""></div><div class="">You're basically there, but you're hitting a couple of subtle issues:</div><div class=""><br class=""></div><div class="">(1) On both platforms you'll want to call llvm::sys::DynamicLibrary::LoadLibraryPermanently(nullptr) at program startup. This makes exported symbols in the main program searchable by RTDyldMemoryManager::getSymbolAddressInProcess (important for making 'sqr' findable on any platform).</div><div class=""><br class=""></div><div class="">(2) On Linux (if I understand correctly) symbols aren't exported from the main process by default, so even if you've called DynamicLibrary::LoadLibraryPermanently, sqr won't show up. To fix this you can add -Wl,export-dynamic to your link line. This will flip the default and export symbols from the main binary, allowing getSymbolAddressInProcess to find sqr. On MacOS this isn't necessary.</div><div class=""><br class=""></div><div class="">(3) On MacOS, C names are mangled by prepending an '_'. To maintain consistency with statically linked programs, the JIT uses these mangled names so you need to search for '_myfunc', rather than 'myfunc'. I usually add a 'mangle' function that uses the DataLayout, then search for 'mangle(<name>)', which does the right thing on all platforms.</div><div class=""><br class=""></div><div class="">I've attached a new version of your code with these changes included. Hope this helps!</div><div class=""><br class=""></div><div class="">- Lang.</div><div class=""><br class=""></div></div><div class="gmail_extra"><br class=""><div class="gmail_quote">On Tue, May 17, 2016 at 12:13 PM, Larry Gritz via llvm-dev <span dir="ltr" class=""><<a href="mailto:llvm-dev@lists.llvm.org" target="_blank" class="">llvm-dev@lists.llvm.org</a>></span> wrote:<br class=""><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">When using ORC JIT, I'm having trouble with external function resolution (that is, of a function defined in the app, with C linkage).<br class="">
<br class="">
I add a declaration for the function to my IR, and when I use MCJIT, it finds it and all is well, But when I use ORC JIT (I *think* correctly, at least it closely matches what I see in the tutorial), I get an LLVM error, "Program used external function 'sqr' which could not be resolved."<br class="">
<br class="">
(Excuse my coming to the ORC party late, I've been stuck on an older LLVM for my DSL, and am only now jumping forward several versions at once, with some growing pains.)<br class="">
<br class="">
I've boiled it down to the minimal example below. When I build with orc=0, thus using MCJIT, it works fine and I get the expected output. When I build with orc=1, it fails as I described.<br class="">
<br class="">
I'm having this trouble with LLVM 3.7 and 3.8, on both Linux and OSX.<br class="">
<br class="">
I figure I'm probably just getting some part of the magic incantation wrong. Can anybody help a poor guy out and spot the error of my ways?<br class="">
<br class="">
-----<br class="">
<br class="">
#include <llvm/ADT/STLExtras.h><br class="">
#include <llvm/Bitcode/ReaderWriter.h><br class="">
#include <llvm/ExecutionEngine/ExecutionEngine.h><br class="">
#include <llvm/ExecutionEngine/Orc/CompileUtils.h><br class="">
#include <llvm/ExecutionEngine/RuntimeDyld.h><br class="">
#include <llvm/ExecutionEngine/Orc/IRCompileLayer.h><br class="">
#include <llvm/ExecutionEngine/Orc/LambdaResolver.h><br class="">
#include <llvm/ExecutionEngine/Orc/ObjectLinkingLayer.h><br class="">
#include <llvm/IR/DataLayout.h><br class="">
#include <llvm/IR/IRBuilder.h><br class="">
#include <llvm/IR/LegacyPassManager.h><br class="">
#include <llvm/IR/LLVMContext.h><br class="">
#include <llvm/IR/Mangler.h><br class="">
#include <llvm/IR/Module.h><br class="">
#include <llvm/IR/Verifier.h><br class="">
#include <llvm/Support/TargetSelect.h><br class="">
#include <llvm/Target/TargetMachine.h><br class="">
#include <llvm/Transforms/IPO.h><br class="">
#include <llvm/Transforms/IPO/PassManagerBuilder.h><br class="">
#include <llvm/Transforms/Scalar.h><br class="">
#include <llvm/Transforms/Utils/UnifyFunctionExitNodes.h><br class="">
<br class="">
template <typename T><br class="">
inline std::vector<T> singletonSet (T t)<br class="">
{<br class="">
std::vector<T> Vec;<br class="">
Vec.push_back(std::move(t));<br class="">
return Vec;<br class="">
}<br class="">
<br class="">
<br class="">
///<br class="">
/// THIS is the function I want my IR to call<br class="">
///<br class="">
extern "C" {<br class="">
float sqr (float x) { return x*x; }<br class="">
}<br class="">
<br class="">
<br class="">
void<br class="">
simple ()<br class="">
{<br class="">
llvm::InitializeAllTargets();<br class="">
llvm::InitializeAllTargetMCs();<br class="">
llvm::InitializeAllAsmPrinters();<br class="">
llvm::InitializeAllAsmParsers();<br class="">
llvm::LLVMContext Context;<br class="">
std::unique_ptr<llvm::TargetMachine> TM (llvm::EngineBuilder().selectTarget());<br class="">
std::unique_ptr<llvm::DataLayout> DL;<br class="">
DL.reset (new llvm::DataLayout (TM->createDataLayout()));<br class="">
std::unique_ptr<llvm::ExecutionEngine> EE;<br class="">
typedef llvm::orc::ObjectLinkingLayer<> ObjLayerT;<br class="">
typedef llvm::orc::IRCompileLayer<ObjLayerT> CompileLayerT;<br class="">
typedef CompileLayerT::ModuleSetHandleT ModuleHandleT;<br class="">
ObjLayerT Objlayer;<br class="">
CompileLayerT Compilelayer (Objlayer, llvm::orc::SimpleCompiler(*TM));<br class="">
std::unique_ptr<llvm::Module> M (new llvm::Module("module", Context));<br class="">
M->setDataLayout (*DL);<br class="">
<br class="">
// Declare stub for external function sqr<br class="">
auto type_float = llvm::Type::getFloatTy (Context);<br class="">
llvm::Type* one_float[] = { type_float };<br class="">
llvm::FunctionType *functype_ff = llvm::FunctionType::get (type_float, one_float, false);<br class="">
llvm::Function::Create (functype_ff, llvm::Function::ExternalLinkage,<br class="">
"sqr", M.get());<br class="">
<br class="">
// Create myfunc and generate its IR, which just calls sqr on its argument<br class="">
llvm::Function *myfunc = llvm::Function::Create (functype_ff,<br class="">
llvm::Function::ExternalLinkage,<br class="">
"myfunc", M.get());<br class="">
llvm::IRBuilder<> builder (Context);<br class="">
auto block = llvm::BasicBlock::Create (Context, "", myfunc);<br class="">
builder.SetInsertPoint (block);<br class="">
llvm::Value *a = llvm::cast<llvm::Value>(myfunc->arg_begin());<br class="">
llvm::Value *asq = builder.CreateCall (M->getFunction ("sqr"), a);<br class="">
builder.CreateRet (asq);<br class="">
<br class="">
// Set up compilation<br class="">
if (orc) {<br class="">
auto Resolver = llvm::orc::createLambdaResolver(<br class="">
// External lookup functor<br class="">
[&](const std::string &name) {<br class="">
if (auto Sym = Compilelayer.findSymbol(name, true))<br class="">
return llvm::RuntimeDyld::SymbolInfo(Sym.getAddress(), Sym.getFlags());<br class="">
// If not found as a symbol, look up in current process.<br class="">
// Why doesn't this work?<br class="">
if (auto Addr = llvm::RTDyldMemoryManager::getSymbolAddressInProcess(name))<br class="">
return llvm::RuntimeDyld::SymbolInfo(Addr, llvm::JITSymbolFlags::Exported);<br class="">
return llvm::RuntimeDyld::SymbolInfo(nullptr);<br class="">
},<br class="">
// Dylib lookup functor<br class="">
[&](const std::string &name) { return nullptr; }<br class="">
);<br class="">
Compilelayer.addModuleSet (singletonSet(std::move(M)),<br class="">
llvm::make_unique<llvm::SectionMemoryManager>(),<br class="">
std::move(Resolver));<br class="">
} else {<br class="">
// MCJIT<br class="">
std::string engine_errors;<br class="">
llvm::EngineBuilder engine_builder (std::move(M));<br class="">
engine_builder.setEngineKind (llvm::EngineKind::JIT)<br class="">
.setOptLevel (llvm::CodeGenOpt::Default) // Aggressive?<br class="">
.setErrorStr (&engine_errors);<br class="">
EE.reset (engine_builder.create());<br class="">
EE->finalizeObject ();<br class="">
}<br class="">
<br class="">
// Ask for a callable function<br class="">
typedef float (*FuncFloatFloat)(float);<br class="">
FuncFloatFloat my_executable_function = NULL;<br class="">
if (orc) {<br class="">
auto ExprSymbol = Compilelayer.findSymbol ("myfunc", true);<br class="">
my_executable_function = (FuncFloatFloat) ExprSymbol.getAddress ();<br class="">
} else {<br class="">
my_executable_function = (FuncFloatFloat) EE->getFunctionAddress ("myfunc");<br class="">
}<br class="">
<br class="">
assert (my_executable_function);<br class="">
printf ("myfunc(42.0f) = %g\n", (*my_executable_function)(42.0f));<br class="">
}<br class="">
<br class="">
<br class="">
<br class="">
--<br class="">
Larry Gritz<br class="">
<a href="mailto:lg@larrygritz.com" class="">lg@larrygritz.com</a><br class="">
<br class="">
<br class="">
_______________________________________________<br class="">
LLVM Developers mailing list<br class="">
<a href="mailto:llvm-dev@lists.llvm.org" class="">llvm-dev@lists.llvm.org</a><br class="">
<a href="http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev" rel="noreferrer" target="_blank" class="">http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev</a><br class="">
</blockquote></div><br class=""></div>
<span id="cid:E7A35E12-29F4-4375-ADDB-E84D6F7BF234@spimageworks.com"><larrys_jit.cpp></span></div></blockquote></div><br class=""><div apple-content-edited="true" class="">
--<br class="">Larry Gritz<br class=""><a href="mailto:lg@larrygritz.com" class="">lg@larrygritz.com</a><br class=""><br class="">
</div>
<br class=""></div></body></html>