dotty.tools.dotc.transform.init.Cache.scala Maven / Gradle / Ivy
package dotty.tools.dotc
package transform
package init
import core.*
import Contexts.*
import ast.tpd
import tpd.Tree
/** The co-inductive cache used for analysis
*
* The cache contains two maps from `(Config, Tree)` to `Res`:
*
* - input cache (`this.last`)
* - output cache (`this.current`)
*
* The two caches are required because we want to make sure in a new iteration,
* an expression is evaluated exactly once. The monotonicity of the analysis
* ensures that the cache state goes up the lattice of the abstract domain,
* consequently the algorithm terminates.
*
* The general skeleton for usage of the cache is as follows
*
* def analysis(entryExp: Expr) = {
* def iterate(entryExp: Expr)(using Cache) =
* eval(entryExp, initConfig)
* if cache.hasChanged && noErrors then
* cache.last = cache.current
* cache.current = Empty
* cache.changed = false
* iterate(entryExp)
* else
* reportErrors
*
*
* def eval(expr: Expr, config: Config)(using Cache) =
* cache.cachedEval(config, expr) {
* // Actual recursive evaluation of expression.
* //
* // Only executed if the entry `(exp, config)` is not in the output cache.
* }
*
* iterate(entryExp)(using new Cache)
* }
*
* See the documentation for the method `Cache.cachedEval` for more information.
*
* What goes to the configuration (`Config`) and what goes to the result (`Res`)
* need to be decided by the specific analysis and justified by reasoning about
* soundness.
*
* @tparam Config The analysis state that matters for evaluating an expression.
* @tparam Res The result from the evaluation the given expression.
*/
class Cache[Config, Res]:
import Cache.*
/** The cache for expression values from last iteration */
protected var last: ExprValueCache[Config, Res] = Map.empty
/** The output cache for expression values
*
* The output cache is computed based on the cache values `last` from the
* last iteration.
*
* Both `last` and `current` are required to make sure an encountered
* expression is evaluated once in each iteration.
*/
protected var current: ExprValueCache[Config, Res] = Map.empty
/** Whether the current heap is different from the last heap?
*
* `changed == false` implies that the fixed point has been reached.
*/
protected var changed: Boolean = false
/** Whether any value in the output cache (this.current) was accessed
* after being added. If no cached values are used after they are added
* for the first time then another iteration of analysis is not needed.
*/
protected var cacheUsed: Boolean = false
/** Used to avoid allocation, its state does not matter */
protected given MutableTreeWrapper = new MutableTreeWrapper
def get(config: Config, expr: Tree): Option[Res] =
val res = current.get(config, expr)
cacheUsed = cacheUsed || res.nonEmpty
res
/** Evaluate an expression with cache
*
* The algorithmic skeleton is as follows:
*
* if don't cache result then
* return eval(expr)
* if this.current.contains(config, expr) then
* return cached value
* else
* val assumed = this.last(config, expr) or bottom value if absent
* this.current(config, expr) = assumed
* val actual = eval(expr)
*
* if assumed != actual then
* this.changed = true
* this.current(config, expr) = actual
*
*/
def cachedEval(config: Config, expr: Tree, cacheResult: Boolean, default: Res)(eval: Tree => Res): Res =
if !cacheResult then
eval(expr)
else
this.get(config, expr) match
case Some(value) => value
case None =>
val assumeValue: Res =
this.last.get(config, expr) match
case Some(value) => value
case None =>
this.last = this.last.updatedNested(config, expr, default)
default
this.current = this.current.updatedNested(config, expr, assumeValue)
val actual = eval(expr)
if actual != assumeValue then
// println("Changed! from = " + assumeValue + ", to = " + actual)
this.changed = true
this.current = this.current.updatedNested(config, expr, actual)
// this.current = this.current.removed(config, expr)
end if
actual
end if
end cachedEval
def hasChanged = changed
def isUsed = cacheUsed
/** Prepare cache for the next iteration
*
* 1. Reset changed flag.
*
* 2. Use current cache as last cache and set current cache to be empty.
*/
def prepareForNextIteration()(using Context) =
this.changed = false
this.cacheUsed = false
this.last = this.current
this.current = Map.empty
end Cache
object Cache:
type ExprValueCache[Config, Res] = Map[Config, Map[TreeWrapper, Res]]
/** A wrapper for trees for storage in maps based on referential equality of trees. */
abstract class TreeWrapper:
def tree: Tree
override final def equals(other: Any): Boolean =
other match
case that: TreeWrapper => this.tree eq that.tree
case _ => false
override final def hashCode = tree.hashCode
/** The immutable wrapper is intended to be stored as key in the heap. */
class ImmutableTreeWrapper(val tree: Tree) extends TreeWrapper
/** For queries on the heap, reuse the same wrapper to avoid unnecessary allocation.
*
* A `MutableTreeWrapper` is only ever used temporarily for querying a map,
* and is never inserted to the map.
*/
class MutableTreeWrapper extends TreeWrapper:
var queryTree: Tree | Null = null
def tree: Tree = queryTree match
case tree: Tree => tree
case null => ???
extension [Config, Res](cache: ExprValueCache[Config, Res])
def get(config: Config, expr: Tree)(using queryWrapper: MutableTreeWrapper): Option[Res] =
queryWrapper.queryTree = expr
cache.get(config).flatMap(_.get(queryWrapper))
def removed(config: Config, expr: Tree)(using queryWrapper: MutableTreeWrapper) =
queryWrapper.queryTree = expr
val innerMap2 = cache(config).removed(queryWrapper)
cache.updated(config, innerMap2)
def updatedNested(config: Config, expr: Tree, result: Res): ExprValueCache[Config, Res] =
val wrapper = new ImmutableTreeWrapper(expr)
updatedNestedWrapper(config, wrapper, result)
def updatedNestedWrapper(config: Config, wrapper: ImmutableTreeWrapper, result: Res): ExprValueCache[Config, Res] =
val innerMap = cache.getOrElse(config, Map.empty[TreeWrapper, Res])
val innerMap2 = innerMap.updated(wrapper, result)
cache.updated(config, innerMap2)
end extension
© 2015 - 2025 Weber Informatics LLC | Privacy Policy