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

org.jetbrains.kotlin.resolve.jvm.checkers.InterfaceDefaultMethodCallChecker.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-RC
Show newest version
/*
 * Copyright 2010-2017 JetBrains s.r.o.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.jetbrains.kotlin.resolve.jvm.checkers

import com.intellij.openapi.project.Project
import com.intellij.psi.PsiElement
import org.jetbrains.kotlin.config.JvmAnalysisFlags
import org.jetbrains.kotlin.config.JvmTarget
import org.jetbrains.kotlin.descriptors.CallableMemberDescriptor
import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.PropertyDescriptor
import org.jetbrains.kotlin.load.java.descriptors.JavaCallableMemberDescriptor
import org.jetbrains.kotlin.load.java.descriptors.JavaPropertyDescriptor
import org.jetbrains.kotlin.psi.KtClassBody
import org.jetbrains.kotlin.psi.KtNamedFunction
import org.jetbrains.kotlin.psi.KtProperty
import org.jetbrains.kotlin.psi.KtSuperExpression
import org.jetbrains.kotlin.resolve.BindingContext
import org.jetbrains.kotlin.resolve.DescriptorUtils.isInterface
import org.jetbrains.kotlin.resolve.DescriptorUtils.unwrapFakeOverride
import org.jetbrains.kotlin.resolve.LanguageVersionSettingsProvider
import org.jetbrains.kotlin.resolve.calls.checkers.CallChecker
import org.jetbrains.kotlin.resolve.calls.checkers.CallCheckerContext
import org.jetbrains.kotlin.resolve.calls.model.ResolvedCall
import org.jetbrains.kotlin.resolve.calls.util.getSuperCallExpression
import org.jetbrains.kotlin.resolve.jvm.diagnostics.ErrorsJvm.INTERFACE_CANT_CALL_DEFAULT_METHOD_VIA_SUPER

class InterfaceDefaultMethodCallChecker(val jvmTarget: JvmTarget, project: Project) : CallChecker {
    private val ideService = LanguageVersionSettingsProvider.getInstance(project)

    override fun check(resolvedCall: ResolvedCall<*>, reportOn: PsiElement, context: CallCheckerContext) {
        val descriptor = resolvedCall.resultingDescriptor as? CallableMemberDescriptor ?: return
        if (descriptor is JavaPropertyDescriptor) return

        val superCallExpression = getSuperCallExpression(resolvedCall.call) ?: return

        if (!isInterface(descriptor.original.containingDeclaration)) return

        val realDescriptor = unwrapFakeOverride(descriptor)
        val realDescriptorOwner = realDescriptor.containingDeclaration as? ClassDescriptor ?: return

        val jvmDefaultMode = context.languageVersionSettings.getFlag(JvmAnalysisFlags.jvmDefaultMode)
        if (isInterface(realDescriptorOwner) && (realDescriptor is JavaCallableMemberDescriptor ||
                    realDescriptor.isCompiledToJvmDefaultWithProperMode(ideService, jvmDefaultMode))
        ) {
            val bindingContext = context.trace.bindingContext
            val thisForSuperCall = getSuperCallLabelTarget(bindingContext, superCallExpression)

            if (thisForSuperCall != null && isInterface(thisForSuperCall)) {
                val declarationWithCall = findInterfaceMember(thisForSuperCall, superCallExpression, bindingContext)
                if (declarationWithCall?.isCompiledToJvmDefaultWithProperMode(ideService, jvmDefaultMode) == false) {
                    context.trace.report(INTERFACE_CANT_CALL_DEFAULT_METHOD_VIA_SUPER.on(reportOn))
                    return
                }
            }
        }
    }

    private fun findInterfaceMember(
        descriptorToSearch: ClassDescriptor,
        startExpression: KtSuperExpression,
        bindingContext: BindingContext
    ): CallableMemberDescriptor? {
        val parents = generateSequence({ startExpression.parent }) { it.parent }
        parents.fold(startExpression) { child, parent ->
            if (parent is KtClassBody &&
                descriptorToSearch == bindingContext.get(BindingContext.CLASS, parent.parent)
            ) {
                return when (child) {
                    is KtNamedFunction -> bindingContext.get(BindingContext.FUNCTION, child)
                    is KtProperty -> bindingContext.get(BindingContext.VARIABLE, child) as? PropertyDescriptor
                    else -> null
                }
            } else parent
        }

        return null
    }


    private fun getSuperCallLabelTarget(
        bindingContext: BindingContext,
        expression: KtSuperExpression
    ): ClassDescriptor? {
        val thisTypeForSuperCall = bindingContext.get(BindingContext.THIS_TYPE_FOR_SUPER_EXPRESSION, expression) ?: return null
        val descriptor = thisTypeForSuperCall.constructor.declarationDescriptor
        return descriptor as? ClassDescriptor
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy