All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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