FIR tree gen: extract interface/class solver to generators
This commit is contained in:
+90
@@ -0,0 +1,90 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2010-2019 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.tree.generator.util
|
||||||
|
|
||||||
|
import org.jetbrains.kotlin.fir.tree.generator.context.AbstractFirTreeBuilder
|
||||||
|
import org.jetbrains.kotlin.fir.tree.generator.model.Element
|
||||||
|
import org.jetbrains.kotlin.fir.tree.generator.model.Implementation
|
||||||
|
import org.jetbrains.kotlin.fir.tree.generator.model.ImplementationWithArg
|
||||||
|
import org.jetbrains.kotlin.fir.tree.generator.model.KindOwner
|
||||||
|
import org.jetbrains.kotlin.generators.util.Node
|
||||||
|
import org.jetbrains.kotlin.generators.util.solveGraphForClassVsInterface
|
||||||
|
|
||||||
|
private class NodeImpl(val element: KindOwner) : Node {
|
||||||
|
override val parents: List<Node>
|
||||||
|
get() = element.allParents.map(::NodeImpl)
|
||||||
|
|
||||||
|
override val origin: NodeImpl
|
||||||
|
get() = if (element.origin == element) this else NodeImpl(element.origin)
|
||||||
|
|
||||||
|
override fun equals(other: Any?): Boolean =
|
||||||
|
other is NodeImpl && element == other.element
|
||||||
|
|
||||||
|
override fun hashCode(): Int =
|
||||||
|
element.hashCode()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun configureInterfacesAndAbstractClasses(builder: AbstractFirTreeBuilder) {
|
||||||
|
val elements = collectElements(builder)
|
||||||
|
val solution = solveGraphForClassVsInterface(
|
||||||
|
elements,
|
||||||
|
elements.filter { it.element.kind?.isInterface == true },
|
||||||
|
elements.filter { it.element.kind?.isInterface == false },
|
||||||
|
)
|
||||||
|
updateKinds(elements, solution)
|
||||||
|
updateSealedKinds(elements)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun collectElements(builder: AbstractFirTreeBuilder): List<NodeImpl> {
|
||||||
|
return (builder.elements + builder.elements.flatMap { it.allImplementations }).map { NodeImpl(it.origin) }
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun updateKinds(nodes: List<NodeImpl>, solution: List<Boolean>) {
|
||||||
|
val allParents = nodes.flatMapTo(mutableSetOf()) { element -> element.parents.map { it.origin } }
|
||||||
|
|
||||||
|
for (index in solution.indices) {
|
||||||
|
val isClass = solution[index]
|
||||||
|
val node = nodes[index].origin
|
||||||
|
val element = node.element
|
||||||
|
val existingKind = element.kind
|
||||||
|
if (isClass) {
|
||||||
|
if (existingKind == Implementation.Kind.Interface)
|
||||||
|
throw IllegalStateException(element.toString())
|
||||||
|
|
||||||
|
if (existingKind == null) {
|
||||||
|
element.kind = when (element) {
|
||||||
|
is Implementation -> {
|
||||||
|
if (node in allParents)
|
||||||
|
Implementation.Kind.AbstractClass
|
||||||
|
else
|
||||||
|
Implementation.Kind.FinalClass
|
||||||
|
}
|
||||||
|
is Element -> Implementation.Kind.AbstractClass
|
||||||
|
else -> throw IllegalStateException()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
element.kind = Implementation.Kind.Interface
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun updateSealedKinds(nodes: Collection<NodeImpl>) {
|
||||||
|
for (node in nodes) {
|
||||||
|
val element = node.element
|
||||||
|
if (element is Element) {
|
||||||
|
if (element.isSealed) {
|
||||||
|
element.kind = when (element.kind) {
|
||||||
|
Implementation.Kind.AbstractClass -> Implementation.Kind.SealedClass
|
||||||
|
Implementation.Kind.Interface -> Implementation.Kind.SealedInterface
|
||||||
|
else -> error("element $element with kind ${element.kind} can not be sealed")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private val KindOwner.origin: KindOwner get() = if (this is ImplementationWithArg) implementation else this
|
||||||
+1
-83
@@ -3,13 +3,7 @@
|
|||||||
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
|
* 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.tree.generator.util
|
package org.jetbrains.kotlin.generators.util
|
||||||
|
|
||||||
import org.jetbrains.kotlin.fir.tree.generator.context.AbstractFirTreeBuilder
|
|
||||||
import org.jetbrains.kotlin.fir.tree.generator.model.Element
|
|
||||||
import org.jetbrains.kotlin.fir.tree.generator.model.Implementation
|
|
||||||
import org.jetbrains.kotlin.fir.tree.generator.model.ImplementationWithArg
|
|
||||||
import org.jetbrains.kotlin.fir.tree.generator.model.KindOwner
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Assume that we have element `E` with parents `P1, P2`
|
* Assume that we have element `E` with parents `P1, P2`
|
||||||
@@ -27,31 +21,6 @@ interface Node {
|
|||||||
val origin: Node
|
val origin: Node
|
||||||
}
|
}
|
||||||
|
|
||||||
private class NodeImpl(val element: KindOwner) : Node {
|
|
||||||
override val parents: List<Node>
|
|
||||||
get() = element.allParents.map(::NodeImpl)
|
|
||||||
|
|
||||||
override val origin: NodeImpl
|
|
||||||
get() = if (element.origin == element) this else NodeImpl(element.origin)
|
|
||||||
|
|
||||||
override fun equals(other: Any?): Boolean =
|
|
||||||
other is NodeImpl && element == other.element
|
|
||||||
|
|
||||||
override fun hashCode(): Int =
|
|
||||||
element.hashCode()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun configureInterfacesAndAbstractClasses(builder: AbstractFirTreeBuilder) {
|
|
||||||
val elements = collectElements(builder)
|
|
||||||
val solution = solveGraphForClassVsInterface(
|
|
||||||
elements,
|
|
||||||
elements.filter { it.element.kind?.isInterface == true },
|
|
||||||
elements.filter { it.element.kind?.isInterface == false },
|
|
||||||
)
|
|
||||||
updateKinds(elements, solution)
|
|
||||||
updateSealedKinds(elements)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun solveGraphForClassVsInterface(
|
fun solveGraphForClassVsInterface(
|
||||||
elements: List<Node>, requiredInterfaces: Collection<Node>, requiredClasses: Collection<Node>,
|
elements: List<Node>, requiredInterfaces: Collection<Node>, requiredClasses: Collection<Node>,
|
||||||
): List<Boolean> {
|
): List<Boolean> {
|
||||||
@@ -72,55 +41,6 @@ private class ElementMapping(val elements: Collection<Node>) {
|
|||||||
val size: Int = elements.size
|
val size: Int = elements.size
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun collectElements(builder: AbstractFirTreeBuilder): List<NodeImpl> {
|
|
||||||
return (builder.elements + builder.elements.flatMap { it.allImplementations }).map { NodeImpl(it.origin) }
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun updateKinds(nodes: List<NodeImpl>, solution: List<Boolean>) {
|
|
||||||
val allParents = nodes.flatMapTo(mutableSetOf()) { element -> element.parents.map { it.origin } }
|
|
||||||
|
|
||||||
for (index in solution.indices) {
|
|
||||||
val isClass = solution[index]
|
|
||||||
val node = nodes[index].origin
|
|
||||||
val element = node.element
|
|
||||||
val existingKind = element.kind
|
|
||||||
if (isClass) {
|
|
||||||
if (existingKind == Implementation.Kind.Interface)
|
|
||||||
throw IllegalStateException(element.toString())
|
|
||||||
|
|
||||||
if (existingKind == null) {
|
|
||||||
element.kind = when (element) {
|
|
||||||
is Implementation -> {
|
|
||||||
if (node in allParents)
|
|
||||||
Implementation.Kind.AbstractClass
|
|
||||||
else
|
|
||||||
Implementation.Kind.FinalClass
|
|
||||||
}
|
|
||||||
is Element -> Implementation.Kind.AbstractClass
|
|
||||||
else -> throw IllegalStateException()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
element.kind = Implementation.Kind.Interface
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun updateSealedKinds(nodes: Collection<NodeImpl>) {
|
|
||||||
for (node in nodes) {
|
|
||||||
val element = node.element
|
|
||||||
if (element is Element) {
|
|
||||||
if (element.isSealed) {
|
|
||||||
element.kind = when (element.kind) {
|
|
||||||
Implementation.Kind.AbstractClass -> Implementation.Kind.SealedClass
|
|
||||||
Implementation.Kind.Interface -> Implementation.Kind.SealedInterface
|
|
||||||
else -> error("element $element with kind ${element.kind} can not be sealed")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun processRequirementsFromConfig(
|
private fun processRequirementsFromConfig(
|
||||||
solution: MutableList<Boolean>,
|
solution: MutableList<Boolean>,
|
||||||
elementMapping: ElementMapping,
|
elementMapping: ElementMapping,
|
||||||
@@ -243,5 +163,3 @@ private fun buildGraphs(elements: Collection<Node>, elementMapping: ElementMappi
|
|||||||
}
|
}
|
||||||
return g to gt
|
return g to gt
|
||||||
}
|
}
|
||||||
|
|
||||||
private val KindOwner.origin: KindOwner get() = if (this is ImplementationWithArg) implementation else this
|
|
||||||
Reference in New Issue
Block a user