org.jetbrains.kotlin.fir.backend.Fir2IrConversionScope.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.backend
import org.jetbrains.kotlin.fir.declarations.FirClass
import org.jetbrains.kotlin.fir.declarations.FirConstructor
import org.jetbrains.kotlin.fir.declarations.FirProperty
import org.jetbrains.kotlin.fir.declarations.FirPropertyAccessor
import org.jetbrains.kotlin.fir.expressions.FirReturnExpression
import org.jetbrains.kotlin.ir.declarations.*
import org.jetbrains.kotlin.ir.util.isSetter
import org.jetbrains.kotlin.ir.util.parentClassOrNull
import org.jetbrains.kotlin.util.PrivateForInline
@OptIn(PrivateForInline::class)
class Fir2IrConversionScope {
@PublishedApi
@PrivateForInline
internal val parentStack = mutableListOf()
@PublishedApi
@PrivateForInline
internal val containingFirClassStack = mutableListOf()
@PublishedApi
@PrivateForInline
internal val currentlyGeneratedDelegatedConstructors = mutableMapOf()
inline fun withParent(parent: T, f: T.() -> R): R {
parentStack += parent
try {
return parent.f()
} finally {
parentStack.removeAt(parentStack.size - 1)
}
}
internal fun forDelegatingConstructorCall(constructor: IrConstructor, irClass: IrClass, f: () -> T): T {
currentlyGeneratedDelegatedConstructors[irClass] = constructor
try {
return f()
} finally {
currentlyGeneratedDelegatedConstructors.remove(irClass)
}
}
fun getConstructorForCurrentlyGeneratedDelegatedConstructor(itClass: IrClass): IrConstructor? =
currentlyGeneratedDelegatedConstructors[itClass]
fun containingFileIfAny(): IrFile? = parentStack.getOrNull(0) as? IrFile
inline fun withContainingFirClass(containingFirClass: FirClass, f: () -> Unit) {
containingFirClassStack += containingFirClass
try {
f()
} finally {
containingFirClassStack.removeAt(containingFirClassStack.size - 1)
}
}
fun parentFromStack(): IrDeclarationParent = parentStack.last()
fun parentAccessorOfPropertyFromStack(property: IrProperty): IrSimpleFunction? {
for (parent in parentStack.asReversed()) {
when (parent) {
property.getter -> return property.getter
property.setter -> return property.setter
}
}
return null
}
fun applyParentFromStackTo(declaration: T): T {
declaration.parent = parentStack.last()
return declaration
}
fun containerFirClass(): FirClass? = containingFirClassStack.lastOrNull()
@PublishedApi
@PrivateForInline
internal val functionStack = mutableListOf()
inline fun withFunction(function: T, f: T.() -> R): R {
functionStack += function
try {
return function.f()
} finally {
functionStack.removeAt(functionStack.size - 1)
}
}
@PublishedApi
@PrivateForInline
internal val propertyStack = mutableListOf>()
inline fun withProperty(property: IrProperty, firProperty: FirProperty? = null, f: IrProperty.() -> R): R {
propertyStack += (property to firProperty)
try {
return property.f()
} finally {
propertyStack.removeAt(propertyStack.size - 1)
}
}
@PublishedApi
@PrivateForInline
internal val classStack = mutableListOf()
inline fun withClass(klass: IrClass, f: IrClass.() -> R): R {
classStack += klass
return try {
klass.f()
} finally {
classStack.removeAt(classStack.size - 1)
}
}
@PublishedApi
@PrivateForInline
internal val whenSubjectVariableStack = mutableListOf()
@PublishedApi
@PrivateForInline
internal val safeCallSubjectVariableStack = mutableListOf()
inline fun withWhenSubject(subject: IrVariable?, f: () -> T): T {
if (subject != null) whenSubjectVariableStack += subject
try {
return f()
} finally {
if (subject != null) whenSubjectVariableStack.removeAt(whenSubjectVariableStack.size - 1)
}
}
inline fun withSafeCallSubject(subject: IrVariable?, f: () -> T): T {
if (subject != null) safeCallSubjectVariableStack += subject
try {
return f()
} finally {
if (subject != null) safeCallSubjectVariableStack.removeAt(safeCallSubjectVariableStack.size - 1)
}
}
fun returnTarget(expression: FirReturnExpression, declarationStorage: Fir2IrDeclarationStorage): IrFunction {
val irTarget = when (val firTarget = expression.target.labeledElement) {
is FirConstructor -> declarationStorage.getCachedIrConstructor(firTarget)
is FirPropertyAccessor -> {
var answer: IrFunction? = null
for ((property, firProperty) in propertyStack.asReversed()) {
if (firProperty?.getter === firTarget) {
answer = property.getter
} else if (firProperty?.setter === firTarget) {
answer = property.setter
}
}
answer
}
else -> declarationStorage.getCachedIrFunction(firTarget)
}
for (potentialTarget in functionStack.asReversed()) {
if (potentialTarget == irTarget) {
return potentialTarget
}
}
return functionStack.last()
}
fun parent(): IrDeclarationParent? = parentStack.lastOrNull()
fun defaultConversionTypeOrigin(): ConversionTypeOrigin =
if ((parent() as? IrFunction)?.isSetter == true) ConversionTypeOrigin.SETTER else ConversionTypeOrigin.DEFAULT
fun dispatchReceiverParameter(irClass: IrClass): IrValueParameter? {
for (function in functionStack.asReversed()) {
if (function.parentClassOrNull == irClass) {
// An inner class's constructor needs an instance of the outer class as a dispatch receiver.
// However, if we are converting `this` receiver inside that constructor, now we should point to the inner class instance.
if (function is IrConstructor && irClass.isInner) {
irClass.thisReceiver?.let { return it }
}
function.dispatchReceiverParameter?.let { return it }
}
}
return irClass.thisReceiver
}
fun lastClass(): IrClass? = classStack.lastOrNull()
fun lastWhenSubject(): IrVariable = whenSubjectVariableStack.last()
fun lastSafeCallSubject(): IrVariable = safeCallSubjectVariableStack.last()
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy