org.jetbrains.kotlin.fir.serialization.FirProvidedDeclarationsForMetadataService.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.serialization
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.FirSessionComponent
import org.jetbrains.kotlin.fir.containingClassLookupTag
import org.jetbrains.kotlin.fir.declarations.*
import org.jetbrains.kotlin.fir.resolve.ScopeSession
import org.jetbrains.kotlin.fir.resolve.toFirRegularClass
import org.jetbrains.kotlin.fir.symbols.impl.FirClassLikeSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirClassSymbol
import org.jetbrains.kotlin.name.FqName
/**
* This extension can be used to provide information about declarations, which should be written in Kotlin metadata (i.e. be part of a module's ABI),
* but for some reason can not (or should not) be generated with [FirDeclarationGenerationExtension]
*
* Example: assume your plugin generates some constructor in [IrGenerationExtension] which have value parameters matching
* all properties of a class, and this constructor is used only as an implementation detail (so the only actor who accesses it
* is a plugin itself). The constructor should be accessible from another module, so it should be present in the metadata. But you
* can not generate this constructor in [FirDeclarationGenerationExtension], because it depends on types of properties,
* which may be not resolved at the moment of constructor creation.
*
*
* // MODULE: a
* open class Base {
* val x: Int = 1
* val y = "hello"
*
* constructor()
*
* // generated
* constructor(x: Int, y: String) { ... } // (1)
* }
*
* // MODULE: b(a)
* class Derived : Base {
* val z = 1.0
*
* constructor() : super()
*
* // generated
* constructor(x: Int, y: String, z: Double) : super(x, y) { ... } // (2)
*
* // constructor (1) should be presented in metadata of class Base, so IR plugin
* // can reference it during generation of constructor (2)
* }
*
* All declarations provided by this extension should be fully resolved and contain all sub declarations if they exist.
* E.g. if you want to provide some class then this class should contain all declarations of this class you want to be
* present in metadata in `declarations` field of a `FirRegularClass`.
* [FirDeclarationGenerationExtension] won't be called for classes provided by this extension.
*/
abstract class FirProvidedDeclarationsForMetadataService : FirSessionComponent {
companion object {
fun create(session: FirSession): FirProvidedDeclarationsForMetadataService {
return FirProvidedDeclarationsForMetadataServiceImpl(session)
}
}
abstract fun getProvidedTopLevelDeclarations(packageFqName: FqName, scopeSession: ScopeSession): List
abstract fun getProvidedConstructors(owner: FirClassSymbol<*>, scopeSession: ScopeSession): List
abstract fun getProvidedCallables(owner: FirClassSymbol<*>, scopeSession: ScopeSession): List
abstract fun registerDeclaration(declaration: FirCallableDeclaration)
}
private class FirProvidedDeclarationsForMetadataServiceImpl(private val session: FirSession) : FirProvidedDeclarationsForMetadataService() {
private val topLevelsCache: MutableMap> =
mutableMapOf()
private val memberCache: MutableMap, ClassDeclarations> =
mutableMapOf()
override fun registerDeclaration(declaration: FirCallableDeclaration) {
val containingClass = declaration.containingClassLookupTag()?.toFirRegularClass(session)
if (containingClass == null) {
val list = topLevelsCache.getOrPut(declaration.symbol.callableId.packageName) { mutableListOf() }
list += declaration
} else {
val declarations = memberCache.getOrPut(containingClass.symbol) { ClassDeclarations() }
when (declaration) {
is FirConstructor -> declarations.providedConstructors += declaration
else -> declarations.providedCallables += declaration
}
}
}
override fun getProvidedTopLevelDeclarations(packageFqName: FqName, scopeSession: ScopeSession): List {
return topLevelsCache[packageFqName] ?: emptyList()
}
override fun getProvidedConstructors(owner: FirClassSymbol<*>, scopeSession: ScopeSession): List {
return memberCache[owner]?.providedConstructors ?: emptyList()
}
override fun getProvidedCallables(owner: FirClassSymbol<*>, scopeSession: ScopeSession): List {
return memberCache[owner]?.providedCallables ?: emptyList()
}
private class ClassDeclarations {
val providedCallables: MutableList = mutableListOf()
val providedConstructors: MutableList = mutableListOf()
}
}
val FirSession.providedDeclarationsForMetadataService: FirProvidedDeclarationsForMetadataService by FirSession.sessionComponentAccessor()
© 2015 - 2025 Weber Informatics LLC | Privacy Policy