f318b5969d
Substitution of type arguments to non-reified type parameters may lead to accidental reification, which should not be done (see ^KT-60174 for examples). So, we should erase them, except the few cases. ^KT-60174: Fixed ^KT-60175: Fixed
411 lines
14 KiB
Kotlin
Vendored
411 lines
14 KiB
Kotlin
Vendored
// The test checks an optimization which is implemented only for JS_IR backend
|
|
// DONT_TARGET_EXACT_BACKEND: JS
|
|
|
|
@file:Suppress("RESERVED_MEMBER_INSIDE_VALUE_CLASS")
|
|
|
|
inline class ClassInt(val x: Int)
|
|
inline class ClassString(val x: String)
|
|
inline class ClassUnderlayingInline(val x: ClassInt)
|
|
inline class ClassNullableInt(val x: Int?)
|
|
inline class ClassNullableUnderlayingInline(val x: ClassInt?)
|
|
inline class ClassNothing(val x: Nothing?)
|
|
|
|
value class ClassWithEqualsOverride(val data: Int) {
|
|
override fun equals(other: Any?): Boolean = other is ClassWithEqualsOverride && data % 256 == other.data % 256
|
|
}
|
|
|
|
class MyClass(val data: Int) {
|
|
override fun equals(other: Any?): Boolean = other is MyClass && data % 10 == other.data % 10
|
|
}
|
|
|
|
inline class ClassUnderlayingWithEquals(val x: MyClass)
|
|
|
|
interface InterfaceForInlineClass
|
|
inline class ClassIntWithInterface(val x: Int) : InterfaceForInlineClass
|
|
inline class ClassStringWithInterface(val x: String) : InterfaceForInlineClass
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testBasicInt function=equals
|
|
// CHECK_NEW_COUNT: function=testBasicInt count=0
|
|
fun testBasicInt() {
|
|
val a1_1 = ClassInt(1)
|
|
val a1_2 = ClassInt(1)
|
|
val a2 = ClassInt(2)
|
|
|
|
assertTrue(a1_1 == a1_1)
|
|
assertTrue(a1_1 == a1_2)
|
|
assertFalse(a1_1 == a2)
|
|
}
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testBasicString function=equals
|
|
// CHECK_NEW_COUNT: function=testBasicString count=0
|
|
fun testBasicString() {
|
|
val b1_1 = ClassString("b1")
|
|
val b1_2 = ClassString("b1")
|
|
val b2 = ClassString("b2")
|
|
|
|
assertTrue(b1_1 == b1_1)
|
|
assertTrue(b1_1 == b1_2)
|
|
assertFalse(b1_1 == b2)
|
|
}
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testFunctionCall function=equals
|
|
// CHECK_NEW_COUNT: function=testFunctionCall count=0
|
|
fun testFunctionCall() {
|
|
// CHECK_NEW_COUNT: function=testFunctionCall$makeClassInt count=0
|
|
fun makeClassInt(x: Int): ClassInt {
|
|
return ClassInt(x)
|
|
}
|
|
|
|
val a1 = ClassInt(1)
|
|
assertTrue(a1 == makeClassInt(1))
|
|
assertTrue(makeClassInt(1) == a1)
|
|
assertTrue(makeClassInt(1) == makeClassInt(1))
|
|
|
|
assertFalse(a1 == makeClassInt(2))
|
|
assertFalse(makeClassInt(2) == a1)
|
|
assertFalse(makeClassInt(2) == makeClassInt(3))
|
|
}
|
|
|
|
// CHECK_NEW_COUNT: function=testTypeErasing count=3
|
|
fun testTypeErasing() {
|
|
val a1 = ClassInt(1)
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testTypeErasing$isEqualsWithA1 function=equals
|
|
// CHECK_NEW_COUNT: function=testTypeErasing$isEqualsWithA1 count=1
|
|
fun <T> isEqualsWithA1(a: T): Boolean {
|
|
return a1 == a
|
|
}
|
|
|
|
assertTrue(isEqualsWithA1(a1))
|
|
assertTrue(isEqualsWithA1(ClassInt(1)))
|
|
|
|
val a2 = ClassInt(2)
|
|
assertFalse(isEqualsWithA1(a2))
|
|
}
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testFunctionCall function=equals
|
|
// CHECK_NEW_COUNT: function=testTypeErasingAndCast count=4
|
|
fun testTypeErasingAndCast() {
|
|
fun boxValue(a: Any): ClassInt {
|
|
return a as ClassInt
|
|
}
|
|
|
|
val a1_1 = ClassInt(1)
|
|
val a1_2 = ClassInt(1)
|
|
|
|
assertTrue(boxValue(a1_1) == a1_2)
|
|
assertTrue(boxValue(a1_1) == boxValue(a1_2))
|
|
assertTrue(ClassInt(if (a1_1 == boxValue(a1_2)) 1 else 0) == a1_1)
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testNullableInstances function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableInstances count=21
|
|
fun testNullableInstances() {
|
|
val a1_1: ClassInt? = ClassInt(1)
|
|
val a1_2 = ClassInt(1)
|
|
val a1_3: ClassInt? = null
|
|
val a1_4: ClassInt? = ClassInt(1)
|
|
|
|
assertTrue(a1_1 == a1_2)
|
|
assertTrue(a1_2 == a1_1)
|
|
assertTrue(a1_1!! == a1_2)
|
|
assertTrue(a1_2 == a1_1!!)
|
|
assertTrue(a1_4 == a1_1!!)
|
|
assertTrue(a1_4!! == a1_1!!)
|
|
|
|
val a1_5 = a1_4!!
|
|
assertTrue(a1_5 == a1_1!!)
|
|
|
|
assertFalse(a1_1 == a1_3)
|
|
assertFalse(a1_2 == a1_3)
|
|
assertFalse(a1_3 == a1_2)
|
|
}
|
|
|
|
fun testUnderlyingInline() {
|
|
val x1: ClassInt? = ClassInt(1)
|
|
|
|
val c1_1 = ClassUnderlayingInline(ClassInt(1))
|
|
val c1_2 = ClassUnderlayingInline(ClassInt(1))
|
|
val c2 = ClassUnderlayingInline(ClassInt(2))
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testUnderlyingInline$caseJsEq function=equals
|
|
// CHECK_NEW_COUNT: function=testUnderlyingInline$caseJsEq count=0
|
|
fun caseJsEq() {
|
|
assertTrue(c1_1 == c1_2)
|
|
assertFalse(c1_1 == c2)
|
|
|
|
val a1 = ClassInt(1)
|
|
assertTrue(c1_1.x == a1)
|
|
assertTrue(c1_1.x == c1_2.x)
|
|
assertFalse(a1 == c2.x)
|
|
}
|
|
caseJsEq()
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testUnderlyingInline$caseEquals function=equals
|
|
// CHECK_NEW_COUNT: function=testUnderlyingInline$caseEquals count=2
|
|
fun caseEquals() {
|
|
assertTrue(c1_1 == ClassUnderlayingInline(x1!!))
|
|
assertTrue(ClassUnderlayingInline(x1!!) == c1_2)
|
|
}
|
|
caseEquals()
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testEqualsOverride function=equals
|
|
// CHECK_NEW_COUNT: function=testEqualsOverride count=14
|
|
fun testEqualsOverride() {
|
|
val d1 = ClassWithEqualsOverride(1)
|
|
val d2 = ClassWithEqualsOverride(2)
|
|
val d257 = ClassWithEqualsOverride(257)
|
|
val d1_1: ClassWithEqualsOverride? = ClassWithEqualsOverride(513)
|
|
val d1_2: ClassWithEqualsOverride? = null
|
|
|
|
assertTrue(d1 == d257)
|
|
assertTrue(d1_1 == d1)
|
|
assertTrue(d1 == d1_1)
|
|
|
|
assertFalse(d1 == d2)
|
|
assertFalse(d1_2 == d1)
|
|
assertFalse(d1 == d1_2)
|
|
assertFalse(d1_1 == d1_2)
|
|
}
|
|
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingType count=1
|
|
fun testNullableUnderlyingType() {
|
|
val x0 = ClassNullableInt(0)
|
|
val x1_1 = ClassNullableInt(1)
|
|
val x1_2 = ClassNullableInt(1)
|
|
val x_null_1 = ClassNullableInt(null)
|
|
val x_null_2 = ClassNullableInt(null)
|
|
val nullable_1: ClassNullableInt? = null
|
|
val nullable_2: ClassNullableInt? = ClassNullableInt(null)
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testNullableUnderlyingType$caseEquals function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingType$caseEquals count=0
|
|
fun caseEquals() {
|
|
assertTrue(x0 == x0)
|
|
assertTrue(x0 == ClassNullableInt(0))
|
|
assertTrue(x1_1 == x1_2)
|
|
assertTrue(x_null_1 == x_null_2)
|
|
assertTrue(x_null_1 == ClassNullableInt(null))
|
|
assertTrue(x_null_1.x == null)
|
|
assertTrue(nullable_1 == null)
|
|
assertTrue(nullable_1?.x == null)
|
|
assertTrue(nullable_2!!.x == null)
|
|
assertTrue(nullable_1?.x == nullable_2!!.x)
|
|
|
|
assertFalse(x0 == x_null_2)
|
|
assertFalse(x1_1 == x0)
|
|
assertFalse(x_null_1 == x1_2)
|
|
assertFalse(ClassNullableInt(null) == ClassNullableInt(0))
|
|
assertFalse(nullable_2 == null)
|
|
}
|
|
caseEquals()
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testNullableUnderlyingType$caseJsEq function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingType$caseJsEq count=0
|
|
fun caseJsEq() {
|
|
assertFalse(nullable_1 == nullable_2)
|
|
}
|
|
caseJsEq()
|
|
}
|
|
|
|
// CHECK_NEW_COUNT: function=testUnderlyingWithEqualsOverride count=4 TARGET_BACKENDS=JS_IR
|
|
// CHECK_CALLED_IN_SCOPE: scope=testUnderlyingWithEqualsOverride function=equals
|
|
fun testUnderlyingWithEqualsOverride() {
|
|
val x0 = ClassUnderlayingWithEquals(MyClass(0))
|
|
val x10 = ClassUnderlayingWithEquals(MyClass(10))
|
|
val x1_1 = ClassUnderlayingWithEquals(MyClass(1))
|
|
val x1_2 = ClassUnderlayingWithEquals(MyClass(1))
|
|
|
|
assertTrue(x0 == x0)
|
|
assertTrue(x0 == x10)
|
|
assertTrue(x1_1 == x1_2)
|
|
|
|
assertFalse(x1_1 == x0)
|
|
assertFalse(x10 == x1_2)
|
|
}
|
|
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingInlineClass count=1
|
|
fun testNullableUnderlyingInlineClass() {
|
|
val i1_1: ClassInt? = ClassInt(1)
|
|
val i1_2: ClassInt? = ClassInt(1)
|
|
|
|
val x_null = ClassNullableUnderlayingInline(null)
|
|
val x0 = ClassNullableUnderlayingInline(ClassInt(0))
|
|
val x1_1 = ClassNullableUnderlayingInline(ClassInt(1))
|
|
val x1_2 = ClassNullableUnderlayingInline(ClassInt(1))
|
|
val x1_3 = ClassNullableUnderlayingInline(i1_1)
|
|
val x1_4 = ClassNullableUnderlayingInline(i1_2!!)
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testNullableUnderlyingInlineClass$caseJsEq function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingInlineClass$caseJsEq count=0
|
|
fun caseJsEq() {
|
|
assertTrue(x_null == x_null)
|
|
assertTrue(x1_1 == x1_2)
|
|
assertTrue(x0 == ClassNullableUnderlayingInline(ClassInt(0)))
|
|
assertTrue(x1_1 == x1_3)
|
|
assertTrue(x1_4 == x1_3)
|
|
assertTrue(x1_4 == x1_2)
|
|
assertTrue(x_null == ClassNullableUnderlayingInline(null))
|
|
assertTrue(ClassNullableUnderlayingInline(null) == x_null)
|
|
|
|
assertFalse(x_null == x0)
|
|
assertFalse(x0 == x_null)
|
|
assertFalse(x1_1 == x_null)
|
|
assertFalse(ClassNullableUnderlayingInline(null) == x1_4)
|
|
}
|
|
caseJsEq()
|
|
|
|
// CHECK_NOT_CALLED_IN_SCOPE: scope=testNullableUnderlyingInlineClass$caseEquals function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableUnderlyingInlineClass$caseEquals count=1
|
|
fun caseEquals() {
|
|
assertTrue(ClassNullableUnderlayingInline(i1_1) == ClassNullableUnderlayingInline(i1_2!!))
|
|
}
|
|
caseEquals()
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testInlineClassWithInterface function=equals
|
|
// CHECK_NEW_COUNT: function=testInlineClassWithInterface count=10
|
|
fun testInlineClassWithInterface() {
|
|
val xi_1_1: InterfaceForInlineClass = ClassIntWithInterface(1)
|
|
val xi_1_2: InterfaceForInlineClass = ClassIntWithInterface(1)
|
|
val x_1 = ClassIntWithInterface(1)
|
|
val x_2 = ClassIntWithInterface(2)
|
|
|
|
val yi_foo: InterfaceForInlineClass = ClassStringWithInterface("foo")
|
|
val y_foo = ClassStringWithInterface("foo")
|
|
val y_bar = ClassStringWithInterface("bar")
|
|
|
|
assertTrue(xi_1_1 == x_1)
|
|
assertTrue(xi_1_1 == xi_1_2)
|
|
assertTrue(yi_foo == y_foo)
|
|
assertTrue(y_foo == yi_foo)
|
|
|
|
assertTrue(xi_1_1 as ClassIntWithInterface == x_1)
|
|
assertTrue(x_1 == xi_1_1 as ClassIntWithInterface)
|
|
assertTrue(xi_1_2 as ClassIntWithInterface == xi_1_1 as ClassIntWithInterface)
|
|
assertTrue(xi_1_2 as? ClassIntWithInterface == xi_1_1 as? ClassIntWithInterface)
|
|
|
|
assertFalse(xi_1_1 == x_2)
|
|
assertFalse(yi_foo == y_bar)
|
|
assertFalse(xi_1_2 == yi_foo)
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testCompareDifferentInstancesInSmartCast function=equals
|
|
// CHECK_NEW_COUNT: function=testCompareDifferentInstancesInSmartCast count=6
|
|
@Suppress("EQUALITY_NOT_APPLICABLE")
|
|
fun testCompareDifferentInstancesInSmartCast() {
|
|
val x1_1: Any = ClassInt(1)
|
|
val x1_2 = ClassInt(1)
|
|
|
|
val y1: Any = ClassString("1")
|
|
val y_foo: Any = ClassString("foo")
|
|
|
|
if (x1_1 is ClassInt && y1 is ClassString) {
|
|
assertTrue(x1_2 == x1_1)
|
|
|
|
assertFalse(x1_1 == y1)
|
|
assertFalse(x1_2 == y1)
|
|
}
|
|
if (x1_1 is ClassInt && y_foo is ClassString) {
|
|
assertFalse(x1_1 == y_foo)
|
|
assertFalse(x1_2 == y_foo)
|
|
assertFalse(y1 == y_foo)
|
|
}
|
|
}
|
|
|
|
fun testCompareDifferentInstncesInInlineTemplate() {
|
|
inline fun <reified T, reified S> myEq(x: T, y: S) = x == y
|
|
|
|
// CHECK_NEW_COUNT: function=testCompareDifferentInstncesInInlineTemplate$caseJsEq count=0
|
|
fun caseJsEq() {
|
|
assertTrue(myEq(ClassInt(1), ClassInt(1)))
|
|
assertTrue(myEq(ClassString("foo"), ClassString("foo")))
|
|
|
|
assertFalse(myEq(ClassInt(1), ClassInt(2)))
|
|
assertFalse(myEq(ClassString("foo"), ClassString("bar")))
|
|
}
|
|
caseJsEq()
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testCompareDifferentInstncesInInlineTemplate$caseEquals function=equals
|
|
// CHECK_NEW_COUNT: function=testCompareDifferentInstncesInInlineTemplate$caseEquals count=4
|
|
fun caseEquals() {
|
|
assertFalse(myEq(ClassInt(1), ClassString("bar")))
|
|
assertFalse(myEq(ClassInt(1), ClassString("1")))
|
|
}
|
|
caseEquals()
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testNothing function=equals
|
|
// CHECK_NEW_COUNT: function=testNothing count=0
|
|
fun testNothing() {
|
|
val x_undefined_1 = ClassNothing(undefined)
|
|
val x_undefined_2 = ClassNothing(undefined)
|
|
val x_null_1 = ClassNothing(null)
|
|
val x_null_2 = ClassNothing(null)
|
|
|
|
assertTrue(x_undefined_1 == x_undefined_1)
|
|
assertTrue(x_undefined_1 == x_undefined_2)
|
|
assertTrue(x_undefined_1 == ClassNothing(undefined))
|
|
|
|
assertTrue(x_null_1 == x_null_1)
|
|
assertTrue(x_null_1 == x_null_2)
|
|
assertTrue(x_null_1 == ClassNothing(null))
|
|
|
|
assertTrue(x_undefined_1 == x_null_1)
|
|
assertTrue(x_null_1 == x_undefined_1)
|
|
|
|
assertTrue(x_undefined_1 == ClassNothing(null))
|
|
assertTrue(ClassNothing(null) == x_undefined_1)
|
|
|
|
assertTrue(ClassNothing(undefined) == x_null_1)
|
|
assertTrue(x_null_1 == ClassNothing(undefined))
|
|
}
|
|
|
|
// CHECK_CALLED_IN_SCOPE: scope=testNullableNothing function=equals
|
|
// CHECK_NEW_COUNT: function=testNullableNothing count=2
|
|
fun testNullableNothing() {
|
|
val x_nothing_undefined: ClassNothing? = ClassNothing(undefined)
|
|
val x_nothing_null: ClassNothing? = ClassNothing(null)
|
|
val x_null: ClassNothing? = null
|
|
|
|
assertTrue(x_nothing_undefined == x_nothing_null)
|
|
assertTrue(x_null == null)
|
|
assertTrue(x_null == undefined)
|
|
|
|
assertTrue(null == x_null)
|
|
assertTrue(undefined == x_null)
|
|
|
|
assertFalse(undefined == x_nothing_null)
|
|
assertFalse(x_nothing_null == undefined)
|
|
|
|
assertFalse(undefined == x_nothing_undefined)
|
|
assertFalse(x_nothing_undefined == undefined)
|
|
|
|
assertFalse(x_null == x_nothing_null)
|
|
assertFalse(x_null == x_nothing_undefined)
|
|
assertFalse(x_nothing_null == x_null)
|
|
assertFalse(x_nothing_undefined == x_null)
|
|
}
|
|
|
|
fun box(): String {
|
|
testBasicInt()
|
|
testBasicString()
|
|
testFunctionCall()
|
|
testTypeErasing()
|
|
testTypeErasingAndCast()
|
|
testNullableInstances()
|
|
testUnderlyingInline()
|
|
testEqualsOverride()
|
|
testNullableUnderlyingType()
|
|
testUnderlyingWithEqualsOverride()
|
|
testNullableUnderlyingInlineClass()
|
|
testInlineClassWithInterface()
|
|
testCompareDifferentInstancesInSmartCast()
|
|
testCompareDifferentInstncesInInlineTemplate()
|
|
testNothing()
|
|
testNullableNothing()
|
|
|
|
return "OK"
|
|
}
|