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

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)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy