parsley.internal.deepembedding.frontend.GeneralisedEmbedding.scala Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2020 Parsley Contributors
*
* SPDX-License-Identifier: BSD-3-Clause
*/
package parsley.internal.deepembedding.frontend
import parsley.internal.deepembedding.ContOps, ContOps.{suspend, ContAdapter}
import parsley.internal.deepembedding.backend, backend.StrictParsley
// These all capture the general structures of combinators to factor out the common patterns for `findLetsAux` and `preprocess`.
private [parsley] sealed abstract class GenericLazyParsley[A] extends LazyParsley[A]
private [frontend] abstract class Unary[A, B](private [frontend] val p: LazyParsley[A]) extends GenericLazyParsley[B] {
def make(p: StrictParsley[A]): StrictParsley[B]
final override def findLetsAux[M[_, +_]: ContOps, R](seen: Set[LazyParsley[_]])(implicit state: LetFinderState): M[R,Unit] =
suspend(p.findLets(seen))
override def preprocess[M[_, +_]: ContOps, R, B_ >: B](implicit lets: LetMap): M[R, StrictParsley[B_]] =
for (p <- suspend(p.optimised[M, R, A])) yield make(p)
}
private [frontend] abstract class Binary[A, B, C](private [frontend] val left: LazyParsley[A],
_right: =>LazyParsley[B]) extends GenericLazyParsley[C] {
private [frontend] lazy val right = _right
def make(p: StrictParsley[A], q: StrictParsley[B]): StrictParsley[C]
final override def findLetsAux[M[_, +_]: ContOps, R](seen: Set[LazyParsley[_]])(implicit state: LetFinderState): M[R,Unit] = {
suspend(left.findLets[M, R](seen)) >> suspend(right.findLets(seen))
}
final override def preprocess[M[_, +_]: ContOps, R, C_ >: C](implicit lets: LetMap): M[R, StrictParsley[C_]] =
for {
left <- suspend(left.optimised[M, R, A])
right <- suspend(right.optimised[M, R, B])
} yield make(left, right)
}
private [frontend] abstract class Ternary[A, B, C, D](private [frontend] val first: LazyParsley[A],
_second: =>LazyParsley[B],
_third: =>LazyParsley[C]) extends GenericLazyParsley[D] {
private [frontend] lazy val second: LazyParsley[B] = _second
private [frontend] lazy val third: LazyParsley[C] = _third
def make(p: StrictParsley[A], q: StrictParsley[B], r: StrictParsley[C]): StrictParsley[D]
final override def findLetsAux[M[_, +_]: ContOps, R](seen: Set[LazyParsley[_]])(implicit state: LetFinderState): M[R, Unit] = {
suspend(first.findLets[M, R](seen)) >> suspend(second.findLets(seen)) >> suspend(third.findLets(seen))
}
final override def preprocess[M[_, +_]: ContOps, R, D_ >: D](implicit lets: LetMap): M[R, StrictParsley[D_]] =
for {
first <- suspend(first.optimised[M, R, A])
second <- suspend(second.optimised[M, R, B])
third <- suspend(third.optimised[M, R, C])
} yield make(first, second, third)
}