Native: add .alloc(value) for primitive types to interop runtime

^KT-50312 Fixed
This commit is contained in:
Svyatoslav Scherbina
2022-03-09 17:48:07 +03:00
parent 11fdd2e4fa
commit 42ed0c5fe6
4 changed files with 234 additions and 1 deletions
@@ -5,5 +5,51 @@
package org.jetbrains.kotlin.generators.native.interopRuntime
fun main() {
import java.io.File
import java.io.FileWriter
fun FileWriter.generateHeader() {
appendLine(File("license/COPYRIGHT_HEADER.txt").readText())
appendLine("package kotlinx.cinterop")
appendLine()
appendLine("//")
appendLine("// NOTE: THIS FILE IS AUTO-GENERATED by the generators/nativeInteropRuntime/NativeInteropRuntimeGenerator.kt")
appendLine("//")
appendLine()
}
enum class PrimitiveInteropType {
Boolean, Byte, Short, Int, Long, UByte, UShort, UInt, ULong, Float, Double;
}
fun FileWriter.generateAllocWithValue(type: PrimitiveInteropType) {
val typeName = type.name
appendLine(
"""
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : $typeName> NativePlacement.alloc(value: T): ${typeName}VarOf<T> =
alloc<${typeName}VarOf<T>> { this.value = value }
""".trimIndent()
)
}
fun generateUtils(targetDir: File) {
FileWriter(targetDir.resolve("_UtilsGenerated.kt")).use { writer ->
writer.generateHeader()
for (type in PrimitiveInteropType.values()) {
writer.generateAllocWithValue(type)
writer.appendLine()
}
}
}
fun main() {
val targetDir = File("kotlin-native/Interop/Runtime/src/main/kotlin/kotlinx/cinterop")
generateUtils(targetDir)
}
@@ -0,0 +1,87 @@
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package kotlinx.cinterop
//
// NOTE: THIS FILE IS AUTO-GENERATED by the generators/nativeInteropRuntime/NativeInteropRuntimeGenerator.kt
//
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Boolean> NativePlacement.alloc(value: T): BooleanVarOf<T> =
alloc<BooleanVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Byte> NativePlacement.alloc(value: T): ByteVarOf<T> =
alloc<ByteVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Short> NativePlacement.alloc(value: T): ShortVarOf<T> =
alloc<ShortVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Int> NativePlacement.alloc(value: T): IntVarOf<T> =
alloc<IntVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Long> NativePlacement.alloc(value: T): LongVarOf<T> =
alloc<LongVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : UByte> NativePlacement.alloc(value: T): UByteVarOf<T> =
alloc<UByteVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : UShort> NativePlacement.alloc(value: T): UShortVarOf<T> =
alloc<UShortVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : UInt> NativePlacement.alloc(value: T): UIntVarOf<T> =
alloc<UIntVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : ULong> NativePlacement.alloc(value: T): ULongVarOf<T> =
alloc<ULongVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Float> NativePlacement.alloc(value: T): FloatVarOf<T> =
alloc<FloatVarOf<T>> { this.value = value }
/**
* Allocates variable with given value type and initializes it with given value.
*/
@Suppress("FINAL_UPPER_BOUND")
public fun <T : Double> NativePlacement.alloc(value: T): DoubleVarOf<T> =
alloc<DoubleVarOf<T>> { this.value = value }
@@ -4705,6 +4705,10 @@ task interop_sourceCodeStruct(type: KonanLocalTest) {
source = "codegen/intrinsics/interop_sourceCodeStruct.kt"
}
task interop_alloc_value(type: KonanLocalTest) {
source = "runtime/interop/interop_alloc_value.kt"
}
standaloneTest("isExperimentalMM") {
source = "codegen/intrinsics/isExperimentalMM.kt"
flags = [ "-tr" ]
@@ -0,0 +1,96 @@
package runtime.interop.interop_alloc_value
import kotlinx.cinterop.*
import kotlin.test.*
@Test
fun testBoolean() = memScoped {
assertEquals(true, alloc(true).value)
assertEquals(false, alloc(false).value)
}
@Test
fun testByte() = memScoped<Unit> {
assertEquals(Byte.MIN_VALUE, alloc(Byte.MIN_VALUE).value)
assertEquals(Byte.MAX_VALUE, alloc(Byte.MAX_VALUE).value)
assertEquals(0.toByte(), alloc(0.toByte()).value)
assertEquals(1.toByte(), alloc(1.toByte()).value)
assertEquals(1, alloc<Byte>(1).value)
}
@Test
fun testShort() = memScoped<Unit> {
assertEquals(Short.MIN_VALUE, alloc(Short.MIN_VALUE).value)
assertEquals(Short.MAX_VALUE, alloc(Short.MAX_VALUE).value)
assertEquals(0.toShort(), alloc(0.toShort()).value)
assertEquals(2.toShort(), alloc(2.toShort()).value)
assertEquals(2, alloc<Short>(2).value)
}
@Test
fun testInt() = memScoped<Unit> {
assertEquals(Int.MIN_VALUE, alloc(Int.MIN_VALUE).value)
assertEquals(Int.MAX_VALUE, alloc(Int.MAX_VALUE).value)
assertEquals(0.toInt(), alloc(0.toInt()).value)
assertEquals(3.toInt(), alloc(3.toInt()).value)
assertEquals(3, alloc<Int>(3).value)
}
@Test
fun testLong() = memScoped<Unit> {
assertEquals(Long.MIN_VALUE, alloc(Long.MIN_VALUE).value)
assertEquals(Long.MAX_VALUE, alloc(Long.MAX_VALUE).value)
assertEquals(0L, alloc(0L).value)
assertEquals(4L, alloc(4L).value)
assertEquals(4, alloc<Long>(4).value)
}
@Test
fun testUByte() = memScoped<Unit> {
assertEquals(UByte.MIN_VALUE, alloc(UByte.MIN_VALUE).value)
assertEquals(UByte.MAX_VALUE, alloc(UByte.MAX_VALUE).value)
assertEquals(5.toUByte(), alloc(5.toUByte()).value)
assertEquals(5u, alloc<UByte>(5u).value)
}
@Test
fun testUShort() = memScoped<Unit> {
assertEquals(UShort.MIN_VALUE, alloc(UShort.MIN_VALUE).value)
assertEquals(UShort.MAX_VALUE, alloc(UShort.MAX_VALUE).value)
assertEquals(6.toUShort(), alloc(6.toUShort()).value)
assertEquals(6u, alloc<UShort>(6u).value)
}
@Test
fun testUInt() = memScoped<Unit> {
assertEquals(UInt.MIN_VALUE, alloc(UInt.MIN_VALUE).value)
assertEquals(UInt.MAX_VALUE, alloc(UInt.MAX_VALUE).value)
assertEquals(7.toUInt(), alloc(7.toUInt()).value)
assertEquals(7u, alloc<UInt>(7u).value)
}
@Test
fun testULong() = memScoped<Unit> {
assertEquals(ULong.MIN_VALUE, alloc(ULong.MIN_VALUE).value)
assertEquals(ULong.MAX_VALUE, alloc(ULong.MAX_VALUE).value)
assertEquals(8uL, alloc(8uL).value)
assertEquals(8u, alloc<ULong>(8u).value)
}
@Test
fun testFloat() = memScoped<Unit> {
assertEquals(Float.MIN_VALUE, alloc(Float.MIN_VALUE).value)
assertEquals(Float.MAX_VALUE, alloc(Float.MAX_VALUE).value)
assertEquals(0.0f, alloc(0.0f).value)
assertEquals(9.0f, alloc(9.0f).value)
assertEquals(9.0f, alloc<Float>(9.0f).value)
}
@Test
fun testDouble() = memScoped<Unit> {
assertEquals(Double.MIN_VALUE, alloc(Double.MIN_VALUE).value)
assertEquals(Double.MAX_VALUE, alloc(Double.MAX_VALUE).value)
assertEquals(0.0, alloc(0.0).value)
assertEquals(10.0, alloc(10.0).value)
assertEquals(10.0, alloc<Double>(10.0).value)
}