dotty.tools.dotc.transform.TryCatchPatterns.scala Maven / Gradle / Ivy
package dotty.tools.dotc
package transform
import core.Symbols.*
import core.StdNames.*
import core.Types.*
import core.NameKinds.ExceptionBinderName
import dotty.tools.dotc.core.Flags
import dotty.tools.dotc.core.Contexts.*
import dotty.tools.dotc.transform.MegaPhase.MiniPhase
import dotty.tools.dotc.util.Spans.Span
/** Compiles the cases that can not be handled by primitive catch cases as a common pattern match.
*
* The following code:
* ```
* try { }
* catch {
* // Cases that can be handled by catch
* // Cases starting with first one that can't be handled by catch
* }
* ```
* will become:
* ```
* try { }
* catch {
*
* case e => e match {
*
* }
* }
* ```
*
* Cases that are not supported include:
* - Applies and unapplies
* - Idents
* - Alternatives
* - `case _: T =>` where `T` is not `Throwable`
*
*/
class TryCatchPatterns extends MiniPhase {
import dotty.tools.dotc.ast.tpd.*
override def phaseName: String = TryCatchPatterns.name
override def description: String = TryCatchPatterns.description
override def runsAfter: Set[String] = Set(ElimRepeated.name)
override def checkPostCondition(tree: Tree)(using Context): Unit = tree match {
case Try(_, cases, _) =>
cases.foreach {
case CaseDef(Typed(_, _), guard, _) => assert(guard.isEmpty, "Try case should not contain a guard.")
case CaseDef(Bind(_, _), guard, _) => assert(guard.isEmpty, "Try case should not contain a guard.")
case c =>
assert(isDefaultCase(c), "Pattern in Try should be Bind, Typed or default case.")
}
case _ =>
}
override def transformTry(tree: Try)(using Context): Tree = {
val (tryCases, patternMatchCases) = tree.cases.span(isCatchCase)
val fallbackCase = mkFallbackPatterMatchCase(patternMatchCases, tree.span)
cpy.Try(tree)(cases = tryCases ++ fallbackCase)
}
/** Is this pattern node a catch-all or type-test pattern? */
private def isCatchCase(cdef: CaseDef)(using Context): Boolean = cdef match {
case CaseDef(Typed(Ident(nme.WILDCARD), tpt), EmptyTree, _) => isSimpleThrowable(tpt.tpe)
case CaseDef(Bind(_, Typed(Ident(nme.WILDCARD), tpt)), EmptyTree, _) => isSimpleThrowable(tpt.tpe)
case _ => isDefaultCase(cdef)
}
private def isSimpleThrowable(tp: Type)(using Context): Boolean = tp.stripped match {
case tp @ TypeRef(pre, _) =>
(pre == NoPrefix || pre.typeSymbol.isStatic) && // Does not require outer class check
!tp.symbol.is(Flags.Trait) && // Traits not supported by JVM
tp.derivesFrom(defn.ThrowableClass)
case tp: AppliedType =>
isSimpleThrowable(tp.tycon)
case _ =>
false
}
private def mkFallbackPatterMatchCase(patternMatchCases: List[CaseDef], span: Span)(
implicit ctx: Context): Option[CaseDef] =
if (patternMatchCases.isEmpty) None
else {
val exName = ExceptionBinderName.fresh()
val fallbackSelector =
newSymbol(ctx.owner, exName, Flags.Synthetic | Flags.Case, defn.ThrowableType, coord = span)
val sel = Ident(fallbackSelector.termRef).withSpan(span)
val rethrow = CaseDef(EmptyTree, EmptyTree, Throw(ref(fallbackSelector)))
Some(CaseDef(
Bind(fallbackSelector, Underscore(fallbackSelector.info).withSpan(span)),
EmptyTree,
transformFollowing(Match(sel, patternMatchCases ::: rethrow :: Nil)))
)
}
}
object TryCatchPatterns:
val name: String = "tryCatchPatterns"
val description: String = "compile cases in try/catch"
© 2015 - 2025 Weber Informatics LLC | Privacy Policy