1use crate::declaration::{QuotKind, RecursorVal};
6use crate::expr_util::{get_app_args, get_app_fn, mk_app};
7use crate::instantiate::instantiate_type_lparams;
8use crate::subst::instantiate;
9use crate::{Environment, Expr, Literal, Name};
10use std::collections::HashMap;
11
12use super::types::{
13 ConfigNode, DecisionNode, Either2, FlatSubstitution, FocusStack, LabelSet, NonEmptyVec,
14 PathBuf, Reducer, ReducerStats, ReducibilityHint, ReductionRule, ReductionTrace, RewriteRule,
15 RewriteRuleSet, SimpleDag, SlidingSum, SmallMap, SparseVec, StackCalc, StatSummary, Stopwatch,
16 StringPool, TokenBucket, TransformStat, TransitiveClosure, TransparencyMode, VersionedRecord,
17 WindowIterator, WriteOnce,
18};
19
20pub(super) fn instantiate_recursor_rhs(
29 rhs: &Expr,
30 rec_val: &RecursorVal,
31 rec_levels: &[crate::Level],
32 rec_args: &[Expr],
33 ctor_args: &[Expr],
34) -> Expr {
35 let mut subst = Vec::new();
36 let np = rec_val.num_params as usize;
37 for item in rec_args.iter().take(np) {
38 subst.push(item.clone());
39 }
40 let nm = rec_val.num_motives as usize;
41 let motive_start = np;
42 for i in 0..nm {
43 if motive_start + i < rec_args.len() {
44 subst.push(rec_args[motive_start + i].clone());
45 }
46 }
47 let nminor = rec_val.num_minors as usize;
48 let minor_start = motive_start + nm;
49 for i in 0..nminor {
50 if minor_start + i < rec_args.len() {
51 subst.push(rec_args[minor_start + i].clone());
52 }
53 }
54 for item in ctor_args.iter().skip(np) {
55 subst.push(item.clone());
56 }
57 let rhs_inst = if rec_val.common.level_params.is_empty() || rec_levels.is_empty() {
58 rhs.clone()
59 } else {
60 instantiate_type_lparams(rhs, &rec_val.common.level_params, rec_levels)
61 };
62 crate::instantiate::instantiate_rev(&rhs_inst, &subst)
63}
64pub(super) fn try_reduce_nat_app(head: &Expr, args: &[Expr]) -> Option<Expr> {
66 if let Expr::Const(name, _) = head {
67 let name_str = name.to_string();
68 match name_str.as_str() {
69 "Nat.zero" => {
70 return Some(Expr::Lit(Literal::Nat(0)));
71 }
72 "Nat.succ" if !args.is_empty() => {
73 if let Expr::Lit(Literal::Nat(n)) = &args[0] {
74 return Some(Expr::Lit(Literal::Nat(n + 1)));
75 }
76 }
77 "Nat.pred" if !args.is_empty() => {
78 if let Expr::Lit(Literal::Nat(n)) = &args[0] {
79 return Some(Expr::Lit(Literal::Nat(n.saturating_sub(1))));
80 }
81 }
82 "Nat.add" if args.len() >= 2 => {
83 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
84 (&args[0], &args[1])
85 {
86 return Some(Expr::Lit(Literal::Nat(m + n)));
87 }
88 }
89 "Nat.mul" if args.len() >= 2 => {
90 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
91 (&args[0], &args[1])
92 {
93 return Some(Expr::Lit(Literal::Nat(m * n)));
94 }
95 }
96 "Nat.sub" if args.len() >= 2 => {
97 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
98 (&args[0], &args[1])
99 {
100 return Some(Expr::Lit(Literal::Nat(m.saturating_sub(*n))));
101 }
102 }
103 "Nat.div" if args.len() >= 2 => {
104 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
105 (&args[0], &args[1])
106 {
107 if *n != 0 {
108 return Some(Expr::Lit(Literal::Nat(m / n)));
109 }
110 return Some(Expr::Lit(Literal::Nat(0)));
111 }
112 }
113 "Nat.mod" if args.len() >= 2 => {
114 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
115 (&args[0], &args[1])
116 {
117 if *n != 0 {
118 return Some(Expr::Lit(Literal::Nat(m % n)));
119 }
120 return Some(Expr::Lit(Literal::Nat(*m)));
121 }
122 }
123 "Nat.pow" if args.len() >= 2 => {
124 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
125 (&args[0], &args[1])
126 {
127 return Some(Expr::Lit(Literal::Nat(m.pow(*n as u32))));
128 }
129 }
130 "Nat.beq" if args.len() >= 2 => {
131 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
132 (&args[0], &args[1])
133 {
134 return Some(nat_bool_result(m == n));
135 }
136 }
137 "Nat.ble" if args.len() >= 2 => {
138 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
139 (&args[0], &args[1])
140 {
141 return Some(nat_bool_result(m <= n));
142 }
143 }
144 "Nat.blt" if args.len() >= 2 => {
145 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
146 (&args[0], &args[1])
147 {
148 return Some(nat_bool_result(m < n));
149 }
150 }
151 "Nat.gcd" if args.len() >= 2 => {
152 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
153 (&args[0], &args[1])
154 {
155 return Some(Expr::Lit(Literal::Nat(gcd(*m, *n))));
156 }
157 }
158 "Nat.land" if args.len() >= 2 => {
159 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
160 (&args[0], &args[1])
161 {
162 return Some(Expr::Lit(Literal::Nat(m & n)));
163 }
164 }
165 "Nat.lor" if args.len() >= 2 => {
166 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
167 (&args[0], &args[1])
168 {
169 return Some(Expr::Lit(Literal::Nat(m | n)));
170 }
171 }
172 "Nat.xor" if args.len() >= 2 => {
173 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
174 (&args[0], &args[1])
175 {
176 return Some(Expr::Lit(Literal::Nat(m ^ n)));
177 }
178 }
179 "Nat.shiftLeft" if args.len() >= 2 => {
180 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
181 (&args[0], &args[1])
182 {
183 if *n < 64 {
184 return Some(Expr::Lit(Literal::Nat(m << n)));
185 }
186 }
187 }
188 "Nat.shiftRight" if args.len() >= 2 => {
189 if let (Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))) =
190 (&args[0], &args[1])
191 {
192 if *n < 64 {
193 return Some(Expr::Lit(Literal::Nat(m >> n)));
194 }
195 return Some(Expr::Lit(Literal::Nat(0)));
196 }
197 }
198 "String.length" if !args.is_empty() => {
199 if let Expr::Lit(Literal::Str(s)) = &args[0] {
200 return Some(Expr::Lit(Literal::Nat(s.len() as u64)));
201 }
202 }
203 "String.append" if args.len() >= 2 => {
204 if let (Expr::Lit(Literal::Str(s1)), Expr::Lit(Literal::Str(s2))) =
205 (&args[0], &args[1])
206 {
207 let mut result = s1.clone();
208 result.push_str(s2);
209 return Some(Expr::Lit(Literal::Str(result)));
210 }
211 }
212 "String.beq" if args.len() >= 2 => {
213 if let (Expr::Lit(Literal::Str(s1)), Expr::Lit(Literal::Str(s2))) =
214 (&args[0], &args[1])
215 {
216 return Some(nat_bool_result(s1 == s2));
217 }
218 }
219 _ => {}
220 }
221 }
222 None
223}
224pub(super) fn gcd(mut a: u64, mut b: u64) -> u64 {
226 while b != 0 {
227 let t = b;
228 b = a % b;
229 a = t;
230 }
231 a
232}
233pub(super) fn nat_bool_result(b: bool) -> Expr {
235 if b {
236 Expr::Const(Name::str("Bool").append_str("true"), vec![])
237 } else {
238 Expr::Const(Name::str("Bool").append_str("false"), vec![])
239 }
240}
241pub(super) fn try_reduce_proj(
245 struct_name: &Name,
246 field_idx: u32,
247 struct_whnf: &Expr,
248 env: &Environment,
249) -> Option<Expr> {
250 let ctor_fn = get_app_fn(struct_whnf);
251 let ctor_name = if let Expr::Const(name, _) = ctor_fn {
252 name
253 } else {
254 return None;
255 };
256 let ctor_val = env.get_constructor_val(ctor_name)?;
257 if &ctor_val.induct != struct_name {
258 return None;
259 }
260 let ctor_args: Vec<&Expr> = get_app_args(struct_whnf);
261 let param_count = ctor_val.num_params as usize;
262 let field_pos = param_count + field_idx as usize;
263 if field_pos < ctor_args.len() {
264 Some(ctor_args[field_pos].clone())
265 } else {
266 None
267 }
268}
269pub(super) fn try_reduce_quot(name: &Name, args: &[Expr], env: &Environment) -> Option<Expr> {
271 let qv = env.get_quotient_val(name)?;
272 match qv.kind {
273 QuotKind::Lift => {
274 if args.len() < 6 {
275 return None;
276 }
277 let quot_val = &args[5];
278 let quot_head = get_app_fn(quot_val);
279 if let Expr::Const(mk_name, _) = quot_head {
280 if let Some(mk_qv) = env.get_quotient_val(mk_name) {
281 if mk_qv.kind == QuotKind::Mk {
282 let mk_args: Vec<&Expr> = get_app_args(quot_val);
283 if mk_args.len() >= 3 {
284 let a = mk_args[2];
285 let f = &args[3];
286 return Some(Expr::App(Box::new(f.clone()), Box::new(a.clone())));
287 }
288 }
289 }
290 }
291 None
292 }
293 QuotKind::Ind => {
294 if args.len() < 6 {
295 return None;
296 }
297 let quot_val = &args[5];
298 let quot_head = get_app_fn(quot_val);
299 if let Expr::Const(mk_name, _) = quot_head {
300 if let Some(mk_qv) = env.get_quotient_val(mk_name) {
301 if mk_qv.kind == QuotKind::Mk {
302 let mk_args: Vec<&Expr> = get_app_args(quot_val);
303 if mk_args.len() >= 3 {
304 let a = mk_args[2];
305 let h = &args[4];
306 return Some(Expr::App(Box::new(h.clone()), Box::new(a.clone())));
307 }
308 }
309 }
310 }
311 None
312 }
313 _ => None,
314 }
315}
316pub fn reduce_nat_op(op: &str, args: &[Expr]) -> Option<Expr> {
318 match (op, args) {
319 ("Nat.succ", [Expr::Lit(Literal::Nat(n))]) => Some(Expr::Lit(Literal::Nat(n + 1))),
320 ("Nat.add", [Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))]) => {
321 Some(Expr::Lit(Literal::Nat(m + n)))
322 }
323 ("Nat.mul", [Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))]) => {
324 Some(Expr::Lit(Literal::Nat(m * n)))
325 }
326 ("Nat.sub", [Expr::Lit(Literal::Nat(m)), Expr::Lit(Literal::Nat(n))]) => {
327 Some(Expr::Lit(Literal::Nat(m.saturating_sub(*n))))
328 }
329 _ => None,
330 }
331}
332#[cfg(test)]
333mod tests {
334 use super::*;
335 use crate::{BinderInfo, FVarId, Level};
336 #[test]
337 fn test_beta_reduction() {
338 let mut reducer = Reducer::new();
339 let nat_ty = Expr::Sort(Level::zero());
340 let body = Expr::BVar(0);
341 let lam = Expr::Lam(
342 BinderInfo::Default,
343 Name::str("x"),
344 Box::new(nat_ty),
345 Box::new(body),
346 );
347 let arg = Expr::Lit(Literal::Nat(42));
348 let app = Expr::App(Box::new(lam), Box::new(arg.clone()));
349 let result = reducer.whnf(&app);
350 assert_eq!(result, arg);
351 }
352 #[test]
353 fn test_beta_reduction_nested() {
354 let mut reducer = Reducer::new();
355 let ty = Expr::Sort(Level::zero());
356 let inner_body = Expr::BVar(1);
357 let inner_lam = Expr::Lam(
358 BinderInfo::Default,
359 Name::str("y"),
360 Box::new(ty.clone()),
361 Box::new(inner_body),
362 );
363 let outer_lam = Expr::Lam(
364 BinderInfo::Default,
365 Name::str("x"),
366 Box::new(ty),
367 Box::new(inner_lam),
368 );
369 let arg_a = Expr::FVar(FVarId(1));
370 let arg_b = Expr::FVar(FVarId(2));
371 let app1 = Expr::App(Box::new(outer_lam), Box::new(arg_a.clone()));
372 let app2 = Expr::App(Box::new(app1), Box::new(arg_b));
373 let result = reducer.whnf(&app2);
374 assert_eq!(result, arg_a);
375 }
376 #[test]
377 fn test_zeta_reduction() {
378 let mut reducer = Reducer::new();
379 let nat_ty = Expr::Sort(Level::zero());
380 let val = Expr::Lit(Literal::Nat(42));
381 let body = Expr::BVar(0);
382 let let_expr = Expr::Let(
383 Name::str("x"),
384 Box::new(nat_ty),
385 Box::new(val.clone()),
386 Box::new(body),
387 );
388 let result = reducer.whnf(&let_expr);
389 assert_eq!(result, val);
390 }
391 #[test]
392 fn test_whnf_already_normal() {
393 let mut reducer = Reducer::new();
394 let lambda = Expr::Lam(
395 BinderInfo::Default,
396 Name::str("x"),
397 Box::new(Expr::Sort(Level::zero())),
398 Box::new(Expr::BVar(0)),
399 );
400 let result = reducer.whnf(&lambda);
401 assert_eq!(result, lambda);
402 }
403 #[test]
404 fn test_nat_arithmetic() {
405 let args = vec![Expr::Lit(Literal::Nat(3)), Expr::Lit(Literal::Nat(4))];
406 let result = reduce_nat_op("Nat.add", &args);
407 assert_eq!(result, Some(Expr::Lit(Literal::Nat(7))));
408 let args = vec![Expr::Lit(Literal::Nat(6)), Expr::Lit(Literal::Nat(7))];
409 let result = reduce_nat_op("Nat.mul", &args);
410 assert_eq!(result, Some(Expr::Lit(Literal::Nat(42))));
411 }
412 #[test]
413 fn test_cache_hit() {
414 let mut reducer = Reducer::new();
415 let expr = Expr::Lit(Literal::Nat(42));
416 let result1 = reducer.whnf(&expr);
417 let result2 = reducer.whnf(&expr);
418 assert_eq!(result1, result2);
419 assert_eq!(reducer.cache.len(), 1);
420 }
421 #[test]
422 fn test_nat_extended_ops() {
423 let head = Expr::Const(Name::str("Nat.div"), vec![]);
424 let args = vec![Expr::Lit(Literal::Nat(10)), Expr::Lit(Literal::Nat(3))];
425 assert_eq!(
426 try_reduce_nat_app(&head, &args),
427 Some(Expr::Lit(Literal::Nat(3)))
428 );
429 let head = Expr::Const(Name::str("Nat.mod"), vec![]);
430 let args = vec![Expr::Lit(Literal::Nat(10)), Expr::Lit(Literal::Nat(3))];
431 assert_eq!(
432 try_reduce_nat_app(&head, &args),
433 Some(Expr::Lit(Literal::Nat(1)))
434 );
435 let head = Expr::Const(Name::str("Nat.gcd"), vec![]);
436 let args = vec![Expr::Lit(Literal::Nat(12)), Expr::Lit(Literal::Nat(8))];
437 assert_eq!(
438 try_reduce_nat_app(&head, &args),
439 Some(Expr::Lit(Literal::Nat(4)))
440 );
441 }
442 #[test]
443 fn test_nat_bitwise_ops() {
444 let head = Expr::Const(Name::str("Nat.land"), vec![]);
445 let args = vec![
446 Expr::Lit(Literal::Nat(0b1100)),
447 Expr::Lit(Literal::Nat(0b1010)),
448 ];
449 assert_eq!(
450 try_reduce_nat_app(&head, &args),
451 Some(Expr::Lit(Literal::Nat(0b1000)))
452 );
453 let head = Expr::Const(Name::str("Nat.lor"), vec![]);
454 let args = vec![
455 Expr::Lit(Literal::Nat(0b1100)),
456 Expr::Lit(Literal::Nat(0b1010)),
457 ];
458 assert_eq!(
459 try_reduce_nat_app(&head, &args),
460 Some(Expr::Lit(Literal::Nat(0b1110)))
461 );
462 let head = Expr::Const(Name::str("Nat.xor"), vec![]);
463 let args = vec![
464 Expr::Lit(Literal::Nat(0b1100)),
465 Expr::Lit(Literal::Nat(0b1010)),
466 ];
467 assert_eq!(
468 try_reduce_nat_app(&head, &args),
469 Some(Expr::Lit(Literal::Nat(0b0110)))
470 );
471 }
472 #[test]
473 fn test_string_ops() {
474 let head = Expr::Const(Name::str("String.length"), vec![]);
475 let args = vec![Expr::Lit(Literal::Str("hello".to_string()))];
476 assert_eq!(
477 try_reduce_nat_app(&head, &args),
478 Some(Expr::Lit(Literal::Nat(5)))
479 );
480 let head = Expr::Const(Name::str("String.append"), vec![]);
481 let args = vec![
482 Expr::Lit(Literal::Str("hello".to_string())),
483 Expr::Lit(Literal::Str(" world".to_string())),
484 ];
485 assert_eq!(
486 try_reduce_nat_app(&head, &args),
487 Some(Expr::Lit(Literal::Str("hello world".to_string())))
488 );
489 }
490 #[test]
491 fn test_transparency_modes() {
492 let mut reducer = Reducer::new();
493 assert!(reducer.should_unfold_hint(ReducibilityHint::Abbrev));
494 assert!(reducer.should_unfold_hint(ReducibilityHint::Regular(1)));
495 assert!(!reducer.should_unfold_hint(ReducibilityHint::Opaque));
496 reducer.set_transparency(TransparencyMode::Reducible);
497 assert!(reducer.should_unfold_hint(ReducibilityHint::Abbrev));
498 assert!(!reducer.should_unfold_hint(ReducibilityHint::Regular(1)));
499 reducer.set_transparency(TransparencyMode::None);
500 assert!(!reducer.should_unfold_hint(ReducibilityHint::Abbrev));
501 }
502}
503#[allow(dead_code)]
506pub fn eval_nat_binop(op: &str, lhs: u64, rhs: u64) -> Option<u64> {
507 match op {
508 "Nat.add" => Some(lhs + rhs),
509 "Nat.sub" => Some(lhs.saturating_sub(rhs)),
510 "Nat.mul" => Some(lhs * rhs),
511 "Nat.div" => {
512 if rhs == 0 {
513 Some(0)
514 } else {
515 Some(lhs / rhs)
516 }
517 }
518 "Nat.mod" => {
519 if rhs == 0 {
520 Some(lhs)
521 } else {
522 Some(lhs % rhs)
523 }
524 }
525 "Nat.pow" => Some(lhs.saturating_pow(rhs as u32)),
526 "Nat.gcd" => {
527 let mut a = lhs;
528 let mut b = rhs;
529 while b != 0 {
530 let t = b;
531 b = a % b;
532 a = t;
533 }
534 Some(a)
535 }
536 "Nat.lcm" => {
537 if lhs == 0 || rhs == 0 {
538 Some(0)
539 } else {
540 let mut a = lhs;
541 let mut b = rhs;
542 while b != 0 {
543 let t = b;
544 b = a % b;
545 a = t;
546 }
547 Some(lhs / a * rhs)
548 }
549 }
550 "Nat.land" => Some(lhs & rhs),
551 "Nat.lor" => Some(lhs | rhs),
552 "Nat.xor" => Some(lhs ^ rhs),
553 "Nat.shiftLeft" => Some(lhs.checked_shl(rhs as u32).unwrap_or(0)),
554 "Nat.shiftRight" => Some(lhs.checked_shr(rhs as u32).unwrap_or(0)),
555 "Nat.min" => Some(lhs.min(rhs)),
556 "Nat.max" => Some(lhs.max(rhs)),
557 _ => None,
558 }
559}
560#[allow(dead_code)]
562pub fn eval_nat_cmp(op: &str, lhs: u64, rhs: u64) -> Option<&'static str> {
563 match op {
564 "Nat.beq" | "Nat.decEq" => Some(if lhs == rhs { "true" } else { "false" }),
565 "Nat.bne" => Some(if lhs != rhs { "true" } else { "false" }),
566 "Nat.ble" => Some(if lhs <= rhs { "true" } else { "false" }),
567 "Nat.blt" => Some(if lhs < rhs { "true" } else { "false" }),
568 _ => None,
569 }
570}
571#[allow(dead_code)]
580pub fn is_whnf(expr: &Expr) -> bool {
581 match expr {
582 Expr::Sort(_)
583 | Expr::Lit(_)
584 | Expr::FVar(_)
585 | Expr::Lam(_, _, _, _)
586 | Expr::Pi(_, _, _, _)
587 | Expr::Let(_, _, _, _) => true,
588 Expr::Const(_, _) => true,
589 Expr::BVar(_) => true,
590 Expr::App(f, _) => match f.as_ref() {
591 Expr::Lam(_, _, _, _) => false,
592 _ => is_whnf(f),
593 },
594 Expr::Proj(_, _, _) => false,
595 }
596}
597#[allow(dead_code)]
599pub fn head_of(expr: &Expr) -> &Expr {
600 match expr {
601 Expr::App(f, _) => head_of(f),
602 _ => expr,
603 }
604}
605#[allow(dead_code)]
607pub fn app_arity(expr: &Expr) -> usize {
608 match expr {
609 Expr::App(f, _) => 1 + app_arity(f),
610 _ => 0,
611 }
612}
613#[cfg(test)]
614mod extra_reduce_tests {
615 use super::*;
616 use crate::{BinderInfo, Level};
617 #[test]
618 fn test_reduction_rule_display() {
619 assert_eq!(ReductionRule::Beta.to_string(), "beta");
620 assert_eq!(ReductionRule::Delta.to_string(), "delta");
621 assert_eq!(ReductionRule::Iota.to_string(), "iota");
622 assert_eq!(ReductionRule::None.to_string(), "none");
623 }
624 #[test]
625 fn test_reduction_trace_enabled() {
626 let mut trace = ReductionTrace::enabled();
627 let before = Expr::Lit(Literal::Nat(1));
628 let after = Expr::Lit(Literal::Nat(2));
629 trace.record(ReductionRule::Beta, before.clone(), after.clone());
630 assert_eq!(trace.step_count(), 1);
631 assert_eq!(trace.count_rule(&ReductionRule::Beta), 1);
632 assert_eq!(trace.count_rule(&ReductionRule::Delta), 0);
633 }
634 #[test]
635 fn test_reduction_trace_disabled() {
636 let mut trace = ReductionTrace::disabled();
637 let e = Expr::Lit(Literal::Nat(1));
638 trace.record(ReductionRule::Beta, e.clone(), e.clone());
639 assert_eq!(trace.step_count(), 0);
640 }
641 #[test]
642 fn test_reducer_stats_total() {
643 let stats = ReducerStats {
644 whnf_calls: 10,
645 cache_hits: 3,
646 beta_count: 4,
647 delta_count: 2,
648 iota_count: 1,
649 zeta_count: 1,
650 nat_lit_count: 2,
651 };
652 assert_eq!(stats.total_reductions(), 10);
653 }
654 #[test]
655 fn test_reducer_stats_cache_hit_rate() {
656 let stats = ReducerStats {
657 whnf_calls: 10,
658 cache_hits: 5,
659 ..Default::default()
660 };
661 assert!((stats.cache_hit_rate() - 0.5).abs() < 1e-10);
662 }
663 #[test]
664 fn test_eval_nat_binop_add() {
665 assert_eq!(eval_nat_binop("Nat.add", 3, 4), Some(7));
666 }
667 #[test]
668 fn test_eval_nat_binop_sub_saturating() {
669 assert_eq!(eval_nat_binop("Nat.sub", 3, 10), Some(0));
670 }
671 #[test]
672 fn test_eval_nat_binop_div_zero() {
673 assert_eq!(eval_nat_binop("Nat.div", 7, 0), Some(0));
674 }
675 #[test]
676 fn test_eval_nat_binop_gcd() {
677 assert_eq!(eval_nat_binop("Nat.gcd", 12, 8), Some(4));
678 }
679 #[test]
680 fn test_eval_nat_binop_lcm() {
681 assert_eq!(eval_nat_binop("Nat.lcm", 4, 6), Some(12));
682 }
683 #[test]
684 fn test_eval_nat_binop_land() {
685 assert_eq!(eval_nat_binop("Nat.land", 0b1100, 0b1010), Some(0b1000));
686 }
687 #[test]
688 fn test_eval_nat_cmp_beq_true() {
689 assert_eq!(eval_nat_cmp("Nat.beq", 5, 5), Some("true"));
690 }
691 #[test]
692 fn test_eval_nat_cmp_beq_false() {
693 assert_eq!(eval_nat_cmp("Nat.beq", 5, 6), Some("false"));
694 }
695 #[test]
696 fn test_eval_nat_cmp_ble() {
697 assert_eq!(eval_nat_cmp("Nat.ble", 3, 5), Some("true"));
698 assert_eq!(eval_nat_cmp("Nat.ble", 5, 3), Some("false"));
699 }
700 #[test]
701 fn test_is_whnf_sort() {
702 assert!(is_whnf(&Expr::Sort(Level::zero())));
703 }
704 #[test]
705 fn test_is_whnf_lit() {
706 assert!(is_whnf(&Expr::Lit(Literal::Nat(42))));
707 }
708 #[test]
709 fn test_is_whnf_lam() {
710 let lam = Expr::Lam(
711 BinderInfo::Default,
712 Name::str("x"),
713 Box::new(Expr::Sort(Level::zero())),
714 Box::new(Expr::BVar(0)),
715 );
716 assert!(is_whnf(&lam));
717 }
718 #[test]
719 fn test_is_whnf_redex() {
720 let lam = Expr::Lam(
721 BinderInfo::Default,
722 Name::str("x"),
723 Box::new(Expr::Sort(Level::zero())),
724 Box::new(Expr::BVar(0)),
725 );
726 let redex = Expr::App(Box::new(lam), Box::new(Expr::Lit(Literal::Nat(1))));
727 assert!(!is_whnf(&redex));
728 }
729 #[test]
730 fn test_head_of_nested_app() {
731 let f = Expr::Const(Name::str("f"), vec![]);
732 let app1 = Expr::App(Box::new(f.clone()), Box::new(Expr::Lit(Literal::Nat(1))));
733 let app2 = Expr::App(Box::new(app1), Box::new(Expr::Lit(Literal::Nat(2))));
734 assert_eq!(*head_of(&app2), f);
735 }
736 #[test]
737 fn test_app_arity() {
738 let f = Expr::Const(Name::str("f"), vec![]);
739 let app1 = Expr::App(Box::new(f.clone()), Box::new(Expr::Lit(Literal::Nat(1))));
740 let app2 = Expr::App(Box::new(app1), Box::new(Expr::Lit(Literal::Nat(2))));
741 assert_eq!(app_arity(&app2), 2);
742 assert_eq!(app_arity(&f), 0);
743 }
744 #[test]
745 fn test_trace_clear() {
746 let mut trace = ReductionTrace::enabled();
747 let e = Expr::Lit(Literal::Nat(0));
748 trace.record(ReductionRule::Beta, e.clone(), e.clone());
749 assert_eq!(trace.step_count(), 1);
750 trace.clear();
751 assert_eq!(trace.step_count(), 0);
752 }
753 #[test]
754 fn test_reducer_stats_display() {
755 let stats = ReducerStats::default();
756 let s = stats.display();
757 assert!(s.contains("whnf:0"));
758 }
759}
760#[cfg(test)]
761mod tests_padding_infra {
762 use super::*;
763 #[test]
764 fn test_stat_summary() {
765 let mut ss = StatSummary::new();
766 ss.record(10.0);
767 ss.record(20.0);
768 ss.record(30.0);
769 assert_eq!(ss.count(), 3);
770 assert!((ss.mean().expect("mean should succeed") - 20.0).abs() < 1e-9);
771 assert_eq!(ss.min().expect("min should succeed") as i64, 10);
772 assert_eq!(ss.max().expect("max should succeed") as i64, 30);
773 }
774 #[test]
775 fn test_transform_stat() {
776 let mut ts = TransformStat::new();
777 ts.record_before(100.0);
778 ts.record_after(80.0);
779 let ratio = ts.mean_ratio().expect("ratio should be present");
780 assert!((ratio - 0.8).abs() < 1e-9);
781 }
782 #[test]
783 fn test_small_map() {
784 let mut m: SmallMap<u32, &str> = SmallMap::new();
785 m.insert(3, "three");
786 m.insert(1, "one");
787 m.insert(2, "two");
788 assert_eq!(m.get(&2), Some(&"two"));
789 assert_eq!(m.len(), 3);
790 let keys = m.keys();
791 assert_eq!(*keys[0], 1);
792 assert_eq!(*keys[2], 3);
793 }
794 #[test]
795 fn test_label_set() {
796 let mut ls = LabelSet::new();
797 ls.add("foo");
798 ls.add("bar");
799 ls.add("foo");
800 assert_eq!(ls.count(), 2);
801 assert!(ls.has("bar"));
802 assert!(!ls.has("baz"));
803 }
804 #[test]
805 fn test_config_node() {
806 let mut root = ConfigNode::section("root");
807 let child = ConfigNode::leaf("key", "value");
808 root.add_child(child);
809 assert_eq!(root.num_children(), 1);
810 }
811 #[test]
812 fn test_versioned_record() {
813 let mut vr = VersionedRecord::new(0u32);
814 vr.update(1);
815 vr.update(2);
816 assert_eq!(*vr.current(), 2);
817 assert_eq!(vr.version(), 2);
818 assert!(vr.has_history());
819 assert_eq!(*vr.at_version(0).expect("value should be present"), 0);
820 }
821 #[test]
822 fn test_simple_dag() {
823 let mut dag = SimpleDag::new(4);
824 dag.add_edge(0, 1);
825 dag.add_edge(1, 2);
826 dag.add_edge(2, 3);
827 assert!(dag.can_reach(0, 3));
828 assert!(!dag.can_reach(3, 0));
829 let order = dag.topological_sort().expect("order should be present");
830 assert_eq!(order, vec![0, 1, 2, 3]);
831 }
832 #[test]
833 fn test_focus_stack() {
834 let mut fs: FocusStack<&str> = FocusStack::new();
835 fs.focus("a");
836 fs.focus("b");
837 assert_eq!(fs.current(), Some(&"b"));
838 assert_eq!(fs.depth(), 2);
839 fs.blur();
840 assert_eq!(fs.current(), Some(&"a"));
841 }
842}
843#[cfg(test)]
844mod tests_extra_iterators {
845 use super::*;
846 #[test]
847 fn test_window_iterator() {
848 let data = vec![1u32, 2, 3, 4, 5];
849 let windows: Vec<_> = WindowIterator::new(&data, 3).collect();
850 assert_eq!(windows.len(), 3);
851 assert_eq!(windows[0], &[1, 2, 3]);
852 assert_eq!(windows[2], &[3, 4, 5]);
853 }
854 #[test]
855 fn test_non_empty_vec() {
856 let mut nev = NonEmptyVec::singleton(10u32);
857 nev.push(20);
858 nev.push(30);
859 assert_eq!(nev.len(), 3);
860 assert_eq!(*nev.first(), 10);
861 assert_eq!(*nev.last(), 30);
862 }
863}
864#[cfg(test)]
865mod tests_padding2 {
866 use super::*;
867 #[test]
868 fn test_sliding_sum() {
869 let mut ss = SlidingSum::new(3);
870 ss.push(1.0);
871 ss.push(2.0);
872 ss.push(3.0);
873 assert!((ss.sum() - 6.0).abs() < 1e-9);
874 ss.push(4.0);
875 assert!((ss.sum() - 9.0).abs() < 1e-9);
876 assert_eq!(ss.count(), 3);
877 }
878 #[test]
879 fn test_path_buf() {
880 let mut pb = PathBuf::new();
881 pb.push("src");
882 pb.push("main");
883 assert_eq!(pb.as_str(), "src/main");
884 assert_eq!(pb.depth(), 2);
885 pb.pop();
886 assert_eq!(pb.as_str(), "src");
887 }
888 #[test]
889 fn test_string_pool() {
890 let mut pool = StringPool::new();
891 let s = pool.take();
892 assert!(s.is_empty());
893 pool.give("hello".to_string());
894 let s2 = pool.take();
895 assert!(s2.is_empty());
896 assert_eq!(pool.free_count(), 0);
897 }
898 #[test]
899 fn test_transitive_closure() {
900 let mut tc = TransitiveClosure::new(4);
901 tc.add_edge(0, 1);
902 tc.add_edge(1, 2);
903 tc.add_edge(2, 3);
904 assert!(tc.can_reach(0, 3));
905 assert!(!tc.can_reach(3, 0));
906 let r = tc.reachable_from(0);
907 assert_eq!(r.len(), 4);
908 }
909 #[test]
910 fn test_token_bucket() {
911 let mut tb = TokenBucket::new(100, 10);
912 assert_eq!(tb.available(), 100);
913 assert!(tb.try_consume(50));
914 assert_eq!(tb.available(), 50);
915 assert!(!tb.try_consume(60));
916 assert_eq!(tb.capacity(), 100);
917 }
918 #[test]
919 fn test_rewrite_rule_set() {
920 let mut rrs = RewriteRuleSet::new();
921 rrs.add(RewriteRule::unconditional(
922 "beta",
923 "App(Lam(x, b), v)",
924 "b[x:=v]",
925 ));
926 rrs.add(RewriteRule::conditional("comm", "a + b", "b + a"));
927 assert_eq!(rrs.len(), 2);
928 assert_eq!(rrs.unconditional_rules().len(), 1);
929 assert_eq!(rrs.conditional_rules().len(), 1);
930 assert!(rrs.get("beta").is_some());
931 let disp = rrs
932 .get("beta")
933 .expect("element at \'beta\' should exist")
934 .display();
935 assert!(disp.contains("→"));
936 }
937}
938#[cfg(test)]
939mod tests_padding3 {
940 use super::*;
941 #[test]
942 fn test_decision_node() {
943 let tree = DecisionNode::Branch {
944 key: "x".into(),
945 val: "1".into(),
946 yes_branch: Box::new(DecisionNode::Leaf("yes".into())),
947 no_branch: Box::new(DecisionNode::Leaf("no".into())),
948 };
949 let mut ctx = std::collections::HashMap::new();
950 ctx.insert("x".into(), "1".into());
951 assert_eq!(tree.evaluate(&ctx), "yes");
952 ctx.insert("x".into(), "2".into());
953 assert_eq!(tree.evaluate(&ctx), "no");
954 assert_eq!(tree.depth(), 1);
955 }
956 #[test]
957 fn test_flat_substitution() {
958 let mut sub = FlatSubstitution::new();
959 sub.add("foo", "bar");
960 sub.add("baz", "qux");
961 assert_eq!(sub.apply("foo and baz"), "bar and qux");
962 assert_eq!(sub.len(), 2);
963 }
964 #[test]
965 fn test_stopwatch() {
966 let mut sw = Stopwatch::start();
967 sw.split();
968 sw.split();
969 assert_eq!(sw.num_splits(), 2);
970 assert!(sw.elapsed_ms() >= 0.0);
971 for &s in sw.splits() {
972 assert!(s >= 0.0);
973 }
974 }
975 #[test]
976 fn test_either2() {
977 let e: Either2<i32, &str> = Either2::First(42);
978 assert!(e.is_first());
979 let mapped = e.map_first(|x| x * 2);
980 assert_eq!(mapped.first(), Some(84));
981 let e2: Either2<i32, &str> = Either2::Second("hello");
982 assert!(e2.is_second());
983 assert_eq!(e2.second(), Some("hello"));
984 }
985 #[test]
986 fn test_write_once() {
987 let wo: WriteOnce<u32> = WriteOnce::new();
988 assert!(!wo.is_written());
989 assert!(wo.write(42));
990 assert!(!wo.write(99));
991 assert_eq!(wo.read(), Some(42));
992 }
993 #[test]
994 fn test_sparse_vec() {
995 let mut sv: SparseVec<i32> = SparseVec::new(100);
996 sv.set(5, 10);
997 sv.set(50, 20);
998 assert_eq!(*sv.get(5), 10);
999 assert_eq!(*sv.get(50), 20);
1000 assert_eq!(*sv.get(0), 0);
1001 assert_eq!(sv.nnz(), 2);
1002 sv.set(5, 0);
1003 assert_eq!(sv.nnz(), 1);
1004 }
1005 #[test]
1006 fn test_stack_calc() {
1007 let mut calc = StackCalc::new();
1008 calc.push(3);
1009 calc.push(4);
1010 calc.add();
1011 assert_eq!(calc.peek(), Some(7));
1012 calc.push(2);
1013 calc.mul();
1014 assert_eq!(calc.peek(), Some(14));
1015 }
1016}