org.jetbrains.kotlin.util.AttributeArrayOwner.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-compiler-embeddable Show documentation
Show all versions of kotlin-compiler-embeddable Show documentation
the Kotlin compiler embeddable
The newest version!
/*
* 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.util
import kotlin.reflect.KClass
/**
* Write access ([registerComponent]/[removeComponent]) is thread **unsafe**.
* Read access is thread **safe** only if there is no concurrent [removeComponent].
*
* [AttributeArrayOwner] based on different implementations of [ArrayMap] and switches them
* depending on array map fullness
* [AttributeArrayOwner] can be used in classes with many instances,
* like user data for Fir elements or attributes for cone types
*
* Note that you can remove attributes from [AttributeArrayOwner] despite
* from components in [ComponentArrayOwner]
*/
abstract class AttributeArrayOwner protected constructor(
arrayMap: ArrayMap,
) : AbstractArrayMapOwner() {
final override var arrayMap: ArrayMap = arrayMap
private set
@Suppress("UNCHECKED_CAST")
constructor() : this(EmptyArrayMap as ArrayMap)
final override fun registerComponent(keyQualifiedName: String, value: T) {
val id = typeRegistry.getId(keyQualifiedName)
when (arrayMap.size) {
0 -> {
val map = arrayMap
if (map !is EmptyArrayMap) {
throw IllegalStateException(buildDiagnosticMessage(map, expectedSize = 0, expectedImplementation = "EmptyArrayMap"))
}
arrayMap = OneElementArrayMap(value, id)
return
}
1 -> {
val mapSnapshot = arrayMap
val map = try {
mapSnapshot as OneElementArrayMap
} catch (e: ClassCastException) {
throw IllegalStateException(
buildDiagnosticMessage(mapSnapshot, expectedSize = 1, expectedImplementation = "OneElementArrayMap"),
/*cause=*/e
)
}
if (map.index == id) {
arrayMap = OneElementArrayMap(value, id)
return
} else {
val newMap = ArrayMapImpl()
newMap[map.index] = map.value
// the assigned map must have the old value to avoid problems in the concurrent scenario
arrayMap = newMap
}
}
}
arrayMap[id] = value
}
private fun buildDiagnosticMessage(map: ArrayMap, expectedSize: Int, expectedImplementation: String): String {
return buildString {
appendLine("Race condition happened, the size of ArrayMap is $expectedSize but it isn't an `$expectedImplementation`")
appendLine("Type: ${map::class.java}")
val content = buildString {
val services = typeRegistry.allValuesThreadUnsafeForRendering()
appendLine("[")
map.mapIndexed { index, value ->
val service = services.entries.firstOrNull { it.value == index }
appendLine(" $service[$index]: $value")
}
appendLine("]")
}
appendLine("Content: $content")
}
}
protected fun removeComponent(tClass: KClass) {
val id = typeRegistry.getId(tClass)
if (arrayMap[id] == null) return
@Suppress("UNCHECKED_CAST")
when (arrayMap.size) {
1 -> arrayMap = EmptyArrayMap as ArrayMap
else -> {
val map = arrayMap as ArrayMapImpl
map.remove(id)
if (map.size == 1) {
val (index, value) = map.entries().first()
arrayMap = OneElementArrayMap(value, index)
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy