org.jetbrains.kotlin.backend.jvm.lower.VarargLowering.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2024 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.backend.jvm.lower
import org.jetbrains.kotlin.backend.common.FileLoweringPass
import org.jetbrains.kotlin.backend.common.IrElementTransformerVoidWithContext
import org.jetbrains.kotlin.backend.common.phaser.PhaseDescription
import org.jetbrains.kotlin.backend.jvm.JvmBackendContext
import org.jetbrains.kotlin.backend.jvm.ir.IrArrayBuilder
import org.jetbrains.kotlin.backend.jvm.ir.createJvmIrBuilder
import org.jetbrains.kotlin.backend.jvm.ir.irArray
import org.jetbrains.kotlin.backend.jvm.ir.irArrayOf
import org.jetbrains.kotlin.builtins.PrimitiveType
import org.jetbrains.kotlin.builtins.StandardNames
import org.jetbrains.kotlin.builtins.UnsignedType
import org.jetbrains.kotlin.ir.UNDEFINED_OFFSET
import org.jetbrains.kotlin.ir.declarations.IrClass
import org.jetbrains.kotlin.ir.declarations.IrFile
import org.jetbrains.kotlin.ir.declarations.IrFunction
import org.jetbrains.kotlin.ir.declarations.IrPackageFragment
import org.jetbrains.kotlin.ir.expressions.*
import org.jetbrains.kotlin.ir.types.makeNotNull
import org.jetbrains.kotlin.ir.util.*
import org.jetbrains.kotlin.util.capitalizeDecapitalize.toLowerCaseAsciiOnly
@PhaseDescription(
name = "VarargLowering",
description = "Replace varargs with array arguments and lower arrayOf and emptyArray calls",
prerequisite = [PolymorphicSignatureLowering::class],
)
internal class VarargLowering(val context: JvmBackendContext) : FileLoweringPass, IrElementTransformerVoidWithContext() {
override fun lower(irFile: IrFile) = irFile.transformChildrenVoid()
// Ignore annotations
override fun visitConstructorCall(expression: IrConstructorCall): IrExpression {
val constructor = expression.symbol.owner
if (constructor.constructedClass.isAnnotationClass)
return expression
return super.visitConstructorCall(expression)
}
override fun visitFunctionAccess(expression: IrFunctionAccessExpression): IrExpression {
expression.transformChildrenVoid()
val function = expression.symbol
// Replace empty varargs with empty arrays
for (i in 0 until expression.valueArgumentsCount) {
if (expression.getValueArgument(i) != null)
continue
val parameter = function.owner.valueParameters[i]
if (parameter.varargElementType != null && !parameter.hasDefaultValue()) {
// Compute the correct type for the array argument.
val arrayType = parameter.type.substitute(expression.typeSubstitutionMap).makeNotNull()
expression.putValueArgument(i, createBuilder().irArrayOf(arrayType))
}
}
return expression
}
override fun visitVararg(expression: IrVararg): IrExpression =
createBuilder(expression.startOffset, expression.endOffset).irArray(expression.type) { addVararg(expression) }
private fun IrArrayBuilder.addVararg(expression: IrVararg) {
loop@ for (element in expression.elements) {
when (element) {
is IrExpression -> +element.transform(this@VarargLowering, null)
is IrSpreadElement -> {
val spread = element.expression
if (spread is IrFunctionAccessExpression && spread.symbol.owner.isArrayOf()) {
// Skip empty arrays and don't copy immediately created arrays
val argument = spread.getValueArgument(0) ?: continue@loop
if (argument is IrVararg) {
addVararg(argument)
continue@loop
}
}
addSpread(spread.transform(this@VarargLowering, null))
}
else -> error("Unexpected IrVarargElement subclass: $element")
}
}
}
private fun createBuilder(startOffset: Int = UNDEFINED_OFFSET, endOffset: Int = UNDEFINED_OFFSET) =
context.createJvmIrBuilder(currentScope!!, startOffset, endOffset)
}
internal val PRIMITIVE_ARRAY_OF_NAMES: Set =
(PrimitiveType.entries.map { type -> type.name } + UnsignedType.entries.map { type -> type.typeName.asString() })
.map { name -> name.toLowerCaseAsciiOnly() + "ArrayOf" }.toSet()
internal const val ARRAY_OF_NAME = "arrayOf"
internal fun IrFunction.isArrayOf(): Boolean {
val parent = when (val directParent = parent) {
is IrClass -> directParent.getPackageFragment()
is IrPackageFragment -> directParent
else -> return false
}
return parent.packageFqName == StandardNames.BUILT_INS_PACKAGE_FQ_NAME &&
name.asString().let { it in PRIMITIVE_ARRAY_OF_NAMES || it == ARRAY_OF_NAME } &&
extensionReceiverParameter == null &&
dispatchReceiverParameter == null &&
valueParameters.size == 1 &&
valueParameters[0].isVararg
}
internal fun IrFunction.isEmptyArray(): Boolean = isTopLevelInPackage("emptyArray", StandardNames.BUILT_INS_PACKAGE_FQ_NAME)
© 2015 - 2025 Weber Informatics LLC | Privacy Policy