1 //===------ OrcV2CBindingsBasicUsage.c - Basic OrcV2 C Bindings Demo ------===//
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
7 //===----------------------------------------------------------------------===//
9 #include "llvm-c/Core.h"
10 #include "llvm-c/Error.h"
11 #include "llvm-c/LLJIT.h"
12 #include "llvm-c/Support.h"
13 #include "llvm-c/Target.h"
17 int handleError(LLVMErrorRef Err
) {
18 char *ErrMsg
= LLVMGetErrorMessage(Err
);
19 fprintf(stderr
, "Error: %s\n", ErrMsg
);
20 LLVMDisposeErrorMessage(ErrMsg
);
24 LLVMOrcThreadSafeModuleRef
createDemoModule(void) {
25 // Create a new ThreadSafeContext and underlying LLVMContext.
26 LLVMOrcThreadSafeContextRef TSCtx
= LLVMOrcCreateNewThreadSafeContext();
28 // Get a reference to the underlying LLVMContext.
29 LLVMContextRef Ctx
= LLVMOrcThreadSafeContextGetContext(TSCtx
);
31 // Create a new LLVM module.
32 LLVMModuleRef M
= LLVMModuleCreateWithNameInContext("demo", Ctx
);
34 // Add a "sum" function":
35 // - Create the function type and function instance.
36 LLVMTypeRef ParamTypes
[] = {LLVMInt32Type(), LLVMInt32Type()};
37 LLVMTypeRef SumFunctionType
=
38 LLVMFunctionType(LLVMInt32Type(), ParamTypes
, 2, 0);
39 LLVMValueRef SumFunction
= LLVMAddFunction(M
, "sum", SumFunctionType
);
41 // - Add a basic block to the function.
42 LLVMBasicBlockRef EntryBB
= LLVMAppendBasicBlock(SumFunction
, "entry");
44 // - Add an IR builder and point it at the end of the basic block.
45 LLVMBuilderRef Builder
= LLVMCreateBuilder();
46 LLVMPositionBuilderAtEnd(Builder
, EntryBB
);
48 // - Get the two function arguments and use them co construct an "add"
50 LLVMValueRef SumArg0
= LLVMGetParam(SumFunction
, 0);
51 LLVMValueRef SumArg1
= LLVMGetParam(SumFunction
, 1);
52 LLVMValueRef Result
= LLVMBuildAdd(Builder
, SumArg0
, SumArg1
, "result");
54 // - Build the return instruction.
55 LLVMBuildRet(Builder
, Result
);
57 // - Free the builder.
58 LLVMDisposeBuilder(Builder
);
60 // Our demo module is now complete. Wrap it and our ThreadSafeContext in a
62 LLVMOrcThreadSafeModuleRef TSM
= LLVMOrcCreateNewThreadSafeModule(M
, TSCtx
);
64 // Dispose of our local ThreadSafeContext value. The underlying LLVMContext
65 // will be kept alive by our ThreadSafeModule, TSM.
66 LLVMOrcDisposeThreadSafeContext(TSCtx
);
72 int main(int argc
, const char *argv
[]) {
76 // Parse command line arguments and initialize LLVM Core.
77 LLVMParseCommandLineOptions(argc
, argv
, "");
79 // Initialize native target codegen and asm printer.
80 LLVMInitializeNativeTarget();
81 LLVMInitializeNativeAsmPrinter();
83 // Create the JIT instance.
87 if ((Err
= LLVMOrcCreateLLJIT(&J
, 0))) {
88 MainResult
= handleError(Err
);
93 // Create our demo module.
94 LLVMOrcThreadSafeModuleRef TSM
= createDemoModule();
96 // Add our demo module to the JIT.
98 LLVMOrcJITDylibRef MainJD
= LLVMOrcLLJITGetMainJITDylib(J
);
100 if ((Err
= LLVMOrcLLJITAddLLVMIRModule(J
, MainJD
, TSM
))) {
101 // If adding the ThreadSafeModule fails then we need to clean it up
102 // ourselves. If adding it succeeds the JIT will manage the memory.
103 LLVMOrcDisposeThreadSafeModule(TSM
);
104 MainResult
= handleError(Err
);
109 // Look up the address of our demo entry point.
110 LLVMOrcJITTargetAddress SumAddr
;
113 if ((Err
= LLVMOrcLLJITLookup(J
, &SumAddr
, "sum"))) {
114 MainResult
= handleError(Err
);
119 // If we made it here then everything succeeded. Execute our JIT'd code.
120 int32_t (*Sum
)(int32_t, int32_t) = (int32_t(*)(int32_t, int32_t))SumAddr
;
121 int32_t Result
= Sum(1, 2);
124 printf("1 + 2 = %i\n", Result
);
127 // Destroy our JIT instance. This will clean up any memory that the JIT has
128 // taken ownership of. This operation is non-trivial (e.g. it may need to
129 // JIT static destructors) and may also fail. In that case we want to render
130 // the error to stderr, but not overwrite any existing return value.
133 if ((Err
= LLVMOrcDisposeLLJIT(J
))) {
134 int NewFailureResult
= handleError(Err
);
136 MainResult
= NewFailureResult
;