[FIR] Unwrap vararg array types for diagnostic rendering
#KT-65770 Fixed
This commit is contained in:
committed by
Space Team
parent
eae72eac54
commit
33648e1f44
+1
-1
@@ -42,7 +42,7 @@ object FirDiagnosticRenderers {
|
||||
propertyAccessorRenderer = null,
|
||||
callArgumentsRenderer = FirCallNoArgumentsRenderer(),
|
||||
modifierRenderer = FirPartialModifierRenderer(),
|
||||
valueParameterRenderer = FirValueParameterRendererNoDefaultValue(),
|
||||
valueParameterRenderer = FirValueParameterRendererForReadability(),
|
||||
declarationRenderer = FirDeclarationRenderer("local "),
|
||||
annotationRenderer = FirAnnotationRendererForReadability(),
|
||||
lineBreakAfterContextReceivers = false,
|
||||
|
||||
+1
-1
@@ -25,7 +25,7 @@ internal object FirExpectActualAnnotationIncompatibilityDiagnosticRenderers {
|
||||
annotationRenderer = null,
|
||||
modifierRenderer = null,
|
||||
contractRenderer = null,
|
||||
valueParameterRenderer = FirValueParameterRendererNoDefaultValue(),
|
||||
valueParameterRenderer = FirValueParameterRendererForReadability(),
|
||||
).renderElementAsString(it.fir, trim = true)
|
||||
// Write property accessors on the same line as the property
|
||||
.run { replace(Printer.LINE_SEPARATOR, "") }
|
||||
|
||||
@@ -6,7 +6,6 @@
|
||||
package org.jetbrains.kotlin.fir.types
|
||||
|
||||
import org.jetbrains.kotlin.name.StandardClassIds
|
||||
import org.jetbrains.kotlin.utils.addToStdlib.runIf
|
||||
|
||||
val ConeKotlinType.isArrayOrPrimitiveArray: Boolean
|
||||
get() = arrayElementTypeArgument() != null
|
||||
@@ -35,30 +34,6 @@ fun ConeTypeProjection.createArrayType(nullable: Boolean = false, createPrimitiv
|
||||
return StandardClassIds.Array.constructClassLikeType(arrayOf(this), nullable)
|
||||
}
|
||||
|
||||
fun ConeKotlinType.arrayElementType(checkUnsignedArrays: Boolean = true): ConeKotlinType? {
|
||||
return when (val argument = arrayElementTypeArgument(checkUnsignedArrays)) {
|
||||
is ConeKotlinTypeProjection -> argument.type
|
||||
else -> null
|
||||
}
|
||||
}
|
||||
|
||||
private fun ConeKotlinType.arrayElementTypeArgument(checkUnsignedArrays: Boolean = true): ConeTypeProjection? {
|
||||
val type = this.lowerBoundIfFlexible()
|
||||
if (type !is ConeClassLikeType) return null
|
||||
val classId = type.lookupTag.classId
|
||||
if (classId == StandardClassIds.Array) {
|
||||
return type.typeArguments.first()
|
||||
}
|
||||
val elementType = StandardClassIds.elementTypeByPrimitiveArrayType[classId] ?: runIf(checkUnsignedArrays) {
|
||||
StandardClassIds.elementTypeByUnsignedArrayType[classId]
|
||||
}
|
||||
if (elementType != null) {
|
||||
return elementType.constructClassLikeType(emptyArray(), isNullable = false)
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
|
||||
fun ConeKotlinType.varargElementType(): ConeKotlinType {
|
||||
return this.arrayElementType() ?: this
|
||||
}
|
||||
|
||||
@@ -78,7 +78,7 @@ class FirRenderer(
|
||||
propertyAccessorRenderer = null,
|
||||
callArgumentsRenderer = FirCallNoArgumentsRenderer(),
|
||||
modifierRenderer = FirPartialModifierRenderer(),
|
||||
valueParameterRenderer = FirValueParameterRendererNoDefaultValue(),
|
||||
valueParameterRenderer = FirValueParameterRendererForReadability(),
|
||||
declarationRenderer = FirDeclarationRenderer("local ")
|
||||
)
|
||||
}
|
||||
|
||||
+7
-1
@@ -15,6 +15,7 @@ open class FirValueParameterRenderer {
|
||||
private val annotationRenderer get() = components.annotationRenderer
|
||||
protected val declarationRenderer get() = components.declarationRenderer
|
||||
private val modifierRenderer get() = components.modifierRenderer
|
||||
protected val typeRenderer get() = components.typeRenderer
|
||||
|
||||
fun renderParameters(valueParameters: List<FirValueParameter>) {
|
||||
printer.print("(")
|
||||
@@ -34,10 +35,15 @@ open class FirValueParameterRenderer {
|
||||
if (valueParameter.name != SpecialNames.NO_NAME_PROVIDED) {
|
||||
printer.print(valueParameter.name.toString() + ": ")
|
||||
}
|
||||
valueParameter.returnTypeRef.accept(visitor)
|
||||
|
||||
renderParameterType(valueParameter)
|
||||
renderDefaultValue(valueParameter)
|
||||
}
|
||||
|
||||
protected open fun renderParameterType(valueParameter: FirValueParameter) {
|
||||
valueParameter.returnTypeRef.accept(visitor)
|
||||
}
|
||||
|
||||
protected open fun renderDefaultValue(valueParameter: FirValueParameter) {
|
||||
valueParameter.defaultValue?.let {
|
||||
printer.print(" = ")
|
||||
|
||||
+32
@@ -0,0 +1,32 @@
|
||||
/*
|
||||
* Copyright 2010-2022 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.fir.renderer
|
||||
|
||||
import org.jetbrains.kotlin.fir.declarations.FirValueParameter
|
||||
import org.jetbrains.kotlin.fir.types.FirResolvedTypeRef
|
||||
import org.jetbrains.kotlin.fir.types.arrayElementType
|
||||
|
||||
class FirValueParameterRendererForReadability : FirValueParameterRenderer() {
|
||||
override fun renderParameterType(valueParameter: FirValueParameter) {
|
||||
val returnTypeRef = valueParameter.returnTypeRef
|
||||
|
||||
if (valueParameter.isVararg && returnTypeRef is FirResolvedTypeRef) {
|
||||
val arrayElementType = returnTypeRef.type.arrayElementType()
|
||||
if (arrayElementType != null) {
|
||||
typeRenderer.render(arrayElementType)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
super.renderParameterType(valueParameter)
|
||||
}
|
||||
|
||||
override fun renderDefaultValue(valueParameter: FirValueParameter) {
|
||||
valueParameter.defaultValue?.let {
|
||||
printer.print(" = ...")
|
||||
}
|
||||
}
|
||||
}
|
||||
-16
@@ -1,16 +0,0 @@
|
||||
/*
|
||||
* Copyright 2010-2022 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.fir.renderer
|
||||
|
||||
import org.jetbrains.kotlin.fir.declarations.FirValueParameter
|
||||
|
||||
class FirValueParameterRendererNoDefaultValue : FirValueParameterRenderer() {
|
||||
override fun renderDefaultValue(valueParameter: FirValueParameter) {
|
||||
valueParameter.defaultValue?.let {
|
||||
printer.print(" = ...")
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -11,6 +11,7 @@ import org.jetbrains.kotlin.fir.utils.exceptions.withFirEntry
|
||||
import org.jetbrains.kotlin.name.ClassId
|
||||
import org.jetbrains.kotlin.name.StandardClassIds
|
||||
import org.jetbrains.kotlin.types.ConstantValueKind
|
||||
import org.jetbrains.kotlin.utils.addToStdlib.runIf
|
||||
import org.jetbrains.kotlin.utils.exceptions.errorWithAttachment
|
||||
import kotlin.contracts.ExperimentalContracts
|
||||
import kotlin.contracts.contract
|
||||
@@ -155,3 +156,27 @@ fun FirTypeProjection.toConeTypeProjection(): ConeTypeProjection = when (this) {
|
||||
}
|
||||
else -> errorWithAttachment("Unexpected ${this::class.simpleName}") { withFirEntry("projection", this@toConeTypeProjection) }
|
||||
}
|
||||
|
||||
fun ConeKotlinType.arrayElementType(checkUnsignedArrays: Boolean = true): ConeKotlinType? {
|
||||
return when (val argument = arrayElementTypeArgument(checkUnsignedArrays)) {
|
||||
is ConeKotlinTypeProjection -> argument.type
|
||||
else -> null
|
||||
}
|
||||
}
|
||||
|
||||
fun ConeKotlinType.arrayElementTypeArgument(checkUnsignedArrays: Boolean = true): ConeTypeProjection? {
|
||||
val type = this.lowerBoundIfFlexible()
|
||||
if (type !is ConeClassLikeType) return null
|
||||
val classId = type.lookupTag.classId
|
||||
if (classId == StandardClassIds.Array) {
|
||||
return type.typeArguments.first()
|
||||
}
|
||||
val elementType = StandardClassIds.elementTypeByPrimitiveArrayType[classId] ?: runIf(checkUnsignedArrays) {
|
||||
StandardClassIds.elementTypeByUnsignedArrayType[classId]
|
||||
}
|
||||
if (elementType != null) {
|
||||
return elementType.constructClassLikeType(emptyArray(), isNullable = false)
|
||||
}
|
||||
|
||||
return null
|
||||
}
|
||||
@@ -0,0 +1,5 @@
|
||||
/AmbiguousVararg.kt:7:5: error: overload resolution ambiguity:
|
||||
public fun foo(vararg t: String): String defined in root package in file AmbiguousVararg.kt
|
||||
public fun foo(vararg t: Int): String defined in root package in file AmbiguousVararg.kt
|
||||
foo()
|
||||
^^^
|
||||
@@ -0,0 +1 @@
|
||||
/AmbiguousVararg.kt:(128,131): error: Overload resolution ambiguity between candidates: [fun foo(vararg t: String): String, fun foo(vararg t: Int): String]
|
||||
@@ -1,4 +1,5 @@
|
||||
// FIR_IDENTICAL
|
||||
// RENDER_DIAGNOSTICS_FULL_TEXT
|
||||
fun foo(vararg t : String) = ""
|
||||
fun foo(vararg t : Int) = ""
|
||||
|
||||
|
||||
Vendored
+3
-3
@@ -5,13 +5,13 @@ build:
|
||||
[mkdir] Created dir: [Temp]/classes
|
||||
[javac] Compiling 2 source files to [Temp]/classes
|
||||
[javac] Compiling [[TestData]] => [[Temp]/classes]
|
||||
[javac] [TestData]/literals.kt:9:9: error: overload resolution ambiguity between candidates: [fun intArrayOf(vararg elements: IntArray): IntArray, fun intArrayOf(vararg elements: IntArray): IntArray]
|
||||
[javac] [TestData]/literals.kt:9:9: error: overload resolution ambiguity between candidates: [fun intArrayOf(vararg elements: Int): IntArray, fun intArrayOf(vararg elements: Int): IntArray]
|
||||
[javac] @AnnInt(intArrayOf(1, 2))
|
||||
[javac] ^^^^^^^^^^
|
||||
[javac] [TestData]/myArrayOf.kt:3:1: error: conflicting overloads: [fun intArrayOf(vararg elements: IntArray): IntArray]
|
||||
[javac] [TestData]/myArrayOf.kt:3:1: error: conflicting overloads: [fun intArrayOf(vararg elements: Int): IntArray]
|
||||
[javac] public fun intArrayOf(vararg elements: Int): IntArray = TODO()
|
||||
[javac] ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
[javac] [TestData]/myArrayOf.kt:4:1: error: conflicting overloads: [fun intArrayOf(vararg elements: IntArray): IntArray]
|
||||
[javac] [TestData]/myArrayOf.kt:4:1: error: conflicting overloads: [fun intArrayOf(vararg elements: Int): IntArray]
|
||||
[javac] public fun intArrayOf(vararg elements: Int): IntArray = TODO()
|
||||
[javac] ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
|
||||
@@ -88,11 +88,11 @@ actual fun f16(s: String = "") {}
|
||||
^^^^^^^^^^^^^^
|
||||
compiler/testData/multiplatform/incompatibleCallables/jvm.kt:28:12: error: 'actual fun f17(s: Array<out String>): Unit' has no corresponding expected declaration
|
||||
The following declaration is incompatible because some value parameter is vararg in one declaration and non-vararg in the other:
|
||||
expect fun f17(vararg s: Array<out String>): Unit
|
||||
expect fun f17(vararg s: String): Unit
|
||||
|
||||
actual fun f17(s: Array<out String>) {}
|
||||
^^^
|
||||
compiler/testData/multiplatform/incompatibleCallables/jvm.kt:29:12: error: 'actual fun f18(vararg s: Array<out String>): Unit' has no corresponding expected declaration
|
||||
compiler/testData/multiplatform/incompatibleCallables/jvm.kt:29:12: error: 'actual fun f18(vararg s: String): Unit' has no corresponding expected declaration
|
||||
The following declaration is incompatible because some value parameter is vararg in one declaration and non-vararg in the other:
|
||||
expect fun f18(s: Array<out String>): Unit
|
||||
|
||||
|
||||
Reference in New Issue
Block a user