LLVM 20.0.0git
ExecutionEngineBindings.cpp
Go to the documentation of this file.
1//===-- ExecutionEngineBindings.cpp - C bindings for EEs ------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file defines the C bindings for the ExecutionEngine library.
10//
11//===----------------------------------------------------------------------===//
12
18#include "llvm/IR/Module.h"
22#include <cstring>
23#include <optional>
24
25using namespace llvm;
26
27#define DEBUG_TYPE "jit"
28
29// Wrapping the C bindings types.
31
32
34 return
35 reinterpret_cast<LLVMTargetMachineRef>(const_cast<TargetMachine*>(P));
36}
37
38/*===-- Operations on generic values --------------------------------------===*/
39
41 unsigned long long N,
42 LLVMBool IsSigned) {
43 GenericValue *GenVal = new GenericValue();
44 GenVal->IntVal = APInt(unwrap<IntegerType>(Ty)->getBitWidth(), N, IsSigned);
45 return wrap(GenVal);
46}
47
49 GenericValue *GenVal = new GenericValue();
50 GenVal->PointerVal = P;
51 return wrap(GenVal);
52}
53
55 GenericValue *GenVal = new GenericValue();
56 switch (unwrap(TyRef)->getTypeID()) {
57 case Type::FloatTyID:
58 GenVal->FloatVal = N;
59 break;
61 GenVal->DoubleVal = N;
62 break;
63 default:
64 llvm_unreachable("LLVMGenericValueToFloat supports only float and double.");
65 }
66 return wrap(GenVal);
67}
68
70 return unwrap(GenValRef)->IntVal.getBitWidth();
71}
72
73unsigned long long LLVMGenericValueToInt(LLVMGenericValueRef GenValRef,
74 LLVMBool IsSigned) {
75 GenericValue *GenVal = unwrap(GenValRef);
76 if (IsSigned)
77 return GenVal->IntVal.getSExtValue();
78 else
79 return GenVal->IntVal.getZExtValue();
80}
81
83 return unwrap(GenVal)->PointerVal;
84}
85
87 switch (unwrap(TyRef)->getTypeID()) {
88 case Type::FloatTyID:
89 return unwrap(GenVal)->FloatVal;
91 return unwrap(GenVal)->DoubleVal;
92 default:
93 llvm_unreachable("LLVMGenericValueToFloat supports only float and double.");
94 }
95}
96
98 delete unwrap(GenVal);
99}
100
101/*===-- Operations on execution engines -----------------------------------===*/
102
105 char **OutError) {
106 std::string Error;
107 EngineBuilder builder(std::unique_ptr<Module>(unwrap(M)));
110 if (ExecutionEngine *EE = builder.create()){
111 *OutEE = wrap(EE);
112 return 0;
113 }
114 *OutError = strdup(Error.c_str());
115 return 1;
116}
117
120 char **OutError) {
121 std::string Error;
122 EngineBuilder builder(std::unique_ptr<Module>(unwrap(M)));
125 if (ExecutionEngine *Interp = builder.create()) {
126 *OutInterp = wrap(Interp);
127 return 0;
128 }
129 *OutError = strdup(Error.c_str());
130 return 1;
131}
132
135 unsigned OptLevel,
136 char **OutError) {
137 std::string Error;
138 EngineBuilder builder(std::unique_ptr<Module>(unwrap(M)));
141 .setOptLevel((CodeGenOptLevel)OptLevel);
142 if (ExecutionEngine *JIT = builder.create()) {
143 *OutJIT = wrap(JIT);
144 return 0;
145 }
146 *OutError = strdup(Error.c_str());
147 return 1;
148}
149
151 size_t SizeOfPassedOptions) {
153 memset(&options, 0, sizeof(options)); // Most fields are zero by default.
154 options.CodeModel = LLVMCodeModelJITDefault;
155
156 memcpy(PassedOptions, &options,
157 std::min(sizeof(options), SizeOfPassedOptions));
158}
159
162 LLVMMCJITCompilerOptions *PassedOptions, size_t SizeOfPassedOptions,
163 char **OutError) {
165 // If the user passed a larger sized options struct, then they were compiled
166 // against a newer LLVM. Tell them that something is wrong.
167 if (SizeOfPassedOptions > sizeof(options)) {
168 *OutError = strdup(
169 "Refusing to use options struct that is larger than my own; assuming "
170 "LLVM library mismatch.");
171 return 1;
172 }
173
174 // Defend against the user having an old version of the API by ensuring that
175 // any fields they didn't see are cleared. We must defend against fields being
176 // set to the bitwise equivalent of zero, and assume that this means "do the
177 // default" as if that option hadn't been available.
178 LLVMInitializeMCJITCompilerOptions(&options, sizeof(options));
179 memcpy(&options, PassedOptions, SizeOfPassedOptions);
180
181 TargetOptions targetOptions;
182 targetOptions.EnableFastISel = options.EnableFastISel;
183 std::unique_ptr<Module> Mod(unwrap(M));
184
185 if (Mod)
186 // Set function attribute "frame-pointer" based on
187 // NoFramePointerElim.
188 for (auto &F : *Mod) {
189 auto Attrs = F.getAttributes();
190 StringRef Value = options.NoFramePointerElim ? "all" : "none";
191 Attrs = Attrs.addFnAttribute(F.getContext(), "frame-pointer", Value);
192 F.setAttributes(Attrs);
193 }
194
195 std::string Error;
196 EngineBuilder builder(std::move(Mod));
199 .setOptLevel((CodeGenOptLevel)options.OptLevel)
200 .setTargetOptions(targetOptions);
201 bool JIT;
202 if (std::optional<CodeModel::Model> CM = unwrap(options.CodeModel, JIT))
203 builder.setCodeModel(*CM);
204 if (options.MCJMM)
205 builder.setMCJITMemoryManager(
206 std::unique_ptr<RTDyldMemoryManager>(unwrap(options.MCJMM)));
207 if (ExecutionEngine *JIT = builder.create()) {
208 *OutJIT = wrap(JIT);
209 return 0;
210 }
211 *OutError = strdup(Error.c_str());
212 return 1;
213}
214
216 delete unwrap(EE);
217}
218
220 unwrap(EE)->finalizeObject();
221 unwrap(EE)->runStaticConstructorsDestructors(false);
222}
223
225 unwrap(EE)->finalizeObject();
226 unwrap(EE)->runStaticConstructorsDestructors(true);
227}
228
230 unsigned ArgC, const char * const *ArgV,
231 const char * const *EnvP) {
232 unwrap(EE)->finalizeObject();
233
234 std::vector<std::string> ArgVec(ArgV, ArgV + ArgC);
235 return unwrap(EE)->runFunctionAsMain(unwrap<Function>(F), ArgVec, EnvP);
236}
237
239 unsigned NumArgs,
240 LLVMGenericValueRef *Args) {
241 unwrap(EE)->finalizeObject();
242
243 std::vector<GenericValue> ArgVec;
244 ArgVec.reserve(NumArgs);
245 for (unsigned I = 0; I != NumArgs; ++I)
246 ArgVec.push_back(*unwrap(Args[I]));
247
248 GenericValue *Result = new GenericValue();
249 *Result = unwrap(EE)->runFunction(unwrap<Function>(F), ArgVec);
250 return wrap(Result);
251}
252
254}
255
257 unwrap(EE)->addModule(std::unique_ptr<Module>(unwrap(M)));
258}
259
261 LLVMModuleRef *OutMod, char **OutError) {
262 Module *Mod = unwrap(M);
263 unwrap(EE)->removeModule(Mod);
264 *OutMod = wrap(Mod);
265 return 0;
266}
267
269 LLVMValueRef *OutFn) {
270 if (Function *F = unwrap(EE)->FindFunctionNamed(Name)) {
271 *OutFn = wrap(F);
272 return 0;
273 }
274 return 1;
275}
276
278 LLVMValueRef Fn) {
279 return nullptr;
280}
281
283 return wrap(&unwrap(EE)->getDataLayout());
284}
285
288 return wrap(unwrap(EE)->getTargetMachine());
289}
290
292 void* Addr) {
293 unwrap(EE)->addGlobalMapping(unwrap<GlobalValue>(Global), Addr);
294}
295
297 unwrap(EE)->finalizeObject();
298
299 return unwrap(EE)->getPointerToGlobal(unwrap<GlobalValue>(Global));
300}
301
303 return unwrap(EE)->getGlobalValueAddress(Name);
304}
305
307 return unwrap(EE)->getFunctionAddress(Name);
308}
309
311 char **OutError) {
312 assert(OutError && "OutError must be non-null");
313 auto *ExecEngine = unwrap(EE);
314 if (ExecEngine->hasError()) {
315 *OutError = strdup(ExecEngine->getErrorMessage().c_str());
316 ExecEngine->clearErrorMessage();
317 return true;
318 }
319 return false;
320}
321
322/*===-- Operations on memory managers -------------------------------------===*/
323
324namespace {
325
326struct SimpleBindingMMFunctions {
331};
332
333class SimpleBindingMemoryManager : public RTDyldMemoryManager {
334public:
335 SimpleBindingMemoryManager(const SimpleBindingMMFunctions& Functions,
336 void *Opaque);
337 ~SimpleBindingMemoryManager() override;
338
339 uint8_t *allocateCodeSection(uintptr_t Size, unsigned Alignment,
340 unsigned SectionID,
341 StringRef SectionName) override;
342
343 uint8_t *allocateDataSection(uintptr_t Size, unsigned Alignment,
344 unsigned SectionID, StringRef SectionName,
345 bool isReadOnly) override;
346
347 bool finalizeMemory(std::string *ErrMsg) override;
348
349private:
350 SimpleBindingMMFunctions Functions;
351 void *Opaque;
352};
353
354SimpleBindingMemoryManager::SimpleBindingMemoryManager(
355 const SimpleBindingMMFunctions& Functions,
356 void *Opaque)
357 : Functions(Functions), Opaque(Opaque) {
358 assert(Functions.AllocateCodeSection &&
359 "No AllocateCodeSection function provided!");
360 assert(Functions.AllocateDataSection &&
361 "No AllocateDataSection function provided!");
362 assert(Functions.FinalizeMemory &&
363 "No FinalizeMemory function provided!");
364 assert(Functions.Destroy &&
365 "No Destroy function provided!");
366}
367
368SimpleBindingMemoryManager::~SimpleBindingMemoryManager() {
369 Functions.Destroy(Opaque);
370}
371
372uint8_t *SimpleBindingMemoryManager::allocateCodeSection(
373 uintptr_t Size, unsigned Alignment, unsigned SectionID,
375 return Functions.AllocateCodeSection(Opaque, Size, Alignment, SectionID,
376 SectionName.str().c_str());
377}
378
379uint8_t *SimpleBindingMemoryManager::allocateDataSection(
380 uintptr_t Size, unsigned Alignment, unsigned SectionID,
381 StringRef SectionName, bool isReadOnly) {
382 return Functions.AllocateDataSection(Opaque, Size, Alignment, SectionID,
383 SectionName.str().c_str(),
384 isReadOnly);
385}
386
387bool SimpleBindingMemoryManager::finalizeMemory(std::string *ErrMsg) {
388 char *errMsgCString = nullptr;
389 bool result = Functions.FinalizeMemory(Opaque, &errMsgCString);
390 assert((result || !errMsgCString) &&
391 "Did not expect an error message if FinalizeMemory succeeded");
392 if (errMsgCString) {
393 if (ErrMsg)
394 *ErrMsg = errMsgCString;
395 free(errMsgCString);
396 }
397 return result;
398}
399
400} // anonymous namespace
401
403 void *Opaque,
408
409 if (!AllocateCodeSection || !AllocateDataSection || !FinalizeMemory ||
410 !Destroy)
411 return nullptr;
412
413 SimpleBindingMMFunctions functions;
414 functions.AllocateCodeSection = AllocateCodeSection;
415 functions.AllocateDataSection = AllocateDataSection;
416 functions.FinalizeMemory = FinalizeMemory;
417 functions.Destroy = Destroy;
418 return wrap(new SimpleBindingMemoryManager(functions, Opaque));
419}
420
422 delete unwrap(MM);
423}
424
425/*===-- JIT Event Listener functions -------------------------------------===*/
426
427
428#if !LLVM_USE_INTEL_JITEVENTS
430{
431 return nullptr;
432}
433#endif
434
435#if !LLVM_USE_OPROFILE
437{
438 return nullptr;
439}
440#endif
441
442#if !LLVM_USE_PERF
444{
445 return nullptr;
446}
447#endif
aarch64 promote const
Lower uses of LDS variables from non kernel functions
#define DEFINE_SIMPLE_CONVERSION_FUNCTIONS(ty, ref)
uint64_t Addr
std::string Name
uint64_t Size
static LLVMTargetMachineRef wrap(const TargetMachine *P)
static char getTypeID(Type *Ty)
Module.h This file contains the declarations for the Module class.
#define F(x, y, z)
Definition: MD5.cpp:55
#define I(x, y, z)
Definition: MD5.cpp:58
#define P(N)
if(auto Err=PB.parsePassPipeline(MPM, Passes)) return wrap(std MPM run * Mod
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
static unsigned getBitWidth(Type *Ty, const DataLayout &DL)
Returns the bitwidth of the given scalar or pointer type.
Class for arbitrary precision integers.
Definition: APInt.h:78
uint64_t getZExtValue() const
Get zero extended value.
Definition: APInt.h:1520
int64_t getSExtValue() const
Get sign extended value.
Definition: APInt.h:1542
Builder class for ExecutionEngines.
EngineBuilder & setTargetOptions(const TargetOptions &Opts)
setTargetOptions - Set the target options that the ExecutionEngine target is using.
EngineBuilder & setMCJITMemoryManager(std::unique_ptr< RTDyldMemoryManager > mcjmm)
setMCJITMemoryManager - Sets the MCJIT memory manager to use.
EngineBuilder & setCodeModel(CodeModel::Model M)
setCodeModel - Set the CodeModel that the ExecutionEngine target data is using.
EngineBuilder & setOptLevel(CodeGenOptLevel l)
setOptLevel - Set the optimization level for the JIT.
EngineBuilder & setErrorStr(std::string *e)
setErrorStr - Set the error string to write to on error.
EngineBuilder & setEngineKind(EngineKind::Kind w)
setEngineKind - Controls whether the user wants the interpreter, the JIT, or whichever engine works.
ExecutionEngine * create()
Lightweight error class with error context and mandatory checking.
Definition: Error.h:160
Abstract interface for implementation execution of LLVM modules, designed to support both interpreter...
A Module instance is used to store all the information related to an LLVM module.
Definition: Module.h:65
virtual uint8_t * allocateDataSection(uintptr_t Size, unsigned Alignment, unsigned SectionID, StringRef SectionName, bool IsReadOnly)=0
Allocate a memory block of (at least) the given size suitable for data.
virtual uint8_t * allocateCodeSection(uintptr_t Size, unsigned Alignment, unsigned SectionID, StringRef SectionName)=0
Allocate a memory block of (at least) the given size suitable for executable code.
virtual bool finalizeMemory(std::string *ErrMsg=nullptr)=0
This method is called when object loading is complete and section page permissions can be applied.
StringRef - Represent a constant reference to a string, i.e.
Definition: StringRef.h:51
Primary interface to the complete machine description for the target machine.
Definition: TargetMachine.h:77
unsigned EnableFastISel
EnableFastISel - This flag enables fast-path instruction selection which trades away generated code q...
@ FloatTyID
32-bit floating point type
Definition: Type.h:58
@ DoubleTyID
64-bit floating point type
Definition: Type.h:59
LLVM Value Representation.
Definition: Value.h:74
LLVMGenericValueRef LLVMCreateGenericValueOfPointer(void *P)
double LLVMGenericValueToFloat(LLVMTypeRef TyRef, LLVMGenericValueRef GenVal)
LLVMJITEventListenerRef LLVMCreateOProfileJITEventListener(void)
LLVMBool LLVMExecutionEngineGetErrMsg(LLVMExecutionEngineRef EE, char **OutError)
Returns true on error, false on success.
LLVMMCJITMemoryManagerRef LLVMCreateSimpleMCJITMemoryManager(void *Opaque, LLVMMemoryManagerAllocateCodeSectionCallback AllocateCodeSection, LLVMMemoryManagerAllocateDataSectionCallback AllocateDataSection, LLVMMemoryManagerFinalizeMemoryCallback FinalizeMemory, LLVMMemoryManagerDestroyCallback Destroy)
Create a simple custom MCJIT memory manager.
void LLVMInitializeMCJITCompilerOptions(LLVMMCJITCompilerOptions *PassedOptions, size_t SizeOfPassedOptions)
void LLVMRunStaticDestructors(LLVMExecutionEngineRef EE)
LLVMBool LLVMCreateJITCompilerForModule(LLVMExecutionEngineRef *OutJIT, LLVMModuleRef M, unsigned OptLevel, char **OutError)
LLVMBool LLVMFindFunction(LLVMExecutionEngineRef EE, const char *Name, LLVMValueRef *OutFn)
struct LLVMOpaqueMCJITMemoryManager * LLVMMCJITMemoryManagerRef
LLVMGenericValueRef LLVMCreateGenericValueOfInt(LLVMTypeRef Ty, unsigned long long N, LLVMBool IsSigned)
void LLVMDisposeMCJITMemoryManager(LLVMMCJITMemoryManagerRef MM)
void(* LLVMMemoryManagerDestroyCallback)(void *Opaque)
LLVMGenericValueRef LLVMRunFunction(LLVMExecutionEngineRef EE, LLVMValueRef F, unsigned NumArgs, LLVMGenericValueRef *Args)
LLVMTargetDataRef LLVMGetExecutionEngineTargetData(LLVMExecutionEngineRef EE)
void LLVMDisposeExecutionEngine(LLVMExecutionEngineRef EE)
LLVMJITEventListenerRef LLVMCreatePerfJITEventListener(void)
void * LLVMRecompileAndRelinkFunction(LLVMExecutionEngineRef EE, LLVMValueRef Fn)
struct LLVMOpaqueExecutionEngine * LLVMExecutionEngineRef
uint64_t LLVMGetFunctionAddress(LLVMExecutionEngineRef EE, const char *Name)
LLVMBool LLVMCreateExecutionEngineForModule(LLVMExecutionEngineRef *OutEE, LLVMModuleRef M, char **OutError)
uint64_t LLVMGetGlobalValueAddress(LLVMExecutionEngineRef EE, const char *Name)
void LLVMAddModule(LLVMExecutionEngineRef EE, LLVMModuleRef M)
LLVMBool LLVMRemoveModule(LLVMExecutionEngineRef EE, LLVMModuleRef M, LLVMModuleRef *OutMod, char **OutError)
LLVMJITEventListenerRef LLVMCreateIntelJITEventListener(void)
struct LLVMOpaqueGenericValue * LLVMGenericValueRef
void * LLVMGenericValueToPointer(LLVMGenericValueRef GenVal)
LLVMBool(* LLVMMemoryManagerFinalizeMemoryCallback)(void *Opaque, char **ErrMsg)
unsigned LLVMGenericValueIntWidth(LLVMGenericValueRef GenValRef)
void LLVMRunStaticConstructors(LLVMExecutionEngineRef EE)
LLVMBool LLVMCreateMCJITCompilerForModule(LLVMExecutionEngineRef *OutJIT, LLVMModuleRef M, LLVMMCJITCompilerOptions *PassedOptions, size_t SizeOfPassedOptions, char **OutError)
Create an MCJIT execution engine for a module, with the given options.
LLVMGenericValueRef LLVMCreateGenericValueOfFloat(LLVMTypeRef TyRef, double N)
uint8_t *(* LLVMMemoryManagerAllocateDataSectionCallback)(void *Opaque, uintptr_t Size, unsigned Alignment, unsigned SectionID, const char *SectionName, LLVMBool IsReadOnly)
unsigned long long LLVMGenericValueToInt(LLVMGenericValueRef GenValRef, LLVMBool IsSigned)
int LLVMRunFunctionAsMain(LLVMExecutionEngineRef EE, LLVMValueRef F, unsigned ArgC, const char *const *ArgV, const char *const *EnvP)
void * LLVMGetPointerToGlobal(LLVMExecutionEngineRef EE, LLVMValueRef Global)
uint8_t *(* LLVMMemoryManagerAllocateCodeSectionCallback)(void *Opaque, uintptr_t Size, unsigned Alignment, unsigned SectionID, const char *SectionName)
void LLVMAddGlobalMapping(LLVMExecutionEngineRef EE, LLVMValueRef Global, void *Addr)
LLVMBool LLVMCreateInterpreterForModule(LLVMExecutionEngineRef *OutInterp, LLVMModuleRef M, char **OutError)
void LLVMFreeMachineCodeForFunction(LLVMExecutionEngineRef EE, LLVMValueRef F)
void LLVMDisposeGenericValue(LLVMGenericValueRef GenVal)
LLVMTargetMachineRef LLVMGetExecutionEngineTargetMachine(LLVMExecutionEngineRef EE)
struct LLVMOpaqueValue * LLVMValueRef
Represents an individual value in LLVM IR.
Definition: Types.h:75
int LLVMBool
Definition: Types.h:28
struct LLVMOpaqueType * LLVMTypeRef
Each value in the LLVM IR has a type, an LLVMTypeRef.
Definition: Types.h:68
struct LLVMOpaqueModule * LLVMModuleRef
The top-level container for all other LLVM Intermediate Representation (IR) objects.
Definition: Types.h:61
struct LLVMOpaqueJITEventListener * LLVMJITEventListenerRef
Definition: Types.h:165
struct LLVMOpaqueTargetMachine * LLVMTargetMachineRef
Definition: TargetMachine.h:35
struct LLVMOpaqueTargetData * LLVMTargetDataRef
Definition: Target.h:37
@ LLVMCodeModelJITDefault
Definition: TargetMachine.h:57
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
static const Kind Either
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
CodeGenOptLevel
Code generation optimization level.
Definition: CodeGen.h:54
@ Global
Append to llvm.global_dtors.
Attribute unwrap(LLVMAttributeRef Attr)
Definition: Attributes.h:332
#define N
PointerTy PointerVal
Definition: GenericValue.h:31