All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.backend.konan.lower.InnerClassLowering.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-RC2
Show newest version
/*
 * Copyright 2010-2018 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license
 * that can be found in the LICENSE file.
 */

package org.jetbrains.kotlin.backend.konan.lower

import org.jetbrains.kotlin.backend.common.ClassLoweringPass
import org.jetbrains.kotlin.backend.common.IrElementTransformerVoidWithContext
import org.jetbrains.kotlin.backend.common.lower.ConstructorDelegationKind
import org.jetbrains.kotlin.backend.common.lower.InnerClassesSupport
import org.jetbrains.kotlin.backend.common.lower.delegationKind
import org.jetbrains.kotlin.backend.konan.Context
import org.jetbrains.kotlin.backend.konan.descriptors.synthesizedName
import org.jetbrains.kotlin.descriptors.DescriptorVisibilities
import org.jetbrains.kotlin.ir.IrStatement
import org.jetbrains.kotlin.ir.builders.declarations.buildField
import org.jetbrains.kotlin.ir.declarations.*
import org.jetbrains.kotlin.ir.expressions.IrBlockBody
import org.jetbrains.kotlin.ir.expressions.IrExpression
import org.jetbrains.kotlin.ir.expressions.IrGetValue
import org.jetbrains.kotlin.ir.expressions.impl.IrGetFieldImpl
import org.jetbrains.kotlin.ir.expressions.impl.IrGetValueImpl
import org.jetbrains.kotlin.ir.expressions.impl.IrSetFieldImpl
import org.jetbrains.kotlin.ir.irAttribute
import org.jetbrains.kotlin.ir.symbols.IrConstructorSymbol
import org.jetbrains.kotlin.ir.symbols.IrFieldSymbol
import org.jetbrains.kotlin.ir.symbols.IrFunctionSymbol
import org.jetbrains.kotlin.ir.types.classifierOrNull
import org.jetbrains.kotlin.ir.util.*
import org.jetbrains.kotlin.ir.visitors.transformChildrenVoid
import org.jetbrains.kotlin.utils.addToStdlib.shouldNotBeCalled
import org.jetbrains.kotlin.utils.addToStdlib.getOrSetIfNull

private var IrClass.outerThisField: IrField? by irAttribute(followAttributeOwner = false)

internal class NativeInnerClassesSupport(private val irFactory: IrFactory) : InnerClassesSupport {
    override fun getOuterThisField(innerClass: IrClass): IrField {
        require(innerClass.isInner) { "Expected an inner class: ${innerClass.render()}" }
        return innerClass::outerThisField.getOrSetIfNull {
            val outerClass = innerClass.parentClassOrNull ?: error("No containing class for inner class ${innerClass.render()}")

            irFactory.buildField {
                startOffset = innerClass.startOffset
                endOffset = innerClass.endOffset
                origin = IrDeclarationOrigin.FIELD_FOR_OUTER_THIS
                name = "this$0".synthesizedName // TODO: other backends have "$this" here.
                type = outerClass.defaultType
                visibility = DescriptorVisibilities.PROTECTED
                isFinal = true
                isExternal = false
                isStatic = false
            }.also {
                it.parent = innerClass
            }
        }
    }

    override fun getInnerClassConstructorWithOuterThisParameter(innerClassConstructor: IrConstructor): IrConstructor = shouldNotBeCalled()

    override fun getInnerClassOriginalPrimaryConstructorOrNull(innerClass: IrClass): IrConstructor = shouldNotBeCalled()
}

internal class OuterThisLowering(val context: Context) : ClassLoweringPass {
    override fun lower(irClass: IrClass) {
        if (!irClass.isInner) return
        irClass.transformChildrenVoid(Transformer(irClass, irClass))
    }

    fun lower(irFunction: IrFunction) {
        var parent = irFunction.parent
        var irClass: IrClass? = null
        while (parent !is IrPackageFragment) {
            irClass = parent as? IrClass
            if (irClass != null) break
            parent = (parent as IrDeclaration).parent
        }
        if (irClass == null || !irClass.isInner) return

        irFunction.body?.transformChildrenVoid(Transformer(irClass, irFunction))
    }

    private inner class Transformer(val irClass: IrClass, val container: IrDeclaration) : IrElementTransformerVoidWithContext() {
        override fun visitClassNew(declaration: IrClass): IrStatement {
            // Skip nested.
            return declaration
        }

        override fun visitGetValue(expression: IrGetValue): IrExpression {
            expression.transformChildrenVoid(this)

            val implicitThisClass = (expression.symbol.owner.parent as? IrClass) ?: return expression

            if (implicitThisClass == irClass) return expression

            val parentScopeSymbols = listOf(container.symbol) + allScopes.map { it.scope.scopeOwnerSymbol }
            var functionSymbol: IrFunctionSymbol? = null
            for (i in parentScopeSymbols.size - 1 downTo 0) {
                val currentSymbol = parentScopeSymbols[i] as? IrFunctionSymbol ?: break
                functionSymbol = currentSymbol
            }
            if (functionSymbol == null) return expression

            val startOffset = expression.startOffset
            val endOffset = expression.endOffset
            val origin = expression.origin

            var irThis: IrExpression
            var innerClass: IrClass
            if ((functionSymbol as? IrConstructorSymbol)?.owner?.parentAsClass == irClass) {
                // For constructor we have outer class as dispatchReceiverParameter.
                innerClass = irClass.parent as? IrClass ?: error("No containing class for inner class ${irClass.render()}")
                val thisParameter = functionSymbol.owner.dispatchReceiverParameter!!
                irThis = IrGetValueImpl(startOffset, endOffset, thisParameter.type, thisParameter.symbol, origin)
            } else {
                innerClass = irClass

                val currentFunctionReceiver = functionSymbol.owner.dispatchReceiverParameter
                val thisParameter =
                        if (currentFunctionReceiver?.type?.classifierOrNull == irClass.symbol)
                            currentFunctionReceiver
                        else
                            irClass.thisReceiver!!

                irThis = IrGetValueImpl(startOffset, endOffset, thisParameter.type, thisParameter.symbol, origin)
            }

            while (innerClass != implicitThisClass) {
                if (!innerClass.isInner) {
                    // Captured 'this' unrelated to inner classes nesting hierarchy, leave it as is -
                    // should be transformed by closures conversion.
                    return expression
                }

                val outerThisField = context.innerClassesSupport.getOuterThisField(innerClass)
                irThis = IrGetFieldImpl(
                        startOffset, endOffset,
                        outerThisField.symbol, outerThisField.type,
                        irThis,
                        origin
                )

                val outer = innerClass.parent
                innerClass = outer as? IrClass
                        ?: throw AssertionError("Unexpected containing declaration for inner class ${innerClass.dump()}: $outer")
            }

            return irThis
        }
    }
}

internal class InnerClassLowering(val context: Context) : ClassLoweringPass {
    override fun lower(irClass: IrClass) {
        OuterThisLowering(context).lower(irClass)
        InnerClassTransformer(irClass).lowerInnerClass()
    }

    private inner class InnerClassTransformer(val irClass: IrClass) {
        lateinit var outerThisFieldSymbol: IrFieldSymbol

        fun lowerInnerClass() {
            if (!irClass.isInner) return

            createOuterThisField()
            lowerConstructors()
        }

        private fun createOuterThisField() {
            val outerThisField = context.innerClassesSupport.getOuterThisField(irClass)
            irClass.declarations += outerThisField
            outerThisFieldSymbol = outerThisField.symbol
        }

        private fun lowerConstructors() {
            irClass.declarations.transformFlat { irMember ->
                if (irMember is IrConstructor)
                    listOf(lowerConstructor(irMember))
                else
                    null
            }
        }

        private fun lowerConstructor(irConstructor: IrConstructor): IrConstructor {
            if (irConstructor.delegationKind(context.irBuiltIns) == ConstructorDelegationKind.CALLS_SUPER) {
                // Initializing constructor: initialize 'this.this$0' with '$outer'.
                val blockBody = irConstructor.body as? IrBlockBody
                        ?: throw AssertionError("Unexpected constructor body: ${irConstructor.body}")
                val startOffset = irConstructor.startOffset
                val endOffset = irConstructor.endOffset
                val thisReceiver = irClass.thisReceiver!!
                val outerReceiver = irConstructor.dispatchReceiverParameter!!
                blockBody.statements.add(
                        0,
                        IrSetFieldImpl(
                                startOffset, endOffset, outerThisFieldSymbol,
                                IrGetValueImpl(startOffset, endOffset, thisReceiver.type, thisReceiver.symbol),
                                IrGetValueImpl(startOffset, endOffset, outerReceiver.type, outerReceiver.symbol),
                                context.irBuiltIns.unitType
                        )
                )
            }

            return irConstructor
        }
    }
}





© 2015 - 2024 Weber Informatics LLC | Privacy Policy