org.jetbrains.kotlin.resolve.multiplatform.ClassicExpectActualMatchingContext.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2023 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 org.jetbrains.kotlin.resolve.multiplatform
import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.descriptors.annotations.AnnotationDescriptor
import org.jetbrains.kotlin.descriptors.annotations.KotlinRetention
import org.jetbrains.kotlin.incremental.components.NoLookupLocation
import org.jetbrains.kotlin.mpp.*
import org.jetbrains.kotlin.name.CallableId
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.resolve.DescriptorUtils
import org.jetbrains.kotlin.resolve.calls.components.ClassicTypeSystemContextForCS
import org.jetbrains.kotlin.resolve.checkers.OptInNames
import org.jetbrains.kotlin.resolve.descriptorUtil.*
import org.jetbrains.kotlin.resolve.multiplatform.K1ExpectActualMatchingContext.AnnotationCallInfo
import org.jetbrains.kotlin.resolve.scopes.DescriptorKindFilter
import org.jetbrains.kotlin.resolve.scopes.MemberScope
import org.jetbrains.kotlin.resolve.scopes.getDescriptorsFiltered
import org.jetbrains.kotlin.types.*
import org.jetbrains.kotlin.types.checker.*
import org.jetbrains.kotlin.types.error.ErrorClassDescriptor
import org.jetbrains.kotlin.types.model.KotlinTypeMarker
import org.jetbrains.kotlin.types.model.TypeConstructorMarker
import org.jetbrains.kotlin.types.model.TypeSubstitutorMarker
import org.jetbrains.kotlin.types.model.TypeSystemContext
import org.jetbrains.kotlin.types.typeUtil.asTypeProjection
import org.jetbrains.kotlin.utils.addToStdlib.UnsafeCastFunction
import org.jetbrains.kotlin.utils.addToStdlib.castAll
import org.jetbrains.kotlin.utils.addToStdlib.shouldNotBeCalled
import org.jetbrains.kotlin.utils.keysToMap
class ClassicExpectActualMatchingContext(
val platformModule: ModuleDescriptor,
/**
* You want to enable this check only in expect-actual matcher checker. And disable everywhere else (especially on backends)
*
* Otherwise, it won't be possible to suppress the compilation error in user code
*/
override val shouldCheckAbsenceOfDefaultParamsInActual: Boolean = false
) : K1ExpectActualMatchingContext,
TypeSystemContext by ClassicTypeSystemContextForCS(platformModule.builtIns, KotlinTypeRefiner.Default) {
override val shouldCheckReturnTypesOfCallables: Boolean
get() = true
private fun DeclarationSymbolMarker.asDescriptor(): DeclarationDescriptor = this as DeclarationDescriptor
private fun CallableSymbolMarker.asDescriptor(): CallableDescriptor = this as CallableDescriptor
private fun FunctionSymbolMarker.asDescriptor(): FunctionDescriptor = this as FunctionDescriptor
private fun PropertySymbolMarker.asDescriptor(): PropertyDescriptor = this as PropertyDescriptor
private fun ValueParameterSymbolMarker.asDescriptor(): ValueParameterDescriptor = this as ValueParameterDescriptor
private fun TypeParameterSymbolMarker.asDescriptor(): TypeParameterDescriptor = this as TypeParameterDescriptor
private fun ClassLikeSymbolMarker.asDescriptor(): ClassifierDescriptorWithTypeParameters = this as ClassifierDescriptorWithTypeParameters
private fun RegularClassSymbolMarker.asDescriptor(): ClassDescriptor = this as ClassDescriptor
private fun TypeAliasSymbolMarker.asDescriptor(): TypeAliasDescriptor = this as TypeAliasDescriptor
private inline fun DeclarationSymbolMarker.safeAsDescriptor(): T? = this as? T
override val RegularClassSymbolMarker.classId: ClassId
get() = (this as ClassifierDescriptor).classId!!
override val TypeAliasSymbolMarker.classId: ClassId
get() = (this as ClassifierDescriptor).classId!!
override val CallableSymbolMarker.callableId: CallableId
get() {
val descriptor = asDescriptor()
return when (val parent = descriptor.containingDeclaration) {
is PackageFragmentDescriptor -> CallableId(parent.fqName, descriptor.name)
is ClassifierDescriptor -> CallableId(parent.classId!!, descriptor.name)
else -> error("Callable descriptor without callableId: $descriptor")
}
}
override val TypeParameterSymbolMarker.parameterName: Name
get() = asDescriptor().name
override val ValueParameterSymbolMarker.parameterName: Name
get() = asDescriptor().name
override fun TypeAliasSymbolMarker.expandToRegularClass(): RegularClassSymbolMarker? {
return asDescriptor().classDescriptor
}
override val RegularClassSymbolMarker.classKind: ClassKind
get() = asDescriptor().kind
override val RegularClassSymbolMarker.isCompanion: Boolean
get() = safeAsDescriptor()?.isCompanionObject == true
override val RegularClassSymbolMarker.isInner: Boolean
get() = asDescriptor().isInner
override val RegularClassSymbolMarker.isInline: Boolean
get() = safeAsDescriptor()?.isInline == true
override val RegularClassSymbolMarker.isValue: Boolean
get() = safeAsDescriptor()?.isValue == true
override val RegularClassSymbolMarker.isFun: Boolean
get() = safeAsDescriptor()?.isFun == true
override val ClassLikeSymbolMarker.typeParameters: List
get() = asDescriptor().declaredTypeParameters
override val ClassLikeSymbolMarker.modality: Modality
get() = asDescriptor().modality
override val ClassLikeSymbolMarker.visibility: Visibility
get() = asDescriptor().visibility.delegate
override val CallableSymbolMarker.modality: Modality?
get() = safeAsDescriptor()?.modality
override val CallableSymbolMarker.visibility: Visibility
get() = asDescriptor().visibility.delegate
override val RegularClassSymbolMarker.superTypes: List
get() = asDescriptor().typeConstructor.supertypes.toList()
override val CallableSymbolMarker.isExpect: Boolean
get() = safeAsDescriptor()?.isExpect == true
override val CallableSymbolMarker.isInline: Boolean
get() = when (this) {
is FunctionDescriptor -> isInline
is PropertyDescriptor -> getter?.isInline == true
else -> false
}
override val CallableSymbolMarker.isSuspend: Boolean
get() = when (this) {
is FunctionDescriptor -> isSuspend
is PropertyDescriptor -> getter?.isSuspend == true
else -> false
}
override val CallableSymbolMarker.isExternal: Boolean
get() = safeAsDescriptor()?.isExternal == true
override val CallableSymbolMarker.isInfix: Boolean
get() = safeAsDescriptor()?.isInfix == true
override val CallableSymbolMarker.isOperator: Boolean
get() = safeAsDescriptor()?.isOperator == true
override val CallableSymbolMarker.isTailrec: Boolean
get() = safeAsDescriptor()?.isTailrec == true
override val PropertySymbolMarker.isVar: Boolean
get() = asDescriptor().isVar
override val PropertySymbolMarker.isLateinit: Boolean
get() = asDescriptor().isLateInit
override val PropertySymbolMarker.isConst: Boolean
get() = asDescriptor().isConst
override val PropertySymbolMarker.getter: FunctionSymbolMarker?
get() = asDescriptor().getter
override val PropertySymbolMarker.setter: FunctionSymbolMarker?
get() = asDescriptor().setter
@OptIn(UnsafeCastFunction::class)
override fun createExpectActualTypeParameterSubstitutor(
expectTypeParameters: List,
actualTypeParameters: List,
parentSubstitutor: TypeSubstitutorMarker?,
): TypeSubstitutorMarker {
val expectParameters = expectTypeParameters.castAll()
val actualParameters = actualTypeParameters.castAll()
val substitutor = TypeSubstitutor.create(
TypeConstructorSubstitution.createByParametersMap(expectParameters.keysToMap {
actualParameters[it.index].defaultType.asTypeProjection()
})
)
return when (parentSubstitutor) {
null -> substitutor
is TypeSubstitutor -> TypeSubstitutor.createChainedSubstitutor(parentSubstitutor.substitution, substitutor.substitution)
else -> error("Unsupported substitutor type: $parentSubstitutor")
}
}
override fun RegularClassSymbolMarker.collectAllMembers(isActualDeclaration: Boolean): List {
return asDescriptor().getMembers(name = null)
}
override fun RegularClassSymbolMarker.getMembersForExpectClass(name: Name): List {
return asDescriptor().getMembers(name)
}
private fun ClassDescriptor.getMembers(name: Name? = null): List {
val nameFilter = if (name != null) { it -> it == name } else MemberScope.ALL_NAME_FILTER
return defaultType.memberScope
.getDescriptorsFiltered(nameFilter = nameFilter)
.filterIsInstance()
.filterNot(DescriptorUtils::isEnumEntry)
.plus(constructors.filter { nameFilter(it.name) })
}
override fun RegularClassSymbolMarker.collectEnumEntryNames(): List {
return collectEnumEntries().map { it.name }
}
override fun RegularClassSymbolMarker.collectEnumEntries(): List {
return asDescriptor()
.unsubstitutedMemberScope
.getDescriptorsFiltered()
.filter(DescriptorUtils::isEnumEntry)
}
override val CallableSymbolMarker.dispatchReceiverType: KotlinTypeMarker?
get() = asDescriptor().dispatchReceiverParameter?.type
override val CallableSymbolMarker.extensionReceiverType: KotlinTypeMarker?
get() = asDescriptor().extensionReceiverParameter?.type
override val CallableSymbolMarker.returnType: KotlinTypeMarker
get() = asDescriptor().returnType!!
override val CallableSymbolMarker.typeParameters: List
get() = asDescriptor().typeParameters
override val FunctionSymbolMarker.valueParameters: List
get() = asDescriptor().valueParameters
override val ValueParameterSymbolMarker.isVararg: Boolean
get() = asDescriptor().varargElementType != null
override val ValueParameterSymbolMarker.isNoinline: Boolean
get() = asDescriptor().isNoinline
override val ValueParameterSymbolMarker.isCrossinline: Boolean
get() = asDescriptor().isCrossinline
override val ValueParameterSymbolMarker.hasDefaultValue: Boolean
get() = asDescriptor().declaresDefaultValue()
override fun FunctionSymbolMarker.allOverriddenDeclarationsRecursive(): Sequence =
(sequenceOf(asDescriptor()) + asDescriptor().overriddenTreeAsSequence(useOriginal = true))
// Tests work even if you don't filter out fake-overrides. Filtering fake-overrides is needed because
// the returned descriptors are compared by `equals`. And `equals` for fake-overrides is weird.
// I didn't manage to invent a test that would check this condition
.filter { it.kind != CallableMemberDescriptor.Kind.FAKE_OVERRIDE && it.kind != CallableMemberDescriptor.Kind.DELEGATION }
override fun CallableSymbolMarker.isAnnotationConstructor(): Boolean {
val descriptor = safeAsDescriptor() ?: return false
return DescriptorUtils.isAnnotationClass(descriptor.constructedClass)
}
override val TypeParameterSymbolMarker.bounds: List
get() = asDescriptor().upperBounds
override val TypeParameterSymbolMarker.variance: Variance
get() = asDescriptor().variance
override val TypeParameterSymbolMarker.isReified: Boolean
get() = asDescriptor().isReified
override fun areCompatibleExpectActualTypes(expectType: KotlinTypeMarker?, actualType: KotlinTypeMarker?): Boolean {
if (expectType == null) return actualType == null
if (actualType == null) return false
require(expectType is KotlinType && actualType is KotlinType)
return if (platformModule.isTypeRefinementEnabled()) {
areCompatibleTypesViaTypeRefinement(expectType, actualType)
} else {
areCompatibleTypesViaTypeContext(expectType, actualType)
}
}
override val RegularClassSymbolMarker.defaultType: KotlinTypeMarker
get() = asDescriptor().defaultType
override fun actualTypeIsSubtypeOfExpectType(expectType: KotlinTypeMarker, actualType: KotlinTypeMarker): Boolean {
shouldNotBeCalled("Checking for subtyping is used only in FIR and IR implementations")
}
@OptIn(TypeRefinement::class)
private fun areCompatibleTypesViaTypeRefinement(a: KotlinType, b: KotlinType): Boolean {
val typeRefinerForPlatformModule = platformModule.getKotlinTypeRefiner().let { moduleRefiner ->
if (moduleRefiner is KotlinTypeRefiner.Default)
KotlinTypeRefinerImpl.createStandaloneInstanceFor(platformModule)
else
moduleRefiner
}
return areCompatibleTypes(
a, b,
typeSystemContext = SimpleClassicTypeSystemContext,
kotlinTypeRefiner = typeRefinerForPlatformModule,
)
}
private fun areCompatibleTypesViaTypeContext(a: KotlinType, b: KotlinType): Boolean {
val typeSystemContext = object : ClassicTypeSystemContext {
override fun areEqualTypeConstructors(c1: TypeConstructorMarker, c2: TypeConstructorMarker): Boolean {
require(c1 is TypeConstructor)
require(c2 is TypeConstructor)
return isExpectedClassAndActualTypeAlias(c1, c2, platformModule) ||
isExpectedClassAndActualTypeAlias(c2, c1, platformModule) ||
super.areEqualTypeConstructors(c1, c2)
}
}
return areCompatibleTypes(
a, b,
typeSystemContext = typeSystemContext,
kotlinTypeRefiner = KotlinTypeRefiner.Default,
)
}
private fun areCompatibleTypes(
a: KotlinType,
b: KotlinType,
typeSystemContext: ClassicTypeSystemContext,
kotlinTypeRefiner: KotlinTypeRefiner,
): Boolean {
with(NewKotlinTypeCheckerImpl(kotlinTypeRefiner)) {
return createClassicTypeCheckerState(
isErrorTypeEqualsToAnything = false,
typeSystemContext = typeSystemContext,
kotlinTypeRefiner = kotlinTypeRefiner,
).equalTypes(a.unwrap(), b.unwrap())
}
}
// For example, expectedTypeConstructor may be the expected class kotlin.text.StringBuilder, while actualTypeConstructor
// is java.lang.StringBuilder. For the purposes of type compatibility checking, we must consider these types equal here.
// Note that the case of an "actual class" works as expected though, because the actual class by definition has the same FQ name
// as the corresponding expected class, so their type constructors are equal as per AbstractClassTypeConstructor#equals
private fun isExpectedClassAndActualTypeAlias(
expectedTypeConstructor: TypeConstructor,
actualTypeConstructor: TypeConstructor,
platformModule: ModuleDescriptor
): Boolean {
val expected = expectedTypeConstructor.declarationDescriptor
val actual = actualTypeConstructor.declarationDescriptor
return expected is ClassifierDescriptorWithTypeParameters &&
expected.isExpect &&
actual is ClassifierDescriptorWithTypeParameters &&
findClassifiersFromModule(expected.classId, platformModule, moduleFilter = ALL_MODULES).any { classifier ->
// Note that it's fine to only check that this "actual typealias" expands to the expected class, without checking
// whether the type arguments in the expansion are in the correct order or have the correct variance, because we only
// allow simple cases like "actual typealias Foo = FooImpl", see DeclarationsChecker#checkActualTypeAlias
(classifier as? TypeAliasDescriptor)?.classDescriptor == actual
}
}
fun findClassifiersFromModule(
classId: ClassId?,
module: ModuleDescriptor,
moduleFilter: (ModuleDescriptor) -> Boolean
): Collection {
if (classId == null) return emptyList()
fun MemberScope.getAllClassifiers(name: Name): Collection =
getDescriptorsFiltered(DescriptorKindFilter.CLASSIFIERS) { it == name }
.filterIsInstance()
val segments = classId.relativeClassName.pathSegments()
var classifiers = module.getPackage(classId.packageFqName).memberScope.getAllClassifiers(segments.first())
classifiers = classifiers.applyFilter(moduleFilter)
for (name in segments.subList(1, segments.size)) {
classifiers = classifiers.mapNotNull { classifier ->
(classifier as? ClassDescriptor)?.unsubstitutedInnerClassesScope?.getContributedClassifier(
name, NoLookupLocation.FOR_ALREADY_TRACKED
) as? ClassifierDescriptorWithTypeParameters
}
}
return classifiers
}
override fun RegularClassSymbolMarker.isNotSamInterface(): Boolean {
val descriptor = asDescriptor()
return descriptor.isDefinitelyNotSamInterface || descriptor.defaultFunctionTypeForSamInterface == null
}
override fun CallableSymbolMarker.shouldSkipMatching(containingExpectClass: RegularClassSymbolMarker): Boolean {
return safeAsDescriptor()?.kind?.isReal == false
}
override val CallableSymbolMarker.hasStableParameterNames: Boolean
get() = asDescriptor().hasStableParameterNames()
override val DeclarationSymbolMarker.annotations: List
get() = asDescriptor().annotations.map(::AnnotationCallInfoImpl)
override fun areAnnotationArgumentsEqual(
expectAnnotation: AnnotationCallInfo,
actualAnnotation: AnnotationCallInfo,
collectionArgumentsCompatibilityCheckStrategy: K1ExpectActualCollectionArgumentsCompatibilityCheckStrategy,
): Boolean {
fun AnnotationCallInfo.getDescriptor(): AnnotationDescriptor = (this as AnnotationCallInfoImpl).annotationDescriptor
return areExpressionConstValuesEqual(
expectAnnotation.getDescriptor(),
actualAnnotation.getDescriptor(),
collectionArgumentsCompatibilityCheckStrategy,
)
}
private inner class AnnotationCallInfoImpl(
val annotationDescriptor: AnnotationDescriptor,
) : AnnotationCallInfo {
override val annotationSymbol: AnnotationDescriptor = annotationDescriptor
override val classId: ClassId?
get() = getAnnotationClassDescriptor()?.classId
override val isRetentionSource: Boolean
get() = getAnnotationClassDescriptor()?.getAnnotationRetention() == KotlinRetention.SOURCE
override val isOptIn: Boolean
get() = getAnnotationClassDescriptor()?.annotations?.hasAnnotation(OptInNames.REQUIRES_OPT_IN_FQ_NAME) ?: false
private fun getAnnotationClassDescriptor(): ClassDescriptor? {
val classDescriptor = annotationDescriptor.annotationClass ?: return null
if (classDescriptor is ErrorClassDescriptor) {
return null
}
if (!classDescriptor.isExpect) {
return classDescriptor
}
val classId = classDescriptor.classId
return findExpandedExpectClassInPlatformModule(classId) ?: classDescriptor
}
}
// For IDE composite module analysis, when actual class may differ
internal fun findExpandedExpectClassInPlatformModule(originalClassId: ClassId): ClassDescriptor? {
val classifier = platformModule.findClassifierAcrossModuleDependencies(originalClassId)
return when (classifier) {
is TypeAliasDescriptor -> classifier.classDescriptor
is ClassDescriptor -> classifier
else -> null
}
}
override val DeclarationSymbolMarker.hasSourceAnnotationsErased: Boolean
get() {
return DescriptorUtils.getContainingSourceFile(asDescriptor()) == SourceFile.NO_SOURCE_FILE &&
this !is K1SyntheticClassifierSymbolMarker &&
!(this is CallableMemberDescriptor && kind == CallableMemberDescriptor.Kind.SYNTHESIZED)
}
override val checkClassScopesForAnnotationCompatibility = true
override fun skipCheckingAnnotationsOfActualClassMember(actualMember: DeclarationSymbolMarker): Boolean =
(actualMember as MemberDescriptor).isActual
override fun findPotentialExpectClassMembersForActual(
expectClass: RegularClassSymbolMarker,
actualClass: RegularClassSymbolMarker,
actualMember: DeclarationSymbolMarker,
checkClassScopesCompatibility: Boolean,
): Map> {
val compatibilityToExpects = ExpectedActualResolver.findExpectForActualClassMember(
actualMember as MemberDescriptor,
actualClass as ClassDescriptor,
expectClass as ClassDescriptor,
checkClassScopesCompatibility,
this,
)
return buildMap {
for ((compatibility, expectMembers) in compatibilityToExpects.entries) {
for (expectMember in expectMembers) {
val oldValue = put(expectMember, compatibility)
if (oldValue != null) {
error(
"Several incompatibilities correspond to the same expect symbol: symbol=$expectMember, " +
"compatibilities=$oldValue, $compatibility"
)
}
}
}
}
}
override fun DeclarationSymbolMarker.getSourceElement(): SourceElementMarker {
return ClassicSourceElement((asDescriptor() as? DeclarationDescriptorWithSource)?.source)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy