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

org.jetbrains.kotlin.backend.konan.serialization.ObjCFunctionNameMangleComputer.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-RC
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.konan.serialization

import org.jetbrains.kotlin.backend.common.serialization.mangle.MangleConstant
import org.jetbrains.kotlin.backend.common.serialization.mangle.PlatformSpecificFunctionNameMangleComputer
import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.ir.ObsoleteDescriptorBasedAPI
import org.jetbrains.kotlin.ir.declarations.IrConstructor
import org.jetbrains.kotlin.ir.declarations.IrFunction
import org.jetbrains.kotlin.ir.declarations.IrSimpleFunction
import org.jetbrains.kotlin.ir.declarations.IrValueParameter
import org.jetbrains.kotlin.ir.objcinterop.*
import org.jetbrains.kotlin.ir.types.getClass
import org.jetbrains.kotlin.ir.util.hasAnnotation
import org.jetbrains.kotlin.name.Name
import org.jetbrains.kotlin.name.NativeStandardInteropNames
import org.jetbrains.kotlin.native.interop.ObjCMethodInfo

private const val OBJC_MARK = "objc:"
private const val OBJC_CONSTRUCTOR_MARK = "#Constructor"
private const val OBJC_PROPERTY_ACCESSOR_MARK = "#Accessor"

abstract class ObjCFunctionNameMangleComputer : PlatformSpecificFunctionNameMangleComputer {

    abstract fun getObjCMethodInfo(): ObjCMethodInfo?

    abstract fun getExtensionReceiverClassName(): Name?

    abstract fun isObjCConstructor(): Boolean

    abstract fun isPropertyAccessor(): Boolean

    abstract fun hasObjCMethodAnnotation(): Boolean

    abstract fun hasObjCFactoryAnnotation(): Boolean

    abstract fun isObjCClassMethod(): Boolean

    abstract fun getValueParameterName(valueParameter: ValueParameter): Name

    final override fun computePlatformSpecificFunctionName(): String? {
        val objcMethodInfo = getObjCMethodInfo() ?: return null
        return buildString {
            getExtensionReceiverClassName()?.let {
                append(it)
                append(MangleConstant.FQN_SEPARATOR)
            }
            append(OBJC_MARK)
            append(objcMethodInfo.selector)
            if (isObjCConstructor()) {
                append(OBJC_CONSTRUCTOR_MARK)
            }
            if (isPropertyAccessor()) {
                append(OBJC_PROPERTY_ACCESSOR_MARK)
            }
        }
    }

    final override fun computePlatformSpecificValueParameterPrefix(valueParameter: ValueParameter): String =
        if (hasObjCMethodAnnotation() || hasObjCFactoryAnnotation() || isObjCClassMethod()) {
            "${getValueParameterName(valueParameter)}:"
        } else {
            ""
        }
}

@ObsoleteDescriptorBasedAPI
class DescriptorObjCFunctionNameMangleComputer(
    private val function: FunctionDescriptor
) : ObjCFunctionNameMangleComputer() {

    override fun getObjCMethodInfo(): ObjCMethodInfo? =
        (if (function is ConstructorDescriptor && function.isObjCConstructor) function.getObjCInitMethod() else function)
            ?.getObjCMethodInfo()

    override fun getExtensionReceiverClassName(): Name? =
        function.extensionReceiverParameter?.run { type.constructor.declarationDescriptor!!.name }

    override fun isObjCConstructor(): Boolean =
        function is ConstructorDescriptor && function.isObjCConstructor

    override fun isPropertyAccessor(): Boolean =
        function is PropertyAccessorDescriptor

    override fun hasObjCMethodAnnotation(): Boolean = function.annotations.hasAnnotation(objCMethodFqName)

    override fun hasObjCFactoryAnnotation(): Boolean = function.annotations.hasAnnotation(objCFactoryFqName)

    override fun isObjCClassMethod(): Boolean = function.containingDeclaration.let { it is ClassDescriptor && it.isObjCClass() }

    override fun getValueParameterName(valueParameter: ValueParameterDescriptor): Name = valueParameter.name
}

class IrObjCFunctionNameMangleComputer(private val function: IrFunction) : ObjCFunctionNameMangleComputer() {
    override fun getObjCMethodInfo(): ObjCMethodInfo? =
        (if (function is IrConstructor && function.isObjCConstructor) function.getObjCInitMethod() else function)?.getObjCMethodInfo()

    override fun getExtensionReceiverClassName(): Name? =
        function.extensionReceiverParameter?.run { type.getClass()!!.name }

    override fun isObjCConstructor(): Boolean =
        function is IrConstructor && function.isObjCConstructor

    override fun isPropertyAccessor(): Boolean =
        (function as? IrSimpleFunction)?.correspondingPropertySymbol != null

    override fun hasObjCMethodAnnotation(): Boolean = function.hasAnnotation(NativeStandardInteropNames.objCMethodClassId)

    override fun hasObjCFactoryAnnotation(): Boolean = function.hasAnnotation(NativeStandardInteropNames.objCFactoryClassId)

    override fun isObjCClassMethod(): Boolean = function.isObjCClassMethod()

    override fun getValueParameterName(valueParameter: IrValueParameter): Name = valueParameter.name
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy