org.jetbrains.kotlin.fir.Utils.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
import com.intellij.psi.PsiElement
import org.jetbrains.kotlin.*
import org.jetbrains.kotlin.builtins.StandardNames
import org.jetbrains.kotlin.descriptors.Modality
import org.jetbrains.kotlin.descriptors.Visibility
import org.jetbrains.kotlin.fir.declarations.*
import org.jetbrains.kotlin.fir.declarations.impl.FirDeclarationStatusImpl
import org.jetbrains.kotlin.fir.declarations.impl.FirDeclarationStatusWithAlteredDefaults
import org.jetbrains.kotlin.fir.declarations.impl.FirResolvedDeclarationStatusImpl
import org.jetbrains.kotlin.fir.declarations.impl.FirResolvedDeclarationStatusWithAlteredDefaults
import org.jetbrains.kotlin.fir.declarations.utils.isEnumClass
import org.jetbrains.kotlin.fir.declarations.utils.isStatic
import org.jetbrains.kotlin.fir.expressions.*
import org.jetbrains.kotlin.fir.renderer.FirRenderer
import org.jetbrains.kotlin.fir.symbols.FirBasedSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirCallableSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirClassLikeSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirClassSymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirPropertyAccessorSymbol
import org.jetbrains.kotlin.fir.types.*
import org.jetbrains.kotlin.fir.types.builder.*
import org.jetbrains.kotlin.fir.types.impl.*
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.util.OperatorNameConventions
import org.jetbrains.kotlin.util.OperatorNameConventions.STATEMENT_LIKE_OPERATORS
import org.jetbrains.kotlin.util.wrapIntoFileAnalysisExceptionIfNeeded
import org.jetbrains.kotlin.util.wrapIntoSourceCodeAnalysisExceptionIfNeeded
// TODO: rewrite
fun FirBlock.returnExpressions(): List = listOfNotNull(statements.lastOrNull() as? FirExpression)
fun R.copyWithNewSourceKind(newKind: KtFakeSourceElementKind): R {
if (source == null) return this
if (source?.kind == newKind) return this
return copyWithNewSource(source?.fakeElement(newKind))
}
// do we need a deep copy here ?
fun R.copyWithNewSource(newSource: KtSourceElement?): R {
if (source?.kind == newSource?.kind) return this
@Suppress("UNCHECKED_CAST")
return when (val typeRef = this) {
is FirResolvedTypeRefImpl -> buildResolvedTypeRefCopy(typeRef) {
source = newSource
}
is FirErrorTypeRef -> buildErrorTypeRefCopy(typeRef) {
source = newSource
}
is FirUserTypeRefImpl -> buildUserTypeRef {
source = newSource
isMarkedNullable = typeRef.isMarkedNullable
qualifier += typeRef.qualifier
annotations += typeRef.annotations
}
is FirFunctionTypeRefImpl -> buildFunctionTypeRefCopy(typeRef) {
source = newSource
}
is FirDynamicTypeRef -> buildDynamicTypeRef {
source = newSource
isMarkedNullable = typeRef.isMarkedNullable
annotations += typeRef.annotations
}
is FirImplicitBuiltinTypeRef -> typeRef.withNewSource(newSource)
is FirIntersectionTypeRef -> buildIntersectionTypeRef {
source = newSource
isMarkedNullable = typeRef.isMarkedNullable
leftType = typeRef.leftType
rightType = typeRef.rightType
}
else -> TODO("Not implemented for ${typeRef::class}")
} as R
}
val FirFile.packageFqName: FqName
get() = packageDirective.packageFqName
val FirElement.psi: PsiElement? get() = (source as? KtPsiSourceElement)?.psi
val FirElement.realPsi: PsiElement? get() = (source as? KtRealPsiSourceElement)?.psi
val FirContextReceiver.labelName: Name? get() = customLabelName ?: labelNameFromTypeRef
fun FirElement.renderWithType(): String =
FirRenderer().renderElementWithTypeAsString(this)
fun FirElement.render(): String =
FirRenderer().renderElementAsString(this)
fun FirDeclarationStatus.copy(
visibility: Visibility? = this.visibility,
modality: Modality? = this.modality,
isExpect: Boolean = this.isExpect,
isActual: Boolean = this.isActual,
isOverride: Boolean = this.isOverride,
isOperator: Boolean = this.isOperator,
isInfix: Boolean = this.isInfix,
isInline: Boolean = this.isInline,
isTailRec: Boolean = this.isTailRec,
isExternal: Boolean = this.isExternal,
isConst: Boolean = this.isConst,
isLateInit: Boolean = this.isLateInit,
isInner: Boolean = this.isInner,
isCompanion: Boolean = this.isCompanion,
isData: Boolean = this.isData,
isSuspend: Boolean = this.isSuspend,
isStatic: Boolean = this.isStatic,
isFromSealedClass: Boolean = this.isFromSealedClass,
isFromEnumClass: Boolean = this.isFromEnumClass,
isFun: Boolean = this.isFun,
hasStableParameterNames: Boolean = this.hasStableParameterNames,
): FirDeclarationStatus {
val newVisibility = visibility ?: this.visibility
val newModality = modality ?: this.modality
val newStatus = if (this is FirResolvedDeclarationStatus) {
FirResolvedDeclarationStatusImpl(newVisibility, newModality!!, effectiveVisibility)
} else {
FirDeclarationStatusImpl(newVisibility, newModality)
}
copyStatusAttributes(
from = this,
to = newStatus,
isExpect = isExpect,
isActual = isActual,
isOverride = isOverride,
isOperator = isOperator,
isInfix = isInfix,
isInline = isInline,
isTailRec = isTailRec,
isExternal = isExternal,
isConst = isConst,
isLateInit = isLateInit,
isInner = isInner,
isCompanion = isCompanion,
isData = isData,
isSuspend = isSuspend,
isStatic = isStatic,
isFromSealedClass = isFromSealedClass,
isFromEnumClass = isFromEnumClass,
isFun = isFun,
hasStableParameterNames = hasStableParameterNames,
)
return newStatus
}
fun FirDeclarationStatus.copyWithNewDefaults(
visibility: Visibility? = this.visibility,
modality: Modality? = this.modality,
defaultVisibility: Visibility = this.defaultVisibility,
defaultModality: Modality = this.defaultModality,
): FirDeclarationStatus {
val newVisibility = visibility ?: this.visibility
val newModality = modality ?: this.modality
val newStatus = when (this) {
is FirResolvedDeclarationStatus -> FirResolvedDeclarationStatusWithAlteredDefaults(
newVisibility, newModality!!,
defaultVisibility, defaultModality,
effectiveVisibility
)
else -> FirDeclarationStatusWithAlteredDefaults(newVisibility, newModality, defaultVisibility, defaultModality)
}
copyStatusAttributes(from = this, to = newStatus)
return newStatus
}
private fun copyStatusAttributes(
from: FirDeclarationStatus,
to: FirDeclarationStatusImpl,
isExpect: Boolean = from.isExpect,
isActual: Boolean = from.isActual,
isOverride: Boolean = from.isOverride,
isOperator: Boolean = from.isOperator,
isInfix: Boolean = from.isInfix,
isInline: Boolean = from.isInline,
isTailRec: Boolean = from.isTailRec,
isExternal: Boolean = from.isExternal,
isConst: Boolean = from.isConst,
isLateInit: Boolean = from.isLateInit,
isInner: Boolean = from.isInner,
isCompanion: Boolean = from.isCompanion,
isData: Boolean = from.isData,
isSuspend: Boolean = from.isSuspend,
isStatic: Boolean = from.isStatic,
isFromSealedClass: Boolean = from.isFromSealedClass,
isFromEnumClass: Boolean = from.isFromEnumClass,
isFun: Boolean = from.isFun,
hasStableParameterNames: Boolean = from.hasStableParameterNames,
) {
to.isExpect = isExpect
to.isActual = isActual
to.isOverride = isOverride
to.isOperator = isOperator
to.isInfix = isInfix
to.isInline = isInline
to.isTailRec = isTailRec
to.isExternal = isExternal
to.isConst = isConst
to.isLateInit = isLateInit
to.isInner = isInner
to.isCompanion = isCompanion
to.isData = isData
to.isSuspend = isSuspend
to.isStatic = isStatic
to.isFromSealedClass = isFromSealedClass
to.isFromEnumClass = isFromEnumClass
to.isFun = isFun
to.hasStableParameterNames = hasStableParameterNames
}
inline fun whileAnalysing(session: FirSession, element: FirElement, block: () -> R): R {
return try {
block()
} catch (throwable: Throwable) {
session.exceptionHandler.handleExceptionOnElementAnalysis(element, throwable)
}
}
inline fun withFileAnalysisExceptionWrapping(file: FirFile, block: () -> R): R {
return try {
block()
} catch (throwable: Throwable) {
file.moduleData.session.exceptionHandler.handleExceptionOnFileAnalysis(file, throwable)
}
}
abstract class FirExceptionHandler : FirSessionComponent {
abstract fun handleExceptionOnElementAnalysis(element: FirElement, throwable: Throwable): Nothing
abstract fun handleExceptionOnFileAnalysis(file: FirFile, throwable: Throwable): Nothing
}
val FirSession.exceptionHandler: FirExceptionHandler by FirSession.sessionComponentAccessor()
object FirCliExceptionHandler : FirExceptionHandler() {
override fun handleExceptionOnElementAnalysis(element: FirElement, throwable: Throwable): Nothing {
throw throwable.wrapIntoSourceCodeAnalysisExceptionIfNeeded(element.source)
}
override fun handleExceptionOnFileAnalysis(file: FirFile, throwable: Throwable): Nothing {
throw throwable.wrapIntoFileAnalysisExceptionIfNeeded(
file.sourceFile?.path,
file.source,
) { file.sourceFileLinesMapping?.getLineAndColumnByOffset(it) }
}
}
@JvmInline
value class MutableOrEmptyList(internal val list: MutableList<@UnsafeVariance T>?) : List {
private constructor(list: Nothing?) : this(list as MutableList?)
override val size: Int
get() = list?.size ?: 0
override fun get(index: Int): T {
return list!![index]
}
override fun isEmpty(): Boolean {
return list?.isEmpty() ?: true
}
override fun iterator(): Iterator {
return list?.iterator() ?: EMPTY_LIST_STUB_ITERATOR
}
override fun listIterator(): ListIterator {
return list?.listIterator() ?: EMPTY_LIST_STUB_LIST_ITERATOR
}
override fun listIterator(index: Int): ListIterator {
return list?.listIterator(index) ?: EMPTY_LIST_STUB_LIST_ITERATOR
}
override fun subList(fromIndex: Int, toIndex: Int): List {
if (list == null && fromIndex == 0 && toIndex == 0) return this
return list!!.subList(fromIndex, toIndex)
}
override fun lastIndexOf(element: @UnsafeVariance T): Int {
return list?.lastIndexOf(element) ?: -1
}
override fun indexOf(element: @UnsafeVariance T): Int {
return list?.indexOf(element) ?: -1
}
override fun containsAll(elements: Collection<@UnsafeVariance T>): Boolean {
return list?.containsAll(elements) ?: elements.isEmpty()
}
override fun contains(element: @UnsafeVariance T): Boolean {
return list?.contains(element) ?: false
}
override fun toString(): String {
return list?.joinToString(prefix = "[", postfix = "]") ?: "[]"
}
companion object {
private val EMPTY = MutableOrEmptyList(null)
private val EMPTY_LIST_STUB = emptyList()
private val EMPTY_LIST_STUB_ITERATOR = EMPTY_LIST_STUB.iterator()
private val EMPTY_LIST_STUB_LIST_ITERATOR = EMPTY_LIST_STUB.listIterator()
fun empty(): MutableOrEmptyList = EMPTY
}
}
fun List.smartPlus(other: List): List = when {
other.isEmpty() -> this
this.isEmpty() -> other
else -> {
val result = ArrayList(this.size + other.size)
result.addAll(this)
result.addAll(other)
result
}
}
// Source element may be missing if the class came from a library
fun FirVariable.isEnumEntries(containingClass: FirClass) = isStatic && name == StandardNames.ENUM_ENTRIES && containingClass.isEnumClass
fun FirVariable.isEnumEntries(containingClassSymbol: FirClassSymbol<*>): Boolean {
return isStatic && name == StandardNames.ENUM_ENTRIES && containingClassSymbol.isEnumClass
}
val FirExpression.isArraySet: Boolean
get() {
val name = (this as? FirFunctionCall)?.calleeReference?.name ?: return false
return origin == FirFunctionCallOrigin.Operator && name == OperatorNameConventions.SET
}
val FirExpression.isStatementLikeExpression: Boolean
get() = when (this) {
is FirFunctionCall -> origin == FirFunctionCallOrigin.Operator && calleeReference.name in STATEMENT_LIKE_OPERATORS
else -> isIndexedAssignment
}
private val FirExpression.isIndexedAssignment: Boolean
get() = this is FirBlock && statements.lastOrNull()?.source?.kind == KtFakeSourceElementKind.ImplicitUnit.IndexedAssignmentCoercion
fun FirBasedSymbol<*>.packageFqName(): FqName {
return when (this) {
is FirClassLikeSymbol<*> -> classId.packageFqName
is FirPropertyAccessorSymbol -> propertySymbol.packageFqName()
is FirCallableSymbol<*> -> callableId.packageName
else -> error("No package fq name for $this")
}
}
fun FirOperation.toAugmentedAssignSourceKind() = when (this) {
FirOperation.PLUS_ASSIGN -> KtFakeSourceElementKind.DesugaredPlusAssign
FirOperation.MINUS_ASSIGN -> KtFakeSourceElementKind.DesugaredMinusAssign
FirOperation.TIMES_ASSIGN -> KtFakeSourceElementKind.DesugaredTimesAssign
FirOperation.DIV_ASSIGN -> KtFakeSourceElementKind.DesugaredDivAssign
FirOperation.REM_ASSIGN -> KtFakeSourceElementKind.DesugaredRemAssign
else -> error("Unexpected operator: $name")
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy