Skip to content

Commit

Permalink
Adds an IntArray pool to MacroEvaluator.Session and modifies Session'…
Browse files Browse the repository at this point in the history
…s ExpansionInfo pool to match the pattern.
  • Loading branch information
tgregg committed Jan 19, 2025
1 parent ffcd659 commit 8161ef7
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 23 deletions.
76 changes: 54 additions & 22 deletions src/main/java/com/amazon/ion/impl/macro/MacroEvaluator.kt
Original file line number Diff line number Diff line change
Expand Up @@ -44,19 +44,35 @@ class MacroEvaluator {
* This state pertains to a single "session" of the macro evaluator, and is reset every time [initExpansion] is called.
* For now, this includes managing the pool of [ExpansionInfo] and tracking the expansion step limit.
*/
private class Session(
class Session(
/** Number of expansion steps at which the evaluation session should be aborted. */
private val expansionLimit: Int = 1_000_000
) {
/** Internal state for tracking the number of expansion steps. */
private var numExpandedExpressions = 0
/** Pool of [ExpansionInfo] to minimize allocation and garbage collection. */
private val expanderPool: ArrayList<ExpansionInfo> = ArrayList(32)
private var expanderPoolIndex = 0

/**
* Gets an [ExpansionInfo] from the pool, or allocates a new one if necessary. The returned ExpansionInfo is
* valid until [reset] is called.
*/
private fun getExpansion(): ExpansionInfo {
val expansion: ExpansionInfo
if (expanderPoolIndex >= expanderPool.size) {
expansion = ExpansionInfo(this)
expanderPool.add(expansion)
} else {
expansion = expanderPool[expanderPoolIndex]
}
expanderPoolIndex++
return expansion
}

/** Gets an [ExpansionInfo] from the pool (or allocates a new one if necessary), initializing it with the provided values. */
fun getExpander(expansionKind: ExpansionKind, expressions: List<Expression>, startInclusive: Int, endExclusive: Int, environment: Environment): ExpansionInfo {
val expansion = expanderPool.removeLastOrNull() ?: ExpansionInfo(this)
expansion.isInPool = false
val expansion = getExpansion()
expansion.expansionKind = expansionKind
expansion.expressions = expressions
expansion.i = startInclusive
Expand All @@ -67,15 +83,6 @@ class MacroEvaluator {
return expansion
}

/** Reclaims an [ExpansionInfo] to the available pool. */
fun reclaimExpander(ex: ExpansionInfo) {
// Ensure that we are not doubly-adding an ExpansionInfo instance to the pool.
if (!ex.isInPool) {
ex.isInPool = true
expanderPool.add(ex)
}
}

fun incrementStepCounter() {
numExpandedExpressions++
if (numExpandedExpressions > expansionLimit) {
Expand All @@ -87,6 +94,34 @@ class MacroEvaluator {

fun reset() {
numExpandedExpressions = 0
expanderPoolIndex = 0
intArrayPoolIndices.fill(0)
}

/** The pool of [IntArray]s for use during this session. */
private val intArrayPools = Array<MutableList<IntArray>>(32) { mutableListOf() }
private val intArrayPoolIndices = IntArray(32) { 0 }

/**
* Gets an [IntArray] from the pool, or allocates a new one if necessary. The returned array is valid until
* [reset] is called.
* */
fun intArrayForSize(size: Int): IntArray {
if (size >= intArrayPools.size) {
// Don't attempt to pool arbitrarily large arrays.
return IntArray(size)
}
val pool = intArrayPools[size]
val index = intArrayPoolIndices[size]
val array: IntArray
if (index >= pool.size) {
array = IntArray(size)
pool.add(array)
} else {
array = pool[index]
}
intArrayPoolIndices[size] = index + 1
return array
}
}

Expand Down Expand Up @@ -116,7 +151,7 @@ class MacroEvaluator {
*/
// TODO(PERF): It might be possible to optimize this by changing it to an enum without any methods (or even a set of
// integer constants) and converting all their implementations to static methods.
private enum class ExpansionKind {
enum class ExpansionKind {
Uninitialized {
override fun produceNext(thisExpansion: ExpansionInfo): Nothing = throw IllegalStateException("ExpansionInfo not initialized.")
},
Expand Down Expand Up @@ -152,7 +187,7 @@ class MacroEvaluator {
is DataModelExpression -> next
is InvokableExpression -> {
val macro = next.macro
val argIndices = calculateArgumentIndices(macro, thisExpansion.expressions, next.startInclusive, next.endExclusive)
val argIndices = calculateArgumentIndices(macro, thisExpansion, next.startInclusive, next.endExclusive)
val newEnvironment = thisExpansion.environment.createChild(thisExpansion.expressions, argIndices)
val expansionKind = ExpansionKind.forMacro(macro)
thisExpansion.childExpansion = thisExpansion.session.getExpander(
Expand Down Expand Up @@ -692,7 +727,7 @@ class MacroEvaluator {
* Alternately, consider ExpansionOperator to reflect the fact that these are
* like operators in an expression tree.
*/
private class ExpansionInfo(@JvmField val session: Session) {
class ExpansionInfo(@JvmField val session: Session) {

/** The [ExpansionKind]. */
@JvmField var expansionKind: ExpansionKind = ExpansionKind.Uninitialized
Expand All @@ -717,9 +752,6 @@ class MacroEvaluator {
@JvmField
var additionalState: Any? = null

@JvmField
var isInPool = false

/**
* Additional state in the form of a child [ExpansionInfo].
*/
Expand Down Expand Up @@ -757,7 +789,6 @@ class MacroEvaluator {
additionalState = null
childExpansion?.close()
childExpansion = null
session.reclaimExpander(this)
}

/**
Expand Down Expand Up @@ -894,7 +925,7 @@ class MacroEvaluator {

/**
* Given a [Macro] (or more specifically, its signature), calculates the position of each of its arguments
* in [encodingExpressions]. The result is a list that can be used to map from a parameter's
* in [expansion].expressions. The result is a list that can be used to map from a parameter's
* signature index to the encoding expression index. Any trailing, optional arguments that are
* elided have a value of -1.
*
Expand All @@ -903,14 +934,15 @@ class MacroEvaluator {
*/
private fun calculateArgumentIndices(
macro: Macro,
encodingExpressions: List<Expression>,
expansion: MacroEvaluator.ExpansionInfo,
argsStartInclusive: Int,
argsEndExclusive: Int
): IntArray {
// TODO: For TDL macro invocations, see if we can calculate this during the "compile" step.
var numArgs = 0
val argsIndices = IntArray(macro.signature.size) // TODO performance: pool these
val argsIndices = expansion.session.intArrayForSize(macro.signature.size)
var currentArgIndex = argsStartInclusive
val encodingExpressions: List<Expression> = expansion.expressions

for (p in macro.signature) {
if (currentArgIndex >= argsEndExclusive) {
Expand Down
2 changes: 1 addition & 1 deletion src/main/java/com/amazon/ion/util/Assumptions.kt
Original file line number Diff line number Diff line change
Expand Up @@ -51,4 +51,4 @@ internal inline fun confirm(assumption: Boolean, lazyMessage: () -> String) {
/**
* Marks a branch as unreachable (for human readability).
*/
internal fun unreachable(reason: String? = null): Nothing = throw IllegalStateException(reason)
fun unreachable(reason: String? = null): Nothing = throw IllegalStateException(reason)

0 comments on commit 8161ef7

Please sign in to comment.