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

main.seskar.compiler.value.backend.ValueTransformer.kt Maven / Gradle / Ivy

There is a newer version: 3.81.0
Show newest version
package seskar.compiler.value.backend

import org.jetbrains.kotlin.backend.common.extensions.IrPluginContext
import org.jetbrains.kotlin.ir.expressions.IrCall
import org.jetbrains.kotlin.ir.expressions.IrExpression
import org.jetbrains.kotlin.ir.expressions.IrVararg
import org.jetbrains.kotlin.ir.expressions.IrVarargElement
import org.jetbrains.kotlin.ir.expressions.impl.IrCallImpl
import org.jetbrains.kotlin.ir.types.getClass
import org.jetbrains.kotlin.ir.types.isNullable
import org.jetbrains.kotlin.ir.util.getArgumentsWithIr
import org.jetbrains.kotlin.ir.util.kotlinFqName
import org.jetbrains.kotlin.ir.util.properties
import org.jetbrains.kotlin.ir.visitors.IrElementTransformerVoid

internal class ValueTransformer(
    private val context: IrPluginContext,
) : IrElementTransformerVoid() {
    override fun visitCall(
        expression: IrCall,
    ): IrExpression {
        if (isHookCall(expression))
            visitHookCall(expression)

        return super.visitCall(expression)
    }

    private fun visitHookCall(
        expression: IrCall,
    ) {
        val dependencies = expression.getArgumentsWithIr()
            .firstOrNull { (parameter) -> parameter.name == DEPENDENCIES && parameter.varargElementType == context.irBuiltIns.anyNType }
            ?.second
            ?: return

        if (dependencies is IrVararg)
            visitVarargElements(dependencies.elements)
    }

    private fun visitVarargElements(
        elements: MutableList,
    ) {
        val valueElements = elements.asSequence()
            .filterIsInstance()
            .filter { downCastRequired(it.type) }
            .toList()

        if (valueElements.isEmpty())
            return

        for (element in valueElements) {
            val index = elements.indexOf(element)
            elements[index] = getValue(element)
        }
    }

    private fun getValue(
        element: IrExpression,
    ): IrExpression {
        val klass = element.type.getClass()!!

        // WA Check how to access `rawValue`
        if (klass.kotlinFqName == DURATION)
            return toString(element, element.type.isNullable())

        val value = klass.properties.first()
        val getter = value.getter!!

        val call = IrCallImpl.fromSymbolOwner(
            startOffset = element.startOffset,
            endOffset = element.endOffset,
            symbol = getter.symbol,
        )

        call.dispatchReceiver = element

        return when (getter.returnType) {
            context.irBuiltIns.longType,
            -> toString(call, element.type.isNullable())

            else -> call
        }
    }

    private fun toString(
        element: IrExpression,
        nullable: Boolean,
    ): IrExpression =
        if (nullable) {
            val call = IrCallImpl.fromSymbolOwner(
                startOffset = element.startOffset,
                endOffset = element.endOffset,
                symbol = context.symbols.extensionToString,
            )

            call.extensionReceiver = element
            call
        } else {
            val call = IrCallImpl.fromSymbolOwner(
                startOffset = element.startOffset,
                endOffset = element.endOffset,
                symbol = context.symbols.memberToString,
            )

            call.dispatchReceiver = element
            call
        }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy