org.jetbrains.kotlin.fir.session.NativeForwardDeclarationsSymbolProvider.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.fir.session
import org.jetbrains.kotlin.descriptors.EffectiveVisibility
import org.jetbrains.kotlin.descriptors.Modality
import org.jetbrains.kotlin.descriptors.Visibilities
import org.jetbrains.kotlin.fir.FirModuleData
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.caches.createCache
import org.jetbrains.kotlin.fir.caches.firCachesFactory
import org.jetbrains.kotlin.fir.caches.getValue
import org.jetbrains.kotlin.fir.declarations.FirDeclarationOrigin
import org.jetbrains.kotlin.fir.declarations.FirResolvePhase
import org.jetbrains.kotlin.fir.declarations.builder.FirRegularClassBuilder
import org.jetbrains.kotlin.fir.declarations.builder.buildRegularClass
import org.jetbrains.kotlin.fir.declarations.getDeprecationsProvider
import org.jetbrains.kotlin.fir.declarations.impl.FirResolvedDeclarationStatusImpl
import org.jetbrains.kotlin.fir.expressions.builder.buildAnnotation
import org.jetbrains.kotlin.fir.expressions.impl.FirEmptyAnnotationArgumentMapping
import org.jetbrains.kotlin.fir.resolve.providers.FirSymbolNamesProvider
import org.jetbrains.kotlin.fir.resolve.providers.FirSymbolNamesProviderWithoutCallables
import org.jetbrains.kotlin.fir.resolve.providers.FirSymbolProvider
import org.jetbrains.kotlin.fir.resolve.providers.FirSymbolProviderInternals
import org.jetbrains.kotlin.fir.scopes.FirKotlinScopeProvider
import org.jetbrains.kotlin.fir.scopes.FirScopeProvider
import org.jetbrains.kotlin.fir.symbols.impl.*
import org.jetbrains.kotlin.fir.types.ConeTypeProjection
import org.jetbrains.kotlin.fir.types.builder.buildResolvedTypeRef
import org.jetbrains.kotlin.fir.types.constructClassType
import org.jetbrains.kotlin.fir.types.toLookupTag
import org.jetbrains.kotlin.library.KotlinLibrary
import org.jetbrains.kotlin.library.includedForwardDeclarations
import org.jetbrains.kotlin.library.isInterop
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.name.NativeForwardDeclarationKind
import org.jetbrains.kotlin.name.NativeStandardInteropNames
import org.jetbrains.kotlin.utils.mapToSetOrEmpty
class NativeForwardDeclarationsSymbolProvider(
session: FirSession,
private val forwardDeclarationsModuleData: FirModuleData,
private val kotlinScopeProvider: FirKotlinScopeProvider,
private val kotlinLibraries: Collection,
) : FirSymbolProvider(session) {
private companion object {
private val validPackages = NativeForwardDeclarationKind.packageFqNameToKind.keys
}
private val includedForwardDeclarations: Set by lazy {
buildSet {
for (library in kotlinLibraries) {
if (!library.isInterop) continue
for (fqName in library.includedForwardDeclarations) {
val classId = ClassId.topLevel(FqName(fqName))
if (classId.packageFqName in validPackages) add(classId)
}
}
}
}
private val includedForwardDeclarationsByPackage: Map> by lazy {
buildMap> {
for (classId in includedForwardDeclarations) {
getOrPut(classId.packageFqName) { mutableSetOf() }
.add(classId.shortClassName)
}
}
}
override fun getClassLikeSymbolByClassId(classId: ClassId): FirClassLikeSymbol<*>? {
if (classId !in includedForwardDeclarations) return null
if (classId.isNestedClass) return null
return syntheticForwardDeclarationClassCache.getValue(classId)
}
private val syntheticForwardDeclarationClassCache =
session.firCachesFactory.createCache(::createSyntheticForwardDeclarationClass)
private fun createSyntheticForwardDeclarationClass(classId: ClassId): FirClassLikeSymbol<*>? {
return createSyntheticForwardDeclarationClass(
classId,
forwardDeclarationsModuleData,
session,
kotlinScopeProvider
)
}
@FirSymbolProviderInternals
override fun getTopLevelCallableSymbolsTo(destination: MutableList>, packageFqName: FqName, name: Name) {
}
@FirSymbolProviderInternals
override fun getTopLevelFunctionSymbolsTo(destination: MutableList, packageFqName: FqName, name: Name) {
}
@FirSymbolProviderInternals
override fun getTopLevelPropertySymbolsTo(destination: MutableList, packageFqName: FqName, name: Name) {
}
override fun getPackage(fqName: FqName): FqName? {
if (fqName in includedForwardDeclarationsByPackage) {
return fqName
}
return null
}
override val symbolNamesProvider: FirSymbolNamesProvider = object : FirSymbolNamesProviderWithoutCallables() {
override val hasSpecificClassifierPackageNamesComputation: Boolean get() = true
override fun getPackageNamesWithTopLevelClassifiers(): Set? =
includedForwardDeclarationsByPackage.keys.mapToSetOrEmpty(FqName::asString)
override fun getTopLevelClassifierNamesInPackage(packageFqName: FqName): Set =
includedForwardDeclarationsByPackage[packageFqName].orEmpty()
}
}
fun createSyntheticForwardDeclarationClass(
classId: ClassId,
firModuleData: FirModuleData,
session: FirSession,
kotlinScopeProvider: FirScopeProvider,
performAdditionalSetup: FirRegularClassBuilder.() -> Unit = {},
): FirRegularClassSymbol? {
val forwardDeclarationKind = NativeForwardDeclarationKind.packageFqNameToKind[classId.packageFqName] ?: return null
val symbol = FirRegularClassSymbol(classId)
buildRegularClass {
moduleData = firModuleData
origin = FirDeclarationOrigin.Synthetic.ForwardDeclaration
check(!classId.isNestedClass) { "Expected top-level class when building forward declaration, got $classId" }
name = classId.shortClassName
status = FirResolvedDeclarationStatusImpl(
Visibilities.Public,
Modality.FINAL,
EffectiveVisibility.Public
).apply {
// This will be wrong if we support exported forward declarations.
// See https://youtrack.jetbrains.com/issue/KT-51377 for more details.
isExpect = false
isActual = false
isCompanion = false
isInner = false
isData = false
isInline = false
isExternal = false
isFun = false
}
classKind = forwardDeclarationKind.classKind
scopeProvider = kotlinScopeProvider
this.symbol = symbol
resolvePhase = FirResolvePhase.ANALYZED_DEPENDENCIES
superTypeRefs += buildResolvedTypeRef {
type = ConeClassLikeLookupTagImpl(forwardDeclarationKind.superClassId)
.constructClassType(ConeTypeProjection.EMPTY_ARRAY, isNullable = false)
}
annotations += buildAnnotation {
annotationTypeRef = buildResolvedTypeRef {
val annotationClassId = ClassId(
NativeStandardInteropNames.cInteropPackage,
NativeStandardInteropNames.ExperimentalForeignApi
)
type = annotationClassId.toLookupTag()
.constructClassType(typeArguments = ConeTypeProjection.EMPTY_ARRAY, isNullable = false)
}
argumentMapping = FirEmptyAnnotationArgumentMapping
}
performAdditionalSetup()
}.apply {
replaceDeprecationsProvider(getDeprecationsProvider(session))
}
return symbol
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy