org.jetbrains.kotlin.ir.overrides.CopyIrTreeWithSymbolsForFakeOverrides.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2020 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.ir.overrides
import org.jetbrains.kotlin.ir.declarations.*
import org.jetbrains.kotlin.ir.symbols.IrClassifierSymbol
import org.jetbrains.kotlin.ir.symbols.IrTypeParameterSymbol
import org.jetbrains.kotlin.ir.symbols.IrValueParameterSymbol
import org.jetbrains.kotlin.ir.types.*
import org.jetbrains.kotlin.ir.types.impl.buildSimpleType
import org.jetbrains.kotlin.ir.types.impl.makeTypeProjection
import org.jetbrains.kotlin.ir.util.TypeRemapper
import org.jetbrains.kotlin.ir.util.copyAnnotations
import org.jetbrains.kotlin.ir.util.findAnnotation
import org.jetbrains.kotlin.ir.util.render
import org.jetbrains.kotlin.name.StandardClassIds.Annotations.EnhancedNullability
import org.jetbrains.kotlin.name.StandardClassIds.Annotations.FlexibleNullability
import org.jetbrains.kotlin.utils.addIfNotNull
import org.jetbrains.kotlin.utils.memoryOptimizedMap
class CopyIrTreeWithSymbolsForFakeOverrides(
private val overridableMember: IrOverridableMember,
private val substitution: Map,
private val parentClass: IrClass,
private val unimplementedOverridesStrategy: IrUnimplementedOverridesStrategy
) {
fun copy(): IrOverridableMember {
val typeParameters = HashMap()
val valueParameters = HashMap()
val copier = FakeOverrideCopier(
valueParameters,
typeParameters,
FakeOverrideTypeRemapper(typeParameters, substitution),
parentClass,
unimplementedOverridesStrategy
)
return when (overridableMember) {
is IrSimpleFunction -> copier.copySimpleFunction(overridableMember)
is IrProperty -> copier.copyProperty(overridableMember)
else -> error("Unsupported member: ${overridableMember.render()}")
}
}
private class FakeOverrideTypeRemapper(
val typeParameters: Map,
val substitution: Map
) : TypeRemapper {
override fun enterScope(irTypeParametersContainer: IrTypeParametersContainer) {}
override fun leaveScope() {}
private fun remapTypeArguments(arguments: List) =
arguments.memoryOptimizedMap { argument ->
(argument as? IrTypeProjection)?.let { makeTypeProjection(remapType(it.type), it.variance) }
?: argument
}
private fun IrType.mergeTypeAnnotations(other: IrType): IrType {
// Flexible types are represented as a type annotation in IR, so we need to keep it when substituting type during override.
// Note that it's incorrect to merge _all_ type annotations though, because for a Collection subclass:
//
// abstract class Z : Collection
//
// `Z.contains` should have the signature `(Int) -> Boolean`, NOT `(@UnsafeVariance Int) -> Boolean` which would occur if we
// copied all type annotations.
return addAnnotations(buildList {
for (fqName in TYPE_ANNOTATIONS_TO_MERGE) {
addIfNotNull(other.annotations.findAnnotation(fqName))
}
})
}
override fun remapType(type: IrType): IrType {
if (type !is IrSimpleType) return type
return when (val substitutedType = substitution[type.classifier]) {
is IrDynamicType -> substitutedType
is IrSimpleType -> substitutedType.mergeNullability(type).mergeTypeAnnotations(type)
else -> type.buildSimpleType {
kotlinType = null
classifier = remapClassifier(type.classifier)
arguments = remapTypeArguments(type.arguments)
annotations = type.copyAnnotations()
}
}
}
private fun remapClassifier(classifier: IrClassifierSymbol): IrClassifierSymbol =
if (classifier is IrTypeParameterSymbol)
typeParameters.getOrElse(classifier) { classifier }
else
classifier
}
private companion object {
// TODO (KT-64715): RawTypeAnnotation, FlexibleMutability, RawType, FlexibleArrayElementVariance?
val TYPE_ANNOTATIONS_TO_MERGE = listOf(
FlexibleNullability.asSingleFqName(),
EnhancedNullability.asSingleFqName(),
)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy