Giter Club home page Giter Club logo

kllvm's Introduction

KLLVM

KLLVM stands for Kotlin-LLVM. It is intended as a library to work with LLVM in Kotlin.

Currently it supports the generation of IR code. The rationale is that there are currently no decent options to use LLVM from the JVM: while there is an autogenerated wrapper of the LLVM C++ API for Java it is unusable. For this reason the best option seems to generate the textual representation of IR code to later compile it to object file (or interpret it directly) using LLVM utilities.

How to use it

In the following example we show how to generate a simple application take takes two parameters, sum them and print the result.

    @test fun generateSum() {
        val EXIT_CODE_OK = 0
        val EXIT_CODE_WRONG_PARAMS = 1
        val N_PARAMS_EXPECTED = 2
        val STRING_TYPE = Pointer(I8Type)

        val module = ModuleBuilder()
        val mainFunction = module.createMainFunction()

        val atoiDeclaration = FunctionDeclaration("atoi", I32Type, listOf(), varargs = true)
        module.addDeclaration(atoiDeclaration)
        module.addDeclaration(FunctionDeclaration("printf", I32Type, listOf(STRING_TYPE), varargs = true))

        val okParamsBlock = mainFunction.createBlock("okParams")
        val koParamsBlock = mainFunction.createBlock("koParams")

        val comparisonResult = mainFunction.tempValue(Comparison(ComparisonType.Equal,
                mainFunction.paramReference(0), IntConst(N_PARAMS_EXPECTED + 1, I32Type)))
        mainFunction.addInstruction(IfInstruction(comparisonResult.reference(), okParamsBlock, koParamsBlock))

        // OK Block : convert to int, sum, and print
        val aAsStringPtr = okParamsBlock.tempValue(GetElementPtr(STRING_TYPE, mainFunction.paramReference(1), IntConst(1, I64Type)))
        val aAsString = okParamsBlock.load(aAsStringPtr.reference())
        val aAsInt = okParamsBlock.tempValue(CallWithBitCast(atoiDeclaration, aAsString))
        val bAsStringPtr = okParamsBlock.tempValue(GetElementPtr(STRING_TYPE, mainFunction.paramReference(1), IntConst(2, I64Type)))
        val bAsString = okParamsBlock.load(bAsStringPtr.reference())
        val bAsInt = okParamsBlock.tempValue(CallWithBitCast(atoiDeclaration, bAsString))
        val sum = okParamsBlock.tempValue(IntAddition(aAsInt.reference(), bAsInt.reference()))
        okParamsBlock.addInstruction(Printf(mainFunction.stringConstForContent("Result: %d\n").reference(), sum.reference()))
        okParamsBlock.addInstruction(ReturnInt(EXIT_CODE_OK))

        // KO Block : error message and exit
        koParamsBlock.addInstruction(Printf(mainFunction.stringConstForContent("Please specify two arguments").reference()))
        koParamsBlock.addInstruction(ReturnInt(EXIT_CODE_WRONG_PARAMS))

        println(module.IRCode())
        assertEquals("""@stringConst0 = private unnamed_addr constant [12 x i8] c"Result: %d\0A\00"
@stringConst1 = private unnamed_addr constant [29 x i8] c"Please specify two arguments\00"



declare i32 @atoi(...)
declare i32 @printf(i8*, ...)

define i32 @main(i32, i8**) {
    
    ; unnamed block
    %tmpValue0 = icmp eq i32 %0, 3
    br i1 %tmpValue0, label %okParams, label %koParams

    okParams:
    %tmpValue1 = getelementptr inbounds i8*, i8** %1, i64 1
    %tmpValue2 = load i8*, i8** %tmpValue1
    %tmpValue3 = call i32 (i8*, ...) bitcast (i32 (...)* @atoi to i32 (i8*, ...)*)(i8* %tmpValue2)
    %tmpValue4 = getelementptr inbounds i8*, i8** %1, i64 2
    %tmpValue5 = load i8*, i8** %tmpValue4
    %tmpValue6 = call i32 (i8*, ...) bitcast (i32 (...)* @atoi to i32 (i8*, ...)*)(i8* %tmpValue5)
    %tmpValue7 = add i32 %tmpValue3, %tmpValue6
    call i32 (i8*, ...) @printf(i8* getelementptr inbounds ([12 x i8], [12 x i8]* @stringConst0, i32 0, i32 0), i32 %tmpValue7)
    ret i32 0

    koParams:
    call i32 (i8*, ...) @printf(i8* getelementptr inbounds ([29 x i8], [29 x i8]* @stringConst1, i32 0, i32 0))
    ret i32 1

}""", module.IRCode().trim())
    }

By using KLLVM is it possible to programmatically build the program and the obtain the IR code. This IR code can then be interpreted using the lli command from LLVM.

It can also be compiled into an executable by running:

llc -filetype=obj example.ll
clang example.o -o example

Status

The library is currently used to compile to IR two Domain Specific Languages. The code is presented in the book How to create pragmatic, lightweight languages.

Future

We plan to add new features following the requests of the community. One possible option could be to create a wrapper around the javacpp-presets for LLVM.

License

This project is released under the Apache License V2.0.

kllvm's People

Contributors

ftomassetti avatar vladocc avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar

kllvm's Issues

Unable to Printf the contents of variables

I found this library for your reddit thread, and decided to implement it into a toy language that I'm in the process of making. I was able to make it handle the printing of string literals, but when I tried to abstract it a bit more, I found that it was having trouble emitting code that printed the contents of variables.

My code can be found at https://github.com/Redrield/ChocoLang/blob/experimental/src/main/kotlin/com/redrield/chocolang/codegen/LLVMIRGenerator.kt#L70 (That line is more associated with defining the variables).

I'm not sure if this is user error of if it's a bug in the API, but I wasn't really able to find any documentation past glances in the source, and from what I found that should work.

Unable to Printf floating point integers.

I got this library working with printing String literals, but I can't seem to get it to print floating point numbers. Delimiting them with %f leads to 0.000000 no matter the contents of the actual constant, and %d just gives me jibberish nothing close to the actual value.

This is the code that I have

val EXIT_CODE_OK = 0
val STRING_TYPE = Pointer(I8Type)

val module = ModuleBuilder()
val mainFunction = module.createMainFunction()

module.addDeclaration(FunctionDeclaration("printf", I32Type, listOf(STRING_TYPE), varargs = true))

val okBlock = mainFunction.entryBlock()

val num = 2.0
val numConst = FloatConst(num, FloatType)

okBlock.addInstruction(Printf(mainFunction.stringConstForContent("%f\n").reference(), numConst))

okBlock.addInstruction(ReturnInt(EXIT_CODE_OK))
println(module.IRCode())

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.