| // CHECK_BYTECODE_LISTING |
| // WITH_STDLIB |
| // WITH_COROUTINES |
| // TARGET_BACKEND: JVM_IR |
| // IGNORE_INLINER: IR |
| // LANGUAGE: +ValueClasses |
| // FIR_IDENTICAL |
| |
| // FILE: caller.kt |
| import kotlin.coroutines.* |
| |
| fun runSuspend(block: suspend () -> Unit) { |
| block.startCoroutine(Continuation(EmptyCoroutineContext) { it.getOrThrow() }) |
| } |
| |
| // FILE: test.kt |
| |
| @JvmInline |
| value class DPoint(val x: Double, val y: Double) { |
| fun f(z: Double) = x + y + z |
| suspend fun suspended() = this |
| inline suspend fun suspendedInline() = this |
| |
| fun functionWithInlineClass(other: DPoint, n: UInt) = x.toUInt() + y.toUInt() + other.x.toUInt() + other.y.toUInt() + n |
| |
| suspend fun suspendFunctionWithLambda(other: DPoint, f: suspend (DPoint) -> DPoint) = DPoint(f(this).x, f(other).y).suspendedInline() |
| |
| suspend fun suspendInlineFunctionWithLambda(other: DPoint, f: suspend (DPoint) -> DPoint) = DPoint(f(this).x, f(other).y).suspendedInline() |
| } |
| |
| fun DPoint.extensionFunction(z: Double) = x + y + z |
| |
| fun g(point: DPoint, z: Double) = point.f(z) |
| |
| class A(val point: DPoint) { |
| fun f(otherDPoint: DPoint, z: Double) = point.f(z) * otherDPoint.f(z) |
| } |
| |
| fun consume(point1: DPoint, point2: DPoint, f: (DPoint, DPoint) -> DPoint) = f(point1, point2) |
| |
| inline fun consumeInline(point1: DPoint, point2: DPoint, f: (DPoint, DPoint) -> DPoint) = f(point1, point2) |
| |
| operator fun DPoint.plus(other: DPoint) = DPoint(this.x + other.x, this.y + other.y) |
| |
| fun makeDPoint(x: Double, y: Double, maker: (Double, Double) -> DPoint) = maker(x, y) |
| |
| inline fun makeDPointInline(x: Double, y: Double, maker: (Double, Double) -> DPoint) = maker(x, y) |
| |
| inline fun <T> id(x: T) = x |
| |
| fun box(): String { |
| val dPoint = DPoint(1.0, 2.0) |
| val a = A(dPoint) |
| |
| require((::DPoint)(1.0, 2.0) == dPoint) |
| require((dPoint::f)(3.0) == 6.0) |
| require((DPoint::f)(dPoint, 3.0) == 6.0) |
| require((dPoint::extensionFunction)(3.0) == 6.0) |
| require((DPoint::extensionFunction)(dPoint, 3.0) == 6.0) |
| |
| runSuspend { require(dPoint.suspended() == dPoint) } |
| runSuspend { require((dPoint::suspended)() == dPoint) } |
| runSuspend { require((DPoint::suspended)(dPoint) == dPoint) } |
| runSuspend { require(dPoint.suspendedInline() == dPoint) } |
| runSuspend { require((dPoint::suspendedInline)() == dPoint) } |
| runSuspend { require((DPoint::suspendedInline)(dPoint) == dPoint) } |
| |
| runSuspend { require(dPoint.suspendFunctionWithLambda(dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require((dPoint::suspendFunctionWithLambda)(dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require((DPoint::suspendFunctionWithLambda)(dPoint, dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require(dPoint.suspendFunctionWithLambda(dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require((dPoint::suspendFunctionWithLambda)(dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require((DPoint::suspendFunctionWithLambda)(dPoint, dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require(dPoint.suspendFunctionWithLambda(dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require((dPoint::suspendFunctionWithLambda)(dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require((DPoint::suspendFunctionWithLambda)(dPoint, dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require(dPoint.suspendFunctionWithLambda(dPoint) { it.suspendedInline() } == dPoint) } |
| runSuspend { require((dPoint::suspendFunctionWithLambda)(dPoint) { it.suspendedInline() } == dPoint) } |
| runSuspend { require((DPoint::suspendFunctionWithLambda)(dPoint, dPoint) { it.suspendedInline() } == dPoint) } |
| |
| runSuspend { require(dPoint.suspendInlineFunctionWithLambda(dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require((dPoint::suspendInlineFunctionWithLambda)(dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require((DPoint::suspendInlineFunctionWithLambda)(dPoint, dPoint, DPoint::suspended) == dPoint) } |
| runSuspend { require(dPoint.suspendInlineFunctionWithLambda(dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require((dPoint::suspendInlineFunctionWithLambda)(dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require((DPoint::suspendInlineFunctionWithLambda)(dPoint, dPoint, DPoint::suspendedInline) == dPoint) } |
| runSuspend { require(dPoint.suspendInlineFunctionWithLambda(dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require((dPoint::suspendInlineFunctionWithLambda)(dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require((DPoint::suspendInlineFunctionWithLambda)(dPoint, dPoint) { it.suspended() } == dPoint) } |
| runSuspend { require(dPoint.suspendInlineFunctionWithLambda(dPoint) { it.suspendedInline() } == dPoint) } |
| runSuspend { require((dPoint::suspendInlineFunctionWithLambda)(dPoint) { it.suspendedInline() } == dPoint) } |
| runSuspend { require((DPoint::suspendInlineFunctionWithLambda)(dPoint, dPoint) { it.suspendedInline() } == dPoint) } |
| |
| require((dPoint::functionWithInlineClass)(dPoint, 100U) == 106U) |
| require((DPoint::functionWithInlineClass)(dPoint, dPoint, 100U) == 106U) |
| |
| require((::g)(dPoint, 3.0) == 6.0) |
| require((a::f)(dPoint, 3.0) == 36.0) |
| require((A::f)(a, dPoint, 3.0) == 36.0) |
| |
| require((::DPoint)(1.0, DPoint(1.0, 2.0).y) == dPoint) |
| require((dPoint::f)(DPoint(1.0, 3.0).y) == 6.0) |
| require((DPoint::f)(dPoint, DPoint(1.0, 3.0).y) == 6.0) |
| require((::g)(dPoint, DPoint(1.0, 3.0).y) == 6.0) |
| require((a::f)(dPoint, DPoint(1.0, 3.0).y) == 36.0) |
| require((A::f)(a, dPoint, DPoint(1.0, 3.0).y) == 36.0) |
| |
| require(consume(DPoint(1.0, 2.0), DPoint(3.0, 4.0), DPoint::plus) == DPoint(4.0, 6.0)) |
| require(consume(DPoint(1.0, 2.0), DPoint(3.0, 4.0)) { dPoint, other -> dPoint.plus(other) } == DPoint(4.0, 6.0)) |
| require(consumeInline(DPoint(1.0, 2.0), DPoint(3.0, 4.0), DPoint::plus) == DPoint(4.0, 6.0)) |
| require(consumeInline(DPoint(1.0, 2.0), DPoint(3.0, 4.0)) { dPoint, other -> dPoint.plus(other) } == DPoint(4.0, 6.0)) |
| require( |
| consume(DPoint(1.0, 2.0), DPoint(3.0, 4.0), fun(dPoint: DPoint, other: DPoint): DPoint = dPoint.plus(other)) == |
| DPoint(4.0, 6.0) |
| ) |
| require( |
| consumeInline(DPoint(1.0, 2.0), DPoint(3.0, 4.0), fun(dPoint: DPoint, other: DPoint): DPoint = dPoint.plus(other)) == |
| DPoint(4.0, 6.0) |
| ) |
| |
| require(makeDPoint(1.0, 2.0, ::DPoint) == DPoint(1.0, 2.0)) |
| require(makeDPointInline(1.0, 2.0, ::DPoint) == DPoint(1.0, 2.0)) |
| |
| require(::DPoint == ::DPoint) |
| require(::DPoint == any1()) |
| require(dPoint::f == dPoint::f) |
| require(DPoint::f == DPoint::f) |
| require(dPoint::f == any2(dPoint)) |
| require(DPoint::f == any2()) |
| require(dPoint::suspended == dPoint::suspended) |
| require(DPoint::suspended == DPoint::suspended) |
| require(dPoint::suspended == any3(dPoint)) |
| require(DPoint::suspended == any3()) |
| require(dPoint::suspendedInline == dPoint::suspendedInline) |
| require(DPoint::suspendedInline == DPoint::suspendedInline) |
| require(dPoint::suspendedInline == any4(dPoint)) |
| require(DPoint::suspendedInline == any4()) |
| require(dPoint::functionWithInlineClass == dPoint::functionWithInlineClass) |
| require(DPoint::functionWithInlineClass == DPoint::functionWithInlineClass) |
| require(dPoint::functionWithInlineClass == any5(dPoint)) |
| require(DPoint::functionWithInlineClass == any5()) |
| require(dPoint::extensionFunction == dPoint::extensionFunction) |
| require(DPoint::extensionFunction == DPoint::extensionFunction) |
| require(dPoint::extensionFunction == any6(dPoint)) |
| require(DPoint::extensionFunction == any6()) |
| require(dPoint::suspendFunctionWithLambda == dPoint::suspendFunctionWithLambda) |
| require(DPoint::suspendFunctionWithLambda == DPoint::suspendFunctionWithLambda) |
| require(dPoint::suspendFunctionWithLambda == any7(dPoint)) |
| require(DPoint::suspendFunctionWithLambda == any7()) |
| require(dPoint::suspendInlineFunctionWithLambda == dPoint::suspendInlineFunctionWithLambda) |
| require(DPoint::suspendInlineFunctionWithLambda == DPoint::suspendInlineFunctionWithLambda) |
| require(dPoint::suspendInlineFunctionWithLambda == any8(dPoint)) |
| require(DPoint::suspendInlineFunctionWithLambda == any8()) |
| require(::g == ::g) |
| require(a::f == a::f) |
| require(A::f == A::f) |
| require(DPoint::plus == DPoint::plus) |
| require(dPoint.let { DPoint(it.x * 2, it.y * 2) } == DPoint(2.0, 4.0)) |
| require(dPoint.let(::id) == DPoint(1.0, 2.0)) |
| |
| runSuspend { require(requiresF(dPoint) { it } == dPoint) } |
| runSuspend { require(requiresF(dPoint, F { it }) == dPoint) } |
| |
| return "OK" |
| } |
| |
| // FILE: another.kt |
| |
| fun any1(): Any = ::DPoint |
| fun any2(dPoint: DPoint): Any = dPoint::f |
| fun any2(): Any = DPoint::f |
| fun any3(dPoint: DPoint): Any = dPoint::suspended |
| fun any3(): Any = DPoint::suspended |
| fun any4(dPoint: DPoint): Any = dPoint::suspendedInline |
| fun any4(): Any = DPoint::suspendedInline |
| fun any5(dPoint: DPoint): Any = dPoint::functionWithInlineClass |
| fun any5(): Any = DPoint::functionWithInlineClass |
| fun any6(dPoint: DPoint): Any = dPoint::extensionFunction |
| fun any6(): Any = DPoint::extensionFunction |
| fun any7(dPoint: DPoint): Any = dPoint::suspendFunctionWithLambda |
| fun any7(): Any = DPoint::suspendFunctionWithLambda |
| fun any8(dPoint: DPoint): Any = dPoint::suspendInlineFunctionWithLambda |
| fun any8(): Any = DPoint::suspendInlineFunctionWithLambda |
| |
| fun interface F { |
| suspend fun run(dPoint: DPoint): DPoint |
| } |
| |
| suspend fun requiresF(x: DPoint, f: F) = f.run(x) |