186c8a785SMehdi Amini //===- ExecutionEngine.cpp - C API for MLIR JIT ---------------------------===// 286c8a785SMehdi Amini // 386c8a785SMehdi Amini // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 486c8a785SMehdi Amini // See https://llvm.org/LICENSE.txt for license information. 586c8a785SMehdi Amini // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 686c8a785SMehdi Amini // 786c8a785SMehdi Amini //===----------------------------------------------------------------------===// 886c8a785SMehdi Amini 986c8a785SMehdi Amini #include "mlir-c/ExecutionEngine.h" 1086c8a785SMehdi Amini #include "mlir/CAPI/ExecutionEngine.h" 1186c8a785SMehdi Amini #include "mlir/CAPI/IR.h" 1286c8a785SMehdi Amini #include "mlir/CAPI/Support.h" 13185ce8cdSUday Bondhugula #include "mlir/ExecutionEngine/OptUtils.h" 1419db802eSAlex Zinenko #include "mlir/Target/LLVMIR/Dialect/LLVMIR/LLVMToLLVMIRTranslation.h" 157a4d6307SMehdi Amini #include "llvm/ExecutionEngine/Orc/Mangling.h" 1686c8a785SMehdi Amini #include "llvm/Support/TargetSelect.h" 1786c8a785SMehdi Amini 1886c8a785SMehdi Amini using namespace mlir; 1986c8a785SMehdi Amini 20*c8b8e8e0SUday Bondhugula extern "C" MlirExecutionEngine 21*c8b8e8e0SUday Bondhugula mlirExecutionEngineCreate(MlirModule op, int optLevel, int numPaths, 22*c8b8e8e0SUday Bondhugula const MlirStringRef *sharedLibPaths) { 23185ce8cdSUday Bondhugula static bool initOnce = [] { 2486c8a785SMehdi Amini llvm::InitializeNativeTarget(); 2586c8a785SMehdi Amini llvm::InitializeNativeTargetAsmPrinter(); 2686c8a785SMehdi Amini return true; 2786c8a785SMehdi Amini }(); 28185ce8cdSUday Bondhugula (void)initOnce; 2986c8a785SMehdi Amini 3013cb4317SMehdi Amini mlir::registerLLVMDialectTranslation(*unwrap(op)->getContext()); 31185ce8cdSUday Bondhugula 32185ce8cdSUday Bondhugula auto tmBuilderOrError = llvm::orc::JITTargetMachineBuilder::detectHost(); 33185ce8cdSUday Bondhugula if (!tmBuilderOrError) { 34185ce8cdSUday Bondhugula llvm::errs() << "Failed to create a JITTargetMachineBuilder for the host\n"; 35185ce8cdSUday Bondhugula return MlirExecutionEngine{nullptr}; 36185ce8cdSUday Bondhugula } 37185ce8cdSUday Bondhugula auto tmOrError = tmBuilderOrError->createTargetMachine(); 38185ce8cdSUday Bondhugula if (!tmOrError) { 39185ce8cdSUday Bondhugula llvm::errs() << "Failed to create a TargetMachine for the host\n"; 40185ce8cdSUday Bondhugula return MlirExecutionEngine{nullptr}; 41185ce8cdSUday Bondhugula } 42185ce8cdSUday Bondhugula 43*c8b8e8e0SUday Bondhugula SmallVector<StringRef> libPaths; 44*c8b8e8e0SUday Bondhugula for (unsigned i = 0; i < static_cast<unsigned>(numPaths); ++i) 45*c8b8e8e0SUday Bondhugula libPaths.push_back(sharedLibPaths[i].data); 46*c8b8e8e0SUday Bondhugula 47185ce8cdSUday Bondhugula // Create a transformer to run all LLVM optimization passes at the 48185ce8cdSUday Bondhugula // specified optimization level. 49185ce8cdSUday Bondhugula auto llvmOptLevel = static_cast<llvm::CodeGenOpt::Level>(optLevel); 50185ce8cdSUday Bondhugula auto transformer = mlir::makeLLVMPassesTransformer( 51185ce8cdSUday Bondhugula /*passes=*/{}, llvmOptLevel, /*targetMachine=*/tmOrError->get()); 52*c8b8e8e0SUday Bondhugula auto jitOrError = 53*c8b8e8e0SUday Bondhugula ExecutionEngine::create(unwrap(op), /*llvmModuleBuilder=*/{}, transformer, 54*c8b8e8e0SUday Bondhugula llvmOptLevel, libPaths); 5586c8a785SMehdi Amini if (!jitOrError) { 5686c8a785SMehdi Amini consumeError(jitOrError.takeError()); 5786c8a785SMehdi Amini return MlirExecutionEngine{nullptr}; 5886c8a785SMehdi Amini } 5986c8a785SMehdi Amini return wrap(jitOrError->release()); 6086c8a785SMehdi Amini } 6186c8a785SMehdi Amini 6286c8a785SMehdi Amini extern "C" void mlirExecutionEngineDestroy(MlirExecutionEngine jit) { 6386c8a785SMehdi Amini delete (unwrap(jit)); 6486c8a785SMehdi Amini } 6586c8a785SMehdi Amini 6686c8a785SMehdi Amini extern "C" MlirLogicalResult 6786c8a785SMehdi Amini mlirExecutionEngineInvokePacked(MlirExecutionEngine jit, MlirStringRef name, 6886c8a785SMehdi Amini void **arguments) { 6986c8a785SMehdi Amini const std::string ifaceName = ("_mlir_ciface_" + unwrap(name)).str(); 7086c8a785SMehdi Amini llvm::Error error = unwrap(jit)->invokePacked( 7186c8a785SMehdi Amini ifaceName, MutableArrayRef<void *>{arguments, (size_t)0}); 7286c8a785SMehdi Amini if (error) 7386c8a785SMehdi Amini return wrap(failure()); 7486c8a785SMehdi Amini return wrap(success()); 7586c8a785SMehdi Amini } 7613cb4317SMehdi Amini 7713cb4317SMehdi Amini extern "C" void *mlirExecutionEngineLookup(MlirExecutionEngine jit, 7813cb4317SMehdi Amini MlirStringRef name) { 7913cb4317SMehdi Amini auto expectedFPtr = unwrap(jit)->lookup(unwrap(name)); 8013cb4317SMehdi Amini if (!expectedFPtr) 8113cb4317SMehdi Amini return nullptr; 8213cb4317SMehdi Amini return reinterpret_cast<void *>(*expectedFPtr); 8313cb4317SMehdi Amini } 847a4d6307SMehdi Amini 857a4d6307SMehdi Amini extern "C" void mlirExecutionEngineRegisterSymbol(MlirExecutionEngine jit, 867a4d6307SMehdi Amini MlirStringRef name, 877a4d6307SMehdi Amini void *sym) { 887a4d6307SMehdi Amini unwrap(jit)->registerSymbols([&](llvm::orc::MangleAndInterner interner) { 897a4d6307SMehdi Amini llvm::orc::SymbolMap symbolMap; 907a4d6307SMehdi Amini symbolMap[interner(unwrap(name))] = 917a4d6307SMehdi Amini llvm::JITEvaluatedSymbol::fromPointer(sym); 927a4d6307SMehdi Amini return symbolMap; 937a4d6307SMehdi Amini }); 947a4d6307SMehdi Amini } 951dc533ceSNicolas Vasilache 961dc533ceSNicolas Vasilache extern "C" void mlirExecutionEngineDumpToObjectFile(MlirExecutionEngine jit, 971dc533ceSNicolas Vasilache MlirStringRef name) { 981dc533ceSNicolas Vasilache unwrap(jit)->dumpToObjectFile(unwrap(name)); 991dc533ceSNicolas Vasilache } 100