org.jetbrains.kotlin.fir.resolve.dfa.Flow.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-compiler-embeddable Show documentation
Show all versions of kotlin-compiler-embeddable Show documentation
the Kotlin compiler embeddable
/*
* Copyright 2010-2021 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.resolve.dfa
import kotlinx.collections.immutable.PersistentList
import kotlinx.collections.immutable.PersistentMap
import kotlinx.collections.immutable.PersistentSet
import kotlinx.collections.immutable.persistentHashMapOf
abstract class Flow {
abstract val knownVariables: Set
abstract fun unwrapVariable(variable: RealVariable): RealVariable
abstract fun getTypeStatement(variable: RealVariable): TypeStatement?
abstract fun getImplications(variable: DataFlowVariable): Collection?
}
class PersistentFlow internal constructor(
private val previousFlow: PersistentFlow?,
private val approvedTypeStatements: PersistentMap,
internal val implications: PersistentMap>,
// RealVariable describes a storage in memory; a pair of RealVariable with its assignment
// index at a particular execution point forms an SSA value corresponding to the result of
// an initializer.
internal val assignmentIndex: PersistentMap,
// RealVariables thus form equivalence sets by values they reference. One is chosen
// as a representative of that set, while the rest are mapped to that representative
// in `directAliasMap`. `backwardsAliasMap` maps each representative to the rest of the set.
internal val directAliasMap: PersistentMap,
private val backwardsAliasMap: PersistentMap>,
) : Flow() {
private val level: Int = if (previousFlow != null) previousFlow.level + 1 else 0
override val knownVariables: Set
get() = approvedTypeStatements.keys + directAliasMap.keys
val allVariablesForDebug: Set
get() = knownVariables + implications.keys + implications.values.flatten().map { it.effect.variable }
override fun unwrapVariable(variable: RealVariable): RealVariable =
directAliasMap[variable] ?: variable
override fun getTypeStatement(variable: RealVariable): TypeStatement? =
approvedTypeStatements[unwrapVariable(variable)]?.copy(variable = variable)
override fun getImplications(variable: DataFlowVariable): Collection? =
implications[variable]
fun lowestCommonAncestor(other: PersistentFlow): PersistentFlow? {
var left = this
var right = other
while (left.level > right.level) {
left = left.previousFlow ?: return null
}
while (right.level > left.level) {
right = right.previousFlow ?: return null
}
while (left != right) {
left = left.previousFlow ?: return null
right = right.previousFlow ?: return null
}
return left
}
fun fork(): MutableFlow = MutableFlow(
this,
approvedTypeStatements.builder(),
implications.builder(),
assignmentIndex.builder(),
directAliasMap.builder(),
backwardsAliasMap.builder(),
)
}
class MutableFlow internal constructor(
private val previousFlow: PersistentFlow?,
internal val approvedTypeStatements: PersistentMap.Builder,
internal val implications: PersistentMap.Builder>,
internal val assignmentIndex: PersistentMap.Builder,
internal val directAliasMap: PersistentMap.Builder,
internal val backwardsAliasMap: PersistentMap.Builder>,
) : Flow() {
constructor() : this(
null,
emptyPersistentHashMapBuilder(),
emptyPersistentHashMapBuilder(),
emptyPersistentHashMapBuilder(),
emptyPersistentHashMapBuilder(),
emptyPersistentHashMapBuilder(),
)
override val knownVariables: Set
get() = approvedTypeStatements.keys + directAliasMap.keys
override fun unwrapVariable(variable: RealVariable): RealVariable =
directAliasMap[variable] ?: variable
override fun getTypeStatement(variable: RealVariable): TypeStatement? =
approvedTypeStatements[unwrapVariable(variable)]?.copy(variable = variable)
override fun getImplications(variable: DataFlowVariable): Collection? =
implications[variable]
fun freeze(): PersistentFlow = PersistentFlow(
previousFlow,
approvedTypeStatements.build(),
implications.build(),
assignmentIndex.build(),
directAliasMap.build(),
backwardsAliasMap.build(),
)
}
private fun emptyPersistentHashMapBuilder(): PersistentMap.Builder =
persistentHashMapOf().builder()