Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ftomassetti/kllvm
Kotlin library to work with LLVM
https://github.com/ftomassetti/kllvm
kotlin llvm llvm-bindings llvm-ir
Last synced: 2 months ago
JSON representation
Kotlin library to work with LLVM
- Host: GitHub
- URL: https://github.com/ftomassetti/kllvm
- Owner: ftomassetti
- License: apache-2.0
- Created: 2017-05-28T09:20:25.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2022-06-24T08:59:10.000Z (over 2 years ago)
- Last Synced: 2024-09-30T03:41:04.988Z (3 months ago)
- Topics: kotlin, llvm, llvm-bindings, llvm-ir
- Language: Kotlin
- Homepage:
- Size: 98.6 KB
- Stars: 57
- Watchers: 5
- Forks: 9
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# 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](https://github.com/bytedeco/javacpp-presets/tree/master/llvm)
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.```kotlin
@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 %koParamsokParams:
%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 0koParams:
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](https://tomassetti.me/create-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.