
org.jetbrains.kotlin.fir.extensions.FirPredicateBasedProvider.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2020 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.extensions
import com.google.common.collect.LinkedHashMultimap
import com.google.common.collect.Multimap
import kotlinx.collections.immutable.PersistentList
import org.jetbrains.kotlin.fir.FirSession
import org.jetbrains.kotlin.fir.FirSessionComponent
import org.jetbrains.kotlin.fir.NoMutableState
import org.jetbrains.kotlin.fir.declarations.FirAnnotatedDeclaration
import org.jetbrains.kotlin.fir.declarations.FirFile
import org.jetbrains.kotlin.fir.extensions.predicate.*
import org.jetbrains.kotlin.fir.resolve.fqName
abstract class FirPredicateBasedProvider : FirSessionComponent {
companion object {
fun create(session: FirSession): FirPredicateBasedProvider {
return FirPredicateBasedProviderImpl(session)
}
}
abstract fun getSymbolsByPredicate(predicate: DeclarationPredicate): List
abstract fun getSymbolsByPredicate(
declarations: Collection,
predicate: DeclarationPredicate
): List
abstract fun getSymbolsWithOwnersByPredicate(
predicate: DeclarationPredicate
): List>>
abstract fun getSymbolsWithOwnersByPredicate(
declarations: Collection,
predicate: DeclarationPredicate
): List>>
abstract fun getOwnersOfDeclaration(declaration: FirAnnotatedDeclaration): List?
abstract fun fileHasPluginAnnotations(file: FirFile): Boolean
abstract fun registerAnnotatedDeclaration(declaration: FirAnnotatedDeclaration, owners: PersistentList)
abstract fun registerGeneratedDeclaration(declaration: FirAnnotatedDeclaration, owner: FirAnnotatedDeclaration)
abstract fun matches(predicate: DeclarationPredicate, declaration: FirAnnotatedDeclaration): Boolean
}
@NoMutableState
private class FirPredicateBasedProviderImpl(private val session: FirSession) : FirPredicateBasedProvider() {
private val registeredPluginAnnotations = session.registeredPluginAnnotations
private val cache = Cache()
override fun getSymbolsByPredicate(predicate: DeclarationPredicate): List {
val annotations = registeredPluginAnnotations.getAnnotationsForPredicate(predicate)
if (annotations.isEmpty()) return emptyList()
val declarations = annotations.flatMapTo(mutableSetOf()) { cache.declarationByAnnotation[it] + cache.declarationsUnderAnnotated[it] }
return getSymbolsByPredicate(declarations, predicate)
}
override fun getSymbolsByPredicate(
declarations: Collection,
predicate: DeclarationPredicate
): List {
return declarations.filter { matches(predicate, it) }
}
override fun getSymbolsWithOwnersByPredicate(predicate: DeclarationPredicate): List>> {
return getSymbolsByPredicate(predicate).zipWithParents()
}
override fun getSymbolsWithOwnersByPredicate(
declarations: Collection,
predicate: DeclarationPredicate
): List>> {
return getSymbolsByPredicate(declarations, predicate).zipWithParents()
}
private fun List.zipWithParents(): List>> {
return this.map { it to cache.ownersForDeclaration.getValue(it) }
}
override fun fileHasPluginAnnotations(file: FirFile): Boolean {
return file in cache.filesWithPluginAnnotations
}
override fun registerAnnotatedDeclaration(declaration: FirAnnotatedDeclaration, owners: PersistentList) {
cache.ownersForDeclaration[declaration] = owners
registerOwnersDeclarations(declaration, owners)
if (declaration.annotations.isEmpty()) return
val matchingAnnotations = declaration.annotations.mapNotNull { it.fqName(session) }
.filter { it in registeredPluginAnnotations.annotations }
if (matchingAnnotations.isEmpty()) return
matchingAnnotations.forEach { cache.declarationByAnnotation.put(it, declaration) }
cache.annotationsOfDeclaration.putAll(declaration, matchingAnnotations)
val file = owners.first() as FirFile
cache.filesWithPluginAnnotations += file
}
override fun getOwnersOfDeclaration(declaration: FirAnnotatedDeclaration): List? {
return cache.ownersForDeclaration[declaration]
}
override fun registerGeneratedDeclaration(declaration: FirAnnotatedDeclaration, owner: FirAnnotatedDeclaration) {
val owners = cache.ownersForDeclaration.getValue(owner).add(owner)
registerAnnotatedDeclaration(declaration, owners)
}
private fun registerOwnersDeclarations(declaration: FirAnnotatedDeclaration, owners: PersistentList) {
val lastOwner = owners.lastOrNull() ?: return
val annotationsFromLastOwner = cache.annotationsOfDeclaration[lastOwner]
val annotationsFromPreviousOwners = cache.parentAnnotationsOfDeclaration[lastOwner]
val allParentDeclarations = annotationsFromLastOwner + annotationsFromPreviousOwners
allParentDeclarations.forEach { cache.declarationsUnderAnnotated.put(it, declaration) }
cache.parentAnnotationsOfDeclaration.putAll(declaration, allParentDeclarations)
}
// ---------------------------------- Matching ----------------------------------
override fun matches(predicate: DeclarationPredicate, declaration: FirAnnotatedDeclaration): Boolean {
return predicate.accept(matcher, declaration)
}
private val matcher = Matcher()
private inner class Matcher : DeclarationPredicateVisitor() {
override fun visitPredicate(predicate: DeclarationPredicate, data: FirAnnotatedDeclaration): Boolean {
throw IllegalStateException("Should not be there")
}
override fun visitAny(predicate: DeclarationPredicate.Any, data: FirAnnotatedDeclaration): Boolean {
return true
}
override fun visitAnd(predicate: DeclarationPredicate.And, data: FirAnnotatedDeclaration): Boolean {
return predicate.a.accept(this, data) && predicate.b.accept(this, data)
}
override fun visitOr(predicate: DeclarationPredicate.Or, data: FirAnnotatedDeclaration): Boolean {
return predicate.a.accept(this, data) || predicate.b.accept(this, data)
}
override fun visitAnnotatedWith(predicate: AnnotatedWith, data: FirAnnotatedDeclaration): Boolean {
return matchWith(data, predicate.annotations)
}
override fun visitUnderAnnotatedWith(predicate: UnderAnnotatedWith, data: FirAnnotatedDeclaration): Boolean {
return matchUnder(data, predicate.annotations)
}
override fun visitAnnotatedWithMeta(predicate: AnnotatedWithMeta, data: FirAnnotatedDeclaration): Boolean {
return matchWith(data, predicate.userDefinedAnnotations)
}
override fun visitUnderMetaAnnotated(predicate: UnderMetaAnnotated, data: FirAnnotatedDeclaration): Boolean {
return matchUnder(data, predicate.userDefinedAnnotations)
}
private val MetaAnnotated.userDefinedAnnotations: Set
get() = metaAnnotations.flatMapTo(mutableSetOf()) { registeredPluginAnnotations.getAnnotationsWithMetaAnnotation(it) }
private fun matchWith(declaration: FirAnnotatedDeclaration, annotations: Set): Boolean {
return cache.annotationsOfDeclaration[declaration].any { it in annotations }
}
private fun matchUnder(declaration: FirAnnotatedDeclaration, annotations: Set): Boolean {
return cache.parentAnnotationsOfDeclaration[declaration].any { it in annotations }
}
}
// ---------------------------------- Cache ----------------------------------
private class Cache {
val declarationByAnnotation: Multimap = LinkedHashMultimap.create()
val annotationsOfDeclaration: LinkedHashMultimap = LinkedHashMultimap.create()
val declarationsUnderAnnotated: Multimap = LinkedHashMultimap.create()
val parentAnnotationsOfDeclaration: LinkedHashMultimap = LinkedHashMultimap.create()
val ownersForDeclaration: MutableMap> = mutableMapOf()
val filesWithPluginAnnotations: MutableSet = mutableSetOf()
}
}
val FirSession.predicateBasedProvider: FirPredicateBasedProvider by FirSession.sessionComponentAccessor()
© 2015 - 2025 Weber Informatics LLC | Privacy Policy