use texcraft_stdext::collections::groupingmap;
use texlang::prelude as txl;
use texlang::traits::*;
use texlang::variable::SupportedType;
use texlang::variable::TypedVariable;
use texlang::*;
pub fn get_advance<S: TexlangState>() -> command::BuiltIn<S> {
get_command::<S, AdvanceOp>()
}
pub fn get_advance_checked<S: TexlangState>() -> command::BuiltIn<S> {
get_command::<S, AdvanceCheckedOp>()
}
pub fn get_multiply<S: TexlangState>() -> command::BuiltIn<S> {
get_command::<S, MultiplyOp>()
}
pub fn get_multiply_wrapped<S: TexlangState>() -> command::BuiltIn<S> {
get_command::<S, MultiplyWrappedOp>()
}
pub fn get_divide<S: TexlangState>() -> command::BuiltIn<S> {
get_command::<S, DivideOp>()
}
fn get_command<S: TexlangState, O: Op>() -> command::BuiltIn<S> {
command::BuiltIn::new_execution(math_primitive_fn::<S, O>)
.with_tag(variable_op_tag())
.with_doc(O::DOC)
}
static VARIABLE_OP_TAG: command::StaticTag = command::StaticTag::new();
pub fn variable_op_tag() -> command::Tag {
VARIABLE_OP_TAG.get()
}
trait Number: Sized + Default + SupportedType + Copy + std::fmt::Display {
fn wrapping_add(lhs: Self, rhs: Self) -> Self;
fn checked_add(lhs: Self, rhs: Self) -> Option<Self>;
fn checked_mul(lhs: Self, rhs: i32) -> Option<Self>;
fn wrapping_mul(lhs: Self, rhs: i32) -> Self;
fn checked_div(lhs: Self, rhs: i32) -> Option<Self>;
}
impl Number for i32 {
fn wrapping_add(lhs: Self, rhs: Self) -> Self {
lhs.wrapping_add(rhs)
}
fn checked_add(lhs: Self, rhs: Self) -> Option<Self> {
lhs.checked_add(rhs)
}
fn checked_mul(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_mul(rhs)
}
fn wrapping_mul(lhs: Self, rhs: i32) -> Self {
lhs.wrapping_mul(rhs)
}
fn checked_div(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_div(rhs)
}
}
impl Number for core::Scaled {
fn wrapping_add(lhs: Self, rhs: Self) -> Self {
lhs.wrapping_add(rhs)
}
fn checked_add(lhs: Self, rhs: Self) -> Option<Self> {
lhs.checked_add(rhs)
}
fn checked_mul(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_mul(rhs)
}
fn wrapping_mul(lhs: Self, rhs: i32) -> Self {
lhs.wrapping_mul(rhs)
}
fn checked_div(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_div(rhs)
}
}
impl Number for core::Glue {
fn wrapping_add(lhs: Self, rhs: Self) -> Self {
lhs.wrapping_add(rhs)
}
fn checked_add(lhs: Self, rhs: Self) -> Option<Self> {
lhs.checked_add(rhs)
}
fn checked_mul(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_mul(rhs)
}
fn wrapping_mul(lhs: Self, rhs: i32) -> Self {
lhs.wrapping_mul(rhs)
}
fn checked_div(lhs: Self, rhs: i32) -> Option<Self> {
lhs.checked_div(rhs)
}
}
trait Op {
const DOC: &'static str;
const RHS_SAME: bool;
type Error: error::TexError;
fn apply<N: Number>(lhs: N, rhs_i: i32, rhs_n: N) -> Result<N, Self::Error>;
fn apply_to_variable<S: TexlangState, N: Number>(
variable: TypedVariable<S, N>,
input: &mut vm::ExecutionInput<S>,
scope: groupingmap::Scope,
) -> txl::Result<()> {
let lhs = *variable.get(input.state());
let (rhs_i, rhs_n) = if Self::RHS_SAME {
(0_i32, N::parse(input)?)
} else {
(i32::parse(input)?, Default::default())
};
let result = match Self::apply(lhs, rhs_i, rhs_n) {
Ok(result) => result,
Err(err) => {
return input.error(err);
}
};
variable.set(input, scope, result);
Ok(())
}
}
struct AdvanceOp;
impl Op for AdvanceOp {
const DOC: &'static str = "Add a quantity to a variable";
const RHS_SAME: bool = true;
type Error = OverflowError;
fn apply<N: Number>(lhs: N, _: i32, rhs_n: N) -> Result<N, Self::Error> {
Ok(N::wrapping_add(lhs, rhs_n))
}
}
struct AdvanceCheckedOp;
impl Op for AdvanceCheckedOp {
const DOC: &'static str = "Add a number to a variable (error on overflow)";
const RHS_SAME: bool = true;
type Error = OverflowError;
fn apply<N: Number>(lhs: N, _: i32, rhs_n: N) -> Result<N, Self::Error> {
match N::checked_add(lhs, rhs_n) {
Some(result) => Ok(result),
None => Err(OverflowError {
op_name: "addition",
lhs: format!["{lhs}"],
rhs: format!["{rhs_n}"],
wrapped_result: format!["{}", N::wrapping_add(lhs, rhs_n)],
}),
}
}
}
struct MultiplyOp;
impl Op for MultiplyOp {
const DOC: &'static str = "Multiply a variable by an integer";
const RHS_SAME: bool = false;
type Error = OverflowError;
fn apply<N: Number>(lhs: N, rhs_i: i32, _: N) -> Result<N, Self::Error> {
match N::checked_mul(lhs, rhs_i) {
Some(result) => Ok(result),
None => Err(OverflowError {
op_name: "multiplication",
lhs: format!["{lhs}"],
rhs: format!["{rhs_i}"],
wrapped_result: format!["{}", N::wrapping_mul(lhs, rhs_i)],
}),
}
}
}
struct MultiplyWrappedOp;
impl Op for MultiplyWrappedOp {
const DOC: &'static str = "Multiply a variable by an integer (wrap on overflow)";
const RHS_SAME: bool = false;
type Error = OverflowError;
fn apply<N: Number>(lhs: N, rhs_i: i32, _: N) -> Result<N, Self::Error> {
Ok(N::wrapping_mul(lhs, rhs_i))
}
}
struct DivideOp;
impl Op for DivideOp {
const DOC: &'static str = "Divide a variable by an integer";
const RHS_SAME: bool = false;
type Error = DivisionByZeroError;
fn apply<N: Number>(lhs: N, rhs_i: i32, _: N) -> Result<N, Self::Error> {
match N::checked_div(lhs, rhs_i) {
Some(result) => Ok(result),
None => Err(DivisionByZeroError {
numerator: format!["{lhs}"],
}),
}
}
}
#[derive(Debug)]
struct OverflowError {
op_name: &'static str,
lhs: String,
rhs: String,
wrapped_result: String,
}
impl error::TexError for OverflowError {
fn kind(&self) -> error::Kind {
error::Kind::FailedPrecondition
}
fn title(&self) -> String {
format!["overflow in checked {}", self.op_name]
}
fn notes(&self) -> Vec<error::display::Note> {
vec![
format!["left hand side evaluated to {}", self.lhs].into(),
format!["right hand side evaluated to {}", self.rhs].into(),
format!["wrapped result would be {}", self.wrapped_result].into(),
]
}
}
#[derive(Debug)]
struct DivisionByZeroError {
numerator: String,
}
impl error::TexError for DivisionByZeroError {
fn kind(&self) -> error::Kind {
error::Kind::FailedPrecondition
}
fn title(&self) -> String {
"division by zero".into()
}
fn notes(&self) -> Vec<error::display::Note> {
vec![format!["numerator evaluated to {}", self.numerator].into()]
}
}
fn math_primitive_fn<S: TexlangState, O: Op>(
_: token::Token,
input: &mut vm::ExecutionInput<S>,
) -> txl::Result<()> {
let scope = TexlangState::variable_assignment_scope_hook(input.state_mut());
let token = input.next_or_err(ArithmeticVariableEndOfInput {})?;
match token.value() {
token::Value::CommandRef(command_ref) => {
match input.commands_map().get_command(&command_ref) {
None => input.error(
parse::Error::new("a variable", Some(token), "")
.with_got_override("got an undefined control sequence")
.with_annotation_override("undefined control sequence"),
),
Some(command::Command::Variable(cmd)) => {
if !cmd.is_arithmetic() {
input.error(
parse::Error::new("an arithmetic variable", Some(token), "")
.with_got_override("got a non-arithmetic variable"),
)?;
return Ok(());
}
let variable = cmd.clone().resolve(token, input.as_mut())?;
OptionalBy::parse(input)?;
match variable {
variable::Variable::Int(variable) => {
O::apply_to_variable(variable, input, scope)
}
variable::Variable::Dimen(variable) => {
O::apply_to_variable(variable, input, scope)
}
variable::Variable::Glue(variable) => {
O::apply_to_variable(variable, input, scope)
}
variable::Variable::CatCode(_)
| variable::Variable::TokenList(_)
| variable::Variable::MathCode(_)
| variable::Variable::Font(_) => {
unreachable!("only arithmetic commands are considered");
}
}
}
Some(cmd) => input.error(
parse::Error::new("a variable", Some(token), "")
.with_got_override("got a non-variable command")
.with_annotation_override(format!["control sequence referencing {cmd}"]),
),
}
}
_ => input.error(
parse::Error::new("a variable", Some(token), "")
.with_got_override("got a character token"),
),
}
}
#[derive(Debug)]
struct ArithmeticVariableEndOfInput;
impl error::EndOfInputError for ArithmeticVariableEndOfInput {
fn doing(&self) -> String {
"parsing an arithmetic variable".into()
}
}
struct OptionalBy;
impl Parsable for OptionalBy {
fn parse_impl<S: TexlangState>(input: &mut vm::ExpandedStream<S>) -> txl::Result<Self> {
texlang::parse::parse_keyword(input, "by")?;
Ok(OptionalBy)
}
}
#[cfg(test)]
mod tests {
use std::collections::HashMap;
use super::*;
use crate::codes;
use crate::prefix;
use crate::registers;
use crate::the;
use texlang::types::CatCode;
use texlang::vm::implement_has_component;
use texlang_testing::*;
#[derive(Default)]
struct State {
catcode: codes::Component<CatCode>,
prefix: prefix::Component,
registers: registers::Component<i32, 256>,
registers_dimen: registers::Component<core::Scaled, 256>,
registers_skip: registers::Component<core::Glue, 256>,
testing: TestingComponent,
}
impl TexlangState for State {
fn variable_assignment_scope_hook(
state: &mut Self,
) -> texcraft_stdext::collections::groupingmap::Scope {
prefix::variable_assignment_scope_hook(state)
}
fn recoverable_error_hook(
&self,
recoverable_error: error::TracedTexError,
) -> Result<(), Box<dyn error::TexError>> {
texlang_testing::TestingComponent::recoverable_error_hook(self, recoverable_error)
}
}
impl the::TheCompatible for State {}
implement_has_component![State{
catcode: codes::Component<CatCode>,
prefix: prefix::Component,
registers: registers::Component<i32, 256>,
registers_dimen: registers::Component<core::Scaled, 256>,
registers_skip: registers::Component<core::Glue, 256>,
testing: TestingComponent,
}];
fn built_in_commands() -> HashMap<&'static str, command::BuiltIn<State>> {
HashMap::from([
("advance", get_advance()),
("advanceChecked", get_advance_checked()),
("multiply", get_multiply()),
("multiplyWrapped", get_multiply_wrapped()),
("divide", get_divide()),
("catcode", codes::get_catcode()),
("count", registers::get_count()),
("dimen", registers::get_dimen()),
("skip", registers::get_skip()),
("global", prefix::get_global()),
("the", the::get_the()),
])
}
macro_rules! arithmetic_tests {
( $register: expr, $( ($name: ident, $op: expr, $lhs: expr, $rhs: expr, $expected: expr) ),* $(,)? ) => {
test_suite![
expansion_equality_tests(
$(
(
$name,
format![r"{} 1 {} {} {} 1 {} \the{} 1", $register, $lhs, $op, $register, $rhs, $register],
$expected
),
)*
),
];
};
}
arithmetic_tests![
r"\count",
(advance_base_case, r"\advance", "1", "2", "3"),
(advance_base_case_with_by, r"\advance", "1", "by 2", "3"),
(advance_negative_summand, r"\advance", "10", "-2", "8"),
(
advance_overflow_case,
r"\advance",
"2147483647",
"1",
"-2147483648"
),
(multiply_base_case, r"\multiply", "5", "4", "20"),
(multiply_base_case_with_by, r"\multiply", "5", "by 4", "20"),
(multiply_pos_neg, r"\multiply", "-5", "4", "-20"),
(multiply_neg_pos, r"\multiply", "5", "-4", "-20"),
(multiply_neg_neg, r"\multiply", "-5", "-4", "20"),
(
multiply_wrapping_overflow,
r"\multiplyWrapped",
"100000",
"100000",
"1410065408"
),
(
multiply_wrapping_base_case,
r"\multiplyWrapped",
"5",
"4",
"20"
),
(divide_base_case, r"\divide", "9", "4", "2"),
(divide_with_by, r"\divide", "9", "by 4", "2"),
(divide_pos_neg, r"\divide", "-9", "4", "-2"),
(divide_neg_pos, r"\divide", "9", "-4", "-2"),
(divide_neg_neg, r"\divide", "-9", "-4", "2"),
(divide_exact, r"\divide", "100", "10", "10"),
(advance_checked_base_case, r"\advanceChecked", "1", "2", "3"),
(
advance_checked_negative_summand,
r"\advanceChecked",
"10",
"-2",
"8"
)
];
arithmetic_tests![
r"\dimen",
(advance_dimen_1, r"\advance", "1pt", "2pt", "3.0pt"),
(advance_dimen_2, r"\advance", "0.025pt", "0.5pt", "0.525pt"),
(mul_dimen_1, r"\multiply", "10pt", "2", "20.0pt"),
(div_dimen_1, r"\divide", "10pt", "2", "5.0pt"),
];
arithmetic_tests![
r"\skip",
(
advance_glue_1,
r"\advance",
"1pt plus 2pt minus 3pt",
"60pt plus 50pt minus 40pt",
"61.0pt plus 52.0pt minus 43.0pt"
),
(advance_glue_2, r"\advance", "0.025pt", "0.5pt", "0.025pt"),
(
advance_glue_3,
r"\advance",
"1pt plus 2fill minus 3fil",
"60pt plus 50pt minus 40filll",
"61.0pt plus 2.0fill minus 40.0filll"
),
(
mul_glue_1,
r"\multiply",
"1pt plus 2pt minus 1.25pt",
"2",
"2.0pt plus 4.0pt minus 2.5pt"
),
(
div_glue_1,
r"\divide",
"10pt plus 20pt minus 3pt",
"2",
"5.0pt plus 10.0pt minus 1.5pt"
),
];
test_suite![
expansion_equality_tests(
(
advance_x_by_x,
r"\count 1 200 \advance \count 1 by \count 1 a\the\count 1",
r"a400"
),
(
global_advance,
r"\count 1 5{\global\advance\count 1 8}\the\count 1",
"13"
),
(
local_advance,
r"\count 1 5{\advance\count 1 8}\the\count 1",
"5"
),
),
recoverable_failure_tests(
(
advance_incorrect_keyword_1,
r"\count 1 1\advance\count 1 fy 2 \the \count 1",
"fy 2 1",
),
(
advance_incorrect_keyword_2,
r"\count 1 1\advance\count 1 be 2 \the \count 1",
"be 2 1",
),
(
advance_catcode_not_supported,
r"\advance\catcode 100 by 2",
"100 by 2",
),
(
advance_checked_overflow,
r"\count 1 2147483647 \advanceChecked\count 1 by 1",
"",
),
(
multiply_overflow,
r"\count 1 100000 \multiply\count 1 by 100000 \the \count 1",
"100000"
),
(
divide_by_zero,
r"\count 1 20 \divide\count 1 by 0 \the\count 1",
"20"
),
),
];
}