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

org.jetbrains.kotlin.backend.common.actualizer.ActualFakeOverridesAdder.kt Maven / Gradle / Ivy

There is a newer version: 2.1.20-Beta1
Show newest version
/*
 * Copyright 2010-2023 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.backend.common.actualizer

import org.jetbrains.kotlin.descriptors.DescriptorVisibilities
import org.jetbrains.kotlin.ir.IrElement
import org.jetbrains.kotlin.ir.declarations.*
import org.jetbrains.kotlin.ir.overrides.IrOverrideChecker
import org.jetbrains.kotlin.ir.symbols.IrClassSymbol
import org.jetbrains.kotlin.ir.symbols.IrPropertySymbol
import org.jetbrains.kotlin.ir.symbols.IrSymbol
import org.jetbrains.kotlin.ir.types.IrTypeSystemContext
import org.jetbrains.kotlin.ir.types.classifierOrFail
import org.jetbrains.kotlin.ir.visitors.IrElementVisitorVoid
import org.jetbrains.kotlin.ir.visitors.acceptChildrenVoid
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.resolve.OverridingUtil

/**
 * It adds fake overrides to non-expect classes inside common or multi-platform module,
 * taken from these non-expect classes actualized super classes.
 *
 * In case when a non-expect class has direct or indirect expect supertypes,
 * it may happen that the actual classes for these supertypes contain additional (non-actual) members that don't exist in their expect counterparts.
 * We still should have fake overrides generated for these members, but FIR2IR isn't able to see their base members in common or multi-platform module.
 * This class is intended to search for such situations and generate such fake overrides.
 */
internal class ActualFakeOverridesAdder(
    private val expectActualMap: Map,
    private val expectToActualClassMap: Map,
    private val typeSystemContext: IrTypeSystemContext
) : IrElementVisitorVoid {
    private val overrideChecker = IrOverrideChecker(typeSystemContext, emptyList())
    private val missingActualMembersMap = mutableMapOf()

    override fun visitClass(declaration: IrClass) {
        extractMissingActualMembersFromSupertypes(declaration)
        visitElement(declaration)
    }

    override fun visitElement(element: IrElement) {
        element.acceptChildrenVoid(this)
    }

    private fun extractMissingActualMembersFromSupertypes(klass: IrClass): FakeOverrideInfo {
        missingActualMembersMap[klass]?.let { return it }

        val missingActualMembers = FakeOverrideInfo()
        missingActualMembersMap[klass] = missingActualMembers

        // New members from supertypes are only relevant for not expect (ordinary) classes
        // New members from the current class are only relevant for actualized expect classes

        val processedMembers = FakeOverrideInfo()
        for (superType in klass.superTypes) {
            val membersFromSupertype = extractMissingActualMembersFromSupertypes(superType.classifierOrFail.owner as IrClass)
            if (!klass.isExpect) {
                appendMissingMembersToNotExpectClass(missingActualMembers, klass, membersFromSupertype.allSymbols(), processedMembers)
            }
        }

        val actualClass = expectActualMap[klass.symbol]?.owner as? IrClass ?: return missingActualMembers

        missingActualMembers.appendMissingMembersFromActualizedExpectClass(klass, actualClass)

        return missingActualMembers
    }

    private fun appendMissingMembersToNotExpectClass(
        fakeOverrideInfo: FakeOverrideInfo,
        klass: IrClass,
        membersFromSupertype: List,
        processedMembers: FakeOverrideInfo
    ) {
        for (symbolFromSupertype in membersFromSupertype) {
            val memberFromSupertype = symbolFromSupertype.owner as IrDeclaration

            if (memberFromSupertype is IrOverridableMember) {
                // We can land here because of a hierarchy like
                // actual A -> common B -> actual C
                // where C defines a member x and A overrides the member x.
                // We will first add a fake-override x to B and then land here.
                // In this case we don't want to add a fake-override on top of the real override to A.
                // Instead, we add the fake-override x to the overridden symbols of A.x.

                @Suppress("UNCHECKED_CAST")
                val override = klass.declarations.firstOrNull {
                    it is IrOverridableMember &&
                            overrideChecker.isOverridableBy(
                                superMember = memberFromSupertype,
                                subMember = it,
                                checkIsInlineFlag = false,
                            ).result == OverridingUtil.OverrideCompatibilityInfo.Result.OVERRIDABLE
                } as? IrOverridableDeclaration

                if (override != null) {
                    override.overriddenSymbols += symbolFromSupertype
                    if (override is IrProperty && symbolFromSupertype is IrPropertySymbol) {
                        override.getter?.let { getter -> symbolFromSupertype.owner.getter?.symbol?.let { getter.overriddenSymbols += it } }
                        override.setter?.let { setter -> symbolFromSupertype.owner.setter?.symbol?.let { setter.overriddenSymbols += it } }
                    }
                    continue
                }
            }

            val newMember = createFakeOverrideMember(listOf(memberFromSupertype), klass)
            val matchingFakeOverrides = collectActualCallablesMatchingToSpecificExpect(
                newMember.symbol,
                fakeOverrideInfo.getMembersForActual(newMember),
                expectToActualClassMap,
                typeSystemContext
            )
            if (matchingFakeOverrides.isEmpty()) {
                processedMembers.addMember(memberFromSupertype as IrOverridableDeclaration<*>)
                fakeOverrideInfo.addMember(newMember)
                klass.addMember(newMember)
            }
        }
    }

    private fun FakeOverrideInfo.appendMissingMembersFromActualizedExpectClass(
        expectClass: IrClass,
        actualClass: IrClass,
    ) {
        val actualWithCorrespondingExpectMembers = hashSetOf().apply {
            expectClass.declarations.mapNotNullTo(this) { expectActualMap[(it as? IrOverridableDeclaration<*>)?.symbol] }
        }

        for (actualMember in actualClass.declarations) {
            if (actualMember is IrOverridableDeclaration<*> &&
                (actualMember as? IrDeclarationWithVisibility)?.visibility != DescriptorVisibilities.PRIVATE &&
                !actualWithCorrespondingExpectMembers.contains(actualMember.symbol)
            ) {
                addMember(actualMember)
            }
        }
    }

    private class FakeOverrideInfo {
        val functionsByName: MutableMap> = mutableMapOf()
        val propertiesByName: MutableMap> = mutableMapOf()

        fun allSymbols(): List {
            return buildList {
                functionsByName.values.flatMapTo(this) { it }
                propertiesByName.values.flatMapTo(this) { it }
            }
        }

        private fun getCorrespondingMap(member: IrOverridableDeclaration<*>): MutableMap> {
            return when (member) {
                is IrFunction -> functionsByName
                is IrProperty -> propertiesByName
                else -> error("Unsupported declaration type: $member")
            }
        }

        fun addMember(member: IrOverridableDeclaration<*>) {
            getCorrespondingMap(member).getOrPut(member.name) { mutableListOf() } += member.symbol
        }

        fun getMembersForActual(actualMember: IrOverridableDeclaration<*>): List {
            return getCorrespondingMap(actualMember)[actualMember.name].orEmpty()
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy