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

kotlin.reflect.jvm.internal.KParameterImpl.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-RC
Show newest version
/*
 * Copyright 2010-2015 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 kotlin.reflect.jvm.internal

import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.resolve.descriptorUtil.declaresOrInheritsDefaultValue
import java.lang.reflect.Type
import kotlin.reflect.KParameter
import kotlin.reflect.KType
import kotlin.reflect.jvm.internal.calls.ValueClassAwareCaller

internal class KParameterImpl(
    val callable: KCallableImpl<*>,
    override val index: Int,
    override val kind: KParameter.Kind,
    computeDescriptor: () -> ParameterDescriptor,
) : KParameter {
    private val descriptor: ParameterDescriptor by ReflectProperties.lazySoft(computeDescriptor)

    override val annotations: List by ReflectProperties.lazySoft { descriptor.computeAnnotations() }

    override val name: String?
        get() {
            val valueParameter = descriptor as? ValueParameterDescriptor ?: return null
            if (valueParameter.containingDeclaration.hasSynthesizedParameterNames()) return null
            val name = valueParameter.name
            return if (name.isSpecial) null else name.asString()
        }


    private fun compoundType(vararg types: Type): Type = when (types.size) {
        0 -> throw KotlinReflectionNotSupportedError("Expected at least 1 type for compound type")
        1 -> types.single()
        else -> CompoundTypeImpl(types)
    }

    private class CompoundTypeImpl(val types: Array) : Type {
        private val hashCode = types.contentHashCode()
        override fun getTypeName(): String {
            return types.joinToString(", ", "[", "]")
        }

        override fun equals(other: Any?): Boolean =
            other is CompoundTypeImpl && this.types contentEquals other.types

        override fun hashCode(): Int = hashCode

        override fun toString(): String = typeName
    }

    override val type: KType
        get() = KTypeImpl(descriptor.type) {
            val descriptor = descriptor

            if (descriptor is ReceiverParameterDescriptor &&
                callable.descriptor.instanceReceiverParameter == descriptor &&
                callable.descriptor.kind == CallableMemberDescriptor.Kind.FAKE_OVERRIDE
            ) {
                // In case of fake overrides, dispatch receiver type should be computed manually because Caller.parameterTypes returns
                // types from Java reflection where receiver is always the declaring class of the original declaration
                // (not the class where the fake override is generated, which is returned by KParameter.type)
                (callable.descriptor.containingDeclaration as ClassDescriptor).toJavaClass()
                    ?: throw KotlinReflectionInternalError("Cannot determine receiver Java type of inherited declaration: $descriptor")
            } else {
                when (val caller = callable.caller) {
                    is ValueClassAwareCaller -> {
                        val parameterTypes = if (callable.isBound) {
                            val slice = caller.getRealSlicesOfParameters(index + 1)
                            val offset = caller.getRealSlicesOfParameters(0).last + 1
                            caller.parameterTypes.slice((slice.first - offset)..(slice.last - offset))
                        } else {
                            val slice = caller.getRealSlicesOfParameters(index)
                            caller.parameterTypes.slice(slice)
                        }
                        compoundType(*parameterTypes.toTypedArray())
                    }
                    is ValueClassAwareCaller.MultiFieldValueClassPrimaryConstructorCaller ->
                        compoundType(*caller.originalParametersGroups[index].toTypedArray())
                    else -> caller.parameterTypes[index]
                }
            }
        }

    override val isOptional: Boolean
        get() = (descriptor as? ValueParameterDescriptor)?.declaresOrInheritsDefaultValue() ?: false

    override val isVararg: Boolean
        get() = descriptor.let { it is ValueParameterDescriptor && it.varargElementType != null }

    override fun equals(other: Any?) =
        other is KParameterImpl && callable == other.callable && index == other.index

    override fun hashCode() =
        (callable.hashCode() * 31) + index.hashCode()

    override fun toString() =
        ReflectionObjectRenderer.renderParameter(this)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy