All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.types.TypeParameterUpperBoundEraser.kt Maven / Gradle / Ivy

There is a newer version: 2.0.0
Show newest version
/*
 * Copyright 2010-2021 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.types

import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.TypeParameterDescriptor
import org.jetbrains.kotlin.storage.LockBasedStorageManager
import org.jetbrains.kotlin.types.TypeUtils.makeStarProjection
import org.jetbrains.kotlin.types.checker.intersectTypes
import org.jetbrains.kotlin.types.error.ErrorUtils
import org.jetbrains.kotlin.types.error.ErrorTypeKind
import org.jetbrains.kotlin.types.typeUtil.*

class TypeParameterUpperBoundEraser(
    val projectionComputer: ErasureProjectionComputer,
    val options: TypeParameterErasureOptions = TypeParameterErasureOptions(leaveNonTypeParameterTypes = false, intersectUpperBounds = false)
) {
    private val storage = LockBasedStorageManager("Type parameter upper bound erasure results")

    private val erroneousErasedBound by lazy {
        ErrorUtils.createErrorType(ErrorTypeKind.CANNOT_COMPUTE_ERASED_BOUND, this.toString())
    }

    private data class DataToEraseUpperBound(
        val typeParameter: TypeParameterDescriptor,
        val typeAttr: ErasureTypeAttributes
    ) {
        override fun equals(other: Any?): Boolean {
            if (other !is DataToEraseUpperBound) return false
            return other.typeParameter == this.typeParameter && other.typeAttr == this.typeAttr
        }

        override fun hashCode(): Int {
            var result = typeParameter.hashCode()
            result += 31 * result + typeAttr.hashCode()
            return result
        }
    }

    private val getErasedUpperBound = storage.createMemoizedFunction {
        with(it) { getErasedUpperBoundInternal(typeParameter, typeAttr) }
    }

    fun getErasedUpperBound(
        typeParameter: TypeParameterDescriptor,
        typeAttr: ErasureTypeAttributes
    ): KotlinType = getErasedUpperBound(DataToEraseUpperBound(typeParameter, typeAttr))

    private fun getDefaultType(typeAttr: ErasureTypeAttributes) =
        typeAttr.defaultType?.replaceArgumentsWithStarProjections() ?: erroneousErasedBound

    // Definition:
    // ErasedUpperBound(T : G) = G<*> // UpperBound(T) is a type G with arguments
    // ErasedUpperBound(T : A) = A // UpperBound(T) is a type A without arguments
    // ErasedUpperBound(T : F) = UpperBound(F) // UB(T) is another type parameter F
    private fun getErasedUpperBoundInternal(
        // Calculation of `potentiallyRecursiveTypeParameter.upperBounds` may recursively depend on `this.getErasedUpperBound`
        // E.g. `class A`
        // To prevent recursive calls return defaultValue() instead
        typeParameter: TypeParameterDescriptor,
        typeAttr: ErasureTypeAttributes
    ): KotlinType {
        val visitedTypeParameters = typeAttr.visitedTypeParameters

        if (visitedTypeParameters != null && typeParameter.original in visitedTypeParameters)
            return getDefaultType(typeAttr)

        /*
         * We should do erasure of containing type parameters with their erasure to avoid creating inconsistent types.
         * E.g. for `class Foo, B>`, we'd have erasure for lower bound: Foo, Any>,
         * but it's wrong type: projection(*) != projection(Any).
         * So we should substitute erasure of the corresponding type parameter: `Foo, Any>` or `Foo, *>`.
         */
        val erasedTypeParameters = typeParameter.defaultType.extractTypeParametersFromUpperBounds(visitedTypeParameters).associate {
            val boundProjection = if (visitedTypeParameters == null || it !in visitedTypeParameters) {
                projectionComputer.computeProjection(
                    it,
                    typeAttr,
                    typeParameterUpperBoundEraser = this,
                    getErasedUpperBound(it, typeAttr.withNewVisitedTypeParameter(typeParameter))
                )
            } else makeStarProjection(it, typeAttr)

            it.typeConstructor to boundProjection
        }
        val erasedTypeParametersSubstitutor =
            TypeSubstitutor.create(TypeConstructorSubstitution.createByConstructorsMap(erasedTypeParameters))
        val erasedUpperBounds =
            erasedTypeParametersSubstitutor.substituteErasedUpperBounds(typeParameter.upperBounds, typeAttr)

        if (erasedUpperBounds.isNotEmpty()) {
            if (!options.intersectUpperBounds) {
                require(erasedUpperBounds.size == 1) { "Should only be one computed upper bound if no need to intersect all bounds" }
                return erasedUpperBounds.single()
            } else {
                @Suppress("UNCHECKED_CAST")
                return intersectTypes(erasedUpperBounds.toList().map { it.unwrap() })
            }
        }

        return getDefaultType(typeAttr)
    }

    private fun TypeSubstitutor.substituteErasedUpperBounds(
        upperBounds: List,
        typeAttr: ErasureTypeAttributes
    ): Set = buildSet {
        for (upperBound in upperBounds) {
            when (val declaration = upperBound.constructor.declarationDescriptor) {
                is ClassDescriptor -> add(
                    upperBound.replaceArgumentsOfUpperBound(
                        substitutor = this@substituteErasedUpperBounds,
                        typeAttr.visitedTypeParameters,
                        options.leaveNonTypeParameterTypes
                    )
                )
                is TypeParameterDescriptor -> {
                    if (typeAttr.visitedTypeParameters?.contains(declaration) == true) {
                        add(getDefaultType(typeAttr))
                    } else {
                        addAll(substituteErasedUpperBounds(declaration.upperBounds, typeAttr))
                    }
                }
            }

            // If no need to intersect, take only the first bound
            if (!options.intersectUpperBounds) break
        }
    }

    companion object {
        fun KotlinType.replaceArgumentsOfUpperBound(
            substitutor: TypeSubstitutor,
            visitedTypeParameters: Set?,
            leaveNonTypeParameterTypes: Boolean = false
        ): KotlinType {
            val replacedArguments = replaceArgumentsByParametersWith { typeParameterDescriptor ->
                val argument = arguments.getOrNull(typeParameterDescriptor.index)
                if (leaveNonTypeParameterTypes && argument?.type?.containsTypeParameter() == false)
                    return@replaceArgumentsByParametersWith argument

                val isTypeParameterVisited = visitedTypeParameters != null && typeParameterDescriptor in visitedTypeParameters

                if (argument == null || isTypeParameterVisited || substitutor.substitution[argument.type] == null) {
                    StarProjectionImpl(typeParameterDescriptor)
                } else {
                    argument
                }
            }

            return substitutor.safeSubstitute(replacedArguments, Variance.OUT_VARIANCE)
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy