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

parsley.internal.deepembedding.backend.SelectiveEmbedding.scala Maven / Gradle / Ivy

/*
 * Copyright 2020 Parsley Contributors 
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */
package parsley.internal.deepembedding.backend

import parsley.internal.deepembedding.ContOps, ContOps.{suspend, ContAdapter}
import parsley.internal.deepembedding.singletons._
import parsley.internal.machine.instructions

import Branch.FlipApp
import StrictParsley.InstrBuffer

private [backend] sealed abstract class BranchLike[A, B, C, D](finaliser: Option[instructions.Instr]) extends StrictParsley[D] {
    val b: StrictParsley[A]
    val p: StrictParsley[B]
    val q: StrictParsley[C]
    def instr(label: Int): instructions.Instr

    def inlinable: Boolean = false
    final override def codeGen[M[_, +_]: ContOps, R](producesResults: Boolean)(implicit instrs: InstrBuffer, state: CodeGenState): M[R, Unit] = {
        val toP = state.freshLabel()
        val end = state.freshLabel()
        suspend(b.codeGen[M, R](producesResults = true)) >> {
            instrs += instr(toP)
            suspend(q.codeGen[M, R](producesResults)) >> {
                for (instr <- finaliser) {
                    if (producesResults) instrs += instr
                    else instrs += instructions.Pop
                }
                instrs += new instructions.Jump(end)
                instrs += new instructions.Label(toP)
                suspend(p.codeGen[M, R](producesResults)) |> {
                    for (instr <- finaliser) {
                        if (producesResults) instrs += instr
                        else instrs += instructions.Pop
                    }
                    instrs += new instructions.Label(end)
                }
            }
        }
    }
}

private [deepembedding] final class Branch[A, B, C](val b: StrictParsley[Either[A, B]], val p: StrictParsley[A => C], val q: StrictParsley[B => C])
    extends BranchLike[Either[A, B], A => C, B => C, C](Some(FlipApp)) {
    override def instr(label: Int): instructions.Instr = new instructions.Case(label)
    override def optimise: StrictParsley[C] = b match {
        case Pure(Left(x)) => <*>(p, new Pure(x)).optimise
        case Pure(Right(y)) => <*>(q, new Pure(y)).optimise
        case _ => (p, q) match {
            case (Pure(f), Pure(g)) => <*>(new Pure((x: Either[A, B]) => x.fold(f, g)), b)
            case _ => this
        }
    }
    // $COVERAGE-OFF$
    final override def pretty: String = s"branch(${b.pretty}, ${p.pretty}, ${q.pretty})"
    // $COVERAGE-ON$
}

private [deepembedding] final class If[A](val b: StrictParsley[Boolean], val p: StrictParsley[A], val q: StrictParsley[A])
    extends BranchLike[Boolean, A, A, A](None) {
    override def instr(label: Int): instructions.Instr = new instructions.If(label)
    override def optimise: StrictParsley[A] = b match {
        case Pure(true) => p
        case Pure(false) => q
        case _ => this
    }
    // $COVERAGE-OFF$
    final override def pretty: String = s"if(${b.pretty}, ${p.pretty}, ${q.pretty})"
    // $COVERAGE-ON$
}

private [backend] sealed abstract class FilterLike[A, B] extends StrictParsley[B] {
    protected val p: StrictParsley[A]
    protected val err: StrictParsley[((A, Int)) => Nothing]
    def inlinable: Boolean = false
    protected def instr(handler: Int, jumpLabel: Int): instructions.Instr

    final override def codeGen[M[_, +_]: ContOps, R](producesResults: Boolean)(implicit instrs: InstrBuffer, state: CodeGenState): M[R, Unit] = {
        val handler1 = state.getLabel(instructions.PopStateAndFail)
        val handler2 = state.getLabel(instructions.AmendAndFail(false))
        val jumpLabel = state.freshLabel()
        instrs += new instructions.PushHandlerAndState(handler1)
        suspend(p.codeGen[M, R](producesResults = true)) >> {
            instrs += instr(handler2, jumpLabel)
            suspend(err.codeGen[M, R](producesResults = true)) |> {
                instrs += new instructions.Label(jumpLabel)
                if (!producesResults) instrs += instructions.Pop
            }
        }
    }

    // $COVERAGE-OFF$
    final override def pretty: String = pretty(p.pretty, err.pretty)
    protected def pretty(p: String, err: String): String
    // $COVERAGE-ON$
}

private [deepembedding] final class Filter[A](val p: StrictParsley[A], pred: A => Boolean, val err: StrictParsley[((A, Int)) => Nothing])
    extends FilterLike[A, A] {
    final override def instr(handler: Int, jumpLabel: Int): instructions.Instr = new instructions.Filter(pred, jumpLabel, handler)
    final override def pretty(p: String, err: String): String = s"filterWith($p, ???, $err)"
}

private [deepembedding] final class MapFilter[A, B](val p: StrictParsley[A], pred: A => Option[B], val err: StrictParsley[((A, Int)) => Nothing])
    extends FilterLike[A, B] {
    final override def instr(handler: Int, jumpLabel: Int): instructions.Instr = new instructions.MapFilter(pred, jumpLabel, handler)
    final override def pretty(p: String, err: String): String = s"mapFilterWith($p, ???, $err)"
}

private [backend] object Branch {
    val FlipApp = instructions.Lift2[Any, Any => Any, Any]((x, f) => f(x))
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy