Reuse captured arguments for flexible type's bounds properly, by equality of type constructors modulo mutability and type argument

^KT-43630 Fixed
This commit is contained in:
Victor Petukhov
2020-12-01 17:24:50 +03:00
parent 9f58e4bcfe
commit d25ad269e0
8 changed files with 200 additions and 23 deletions
@@ -10659,6 +10659,16 @@ public class FirOldFrontendDiagnosticsTestGenerated extends AbstractFirOldFronte
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/captureTypeOnlyOnTopLevel.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentBounds.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentBounds() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentBounds.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentConstructors.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentConstructors() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentConstructors.kt");
}
@TestMetadata("capturedType.kt")
public void testCapturedType() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedType.kt");
@@ -0,0 +1,23 @@
// FILE: Bar.java
// !DIAGNOSTICS: -UNUSED_PARAMETER
// SKIP_TXT
public class Bar<K, N> { }
// FILE: Foo.java
public class Foo<P> extends Bar<Integer, Integer> {
public static final Bar bar = null;
}
// FILE: main.kt
fun <P> takeFoo(foo: Foo<P>) {}
fun main(x: Foo<*>?) {
val y = Foo.bar
if (y !is Foo<*>?) return
if (y == null) return
if (x != y) return
takeFoo(x) // Here we capture `{Bar<Any!, Any!> & Foo<*>}..Foo<*>?`
}
@@ -0,0 +1,23 @@
// FILE: Bar.java
// !DIAGNOSTICS: -UNUSED_PARAMETER
// SKIP_TXT
public class Bar<K, N> { }
// FILE: Foo.java
public class Foo<P> extends Bar<Integer, Integer> {
public static final Bar bar = null;
}
// FILE: main.kt
fun <P> takeFoo(foo: Foo<P>) {}
fun main(x: Foo<*>?) {
val y = Foo.bar
if (y !is Foo<*>?) return
if (y == null) return
if (x != y) return
takeFoo(<!DEBUG_INFO_SMARTCAST!>x<!>) // Here we capture `{Bar<Any!, Any!> & Foo<*>}..Foo<*>?`
}
@@ -0,0 +1,29 @@
// FULL_JDK
// !DIAGNOSTICS: -UNUSED_PARAMETER
// SKIP_TXT
// FILE: Bar.java
public class Bar<K, N> { }
// FILE: Foo.java
import java.util.List;
public class Foo<P> extends Bar<Integer, Integer> {
public static final List<?> bar = null;
}
// FILE: main.kt
fun <P> takeFoo(foo: Foo<P>) {}
fun main(x: Foo<*>?) {
val y = Foo.bar
if (y !is Foo<*>?) return
if (y == null) return
if (x != y) return
// Here we capture `({Foo<*> & MutableList<*>}..{Foo<*>? & List<*>?})`
// `*` inside `MutableList` and `List` have to become the same captured type
takeFoo(x)
}
@@ -0,0 +1,29 @@
// FULL_JDK
// !DIAGNOSTICS: -UNUSED_PARAMETER
// SKIP_TXT
// FILE: Bar.java
public class Bar<K, N> { }
// FILE: Foo.java
import java.util.List;
public class Foo<P> extends Bar<Integer, Integer> {
public static final List<?> bar = null;
}
// FILE: main.kt
fun <P> takeFoo(foo: Foo<P>) {}
fun main(x: Foo<*>?) {
val y = Foo.bar
if (y !is Foo<*>?) return
if (y == null) return
if (x != y) return
// Here we capture `({Foo<*> & MutableList<*>}..{Foo<*>? & List<*>?})`
// `*` inside `MutableList` and `List` have to become the same captured type
takeFoo(<!DEBUG_INFO_SMARTCAST!>x<!>)
}
@@ -10666,6 +10666,16 @@ public class DiagnosticsTestGenerated extends AbstractDiagnosticsTestWithFirVali
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/captureTypeOnlyOnTopLevel.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentBounds.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentBounds() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentBounds.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentConstructors.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentConstructors() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentConstructors.kt");
}
@TestMetadata("capturedType.kt")
public void testCapturedType() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedType.kt");
@@ -10661,6 +10661,16 @@ public class DiagnosticsUsingJavacTestGenerated extends AbstractDiagnosticsUsing
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/captureTypeOnlyOnTopLevel.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentBounds.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentBounds() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentBounds.kt");
}
@TestMetadata("capturedFlexibleIntersectionTypesWithDifferentConstructors.kt")
public void testCapturedFlexibleIntersectionTypesWithDifferentConstructors() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedFlexibleIntersectionTypesWithDifferentConstructors.kt");
}
@TestMetadata("capturedType.kt")
public void testCapturedType() throws Exception {
runTest("compiler/testData/diagnostics/tests/inference/capturedTypes/capturedType.kt");
@@ -23,12 +23,30 @@ import org.jetbrains.kotlin.descriptors.annotations.Annotations
import org.jetbrains.kotlin.resolve.calls.inference.CapturedTypeConstructor
import org.jetbrains.kotlin.resolve.scopes.MemberScope
import org.jetbrains.kotlin.types.*
import org.jetbrains.kotlin.types.FlexibleTypeBoundsChecker.areTypesMayBeLowerAndUpperBoundsOfSameFlexibleTypeByMutability
import org.jetbrains.kotlin.types.model.CaptureStatus
import org.jetbrains.kotlin.types.model.CapturedTypeMarker
import org.jetbrains.kotlin.types.refinement.TypeRefinement
import org.jetbrains.kotlin.types.typeUtil.asTypeProjection
import org.jetbrains.kotlin.types.typeUtil.builtIns
private class CapturedArguments(val capturedArguments: List<TypeProjection>, private val originalType: KotlinType) {
fun isSuitableForType(type: KotlinType): Boolean {
val areArgumentsMatched = type.arguments.withIndex().all { (i, typeArgumentsType) ->
originalType.arguments.size > i && typeArgumentsType == originalType.arguments[i]
}
if (!areArgumentsMatched) return false
val areConstructorsMatched = originalType.constructor == type.constructor
|| areTypesMayBeLowerAndUpperBoundsOfSameFlexibleTypeByMutability(originalType, type)
if (!areConstructorsMatched) return false
return true
}
}
// null means that type should be leaved as is
fun prepareArgumentTypeRegardingCaptureTypes(argumentType: UnwrappedType): UnwrappedType? {
return if (argumentType is NewCapturedType) null else captureFromExpression(argumentType)
@@ -42,33 +60,44 @@ fun captureFromExpression(type: UnwrappedType): UnwrappedType? {
}
/*
* We capture intersection types in two stages:
* capture type arguments for each component and replace it in the original type after that.
* This is to substitute captured types into flexible types properly:
* we should have the same captured types both for lower bound and for upper one.
*
* Example:
* The original type: ({Comparable<*> & java.io.Serializable}..{Comparable<*>? & java.io.Serializable?})
* Result of capturing arguments by components: [[CapturedType(*)], null]
* The resulting type: ({Comparable<CapturedType(*)> & java.io.Serializable}..{Comparable<CapturedType(*)>? & java.io.Serializable?})
* We capture arguments in the intersection types in specific way:
* 1) Firstly, we create captured arguments for all type arguments grouped by a type constructor* and a type argument's type.
* It means, that we create only one captured argument for two types `Foo<*>` and `Foo<*>?` within a flexible type, for instance.
* * In addition to grouping by type constructors, we look at possibility locating of two types in different bounds of the same flexible type.
* This is necessary in order to create the same captured arguments,
* for example, for `MutableList` in the lower bound of the flexible type and for `List` in the upper one.
* Example: MutableList<*>..List<*>? -> MutableList<Captured1(*)>..List<Captured2(*)>?, Captured1(*) and Captured2(*) are the same.
* 2) Secondly, we replace type arguments with captured arguments by given a type constructor and type arguments.
*/
val capturedArgumentsByComponents = captureArgumentsForIntersectionType(typeConstructor) ?: return null
val capturedArgumentsByComponents = captureArgumentsForIntersectionType(type) ?: return null
fun replaceArgumentsByComponents(typeToReplace: UnwrappedType) =
typeToReplace.constructor.supertypes.mapIndexed { i, componentType ->
val capturedArguments = capturedArgumentsByComponents[i] ?: return@mapIndexed componentType.asSimpleType()
componentType.unwrap().replaceArguments(capturedArguments)
// We reuse `TypeToCapture` for some types, suitability to reuse defines by `isSuitableForType`
fun findCorrespondingCapturedArgumentsForType(type: KotlinType) =
capturedArgumentsByComponents.find { typeToCapture -> typeToCapture.isSuitableForType(type) }?.capturedArguments
fun replaceArgumentsWithCapturedArgumentsByIntersectionComponents(typeToReplace: UnwrappedType): List<SimpleType> {
return if (typeToReplace.constructor is IntersectionTypeConstructor) {
typeToReplace.constructor.supertypes.map { componentType ->
val capturedArguments = findCorrespondingCapturedArgumentsForType(componentType)
?: return@map componentType.asSimpleType()
componentType.unwrap().replaceArguments(capturedArguments)
}
} else {
val capturedArguments = findCorrespondingCapturedArgumentsForType(typeToReplace)
?: return listOf(typeToReplace.asSimpleType())
listOf(typeToReplace.unwrap().replaceArguments(capturedArguments))
}
}
return if (type is FlexibleType) {
val lowerIntersectedType =
intersectTypes(replaceArgumentsByComponents(type.lowerBound)).makeNullableAsSpecified(type.lowerBound.isMarkedNullable)
val upperIntersectedType =
intersectTypes(replaceArgumentsByComponents(type.upperBound)).makeNullableAsSpecified(type.upperBound.isMarkedNullable)
val lowerIntersectedType = intersectTypes(replaceArgumentsWithCapturedArgumentsByIntersectionComponents(type.lowerBound))
.makeNullableAsSpecified(type.lowerBound.isMarkedNullable)
val upperIntersectedType = intersectTypes(replaceArgumentsWithCapturedArgumentsByIntersectionComponents(type.upperBound))
.makeNullableAsSpecified(type.upperBound.isMarkedNullable)
KotlinTypeFactory.flexibleType(lowerIntersectedType, upperIntersectedType)
} else {
intersectTypes(replaceArgumentsByComponents(type)).makeNullableAsSpecified(type.isMarkedNullable)
intersectTypes(replaceArgumentsWithCapturedArgumentsByIntersectionComponents(type)).makeNullableAsSpecified(type.isMarkedNullable)
}
}
@@ -76,15 +105,29 @@ fun captureFromExpression(type: UnwrappedType): UnwrappedType? {
internal fun captureFromArguments(type: SimpleType, status: CaptureStatus) =
captureArguments(type, status)?.let { type.replaceArguments(it) }
private fun captureArgumentsForIntersectionType(typeConstructor: TypeConstructor): List<List<TypeProjection>?>? {
private fun captureArgumentsForIntersectionType(type: KotlinType): List<CapturedArguments>? {
// It's possible to have one of the bounds as non-intersection type
fun getTypesToCapture(type: KotlinType) =
if (type.constructor is IntersectionTypeConstructor) type.constructor.supertypes else listOf(type)
val filteredTypesToCapture =
if (type is FlexibleType) {
val typesToCapture = getTypesToCapture(type.lowerBound) + getTypesToCapture(type.upperBound)
typesToCapture.distinctBy { (FlexibleTypeBoundsChecker.getBaseBoundFqNameByMutability(it) ?: it.constructor) to it.arguments }
} else type.constructor.supertypes
var changed = false
val capturedArgumentsByComponents = typeConstructor.supertypes.map { supertype ->
captureArguments(supertype.unwrap(), CaptureStatus.FROM_EXPRESSION)?.apply { changed = true }
val capturedArgumentsByTypes = filteredTypesToCapture.mapNotNull { typeToCapture ->
val capturedArguments = captureArguments(typeToCapture.unwrap(), CaptureStatus.FROM_EXPRESSION)
?: return@mapNotNull null
changed = true
CapturedArguments(capturedArguments, originalType = typeToCapture)
}
if (!changed) return null
return capturedArgumentsByComponents
return capturedArgumentsByTypes
}
private fun captureFromArguments(type: UnwrappedType, status: CaptureStatus): UnwrappedType? {