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

org.jetbrains.kotlin.builtins.functions.FunctionTypeKindExtractor.kt Maven / Gradle / Ivy

There is a newer version: 2.0.20-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.builtins.functions

import org.jetbrains.kotlin.name.FqName

@RequiresOptIn
annotation class AllowedToUsedOnlyInK1

class FunctionTypeKindExtractor(private val kinds: List) {
    companion object {
        /**
         * This instance should be used only in:
         *  - FE 1.0, since it does not support custom functional kinds from plugins
         *  - places in FIR where session is not accessible by design (like in renderer of FIR elements)
         */
        @JvmStatic
        @AllowedToUsedOnlyInK1
        val Default = FunctionTypeKindExtractor(
            listOf(
                FunctionTypeKind.Function,
                FunctionTypeKind.SuspendFunction,
                FunctionTypeKind.KFunction,
                FunctionTypeKind.KSuspendFunction,
            )
        )
    }

    private val knownKindsByPackageFqName = kinds.groupBy { it.packageFqName }

    fun getFunctionalClassKind(packageFqName: FqName, className: String): FunctionTypeKind? {
        return getFunctionalClassKindWithArity(packageFqName, className)?.kind
    }

    fun getFunctionalClassKindWithArity(packageFqName: FqName, className: String): KindWithArity? {
        val kinds = knownKindsByPackageFqName[packageFqName] ?: return null
        for (kind in kinds) {
            if (!className.startsWith(kind.classNamePrefix)) continue
            val arity = toInt(className.substring(kind.classNamePrefix.length)) ?: continue
            return KindWithArity(kind, arity)
        }
        return null
    }

    fun hasKindWithSpecificPackage(packageFqName: FqName): Boolean {
        return packageFqName in knownKindsByPackageFqName
    }

    fun getFunctionKindPackageNames(): Set = knownKindsByPackageFqName.keys

    fun hasExtensionKinds(): Boolean = kinds.any { !it.isBuiltin }

    data class KindWithArity(val kind: FunctionTypeKind, val arity: Int)

    private fun toInt(s: String): Int? {
        if (s.isEmpty()) return null

        var result = 0
        for (c in s) {
            val d = c - '0'
            if (d !in 0..9) return null
            result = result * 10 + d
        }
        return result
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy