[FIR generator] Use a single list of parent TypeRefs for elements

(Instead of storing the list of parent elements + the map of parent
arguments)
This commit is contained in:
Sergej Jaskiewicz
2023-09-14 17:05:42 +03:00
committed by Space Team
parent 5ee7aa7077
commit df0f86bf8d
10 changed files with 46 additions and 37 deletions
@@ -166,11 +166,11 @@ object NodeConfigurator : AbstractFieldConfigurator<FirTreeBuilder>(FirTreeBuild
}
loopJump.configure {
parentArg(jump, "E", loop)
parentArgs(jump, "E" to loop)
}
returnExpression.configure {
parentArg(jump, "E", function)
parentArgs(jump, "E" to function)
+field("result", expression).withTransform()
needTransformOtherChildren()
}
@@ -34,19 +34,25 @@ abstract class AbstractFieldConfigurator<T : AbstractFirTreeBuilder>(private val
return TypeVariable(name, upperBounds, variance).also(element.params::add)
}
fun parentArg(parent: Element, argument: String, type: TypeRef) {
parentArg(parent, NamedTypeParameterRef(argument), type)
fun parentArgs(parent: Element, vararg arguments: Pair<String, TypeRef>) {
parentArgs(parent, listOf(*arguments))
}
fun parentArg(parent: Element, argument: NamedTypeParameterRef, type: TypeRef) {
require(parent in element.parents) {
private fun parentArgs(parent: Element, arguments: List<Pair<String, TypeRef>>) {
parentArgs(parent, arguments.map { (name, arg) -> NamedTypeParameterRef(name) to arg })
}
@JvmName("parentArgs2")
private fun parentArgs(parent: Element, arguments: List<Pair<NamedTypeParameterRef, TypeRef>>) {
val parentIndex = element.parentRefs.indexOfFirst { it.element == parent }
require(parentIndex >= 0) {
"$parent is not parent of $element"
}
val argMap = element.parentsArguments.getOrPut(parent) { mutableMapOf() }
require(argument !in argMap) {
"Argument $argument already defined for parent $parent of $element"
val parentRef = element.parentRefs[parentIndex]
require(parentRef.args.isEmpty()) {
"Parent $parent of element $element already has type arguments: $parentRef"
}
argMap[argument] = type
element.parentRefs[parentIndex] = parentRef.copy(arguments.toMap())
}
fun needTransformOtherChildren() {
@@ -47,9 +47,9 @@ abstract class AbstractFirTreeBuilder {
private fun createElement(name: String, kind: Element.Kind, vararg dependencies: Element): Element =
Element(name, kind).also {
if (dependencies.isEmpty()) {
it.parents.add(baseFirElement)
it.parentRefs.add(baseFirElement)
}
it.parents.addAll(dependencies)
it.parentRefs.addAll(dependencies)
elements += it
}
@@ -57,7 +57,7 @@ abstract class AbstractFirTreeImplementationConfigurator {
private fun collectLeafsWithoutImplementation(builder: AbstractFirTreeBuilder): Set<Element> {
val elements = builder.elements.toMutableSet()
builder.elements.forEach {
elements.removeAll(it.parents)
elements.removeAll(it.parentRefs.map { it.element }.toSet())
}
elements.removeAll(elementsWithImpl)
return elements
@@ -8,6 +8,7 @@ package org.jetbrains.kotlin.fir.tree.generator.model
import org.jetbrains.kotlin.fir.tree.generator.printer.BASE_PACKAGE
import org.jetbrains.kotlin.fir.tree.generator.util.set
import org.jetbrains.kotlin.generators.tree.*
import org.jetbrains.kotlin.generators.tree.ElementOrRef
import org.jetbrains.kotlin.generators.tree.ElementOrRef as GenericElementOrRef
import org.jetbrains.kotlin.generators.tree.ElementRef as GenericElementRef
@@ -38,13 +39,12 @@ class Element(override val name: String, kind: Kind) : AbstractElement<Element,
override val type: String = "Fir$name"
override val packageName: String = BASE_PACKAGE + kind.packageName.let { if (it.isBlank()) it else "." + it }
override val fullQualifiedName: String get() = super.fullQualifiedName!!
override val parents = mutableListOf<Element>()
override val parentRefs = mutableListOf<ElementOrRef<Element, Field>>()
var defaultImplementation: Implementation? = null
val customImplementations = mutableListOf<Implementation>()
override val params = mutableListOf<TypeVariable>()
override val parentsArguments = mutableMapOf<Element, MutableMap<TypeRef, TypeRef>>()
override var kind: ImplementationKind? = null
set(value) {
if (value !in allowedKinds) {
@@ -61,7 +61,7 @@ class Element(override val name: String, kind: Kind) : AbstractElement<Element,
var doesNotNeedImplementation: Boolean = false
val needTransformOtherChildren: Boolean get() = _needTransformOtherChildren || parents.any { it.needTransformOtherChildren }
val needTransformOtherChildren: Boolean get() = _needTransformOtherChildren || parentRefs.any { it.element.needTransformOtherChildren }
override val overridenFields: MutableMap<Field, MutableMap<Field, Boolean>> = mutableMapOf()
val useNullableForReplace: MutableSet<Field> = mutableSetOf()
val allImplementations: List<Implementation> by lazy {
@@ -105,16 +105,18 @@ class Element(override val name: String, kind: Kind) : AbstractElement<Element,
val parentFields: List<Field> by lazy {
val result = LinkedHashMap<String, Field>()
parents.forEach { parent ->
parentRefs.forEach { parentRef ->
val parent = parentRef.element
val fields = parent.allFields.map { field ->
val copy = (field as? SimpleField)?.let { simpleField ->
parentsArguments[parent]?.get(NamedTypeParameterRef(simpleField.type))?.let {
// FIXME: Replace with parentRef.args[simpleField.typeRef]
parentRef.args[NamedTypeParameterRef(simpleField.type)]?.let {
simpleField.replaceType(it)
}
} ?: field.copy()
copy.apply {
arguments.replaceAll {
parentsArguments[parent]?.get(it) ?: it
parentRef.args[it] ?: it
}
fromParent = true
}
@@ -56,21 +56,22 @@ fun SmartPrinter.printElement(element: Element) {
print(typeParameters())
val needPureAbstractElement = !isInterface && !allParents.any { it.kind == ImplementationKind.AbstractClass || it.kind == ImplementationKind.SealedClass }
if (parents.isNotEmpty() || needPureAbstractElement) {
if (parentRefs.isNotEmpty() || needPureAbstractElement) {
print(" : ")
if (needPureAbstractElement) {
print("${pureAbstractElementType.type}()")
if (parents.isNotEmpty()) {
if (parentRefs.isNotEmpty()) {
print(", ")
}
}
print(
parents.joinToString(", ") {
var result = it.type
parentsArguments[it]?.let { arguments ->
result += arguments.values.joinToString(", ", "<", ">") { it.typeWithArguments }
parentRefs.joinToString(", ") {
// TODO: Factor out
var result = it.element.type
if (it.args.isNotEmpty()) {
result += it.args.values.joinToString(", ", "<", ">") { it.typeWithArguments }
}
result + it.kind.braces()
result + it.element.kind.braces()
},
)
}
@@ -134,14 +135,16 @@ fun SmartPrinter.printElement(element: Element) {
if (needTransformOtherChildren) {
println()
abstract()
if (element.parents.any { it.needTransformOtherChildren }) {
if (element.parentRefs.any { it.element.needTransformOtherChildren }) {
print("override ")
}
println(transformFunctionDeclaration("OtherChildren", typeWithArguments))
}
if (element == AbstractFirTreeBuilder.baseFirElement) {
require(isInterface)
require(isInterface) {
"$element must be an interface"
}
println()
println("fun accept(visitor: FirVisitorVoid) = accept(visitor, null)")
println()
@@ -61,9 +61,9 @@ fun Implementation.collectImports(base: List<String> = emptyList(), kind: Import
}
fun Element.collectImports(): List<String> {
val baseTypes = parents.mapTo(mutableListOf()) { it.fullQualifiedName }
val baseTypes = parentRefs.mapTo(mutableListOf()) { it.fullQualifiedName!! }
baseTypes += AbstractFirTreeBuilder.baseFirElement.fullQualifiedName
baseTypes += parentsArguments.values.flatMap { it.values }.mapNotNull { it.fullQualifiedName }
baseTypes += parentRefs.flatMap { it.args.values }.mapNotNull { it.fullQualifiedName } // TODO: Use recursive import collection for TypeRefs
if (needPureAbstractElement) {
baseTypes += pureAbstractElementType.fullQualifiedName
}
@@ -21,12 +21,12 @@ private val elementsWithMultipleSupertypesForDefaultVisitor = mapOf(
private fun Element.isAcceptableForDefaultVisiting(): Boolean {
if (this == AbstractFirTreeBuilder.baseFirElement) return false
val hasSingleSupertype = parents.size == 1 && parents.single().name != "Element"
val hasSingleSupertype = parentRefs.size == 1 && parentRefs.single().element.name != "Element"
return hasSingleSupertype || this in elementsWithMultipleSupertypesForDefaultVisitor
}
private fun Element.getNameOfSupertypeForDefaultVisiting(): String {
val parentForDefaultVisiting = parents.singleOrNull() ?: elementsWithMultipleSupertypesForDefaultVisitor.getValue(this)
val parentForDefaultVisiting = parentRefs.singleOrNull()?.element ?: elementsWithMultipleSupertypesForDefaultVisitor.getValue(this)
return parentForDefaultVisiting.name
}
@@ -12,7 +12,7 @@ import java.io.File
fun Element.traverseParents(block: (Element) -> Unit) {
block(this)
parents.forEach { it.traverseParents(block) }
parentRefs.forEach { it.element.traverseParents(block) }
}
operator fun <K, V, U> MutableMap<K, MutableMap<V, U>>.set(k1: K, k2: V, value: U) {
@@ -20,9 +20,7 @@ interface AbstractElement<Element, Field> : ElementOrRef<Element, Field>, FieldC
val params: List<TypeVariable>
val parents: List<Element>
val parentsArguments: Map<Element, Map<TypeRef, TypeRef>>
val parentRefs: List<ElementOrRef<Element, Field>>
val overridenFields: Map<Field, Map<Field, Boolean>>
@@ -30,7 +28,7 @@ interface AbstractElement<Element, Field> : ElementOrRef<Element, Field>, FieldC
get() = false
override val allParents: List<ImplementationKindOwner>
get() = parents
get() = parentRefs.map { it.element }
override fun getTypeWithArguments(notNull: Boolean): String = type + generics
}