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

dotty.tools.dotc.transform.EtaReduce.scala Maven / Gradle / Ivy

package dotty.tools
package dotc
package transform

import MegaPhase.MiniPhase
import core.*
import Symbols.*, Contexts.*, Types.*, Decorators.*
import StdNames.nme

import NameKinds.AdaptedClosureName

/** Rewrite `(x1, ... xN) => f(x1, ... xN)` for N >= 0 to `f`,
 *  provided `f` is a pure path of function type.
 *
 *  This optimization is crucial for context functions. The compiler
 *  produces a contextual closure around values passed as arguments
 *  where a context function is expected, unless that value has the
 *  syntactic form of a context function literal.
 *
 *  Also handle variants of eta-expansions where
 *   - result f.apply(X_1,...,X_n) is subject to a synthetic cast, or
 *   - the application uses a specialized apply method, or
 *   - the closure is adapted (see Erasure#adaptClosure)
 *
 *  Without this phase, when a contextual function is passed as an argument to a
 *  recursive function, that would have the unfortunate effect of a linear growth
 *  in transient thunks of identical type wrapped around each other, leading
 *  to performance degradation, and in some cases, stack overflows.
 */
class EtaReduce extends MiniPhase:
  import ast.tpd.*

  override def phaseName: String = EtaReduce.name

  override def description: String = EtaReduce.description

  override def transformBlock(tree: Block)(using Context): Tree =

    def tryReduce(mdef: DefDef, rhs: Tree): Tree = rhs match
      case Apply(Select(fn, name), args)
      if (name == nme.apply || defn.FunctionSpecializedApplyNames.contains(name))
          && mdef.paramss.head.corresponds(args)((param, arg) =>
              arg.isInstanceOf[Ident] && arg.symbol == param.symbol)
          && isPurePath(fn)
          && fn.tpe <:< tree.tpe
          && !(fn.symbol.is(Flags.Param) && fn.symbol.owner == mdef.symbol) // Do not eta-educe `(..., f: T => R, ...) => f.apply(..)` into `f`
          && defn.isFunctionClass(fn.tpe.widen.typeSymbol) =>
        report.log(i"eta reducing $tree --> $fn")
        fn
      case TypeApply(Select(qual, _), _) if rhs.symbol.isTypeCast && rhs.span.isSynthetic =>
        tryReduce(mdef, qual)
      case Apply(_, arg :: Nil) if Erasure.Boxing.isUnbox(rhs.symbol) && rhs.span.isSynthetic =>
        tryReduce(mdef, arg)
      case Block(call :: Nil, unit @ Literal(Constants.Constant(()))) if unit.span.isSynthetic =>
        tryReduce(mdef, call)
      case _ =>
        tree

    tree match
      case Block((meth: DefDef) :: Nil, expr) if meth.symbol.isAnonymousFunction =>
        expr match
          case closure: Closure if meth.symbol == closure.meth.symbol =>
            tryReduce(meth, meth.rhs)
          case Block((adapted: DefDef) :: Nil, closure: Closure)
          if adapted.name.is(AdaptedClosureName) && adapted.symbol == closure.meth.symbol =>
            tryReduce(meth, meth.rhs)
          case _ =>
            tree
      case _ =>
        tree
  end transformBlock

end EtaReduce

object EtaReduce:
  val name: String = "etaReduce"
  val description: String = "reduce eta expansions of pure paths"




© 2015 - 2025 Weber Informatics LLC | Privacy Policy