From e784fa40c5808bfb88480f0ab3746ed378a939df Mon Sep 17 00:00:00 2001 From: Reid Spencer Date: Thu, 19 Aug 2004 20:10:04 +0000 Subject: Add the fibonacci example provided by Valery Khamenya. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@15924 91177308-0d34-0410-b5e6-96231b3b80d8 --- examples/Fibonacci/Makefile | 15 ++++ examples/Fibonacci/fibonacci.cpp | 188 +++++++++++++++++++++++++++++++++++++++ 2 files changed, 203 insertions(+) create mode 100644 examples/Fibonacci/Makefile create mode 100644 examples/Fibonacci/fibonacci.cpp (limited to 'examples/Fibonacci') diff --git a/examples/Fibonacci/Makefile b/examples/Fibonacci/Makefile new file mode 100644 index 0000000..df1e128 --- /dev/null +++ b/examples/Fibonacci/Makefile @@ -0,0 +1,15 @@ +##===- projects/HowToUseJIT/Makefile -----------------------*- Makefile -*-===## +# +# The LLVM Compiler Infrastructure +# +# This file was developed by Valery A. Khamenya and is distributed under +# the University of Illinois Open Source License. See LICENSE.TXT for details. +# +##===----------------------------------------------------------------------===## +LEVEL = ../../.. +TOOLNAME = Fibonacci +USEDLIBS = lli-jit lli-interpreter codegen executionengine x86 selectiondag \ + scalaropts analysis.a transformutils.a bcreader target.a vmcore \ + support.a + +include $(LEVEL)/Makefile.common diff --git a/examples/Fibonacci/fibonacci.cpp b/examples/Fibonacci/fibonacci.cpp new file mode 100644 index 0000000..776378d --- /dev/null +++ b/examples/Fibonacci/fibonacci.cpp @@ -0,0 +1,188 @@ +//===--- fibonacci.cpp - An example use of the JIT ----------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file was developed by Valery A. Khamenya and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This small program provides an example of how to build quickly a small +// module with function Fibonacci and execute it with the JIT. +// +// This simple example shows as well 30% speed up with LLVM 1.3 +// in comparison to gcc 3.3.3 at AMD Athlon XP 1500+ . +// +// (Modified from HowToUseJIT.cpp and Stacker/lib/compiler/StackerCompiler.cpp) +// +//===------------------------------------------------------------------------=== +// Goal: +// The goal of this snippet is to create in the memory +// the LLVM module consisting of one function as follow: +// +// int fib(int x) { +// if(x<=2) return 1; +// return fib(x-1)+fib(x-2); +// } +// +// then compile the module via JIT, then execute the `fib' +// function and return result to a driver, i.e. to a "host program". +// + +#include + +#include +#include +#include +#include +#include +#include +#include "llvm/ExecutionEngine/ExecutionEngine.h" +#include "llvm/ExecutionEngine/GenericValue.h" + + +using namespace llvm; + +int main(int argc, char**argv) { + + int n = argc > 1 ? atol(argv[1]) : 44; + + // Create some module to put our function into it. + Module *M = new Module("test"); + + + // We are about to create the "fib" function: + Function *FibF; + + { + // first create type for the single argument of fib function: + // the type is 'int ()' + std::vector ArgT(1); + ArgT[0] = Type::IntTy; + + // now create full type of the "fib" function: + FunctionType *FibT = FunctionType::get(Type::IntTy, // type of result + ArgT, + /*not vararg*/false); + + // Now create the fib function entry and + // insert this entry into module M + // (By passing a module as the last parameter to the Function constructor, + // it automatically gets appended to the Module.) + FibF = new Function(FibT, + Function::ExternalLinkage, // maybe too much + "fib", M); + + // Add a basic block to the function... (again, it automatically inserts + // because of the last argument.) + BasicBlock *BB = new BasicBlock("EntryBlock of fib function", FibF); + + // Get pointers to the constants ... + Value *One = ConstantSInt::get(Type::IntTy, 1); + Value *Two = ConstantSInt::get(Type::IntTy, 2); + + // Get pointers to the integer argument of the add1 function... + assert(FibF->abegin() != FibF->aend()); // Make sure there's an arg + + Argument &ArgX = FibF->afront(); // Get the arg + ArgX.setName("AnArg"); // Give it a nice symbolic name for fun. + + SetCondInst* CondInst + = new SetCondInst( Instruction::SetLE, + &ArgX, Two ); + + BB->getInstList().push_back(CondInst); + + // Create the true_block + BasicBlock* true_bb = new BasicBlock("arg<=2"); + + + // Create the return instruction and add it + // to the basic block for true case: + true_bb->getInstList().push_back(new ReturnInst(One)); + + // Create an exit block + BasicBlock* exit_bb = new BasicBlock("arg>2"); + + { + + // create fib(x-1) + CallInst* CallFibX1; + { + // Create the sub instruction... does not insert... + Instruction *Sub + = BinaryOperator::create(Instruction::Sub, &ArgX, One, + "arg"); + + exit_bb->getInstList().push_back(Sub); + + CallFibX1 = new CallInst(FibF, Sub, "fib(x-1)"); + exit_bb->getInstList().push_back(CallFibX1); + + } + + // create fib(x-2) + CallInst* CallFibX2; + { + // Create the sub instruction... does not insert... + Instruction * Sub + = BinaryOperator::create(Instruction::Sub, &ArgX, Two, + "arg"); + + exit_bb->getInstList().push_back(Sub); + CallFibX2 = new CallInst(FibF, Sub, "fib(x-2)"); + exit_bb->getInstList().push_back(CallFibX2); + + } + + // Create the add instruction... does not insert... + Instruction *Add = + BinaryOperator::create(Instruction::Add, + CallFibX1, CallFibX2, "addresult"); + + // explicitly insert it into the basic block... + exit_bb->getInstList().push_back(Add); + + // Create the return instruction and add it to the basic block + exit_bb->getInstList().push_back(new ReturnInst(Add)); + } + + // Create a branch on the SetCond + BranchInst* br_inst = + new BranchInst( true_bb, exit_bb, CondInst ); + + BB->getInstList().push_back( br_inst ); + FibF->getBasicBlockList().push_back(true_bb); + FibF->getBasicBlockList().push_back(exit_bb); + } + + // Now we going to create JIT + ExistingModuleProvider* MP = new ExistingModuleProvider(M); + ExecutionEngine* EE = ExecutionEngine::create( MP, false ); + + // Call the `foo' function with argument n: + std::vector args(1); + args[0].IntVal = n; + + + std::clog << "verifying... "; + if (verifyModule(*M)) { + std::cerr << argv[0] + << ": assembly parsed, but does not verify as correct!\n"; + return 1; + } + else + std::clog << "OK\n"; + + + std::clog << "We just constructed this LLVM module:\n\n---------\n" << *M; + std::clog << "---------\nstarting fibonacci(" + << n << ") with JIT...\n" << std::flush; + + GenericValue gv = EE->runFunction(FibF, args); + + // import result of execution: + std::cout << "Result: " << gv.IntVal << std:: endl; + + return 0; +} -- cgit v1.1