1 //===-------- BasicOrcV2CBindings.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();
95 LLVMOrcResourceTrackerRef RT
;
97 // Add our demo module to the JIT.
99 LLVMOrcJITDylibRef MainJD
= LLVMOrcLLJITGetMainJITDylib(J
);
100 RT
= LLVMOrcJITDylibCreateResourceTracker(MainJD
);
102 if ((Err
= LLVMOrcLLJITAddLLVMIRModuleWithRT(J
, RT
, TSM
))) {
103 // If adding the ThreadSafeModule fails then we need to clean it up
104 // ourselves. If adding it succeeds the JIT will manage the memory.
105 LLVMOrcDisposeThreadSafeModule(TSM
);
106 MainResult
= handleError(Err
);
111 // Look up the address of our demo entry point.
112 printf("Looking up before removal...\n");
113 LLVMOrcJITTargetAddress SumAddr
;
116 if ((Err
= LLVMOrcLLJITLookup(J
, &SumAddr
, "sum"))) {
117 MainResult
= handleError(Err
);
122 // If we made it here then everything succeeded. Execute our JIT'd code.
123 int32_t (*Sum
)(int32_t, int32_t) = (int32_t(*)(int32_t, int32_t))SumAddr
;
124 int32_t Result
= Sum(1, 2);
127 printf("1 + 2 = %i\n", Result
);
132 if ((Err
= LLVMOrcResourceTrackerRemove(RT
))) {
133 MainResult
= handleError(Err
);
138 // Attempt a second lookup. Here we expect an error as the code and symbols
139 // should have been removed.
140 printf("Attempting to remove code / symbols...\n");
142 LLVMOrcJITTargetAddress ThrowAwayAddress
;
143 LLVMErrorRef Err
= LLVMOrcLLJITLookup(J
, &ThrowAwayAddress
, "sum");
145 printf("Received error as expected:\n");
148 printf("Failure: Second lookup should have generated an error.\n");
154 // Destroy our JIT instance. This will clean up any memory that the JIT has
155 // taken ownership of. This operation is non-trivial (e.g. it may need to
156 // JIT static destructors) and may also fail. In that case we want to render
157 // the error to stderr, but not overwrite any existing return value.
159 printf("Releasing resource tracker...\n");
160 LLVMOrcReleaseResourceTracker(RT
);
162 printf("Destroying LLJIT instance and exiting.\n");
165 if ((Err
= LLVMOrcDisposeLLJIT(J
))) {
166 int NewFailureResult
= handleError(Err
);
168 MainResult
= NewFailureResult
;