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

org.jetbrains.kotlin.javac.wrappers.symbols.symbolBasedTypes.kt Maven / Gradle / Ivy

There is a newer version: 2.0.0
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.javac.wrappers.symbols

import com.sun.tools.javac.code.Symbol
import org.jetbrains.kotlin.builtins.PrimitiveType
import org.jetbrains.kotlin.javac.JavacWrapper
import org.jetbrains.kotlin.load.java.structure.*
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.resolve.jvm.JvmPrimitiveType
import javax.lang.model.element.TypeParameterElement
import javax.lang.model.type.*

sealed class SymbolBasedType(
    val typeMirror: T,
    val javac: JavacWrapper
) : JavaType, JavaAnnotationOwner {

    companion object {
        fun  create(t: T, javac: JavacWrapper) = when {
            t.kind.isPrimitive || t.kind == TypeKind.VOID -> SymbolBasedPrimitiveType(t, javac)
            t.kind == TypeKind.DECLARED || t.kind == TypeKind.TYPEVAR -> SymbolBasedClassifierType(t, javac)
            t.kind == TypeKind.WILDCARD -> SymbolBasedWildcardType(t as WildcardType, javac)
            t.kind == TypeKind.ARRAY -> SymbolBasedArrayType(t as ArrayType, javac)
            else -> throw UnsupportedOperationException("Unsupported type: $t")
        }
    }

    override val annotations: Collection
        get() = typeMirror.annotationMirrors.map { SymbolBasedAnnotation(it, javac) }

    override val isDeprecatedInJavaDoc: Boolean
        get() = javac.isDeprecated(typeMirror)

    override fun findAnnotation(fqName: FqName) = typeMirror.findAnnotation(fqName, javac)

    override fun equals(other: Any?) = (other as? SymbolBasedType<*>)?.typeMirror == typeMirror

    override fun hashCode() = typeMirror.hashCode()

    override fun toString() = typeMirror.toString()

}

class SymbolBasedPrimitiveType(
    typeMirror: TypeMirror,
    javac: JavacWrapper
) : SymbolBasedType(typeMirror, javac), JavaPrimitiveType {

    override val type: PrimitiveType?
        get() = if (typeMirror.kind == TypeKind.VOID) null else JvmPrimitiveType.get(typeMirror.toString()).primitiveType

}

class SymbolBasedClassifierType(
    typeMirror: T,
    javac: JavacWrapper
) : SymbolBasedType(typeMirror, javac), JavaClassifierType {

    private val isFake get() = classifier is FakeSymbolBasedClass

    // TODO: we should replace this with a "link" to classifier, not an actual classifier itself
    // It should be something like ConeClassifierLookupTag (see compiler:fir:cones)
    override val classifier: JavaClassifier?
            by lazy {
                when (typeMirror.kind) {
                    TypeKind.DECLARED -> ((typeMirror as DeclaredType).asElement() as Symbol.ClassSymbol).let { symbol ->
                        // try to find cached javaClass
                        val classId = symbol.computeClassId()
                        classId?.let { javac.findClass(it) } ?: FakeSymbolBasedClass(symbol, javac, classId, symbol.classfile)
                    }
                    TypeKind.TYPEVAR -> SymbolBasedTypeParameter((typeMirror as TypeVariable).asElement() as TypeParameterElement, javac)
                    else -> null
                }
            }

    override val typeArguments: List
        get() {
            if (typeMirror.kind != TypeKind.DECLARED || isFake) return emptyList()

            val arguments = arrayListOf()
            var type = typeMirror as DeclaredType
            var staticType = false

            while (!staticType) {
                if (type.asElement().isStatic) {
                    staticType = true
                }
                arguments.addAll(type.typeArguments.map { create(it, javac) })
                type = type.enclosingType as? DeclaredType ?: return arguments
            }

            return arguments
        }

    override val isRaw: Boolean
        get() = when {
            typeMirror !is DeclaredType -> false
            isFake -> false
            (classifier as? JavaClass)?.typeParameters?.isEmpty() == true -> false
            else -> typeMirror.typeArguments.isEmpty() || (classifier as? JavaClass)?.typeParameters?.size != typeMirror.typeArguments.size
        }

    override val classifierQualifiedName: String
        get() = typeMirror.toString()

    override val presentableText: String
        get() = typeMirror.toString()

    override val isDeprecatedInJavaDoc: Boolean
        get() = !isFake && super.isDeprecatedInJavaDoc
}

class SymbolBasedWildcardType(
    typeMirror: WildcardType,
    javac: JavacWrapper
) : SymbolBasedType(typeMirror, javac), JavaWildcardType {

    override val bound: JavaType?
        get() {
            val boundMirror = typeMirror.extendsBound ?: typeMirror.superBound
            return boundMirror?.let { create(it, javac) }
        }

    override val isExtends: Boolean
        get() = typeMirror.extendsBound != null

}

class SymbolBasedArrayType(
    typeMirror: ArrayType,
    javac: JavacWrapper
) : SymbolBasedType(typeMirror, javac), JavaArrayType {

    override val componentType: JavaType
        get() = create(typeMirror.componentType, javac)

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy