aboutsummaryrefslogtreecommitdiff
path: root/compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt
diff options
context:
space:
mode:
Diffstat (limited to 'compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt')
-rw-r--r--compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt129
1 files changed, 129 insertions, 0 deletions
diff --git a/compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt b/compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt
new file mode 100644
index 00000000..9f751700
--- /dev/null
+++ b/compiler-plugin/src/main/kotlin/com/google/devtools/ksp/symbol/impl/kotlin/KSTypeReferenceImpl.kt
@@ -0,0 +1,129 @@
+/*
+ * Copyright 2020 Google LLC
+ * Copyright 2010-2020 JetBrains s.r.o. and Kotlin Programming Language contributors.
+ *
+ * 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 com.google.devtools.ksp.symbol.impl.kotlin
+
+import com.google.devtools.ksp.ExceptionMessage
+import com.google.devtools.ksp.KSObjectCache
+import com.google.devtools.ksp.memoized
+import com.google.devtools.ksp.processing.impl.ResolverImpl
+import com.google.devtools.ksp.symbol.KSAnnotation
+import com.google.devtools.ksp.symbol.KSNode
+import com.google.devtools.ksp.symbol.KSReferenceElement
+import com.google.devtools.ksp.symbol.KSType
+import com.google.devtools.ksp.symbol.KSTypeReference
+import com.google.devtools.ksp.symbol.KSVisitor
+import com.google.devtools.ksp.symbol.Location
+import com.google.devtools.ksp.symbol.Modifier
+import com.google.devtools.ksp.symbol.Origin
+import com.google.devtools.ksp.symbol.impl.toLocation
+import com.google.devtools.ksp.toKSModifiers
+import org.jetbrains.kotlin.psi.*
+
+class KSTypeReferenceImpl private constructor(val ktTypeReference: KtTypeReference) : KSTypeReference {
+ companion object : KSObjectCache<KtTypeReference, KSTypeReferenceImpl>() {
+ fun getCached(ktTypeReference: KtTypeReference) = cache.getOrPut(ktTypeReference) {
+ KSTypeReferenceImpl(ktTypeReference)
+ }
+ }
+
+ override val origin = Origin.KOTLIN
+
+ override val location: Location by lazy {
+ ktTypeReference.toLocation()
+ }
+ override val parent: KSNode? by lazy {
+ var parentPsi = ktTypeReference.parent
+ while (
+ parentPsi != null && parentPsi !is KtAnnotationEntry && parentPsi !is KtFunctionType &&
+ parentPsi !is KtClassOrObject && parentPsi !is KtFunction && parentPsi !is KtUserType &&
+ parentPsi !is KtProperty && parentPsi !is KtTypeAlias && parentPsi !is KtTypeProjection &&
+ parentPsi !is KtTypeParameter && parentPsi !is KtParameter
+ ) {
+ parentPsi = parentPsi.parent
+ }
+ when (parentPsi) {
+ is KtAnnotationEntry -> KSAnnotationImpl.getCached(parentPsi)
+ is KtFunctionType -> KSCallableReferenceImpl.getCached(parentPsi)
+ is KtClassOrObject -> KSClassDeclarationImpl.getCached(parentPsi)
+ is KtFunction -> KSFunctionDeclarationImpl.getCached(parentPsi)
+ is KtUserType -> KSClassifierReferenceImpl.getCached(parentPsi)
+ is KtProperty -> KSPropertyDeclarationImpl.getCached(parentPsi)
+ is KtTypeAlias -> KSTypeAliasImpl.getCached(parentPsi)
+ is KtTypeProjection -> KSTypeArgumentKtImpl.getCached(parentPsi)
+ is KtTypeParameter -> KSTypeParameterImpl.getCached(parentPsi)
+ is KtParameter -> KSValueParameterImpl.getCached(parentPsi)
+ else -> null
+ }
+ }
+
+ // Parenthesized type in grammar seems to be implemented as KtNullableType.
+ private fun visitNullableType(visit: (KtNullableType) -> Unit) {
+ var typeElement = ktTypeReference.typeElement
+ while (typeElement is KtNullableType) {
+ visit(typeElement)
+ typeElement = typeElement.innerType
+ }
+ }
+
+ // Annotations and modifiers are only allowed in one of the parenthesized type.
+ // https://github.com/JetBrains/kotlin/blob/50e12239ef8141a45c4dca2bf0544be6191ecfb6/compiler/frontend/src/org/jetbrains/kotlin/diagnostics/rendering/DefaultErrorMessages.java#L608
+ override val annotations: Sequence<KSAnnotation> by lazy {
+ fun List<KtAnnotationEntry>.toKSAnnotations(): Sequence<KSAnnotation> =
+ asSequence().map {
+ KSAnnotationImpl.getCached(it)
+ }
+
+ val innerAnnotations = mutableListOf<Sequence<KSAnnotation>>()
+ visitNullableType {
+ innerAnnotations.add(it.annotationEntries.toKSAnnotations())
+ }
+
+ (ktTypeReference.annotationEntries.toKSAnnotations() + innerAnnotations.asSequence().flatten()).memoized()
+ }
+
+ override val modifiers: Set<Modifier> by lazy {
+ val innerModifiers = mutableSetOf<Modifier>()
+ visitNullableType {
+ innerModifiers.addAll(it.modifierList.toKSModifiers())
+ }
+ ktTypeReference.toKSModifiers() + innerModifiers
+ }
+
+ override val element: KSReferenceElement by lazy {
+ var typeElement = ktTypeReference.typeElement
+ while (typeElement is KtNullableType)
+ typeElement = typeElement.innerType
+ when (typeElement) {
+ is KtFunctionType -> KSCallableReferenceImpl.getCached(typeElement)
+ is KtUserType -> KSClassifierReferenceImpl.getCached(typeElement)
+ is KtDynamicType -> KSDynamicReferenceImpl.getCached(this)
+ is KtIntersectionType -> KSDefNonNullReferenceImpl.getCached(typeElement)
+ else -> throw IllegalStateException("Unexpected type element ${typeElement?.javaClass}, $ExceptionMessage")
+ }
+ }
+
+ override fun <D, R> accept(visitor: KSVisitor<D, R>, data: D): R {
+ return visitor.visitTypeReference(this, data)
+ }
+
+ override fun resolve(): KSType = ResolverImpl.instance!!.resolveUserType(this)
+
+ override fun toString(): String {
+ return element.toString()
+ }
+}