org.jetbrains.kotlin.fir.analysis.cfa.FirPropertyInitializationAnalyzer.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.analysis.cfa
import org.jetbrains.kotlin.contracts.description.canBeRevisited
import org.jetbrains.kotlin.contracts.description.isDefinitelyVisited
import org.jetbrains.kotlin.contracts.description.isInPlace
import org.jetbrains.kotlin.descriptors.ClassKind
import org.jetbrains.kotlin.diagnostics.DiagnosticReporter
import org.jetbrains.kotlin.diagnostics.reportOn
import org.jetbrains.kotlin.fir.analysis.cfa.util.PropertyInitializationInfoData
import org.jetbrains.kotlin.fir.analysis.cfa.util.previousCfgNodes
import org.jetbrains.kotlin.fir.analysis.checkers.MppCheckerKind
import org.jetbrains.kotlin.fir.analysis.checkers.context.CheckerContext
import org.jetbrains.kotlin.fir.analysis.checkers.getContainingSymbol
import org.jetbrains.kotlin.fir.analysis.checkers.hasDiagnosticKind
import org.jetbrains.kotlin.fir.analysis.diagnostics.FirErrors
import org.jetbrains.kotlin.fir.declarations.*
import org.jetbrains.kotlin.fir.declarations.utils.*
import org.jetbrains.kotlin.fir.diagnostics.DiagnosticKind
import org.jetbrains.kotlin.fir.expressions.*
import org.jetbrains.kotlin.fir.isCatchParameter
import org.jetbrains.kotlin.fir.references.toResolvedPropertySymbol
import org.jetbrains.kotlin.fir.resolve.dfa.cfg.*
import org.jetbrains.kotlin.fir.resolve.dfa.cfg.ControlFlowGraph.Kind
import org.jetbrains.kotlin.fir.symbols.FirBasedSymbol
import org.jetbrains.kotlin.fir.symbols.SymbolInternals
import org.jetbrains.kotlin.fir.symbols.impl.FirPropertySymbol
import org.jetbrains.kotlin.fir.symbols.impl.FirSyntheticPropertySymbol
import org.jetbrains.kotlin.fir.types.resolvedType
object FirPropertyInitializationAnalyzer : AbstractFirPropertyInitializationChecker(MppCheckerKind.Common) {
override fun analyze(data: PropertyInitializationInfoData, reporter: DiagnosticReporter, context: CheckerContext) {
data.checkPropertyAccesses(isForInitialization = false, context, reporter)
}
}
val FirDeclaration.evaluatedInPlace: Boolean
get() = when (this) {
is FirAnonymousFunction -> invocationKind.isInPlace
is FirAnonymousObject -> classKind != ClassKind.ENUM_ENTRY
is FirConstructor -> true // child of class initialization graph
is FirFunction, is FirClass -> false
else -> true // property initializer, etc.
}
/**
* [isForInitialization] means that caller is interested in member property in the scope
* of file or class initialization section. In this case the fact that property has
* initializer does not mean that it's safe to access this property in any place:
*
* ```
* class A {
* val b = a // a is not initialized here
* val a = 10
* val c = a // but initialized here
* }
* ```
*/
@OptIn(SymbolInternals::class)
fun FirPropertySymbol.requiresInitialization(isForInitialization: Boolean): Boolean {
val hasImplicitBackingField = !hasExplicitBackingField && hasBackingField
return when {
this is FirSyntheticPropertySymbol -> false
isForInitialization -> hasDelegate || hasImplicitBackingField
else -> !hasInitializer && hasImplicitBackingField && fir.isCatchParameter != true
}
}
fun PropertyInitializationInfoData.checkPropertyAccesses(
isForInitialization: Boolean,
context: CheckerContext,
reporter: DiagnosticReporter
) {
// If a property has an initializer (or does not need one), then any reads are OK while any writes are OK
// if it's a `var` and bad if it's a `val`. `FirReassignmentAndInvisibleSetterChecker` does this without a CFG.
val filtered = properties.filterTo(mutableSetOf()) {
it.requiresInitialization(isForInitialization) || it in conditionallyInitializedProperties
}
if (filtered.isEmpty()) return
checkPropertyAccesses(
graph, filtered, context, reporter, scope = null,
isForInitialization,
doNotReportUninitializedVariable = false,
doNotReportConstantUninitialized = true,
scopes = mutableMapOf(),
)
}
private fun PropertyInitializationInfoData.checkPropertyAccesses(
graph: ControlFlowGraph,
properties: Set,
context: CheckerContext,
reporter: DiagnosticReporter,
scope: FirDeclaration?,
isForInitialization: Boolean,
doNotReportUninitializedVariable: Boolean,
doNotReportConstantUninitialized: Boolean,
scopes: MutableMap,
) {
val capturedInitializationError = if (receiver != null)
FirErrors.CAPTURED_MEMBER_VAL_INITIALIZATION
else
FirErrors.CAPTURED_VAL_INITIALIZATION
// TODO: move this to PropertyInitializationInfoData (the collector also does this check when visiting assignments)
fun FirQualifiedAccessExpression.hasMatchingReceiver(): Boolean {
val expression = dispatchReceiver?.unwrapSmartcastExpression()
return (expression as? FirThisReceiverExpression)?.calleeReference?.boundSymbol == receiver ||
(expression as? FirResolvedQualifier)?.symbol == receiver
}
fun CFGNode<*>.reportErrorsOnInitializationsInInputs(symbol: FirPropertySymbol, path: EdgeLabel) {
for (previousNode in previousCfgNodes) {
if (edgeFrom(previousNode).kind.isBack) continue
when (val assignmentNode = getValue(previousNode)[path]?.get(symbol)?.location) {
is VariableDeclarationNode -> {} // unreachable - `val`s with initializers do not require hindsight
is VariableAssignmentNode ->
reporter.reportOn(assignmentNode.fir.lValue.source, capturedInitializationError, symbol, context)
else -> // merge node for a branching construct, e.g. `if (p) { x = 1 } else { x = 2 }` - report on all branches
assignmentNode?.reportErrorsOnInitializationsInInputs(symbol, path)
}
}
}
for (node in graph.nodes) {
if (node.isUnion) {
for ((path, data) in getValue(node)) {
for ((symbol, range) in data) {
if (!symbol.isVal || !range.canBeRevisited() || symbol !in properties) continue
// This can be something like `f({ x = 1 }, { x = 2 })` where `f` calls both lambdas in-place.
// At each assignment it was only considered in isolation, but now that we're merging their control flows,
// we can see that the assignments clash, so we need to go back and emit errors on these nodes.
if (node.previousCfgNodes.all { getValue(it)[path]?.get(symbol)?.canBeRevisited() != true }) {
node.reportErrorsOnInitializationsInInputs(symbol, path)
}
}
}
}
when {
node is VariableDeclarationNode -> {
val symbol = node.fir.symbol
if (scope != null && receiver == null && node.fir.isVal && symbol in properties) {
// It's OK to initialize this variable from a nested called-in-place function, but not from
// a non-called-in-place function or a non-anonymous-object class initializer.
scopes[symbol] = scope
}
}
node is VariableAssignmentNode -> {
val symbol = node.fir.calleeReference?.toResolvedPropertySymbol() ?: continue
if (!symbol.isVal || node.fir.unwrapLValue()?.hasMatchingReceiver() != true || symbol !in properties) continue
val info = getValue(node)
if (info.values.any { it[symbol]?.canBeRevisited() == true }) {
reporter.reportOn(node.fir.lValue.source, FirErrors.VAL_REASSIGNMENT, symbol, context)
} else if (scope != scopes[symbol]) {
reporter.reportOn(node.fir.lValue.source, capturedInitializationError, symbol, context)
} else if (!symbol.isLocal && !node.owner.isInline(until = symbol.getContainingSymbol(context.session))) {
// If the assignment is inside INVOKE_ONCE lambda and the lambda is not inlined,
// backend generates either separate function or separate class for the lambda.
// If we try to initialize non-static final field there, we will get exception at
// runtime, since we can initialize such fields only inside constructors.
reporter.reportOn(node.fir.lValue.source, FirErrors.NON_INLINE_MEMBER_VAL_INITIALIZATION, symbol, context)
}
}
node is QualifiedAccessNode -> {
if (doNotReportUninitializedVariable) continue
if (node.fir.resolvedType.hasDiagnosticKind(DiagnosticKind.RecursionInImplicitTypes)) continue
val symbol = node.fir.calleeReference.toResolvedPropertySymbol() ?: continue
if (doNotReportConstantUninitialized && symbol.isConst) continue
if (!symbol.isLateInit && !symbol.isExternal && node.fir.hasMatchingReceiver() && symbol in properties &&
getValue(node).values.any { it[symbol]?.isDefinitelyVisited() != true }
) {
reporter.reportOn(node.fir.source, FirErrors.UNINITIALIZED_VARIABLE, symbol, context)
}
}
// In the class case, subgraphs of the exit node are member functions, which are considered to not
// be part of initialization, so any val is considered to be initialized there and the CFG is not
// needed. The errors on reassignments will be emitted by `FirReassignmentAndInvisibleSetterChecker`.
node is CFGNodeWithSubgraphs<*> && (receiver == null || node !== graph.exitNode) -> {
for (subGraph in node.subGraphs) {
/*
* For class initialization graph we allow to read properties in non-in-place lambdas
* even if they may be not initialized at this point, because if lambda is not in-place,
* then it most likely will be called after object will be initialized
*/
val doNotReportForSubGraph = isForInitialization && subGraph.kind.doNotReportUninitializedVariableForInitialization
// Must report uninitialized variable if we start initializing a constant property. This
// allows "regular" properties to reference constant properties out-of-order, but all other
// property references must be in-order.
val isSubGraphConstProperty = (subGraph.declaration as? FirProperty)?.isConst == true
val newScope = subGraph.declaration?.takeIf { !it.evaluatedInPlace } ?: scope
checkPropertyAccesses(
subGraph, properties, context, reporter, newScope,
isForInitialization,
doNotReportUninitializedVariable = doNotReportUninitializedVariable || doNotReportForSubGraph,
doNotReportConstantUninitialized = doNotReportConstantUninitialized && !isSubGraphConstProperty,
scopes
)
}
}
}
}
}
private val Kind.doNotReportUninitializedVariableForInitialization: Boolean
get() = when (this) {
Kind.Function, Kind.AnonymousFunction, Kind.LocalFunction -> true
else -> false
}
/**
* Determine if this declaration is evaluated inline. This is distinct from [evaluatedInPlace], as the declaration must also be inlined by
* the compiler.
*/
private val FirDeclaration.evaluatedInline: Boolean
get() = when (this) {
is FirAnonymousFunction -> inlineStatus == InlineStatus.Inline
is FirConstructor -> true // child of class initialization graph
is FirFunction, is FirClass -> false
else -> true // property initializer, etc.
}
/**
* Checks that [ControlFlowGraph.declaration] is [evaluatedInline], and also recursively check all
* parent [ControlFlowGraph]s.
*
* @param until will stop recursion if [ControlFlowGraph.declaration] matches the specified symbol.
* This is used to stop recursion when there are nested declarations (like a local class), and we
* only need to check until that nested declaration.
*/
private fun ControlFlowGraph.isInline(until: FirBasedSymbol<*>?): Boolean {
val declaration = declaration
if (declaration?.symbol == until) return true
if (declaration?.evaluatedInline != true) return false
return enterNode.previousNodes.all { it.owner.isInline(until) }
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy