1*9880d681SAndroid Build Coastguard Worker //===--------------- OrcCAPITest.cpp - Unit tests Orc C API ---------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker // The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker
10*9880d681SAndroid Build Coastguard Worker #include "OrcTestCommon.h"
11*9880d681SAndroid Build Coastguard Worker #include "gtest/gtest.h"
12*9880d681SAndroid Build Coastguard Worker #include "llvm-c/Core.h"
13*9880d681SAndroid Build Coastguard Worker #include "llvm-c/OrcBindings.h"
14*9880d681SAndroid Build Coastguard Worker #include "llvm-c/Target.h"
15*9880d681SAndroid Build Coastguard Worker #include "llvm-c/TargetMachine.h"
16*9880d681SAndroid Build Coastguard Worker
17*9880d681SAndroid Build Coastguard Worker #include <stdio.h>
18*9880d681SAndroid Build Coastguard Worker #include <stdlib.h>
19*9880d681SAndroid Build Coastguard Worker #include <string.h>
20*9880d681SAndroid Build Coastguard Worker
21*9880d681SAndroid Build Coastguard Worker namespace llvm {
22*9880d681SAndroid Build Coastguard Worker
23*9880d681SAndroid Build Coastguard Worker DEFINE_SIMPLE_CONVERSION_FUNCTIONS(TargetMachine, LLVMTargetMachineRef)
24*9880d681SAndroid Build Coastguard Worker
25*9880d681SAndroid Build Coastguard Worker class OrcCAPIExecutionTest : public testing::Test, public OrcExecutionTest {
26*9880d681SAndroid Build Coastguard Worker protected:
createTestModule(const Triple & TT)27*9880d681SAndroid Build Coastguard Worker std::unique_ptr<Module> createTestModule(const Triple &TT) {
28*9880d681SAndroid Build Coastguard Worker ModuleBuilder MB(Context, TT.str(), "");
29*9880d681SAndroid Build Coastguard Worker Function *TestFunc = MB.createFunctionDecl<int()>("testFunc");
30*9880d681SAndroid Build Coastguard Worker Function *Main = MB.createFunctionDecl<int(int, char*[])>("main");
31*9880d681SAndroid Build Coastguard Worker
32*9880d681SAndroid Build Coastguard Worker Main->getBasicBlockList().push_back(BasicBlock::Create(Context));
33*9880d681SAndroid Build Coastguard Worker IRBuilder<> B(&Main->back());
34*9880d681SAndroid Build Coastguard Worker Value* Result = B.CreateCall(TestFunc);
35*9880d681SAndroid Build Coastguard Worker B.CreateRet(Result);
36*9880d681SAndroid Build Coastguard Worker
37*9880d681SAndroid Build Coastguard Worker return MB.takeModule();
38*9880d681SAndroid Build Coastguard Worker }
39*9880d681SAndroid Build Coastguard Worker
40*9880d681SAndroid Build Coastguard Worker typedef int (*MainFnTy)();
41*9880d681SAndroid Build Coastguard Worker
myTestFuncImpl()42*9880d681SAndroid Build Coastguard Worker static int myTestFuncImpl() {
43*9880d681SAndroid Build Coastguard Worker return 42;
44*9880d681SAndroid Build Coastguard Worker }
45*9880d681SAndroid Build Coastguard Worker
46*9880d681SAndroid Build Coastguard Worker static char *testFuncName;
47*9880d681SAndroid Build Coastguard Worker
myResolver(const char * Name,void * Ctx)48*9880d681SAndroid Build Coastguard Worker static uint64_t myResolver(const char *Name, void *Ctx) {
49*9880d681SAndroid Build Coastguard Worker if (!strncmp(Name, testFuncName, 8))
50*9880d681SAndroid Build Coastguard Worker return (uint64_t)&myTestFuncImpl;
51*9880d681SAndroid Build Coastguard Worker return 0;
52*9880d681SAndroid Build Coastguard Worker }
53*9880d681SAndroid Build Coastguard Worker
54*9880d681SAndroid Build Coastguard Worker struct CompileContext {
CompileContextllvm::OrcCAPIExecutionTest::CompileContext55*9880d681SAndroid Build Coastguard Worker CompileContext() : Compiled(false) { }
56*9880d681SAndroid Build Coastguard Worker
57*9880d681SAndroid Build Coastguard Worker OrcCAPIExecutionTest* APIExecTest;
58*9880d681SAndroid Build Coastguard Worker std::unique_ptr<Module> M;
59*9880d681SAndroid Build Coastguard Worker LLVMOrcModuleHandle H;
60*9880d681SAndroid Build Coastguard Worker bool Compiled;
61*9880d681SAndroid Build Coastguard Worker };
62*9880d681SAndroid Build Coastguard Worker
myCompileCallback(LLVMOrcJITStackRef JITStack,void * Ctx)63*9880d681SAndroid Build Coastguard Worker static LLVMOrcTargetAddress myCompileCallback(LLVMOrcJITStackRef JITStack,
64*9880d681SAndroid Build Coastguard Worker void *Ctx) {
65*9880d681SAndroid Build Coastguard Worker CompileContext *CCtx = static_cast<CompileContext*>(Ctx);
66*9880d681SAndroid Build Coastguard Worker auto *ET = CCtx->APIExecTest;
67*9880d681SAndroid Build Coastguard Worker CCtx->M = ET->createTestModule(ET->TM->getTargetTriple());
68*9880d681SAndroid Build Coastguard Worker CCtx->H = LLVMOrcAddEagerlyCompiledIR(JITStack, wrap(CCtx->M.get()),
69*9880d681SAndroid Build Coastguard Worker myResolver, nullptr);
70*9880d681SAndroid Build Coastguard Worker CCtx->Compiled = true;
71*9880d681SAndroid Build Coastguard Worker LLVMOrcTargetAddress MainAddr = LLVMOrcGetSymbolAddress(JITStack, "main");
72*9880d681SAndroid Build Coastguard Worker LLVMOrcSetIndirectStubPointer(JITStack, "foo", MainAddr);
73*9880d681SAndroid Build Coastguard Worker return MainAddr;
74*9880d681SAndroid Build Coastguard Worker }
75*9880d681SAndroid Build Coastguard Worker };
76*9880d681SAndroid Build Coastguard Worker
77*9880d681SAndroid Build Coastguard Worker char *OrcCAPIExecutionTest::testFuncName = nullptr;
78*9880d681SAndroid Build Coastguard Worker
TEST_F(OrcCAPIExecutionTest,TestEagerIRCompilation)79*9880d681SAndroid Build Coastguard Worker TEST_F(OrcCAPIExecutionTest, TestEagerIRCompilation) {
80*9880d681SAndroid Build Coastguard Worker if (!TM)
81*9880d681SAndroid Build Coastguard Worker return;
82*9880d681SAndroid Build Coastguard Worker
83*9880d681SAndroid Build Coastguard Worker LLVMOrcJITStackRef JIT =
84*9880d681SAndroid Build Coastguard Worker LLVMOrcCreateInstance(wrap(TM.get()));
85*9880d681SAndroid Build Coastguard Worker
86*9880d681SAndroid Build Coastguard Worker std::unique_ptr<Module> M = createTestModule(TM->getTargetTriple());
87*9880d681SAndroid Build Coastguard Worker
88*9880d681SAndroid Build Coastguard Worker LLVMOrcGetMangledSymbol(JIT, &testFuncName, "testFunc");
89*9880d681SAndroid Build Coastguard Worker
90*9880d681SAndroid Build Coastguard Worker LLVMOrcModuleHandle H =
91*9880d681SAndroid Build Coastguard Worker LLVMOrcAddEagerlyCompiledIR(JIT, wrap(M.get()), myResolver, nullptr);
92*9880d681SAndroid Build Coastguard Worker MainFnTy MainFn = (MainFnTy)LLVMOrcGetSymbolAddress(JIT, "main");
93*9880d681SAndroid Build Coastguard Worker int Result = MainFn();
94*9880d681SAndroid Build Coastguard Worker EXPECT_EQ(Result, 42)
95*9880d681SAndroid Build Coastguard Worker << "Eagerly JIT'd code did not return expected result";
96*9880d681SAndroid Build Coastguard Worker
97*9880d681SAndroid Build Coastguard Worker LLVMOrcRemoveModule(JIT, H);
98*9880d681SAndroid Build Coastguard Worker
99*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeMangledSymbol(testFuncName);
100*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeInstance(JIT);
101*9880d681SAndroid Build Coastguard Worker }
102*9880d681SAndroid Build Coastguard Worker
TEST_F(OrcCAPIExecutionTest,TestLazyIRCompilation)103*9880d681SAndroid Build Coastguard Worker TEST_F(OrcCAPIExecutionTest, TestLazyIRCompilation) {
104*9880d681SAndroid Build Coastguard Worker if (!TM)
105*9880d681SAndroid Build Coastguard Worker return;
106*9880d681SAndroid Build Coastguard Worker
107*9880d681SAndroid Build Coastguard Worker LLVMOrcJITStackRef JIT =
108*9880d681SAndroid Build Coastguard Worker LLVMOrcCreateInstance(wrap(TM.get()));
109*9880d681SAndroid Build Coastguard Worker
110*9880d681SAndroid Build Coastguard Worker std::unique_ptr<Module> M = createTestModule(TM->getTargetTriple());
111*9880d681SAndroid Build Coastguard Worker
112*9880d681SAndroid Build Coastguard Worker LLVMOrcGetMangledSymbol(JIT, &testFuncName, "testFunc");
113*9880d681SAndroid Build Coastguard Worker
114*9880d681SAndroid Build Coastguard Worker LLVMOrcModuleHandle H =
115*9880d681SAndroid Build Coastguard Worker LLVMOrcAddLazilyCompiledIR(JIT, wrap(M.get()), myResolver, nullptr);
116*9880d681SAndroid Build Coastguard Worker MainFnTy MainFn = (MainFnTy)LLVMOrcGetSymbolAddress(JIT, "main");
117*9880d681SAndroid Build Coastguard Worker int Result = MainFn();
118*9880d681SAndroid Build Coastguard Worker EXPECT_EQ(Result, 42)
119*9880d681SAndroid Build Coastguard Worker << "Lazily JIT'd code did not return expected result";
120*9880d681SAndroid Build Coastguard Worker
121*9880d681SAndroid Build Coastguard Worker LLVMOrcRemoveModule(JIT, H);
122*9880d681SAndroid Build Coastguard Worker
123*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeMangledSymbol(testFuncName);
124*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeInstance(JIT);
125*9880d681SAndroid Build Coastguard Worker }
126*9880d681SAndroid Build Coastguard Worker
TEST_F(OrcCAPIExecutionTest,TestDirectCallbacksAPI)127*9880d681SAndroid Build Coastguard Worker TEST_F(OrcCAPIExecutionTest, TestDirectCallbacksAPI) {
128*9880d681SAndroid Build Coastguard Worker if (!TM)
129*9880d681SAndroid Build Coastguard Worker return;
130*9880d681SAndroid Build Coastguard Worker
131*9880d681SAndroid Build Coastguard Worker LLVMOrcJITStackRef JIT =
132*9880d681SAndroid Build Coastguard Worker LLVMOrcCreateInstance(wrap(TM.get()));
133*9880d681SAndroid Build Coastguard Worker
134*9880d681SAndroid Build Coastguard Worker LLVMOrcGetMangledSymbol(JIT, &testFuncName, "testFunc");
135*9880d681SAndroid Build Coastguard Worker
136*9880d681SAndroid Build Coastguard Worker CompileContext C;
137*9880d681SAndroid Build Coastguard Worker C.APIExecTest = this;
138*9880d681SAndroid Build Coastguard Worker LLVMOrcCreateIndirectStub(JIT, "foo",
139*9880d681SAndroid Build Coastguard Worker LLVMOrcCreateLazyCompileCallback(JIT,
140*9880d681SAndroid Build Coastguard Worker myCompileCallback,
141*9880d681SAndroid Build Coastguard Worker &C));
142*9880d681SAndroid Build Coastguard Worker MainFnTy FooFn = (MainFnTy)LLVMOrcGetSymbolAddress(JIT, "foo");
143*9880d681SAndroid Build Coastguard Worker int Result = FooFn();
144*9880d681SAndroid Build Coastguard Worker EXPECT_TRUE(C.Compiled)
145*9880d681SAndroid Build Coastguard Worker << "Function wasn't lazily compiled";
146*9880d681SAndroid Build Coastguard Worker EXPECT_EQ(Result, 42)
147*9880d681SAndroid Build Coastguard Worker << "Direct-callback JIT'd code did not return expected result";
148*9880d681SAndroid Build Coastguard Worker
149*9880d681SAndroid Build Coastguard Worker C.Compiled = false;
150*9880d681SAndroid Build Coastguard Worker FooFn();
151*9880d681SAndroid Build Coastguard Worker EXPECT_FALSE(C.Compiled)
152*9880d681SAndroid Build Coastguard Worker << "Direct-callback JIT'd code was JIT'd twice";
153*9880d681SAndroid Build Coastguard Worker
154*9880d681SAndroid Build Coastguard Worker LLVMOrcRemoveModule(JIT, C.H);
155*9880d681SAndroid Build Coastguard Worker
156*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeMangledSymbol(testFuncName);
157*9880d681SAndroid Build Coastguard Worker LLVMOrcDisposeInstance(JIT);
158*9880d681SAndroid Build Coastguard Worker }
159*9880d681SAndroid Build Coastguard Worker
160*9880d681SAndroid Build Coastguard Worker } // namespace llvm
161