parsley.internal.machine.instructions.OptInstrs.scala Maven / Gradle / Ivy
/*
* Copyright 2020 Parsley Contributors
*
* SPDX-License-Identifier: BSD-3-Clause
*/
package parsley.internal.machine.instructions
import scala.collection.mutable
import parsley.XCompat._
import parsley.token.errors.LabelConfig
import parsley.internal.errors.ExpectItem
import parsley.internal.machine.Context
import parsley.internal.machine.XAssert._
import parsley.internal.machine.errors.{EmptyHints, ExpectedError}
import parsley.internal.machine.stacks.ErrorStack
private [internal] final class Lift1(f: Any => Any) extends Instr {
override def apply(ctx: Context): Unit = {
ensureRegularInstruction(ctx)
ctx.exchangeAndContinue(f(ctx.stack.upeek))
}
// $COVERAGE-OFF$
override def toString: String = "Perform(?)"
// $COVERAGE-ON$
}
private [internal] object Lift1 {
def apply[A, B](f: A => B): Lift1 = new Lift1(f.asInstanceOf[Any => Any])
}
private [internal] final class Exchange[A](private [Exchange] val x: A) extends Instr {
override def apply(ctx: Context): Unit = {
ensureRegularInstruction(ctx)
ctx.exchangeAndContinue(x)
}
// $COVERAGE-OFF$
override def toString: String = s"Ex($x)"
// $COVERAGE-ON$
}
private [internal] final class SatisfyExchange[A](f: Char => Boolean, x: A, _expected: LabelConfig) extends Instr {
private [this] final val expected = _expected.asExpectDescs
override def apply(ctx: Context): Unit = {
ensureRegularInstruction(ctx)
if (ctx.moreInput && f(ctx.peekChar)) {
ctx.consumeChar()
ctx.pushAndContinue(x)
}
else ctx.expectedFail(expected, unexpectedWidth = 1)
}
// $COVERAGE-OFF$
override def toString: String = s"SatEx(?, $x)"
// $COVERAGE-ON$
}
private [internal] final class RecoverWith[A](x: A) extends Instr {
override def apply(ctx: Context): Unit = {
ensureHandlerInstruction(ctx)
ctx.restoreHints() // This must be before adding the error to hints
ctx.catchNoConsumed(ctx.handlers.check) {
ctx.handlers = ctx.handlers.tail
ctx.addErrorToHintsAndPop()
ctx.pushAndContinue(x)
}
}
// $COVERAGE-OFF$
override def toString: String = s"RecoverWith($x)"
// $COVERAGE-ON$
}
private [internal] final class AlwaysRecoverWith[A](x: A) extends Instr {
override def apply(ctx: Context): Unit = {
ensureHandlerInstruction(ctx)
ctx.restoreState()
ctx.restoreHints() // This must be before adding the error to hints
ctx.handlers = ctx.handlers.tail
ctx.addErrorToHintsAndPop()
ctx.good = true
ctx.pushAndContinue(x)
}
// $COVERAGE-OFF$
override def toString: String = s"AlwaysRecoverWith($x)"
// $COVERAGE-ON$
}
private [internal] final class JumpTable(jumpTable: mutable.LongMap[(Int, Iterable[ExpectItem])],
private [this] var default: Int,
private [this] var merge: Int,
size: Int,
allErrorItems: Iterable[ExpectItem]) extends Instr {
def this(prefixes: List[Char], labels: List[Int], default: Int, merge: Int,
size: Int, allErrorItems: Iterable[ExpectItem], errorItemss: List[Iterable[ExpectItem]]) = {
this(mutable.LongMap(prefixes.view.map(_.toLong).zip(labels.zip(errorItemss)).toSeq: _*), default, merge, size, allErrorItems)
}
private [this] var defaultPreamble: Int = _
override def apply(ctx: Context): Unit = {
ensureRegularInstruction(ctx)
if (ctx.moreInput) {
val (dest, errorItems) = jumpTable.getOrElse(ctx.peekChar.toLong, (default, allErrorItems))
ctx.pc = dest
if (dest != default) {
ctx.pushHandler(defaultPreamble)
ctx.hints = EmptyHints
}
addErrors(ctx, errorItems) // adds a handler
}
else {
addErrors(ctx, allErrorItems)
ctx.pc = default
}
}
private def addErrors(ctx: Context, errorItems: Iterable[ExpectItem]): Unit = {
// FIXME: the more appropriate way of demanding input may be to pick 1 character, for same rationale with StringTok
ctx.errs = new ErrorStack(new ExpectedError(ctx.offset, ctx.line, ctx.col, errorItems, unexpectedWidth = size), ctx.errs)
ctx.pushHandler(merge)
}
override def relabel(labels: Array[Int]): this.type = {
jumpTable.mapValuesInPlaceCompat {
case (_, (i, errs)) => (labels(i), errs)
}
default = labels(default)
merge = labels(merge)
defaultPreamble = default - 1
this
}
// $COVERAGE-OFF$
override def toString: String = s"JumpTable(${jumpTable.map{case (k, v) => k.toChar -> v._1}.mkString(", ")}, _ -> $default, $merge)"
// $COVERAGE-ON$
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy