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

org.jetbrains.kotlin.resolve.sam.SamConversionResolverImpl.kt Maven / Gradle / Ivy

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

import org.jetbrains.kotlin.builtins.createFunctionType
import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.descriptors.annotations.Annotations.Companion.EMPTY
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.name.SpecialNames
import org.jetbrains.kotlin.resolve.DescriptorUtils
import org.jetbrains.kotlin.resolve.descriptorUtil.builtIns
import org.jetbrains.kotlin.resolve.descriptorUtil.fqNameSafe
import org.jetbrains.kotlin.storage.StorageManager
import org.jetbrains.kotlin.types.*
import org.jetbrains.kotlin.types.typeUtil.asTypeProjection
import org.jetbrains.kotlin.types.typeUtil.contains
import java.util.*

class SamConversionResolverImpl(
    storageManager: StorageManager,
    private val samWithReceiverResolvers: Iterable
) : SamConversionResolver {
    class SamConversionResolverWithoutReceiverConversion(storageManager: StorageManager) : SamConversionResolver {
        val resolver = SamConversionResolverImpl(storageManager, emptyList())

        override fun resolveFunctionTypeIfSamInterface(classDescriptor: ClassDescriptor): SimpleType? {
            return resolver.resolveFunctionTypeIfSamInterface(classDescriptor)
        }
    }

    private val functionTypesForSamInterfaces = storageManager.createCacheWithNullableValues()

    override fun resolveFunctionTypeIfSamInterface(classDescriptor: ClassDescriptor): SimpleType? {
        return functionTypesForSamInterfaces.computeIfAbsent(classDescriptor) {
            val abstractMethod = getSingleAbstractMethodOrNull(classDescriptor) ?: return@computeIfAbsent null
            val shouldConvertFirstParameterToDescriptor = samWithReceiverResolvers.any { it.shouldConvertFirstSamParameterToReceiver(abstractMethod) }
            getFunctionTypeForAbstractMethod(abstractMethod, shouldConvertFirstParameterToDescriptor)
        }
    }
}

fun getSingleAbstractMethodOrNull(klass: ClassDescriptor): FunctionDescriptor? {
    // NB: this check MUST BE at start. Please do not touch until following to-do is resolved
    // Otherwise android data binding can cause resolve re-entrance
    // For details see KT-18687, KT-16149
    // TODO: prevent resolve re-entrance on architecture level, or (alternatively) ask data binding owners not to do it
    if (klass.fqNameSafe.asString().endsWith(".databinding.DataBindingComponent")) return null

    if (klass.isDefinitelyNotSamInterface) return null

    val abstractMember = getAbstractMembers(klass).singleOrNull() ?: return null

    return if (abstractMember is SimpleFunctionDescriptor && abstractMember.typeParameters.isEmpty())
        abstractMember
    else
        null
}

@Suppress("UNCHECKED_CAST")
fun getAbstractMembers(classDescriptor: ClassDescriptor): List {
    return DescriptorUtils
        .getAllDescriptors(classDescriptor.unsubstitutedMemberScope)
        .filter { it is CallableMemberDescriptor && it.modality == Modality.ABSTRACT } as List
}

fun getFunctionTypeForAbstractMethod(
    function: FunctionDescriptor,
    shouldConvertFirstParameterToDescriptor: Boolean
): SimpleType {
    val returnType = function.returnType ?: error("function is not initialized: $function")
    val valueParameters = function.valueParameters

    val parameterTypes = ArrayList(valueParameters.size)
    val parameterNames = ArrayList(valueParameters.size)

    val contextReceiversTypes = function.contextReceiverParameters.map { it.type }
    var startIndex = contextReceiversTypes.size
    var receiverType: KotlinType? = null
    val extensionReceiver = function.extensionReceiverParameter
    if (extensionReceiver != null) {
        receiverType = extensionReceiver.type
    } else if (shouldConvertFirstParameterToDescriptor && function.valueParameters.isNotEmpty()) {
        receiverType = valueParameters[0].type
        startIndex += 1
    }

    for (i in startIndex until valueParameters.size) {
        val parameter = valueParameters[i]
        parameterTypes.add(parameter.type)
        parameterNames.add(if (function.hasSynthesizedParameterNames()) SpecialNames.NO_NAME_PROVIDED else parameter.name)
    }

    return createFunctionType(
        function.builtIns, EMPTY, receiverType, contextReceiversTypes, parameterTypes,
        parameterNames, returnType, function.isSuspend
    )
}

fun SamConversionResolver.getFunctionTypeForPossibleSamType(
    possibleSamType: UnwrappedType,
    samConversionOracle: SamConversionOracle
): UnwrappedType? = getFunctionTypeForSamType(possibleSamType, this, samConversionOracle)?.unwrap()

fun getFunctionTypeForSamType(
    samType: KotlinType,
    samResolver: SamConversionResolver,
    samConversionOracle: SamConversionOracle
): KotlinType? {
    val unwrappedType = samType.unwrap()
    if (unwrappedType is FlexibleType) {
        val lower = getFunctionTypeForSamType(unwrappedType.lowerBound, samResolver, samConversionOracle)
        val upper = getFunctionTypeForSamType(unwrappedType.upperBound, samResolver, samConversionOracle)

        assert((lower == null) == (upper == null)) { "Illegal flexible type: $unwrappedType" }

        if (lower == null || upper == null) return null

        return KotlinTypeFactory.flexibleType(lower, upper)
    } else {
        return getFunctionTypeForSamType(unwrappedType as SimpleType, samResolver, samConversionOracle)
    }
}

private fun getFunctionTypeForSamType(
    samType: SimpleType,
    samResolver: SamConversionResolver,
    samConversionOracle: SamConversionOracle
): SimpleType? {
    // e.g. samType == Comparator?
    val classifier = samType.constructor.declarationDescriptor
    if (classifier !is ClassDescriptor) return null

    if (!samConversionOracle.isPossibleSamType(samType)) return null

    // Function2
    val functionTypeDefault = samResolver.resolveFunctionTypeIfSamInterface(classifier) ?: return null
    val noProjectionsSamType = nonProjectionParametrization(samType) ?: return null

    // Function2?
    val type = TypeSubstitutor.create(noProjectionsSamType).substitute(functionTypeDefault, Variance.IN_VARIANCE)
    assert(type != null) {
        "Substitution based on type with no projections '$noProjectionsSamType' should not end with conflict"
    }

    val simpleType = type!!.asSimpleType()
    return simpleType.makeNullableAsSpecified(samType.isMarkedNullable)
}

// If type 'samType' contains no projection, then it's non-projection parametrization is 'samType' itself
// Else each projection type argument 'out/in A_i' (but star projections) is replaced with it's bound 'A_i'
// Star projections are treated specially:
// - If first upper bound of corresponding type parameter does not contain any type parameter of 'samType' class,
//   then use this upper bound instead of star projection
// - Otherwise no non-projection parametrization exists for such 'samType'
//
// See Non-wildcard parametrization in JLS 8 p.9.9 for clarification
fun nonProjectionParametrization(samType: SimpleType): SimpleType? {
    if (samType.arguments.none { it.projectionKind != Variance.INVARIANT }) return samType
    val parameters = samType.constructor.parameters
    val parametersSet = parameters.toSet()

    return samType.replace(
        newArguments = samType.arguments.zip(parameters).map {
            val (projection, parameter) = it
            when {
                projection.projectionKind == Variance.INVARIANT -> projection

                projection.isStarProjection ->
                    parameter.upperBounds.first().takeUnless { t ->
                        t.contains { it.constructor.declarationDescriptor in parametersSet }
                    }?.asTypeProjection() ?: return@nonProjectionParametrization null

                else -> projection.type.asTypeProjection()
            }
        })
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy