org.jetbrains.kotlin.fir.scopes.impl.AbstractFirUseSiteMemberScope.kt Maven / Gradle / Ivy
/*
* 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.fir.scopes.impl
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.declarations.utils.isStatic
import org.jetbrains.kotlin.fir.resolve.substitution.ConeSubstitutor
import org.jetbrains.kotlin.fir.scopes.*
import org.jetbrains.kotlin.fir.scopes.impl.FirTypeIntersectionScopeContext.ResultOfIntersection
import org.jetbrains.kotlin.fir.symbols.ConeClassLikeLookupTag
import org.jetbrains.kotlin.fir.symbols.impl.*
import org.jetbrains.kotlin.fir.types.ConeSimpleKotlinType
import org.jetbrains.kotlin.name.Name
abstract class AbstractFirUseSiteMemberScope(
val ownerClassLookupTag: ConeClassLikeLookupTag,
session: FirSession,
overrideChecker: FirOverrideChecker,
protected val superTypeScopes: List,
dispatchReceiverType: ConeSimpleKotlinType,
protected val declaredMemberScope: FirContainingNamesAwareScope
) : AbstractFirOverrideScope(session, overrideChecker) {
protected val supertypeScopeContext =
FirTypeIntersectionScopeContext(session, overrideChecker, superTypeScopes, dispatchReceiverType, forClassUseSiteScope = true)
private val functions: MutableMap> = hashMapOf()
private val properties: MutableMap>> = hashMapOf()
protected val directOverriddenFunctions: MutableMap>> =
hashMapOf()
protected val directOverriddenProperties: MutableMap>> = hashMapOf()
protected val functionsFromSupertypes: MutableMap>> = mutableMapOf()
protected val propertiesFromSupertypes: MutableMap>> = mutableMapOf()
protected val fieldsFromSupertypes: MutableMap> = mutableMapOf()
private val callableNamesCached by lazy(LazyThreadSafetyMode.PUBLICATION) {
buildSet {
addAll(declaredMemberScope.getCallableNames())
superTypeScopes.flatMapTo(this) { it.getCallableNames() }
}
}
private val classifierNamesCached by lazy(LazyThreadSafetyMode.PUBLICATION) {
buildSet {
addAll(declaredMemberScope.getClassifierNames())
superTypeScopes.flatMapTo(this) { it.getClassifierNames() }
}
}
final override fun processFunctionsByName(name: Name, processor: (FirNamedFunctionSymbol) -> Unit) {
// Important optimization: avoid creating cache keys for names that are definitely absent
if (name !in getCallableNames()) return
functions.getOrPut(name) {
collectFunctions(name)
}.forEach {
processor(it)
}
}
protected open fun collectFunctions(
name: Name
): Collection = mutableListOf().apply {
collectDeclaredFunctions(name, this)
val explicitlyDeclaredFunctions = this.toSet()
collectFunctionsFromSupertypes(name, this, explicitlyDeclaredFunctions)
}
protected fun collectDeclaredFunctions(name: Name, destination: MutableList) {
declaredMemberScope.processFunctionsByName(name) { symbol ->
if (symbol.isStatic) return@processFunctionsByName
if (!symbol.isVisibleInCurrentClass()) return@processFunctionsByName
val directOverridden = computeDirectOverriddenForDeclaredFunction(symbol)
directOverriddenFunctions[symbol] = directOverridden
destination += symbol.replaceWithWrapperSymbolIfNeeded()
}
}
protected abstract fun FirNamedFunctionSymbol.isVisibleInCurrentClass(): Boolean
protected fun collectFunctionsFromSupertypes(
name: Name,
destination: MutableList,
explicitlyDeclaredFunctions: Set
) {
for (chosenSymbolFromSupertype in getFunctionsFromSupertypesByName(name)) {
val superSymbol = chosenSymbolFromSupertype.extractSomeSymbolFromSuperType()
if (!superSymbol.isVisibleInCurrentClass()) continue
val overriddenBy = superSymbol.getOverridden(explicitlyDeclaredFunctions)
if (overriddenBy == null) {
destination += chosenSymbolFromSupertype.chosenSymbol
}
}
}
private fun getFunctionsFromSupertypesByName(name: Name): List> {
return functionsFromSupertypes.getOrPut(name) {
supertypeScopeContext.collectIntersectionResultsForCallables(name, FirScope::processFunctionsByName)
}
}
final override fun processPropertiesByName(name: Name, processor: (FirVariableSymbol<*>) -> Unit) {
// Important optimization: avoid creating cache keys for names that are definitely absent
if (name !in getCallableNames()) return
properties.getOrPut(name) {
collectProperties(name)
}.forEach {
processor(it)
}
}
protected abstract fun collectProperties(name: Name): Collection>
private fun computeDirectOverriddenForDeclaredFunction(declaredFunctionSymbol: FirNamedFunctionSymbol): List> {
val result = mutableListOf>()
val declaredFunction = declaredFunctionSymbol.fir
for (resultOfIntersection in getFunctionsFromSupertypesByName(declaredFunctionSymbol.name)) {
val symbolFromSupertype = resultOfIntersection.extractSomeSymbolFromSuperType()
if (overrideChecker.isOverriddenFunction(declaredFunction, symbolFromSupertype.fir)) {
result.add(resultOfIntersection)
}
}
return result
}
protected fun > ResultOfIntersection.extractSomeSymbolFromSuperType(): D {
return if (this.isIntersectionOverride()) {
/*
* we don't want to create intersection override if some declared function actually overrides some functions
* from supertypes, so instead of intersection override symbol we check actual symbol from supertype
*
* TODO: is it enough to check only one function?
*/
keySymbol
} else {
chosenSymbol
}
}
override fun processDirectOverriddenFunctionsWithBaseScope(
functionSymbol: FirNamedFunctionSymbol,
processor: (FirNamedFunctionSymbol, FirTypeScope) -> ProcessorAction
): ProcessorAction {
return processDirectOverriddenMembersWithBaseScopeImpl(
directOverriddenFunctions,
functionsFromSupertypes,
functionSymbol,
processor
)
}
override fun processDirectOverriddenPropertiesWithBaseScope(
propertySymbol: FirPropertySymbol,
processor: (FirPropertySymbol, FirTypeScope) -> ProcessorAction
): ProcessorAction {
return processDirectOverriddenMembersWithBaseScopeImpl(
directOverriddenProperties,
propertiesFromSupertypes,
propertySymbol,
processor
)
}
private fun > processDirectOverriddenMembersWithBaseScopeImpl(
directOverriddenMap: Map>>,
callablesFromSupertypes: Map>>,
callableSymbol: D,
processor: (D, FirTypeScope) -> ProcessorAction
): ProcessorAction {
when (val directOverridden = directOverriddenMap[callableSymbol]) {
null -> {
val resultOfIntersection = callablesFromSupertypes[callableSymbol.name]
?.firstOrNull { it.chosenSymbol == callableSymbol }
?: return ProcessorAction.NONE
for ((overridden, baseScope) in resultOfIntersection.overriddenMembers) {
if (!processor(overridden, baseScope)) return ProcessorAction.STOP
}
return ProcessorAction.NONE
}
else -> {
for (resultOfIntersection in directOverridden) {
for ((overridden, baseScope) in resultOfIntersection.overriddenMembers) {
if (!processor(overridden, baseScope)) return ProcessorAction.STOP
}
}
return ProcessorAction.NONE
}
}
}
override fun processClassifiersByNameWithSubstitution(name: Name, processor: (FirClassifierSymbol<*>, ConeSubstitutor) -> Unit) {
// Important optimization: avoid creating cache keys for names that are definitely absent
if (name !in getClassifierNames()) return
var shadowed = false
declaredMemberScope.processClassifiersByNameWithSubstitution(name) { classifier, substitutor ->
shadowed = true
processor(classifier, substitutor)
}
if (!shadowed) {
supertypeScopeContext.processClassifiersByNameWithSubstitution(name, processor)
}
}
override fun processDeclaredConstructors(processor: (FirConstructorSymbol) -> Unit) {
declaredMemberScope.processDeclaredConstructors(processor)
}
override fun getCallableNames(): Set {
return callableNamesCached
}
override fun getClassifierNames(): Set {
return classifierNamesCached
}
/**
* This function is currently used only for creating suspend views in Java.
*/
protected open fun FirNamedFunctionSymbol.replaceWithWrapperSymbolIfNeeded(): FirNamedFunctionSymbol {
return this
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy