use alloc::collections::BTreeSet;
use alloc::string::String;
#[cfg(feature = "variable-lookbehinds")]
use alloc::sync::Arc;
use alloc::vec;
use alloc::vec::Vec;
use regex_automata::meta::Regex;
use regex_automata::util::look::LookMatcher;
use regex_automata::util::primitives::NonMaxUsize;
use regex_automata::Anchored;
use regex_automata::Input;
#[cfg(feature = "variable-lookbehinds")]
use regex_automata::util::pool::Pool;
#[cfg(feature = "variable-lookbehinds")]
pub(crate) type CachePoolFn = alloc::boxed::Box<
dyn Fn() -> regex_automata::hybrid::dfa::Cache
+ Send
+ Sync
+ core::panic::UnwindSafe
+ core::panic::RefUnwindSafe,
>;
use crate::error::RuntimeError;
use crate::prev_codepoint_ix;
use crate::Assertion;
use crate::Error;
use crate::Formatter;
use crate::Result;
use crate::{codepoint_len, HardRegexRuntimeOptions};
const OPTION_TRACE: u32 = 1 << 0;
pub(crate) const OPTION_SKIPPED_EMPTY_MATCH: u32 = 1 << 1;
pub(crate) const OPTION_FIND_NOT_EMPTY: u32 = 1 << 2;
const MAX_STACK: usize = 1_000_000;
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub struct CaptureGroupRange(pub usize, pub usize);
impl CaptureGroupRange {
pub fn start(&self) -> usize {
self.0
}
pub fn end(&self) -> usize {
self.1
}
pub fn to_option_if_non_empty(self) -> Option<Self> {
if self.start() == self.end() {
None
} else {
Some(self)
}
}
}
#[derive(Clone)]
pub struct Delegate {
pub inner: Regex,
pub pattern: String,
pub capture_groups: Option<CaptureGroupRange>,
}
impl core::fmt::Debug for Delegate {
fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result {
let Self {
inner: _,
pattern,
capture_groups,
} = self;
f.debug_struct("Delegate")
.field("pattern", pattern)
.field("capture_groups", capture_groups)
.finish()
}
}
#[cfg(feature = "variable-lookbehinds")]
pub struct ReverseBackwardsDelegate {
pub pattern: String,
pub(crate) dfa: Arc<regex_automata::hybrid::dfa::DFA>,
pub(crate) cache_pool: Pool<regex_automata::hybrid::dfa::Cache, CachePoolFn>,
pub(crate) capture_group_extraction_inner: Option<Regex>,
pub capture_groups: Option<CaptureGroupRange>,
}
#[cfg(feature = "variable-lookbehinds")]
impl Clone for ReverseBackwardsDelegate {
fn clone(&self) -> Self {
let dfa_for_closure = Arc::clone(&self.dfa);
let create: CachePoolFn = alloc::boxed::Box::new(move || dfa_for_closure.create_cache());
Self {
pattern: self.pattern.clone(),
cache_pool: Pool::new(create),
dfa: Arc::clone(&self.dfa),
capture_group_extraction_inner: self.capture_group_extraction_inner.clone(),
capture_groups: self.capture_groups,
}
}
}
#[cfg(feature = "variable-lookbehinds")]
impl core::fmt::Debug for ReverseBackwardsDelegate {
fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result {
let Self {
pattern,
dfa: _,
cache_pool: _,
capture_group_extraction_inner: _,
capture_groups,
} = self;
f.debug_struct("ReverseBackwardsDelegate")
.field("pattern", pattern)
.field("capture_groups", capture_groups)
.finish()
}
}
#[derive(Debug)]
pub enum Insn {
End,
Any,
AnyNoNL,
AnyNoCRLF,
Assertion(Assertion),
Lit(String), Split(usize, usize),
SplitUnanchored(usize, usize),
Jmp(usize),
Save(usize),
Save0(usize),
SaveCaptureGroupStart(usize),
Restore(usize),
RepeatGr {
lo: usize,
hi: usize,
next: usize,
repeat: usize,
},
RepeatNg {
lo: usize,
hi: usize,
next: usize,
repeat: usize,
},
RepeatEpsilonGr {
lo: usize,
next: usize,
repeat: usize,
check: usize,
},
RepeatEpsilonNg {
lo: usize,
next: usize,
repeat: usize,
check: usize,
},
FailNegativeLookAround,
GoBack(usize),
Backref {
slot: usize,
casei: bool,
},
BeginAtomic,
EndAtomic,
Delegate(Delegate),
ContinueFromPreviousMatchEnd {
at_start: bool,
},
BackrefExistsCondition(usize),
Fail,
#[cfg(feature = "variable-lookbehinds")]
BackwardsDelegate(ReverseBackwardsDelegate),
AbsentRepeater(Delegate),
}
#[derive(Debug)]
pub struct Prog {
pub body: Vec<Insn>,
n_saves: usize,
}
impl Prog {
pub(crate) fn new(body: Vec<Insn>, n_saves: usize) -> Prog {
Prog { body, n_saves }
}
#[doc(hidden)]
pub(crate) fn debug_print(&self, writer: &mut Formatter<'_>) -> core::fmt::Result {
for (i, insn) in self.body.iter().enumerate() {
writeln!(writer, "{:3}: {:?}", i, insn)?;
}
Ok(())
}
}
#[derive(Debug)]
struct Branch {
pc: usize,
ix: usize,
nsave: usize,
}
#[derive(Debug)]
struct Save {
slot: usize,
value: usize,
}
struct State {
saves: Vec<usize>,
stack: Vec<Branch>,
oldsave: Vec<Save>,
nsave: usize,
explicit_sp: usize,
max_stack: usize,
#[allow(dead_code)]
options: u32,
}
impl State {
fn new(n_saves: usize, max_stack: usize, options: u32) -> State {
State {
saves: vec![usize::MAX; n_saves],
stack: Vec::new(),
oldsave: Vec::new(),
nsave: 0,
explicit_sp: n_saves,
max_stack,
options,
}
}
fn push(&mut self, pc: usize, ix: usize) -> Result<()> {
if self.stack.len() < self.max_stack {
let nsave = self.nsave;
self.stack.push(Branch { pc, ix, nsave });
self.nsave = 0;
self.trace_stack("push");
Ok(())
} else {
Err(Error::RuntimeError(RuntimeError::StackOverflow))
}
}
fn pop(&mut self) -> (usize, usize) {
for _ in 0..self.nsave {
let Save { slot, value } = self.oldsave.pop().unwrap();
self.saves[slot] = value;
}
let Branch { pc, ix, nsave } = self.stack.pop().unwrap();
self.nsave = nsave;
self.trace_stack("pop");
(pc, ix)
}
fn save(&mut self, slot: usize, val: usize) {
for i in 0..self.nsave {
if self.oldsave[self.oldsave.len() - i - 1].slot == slot {
self.saves[slot] = val;
return;
}
}
self.oldsave.push(Save {
slot,
value: self.saves[slot],
});
self.nsave += 1;
self.saves[slot] = val;
#[cfg(feature = "std")]
if self.options & OPTION_TRACE != 0 {
println!("saves: {:?}", self.saves);
}
}
fn get(&self, slot: usize) -> usize {
self.saves[slot]
}
fn stack_push(&mut self, val: usize) {
if self.saves.len() == self.explicit_sp {
self.saves.push(self.explicit_sp + 1);
}
let explicit_sp = self.explicit_sp;
let sp = self.get(explicit_sp);
if self.saves.len() == sp {
self.saves.push(val);
} else {
self.save(sp, val);
}
self.save(explicit_sp, sp + 1);
}
fn stack_pop(&mut self) -> usize {
let explicit_sp = self.explicit_sp;
let sp = self.get(explicit_sp) - 1;
let result = self.get(sp);
self.save(explicit_sp, sp);
result
}
fn backtrack_count(&self) -> usize {
self.stack.len()
}
fn backtrack_cut(&mut self, count: usize) {
if self.stack.len() == count {
return;
}
let (oldsave_start, oldsave_end) = {
let mut end = self.oldsave.len() - self.nsave;
for &Branch { nsave, .. } in &self.stack[count + 1..] {
end -= nsave;
}
let start = end - self.stack[count].nsave;
(start, end)
};
let mut saved = BTreeSet::new();
for &Save { slot, .. } in &self.oldsave[oldsave_start..oldsave_end] {
saved.insert(slot);
}
let mut oldsave_ix = oldsave_end;
for ix in oldsave_end..self.oldsave.len() {
let Save { slot, .. } = self.oldsave[ix];
let new_slot = saved.insert(slot);
if new_slot {
self.oldsave.swap(oldsave_ix, ix);
oldsave_ix += 1;
}
}
self.stack.truncate(count);
self.oldsave.truncate(oldsave_ix);
self.nsave = oldsave_ix - oldsave_start;
}
#[inline]
#[allow(unused_variables)]
fn trace_stack(&self, operation: &str) {
#[cfg(feature = "std")]
if self.options & OPTION_TRACE != 0 {
println!("stack after {}: {:?}", operation, self.stack);
}
}
}
fn codepoint_len_at(s: &str, ix: usize) -> usize {
codepoint_len(s.as_bytes()[ix])
}
#[inline]
fn matches_literal(s: &str, ix: usize, end: usize, literal: &str) -> bool {
end <= s.len() && &s.as_bytes()[ix..end] == literal.as_bytes()
}
fn matches_literal_casei(s: &str, ix: usize, end: usize, literal: &str) -> bool {
if end > s.len() {
return false;
}
if matches_literal(s, ix, end, literal) {
return true;
}
if !s.is_char_boundary(ix) || !s.is_char_boundary(end) {
return false;
}
if s[ix..end].is_ascii() {
return s[ix..end].eq_ignore_ascii_case(literal);
}
use regex_syntax::ast::*;
let span = Span::splat(Position::new(0, 0, 0));
let literals = literal
.chars()
.map(|c| {
Ast::literal(Literal {
span,
kind: LiteralKind::Verbatim,
c,
})
})
.collect();
let ast = Ast::concat(Concat {
span,
asts: literals,
});
let mut translator = regex_syntax::hir::translate::TranslatorBuilder::new()
.case_insensitive(true)
.build();
let hir = translator.translate(literal, &ast).unwrap();
use regex_automata::meta::Builder as RaBuilder;
let re = RaBuilder::new()
.build_from_hir(&hir)
.expect("literal hir should get built successfully");
re.find(&s[ix..end]).is_some()
}
#[inline]
fn store_capture_groups(
state: &mut State,
inner_slots: &[Option<NonMaxUsize>],
range: CaptureGroupRange,
skip_earlier_captures: bool,
) {
let start_group = range.start();
let end_group = range.end();
for i in 0..(end_group - start_group) {
let slot = (start_group + i) * 2;
if let Some(start) = inner_slots[(i + 1) * 2] {
let end = inner_slots[(i + 1) * 2 + 1].unwrap();
let mut save = !skip_earlier_captures;
if skip_earlier_captures {
let existing_start = state.get(slot);
let existing_end = state.get(slot + 1);
save = (start.get() >= existing_start || existing_start == usize::MAX)
&& (end.get() >= existing_end || existing_end == usize::MAX);
}
if save {
state.save(slot, start.get());
state.save(slot + 1, end.get());
}
}
}
}
pub fn run_trace(prog: &Prog, s: &str, pos: usize) -> Result<Option<Vec<usize>>> {
run(
prog,
s,
pos,
OPTION_TRACE,
&HardRegexRuntimeOptions::default(),
)
}
pub fn run_default(prog: &Prog, s: &str, pos: usize) -> Result<Option<Vec<usize>>> {
run(prog, s, pos, 0, &HardRegexRuntimeOptions::default())
}
#[allow(clippy::cognitive_complexity)]
pub(crate) fn run(
prog: &Prog,
s: &str,
pos: usize,
option_flags: u32,
options: &HardRegexRuntimeOptions,
) -> Result<Option<Vec<usize>>> {
let mut state = State::new(prog.n_saves, MAX_STACK, option_flags);
let mut inner_slots: Vec<Option<NonMaxUsize>> = Vec::new();
let look_matcher = LookMatcher::new();
#[cfg(feature = "std")]
if option_flags & OPTION_TRACE != 0 {
println!("pos\tinstruction");
}
let mut backtrack_count = 0;
let mut pc = 0;
let mut ix = pos;
let mut match_attempt_start = pos;
loop {
'fail: loop {
#[cfg(feature = "std")]
if option_flags & OPTION_TRACE != 0 {
println!("{}\t{} {:?}", ix, pc, prog.body[pc]);
}
match prog.body[pc] {
Insn::End => {
#[cfg(feature = "std")]
if option_flags & OPTION_TRACE != 0 {
println!("saves: {:?}", state.saves);
}
if option_flags & OPTION_FIND_NOT_EMPTY != 0 && ix == match_attempt_start {
break 'fail;
}
if let Some(&slot1) = state.saves.get(1) {
if state.get(0) > slot1 {
state.save(0, slot1);
}
}
return Ok(Some(state.saves));
}
Insn::Any => {
if ix < s.len() {
ix += codepoint_len_at(s, ix);
} else {
break 'fail;
}
}
Insn::AnyNoNL => {
if ix < s.len() && s.as_bytes()[ix] != b'\n' {
ix += codepoint_len_at(s, ix);
} else {
break 'fail;
}
}
Insn::AnyNoCRLF => {
if ix < s.len() && s.as_bytes()[ix] != b'\r' && s.as_bytes()[ix] != b'\n' {
ix += codepoint_len_at(s, ix);
} else {
break 'fail;
}
}
Insn::Lit(ref val) => {
let ix_end = ix + val.len();
if !matches_literal(s, ix, ix_end, val) {
break 'fail;
}
ix = ix_end
}
Insn::Assertion(assertion) => {
if !match assertion {
Assertion::StartText => look_matcher.is_start(s.as_bytes(), ix),
Assertion::EndText => look_matcher.is_end(s.as_bytes(), ix),
Assertion::EndTextIgnoreTrailingNewlines { crlf } => {
let bytes = s.as_bytes();
if ix == bytes.len() {
true
} else if crlf {
bytes[ix..].iter().all(|&b| b == b'\n' || b == b'\r')
} else {
bytes[ix..].iter().all(|&b| b == b'\n')
}
}
Assertion::StartLine { crlf: false } => {
look_matcher.is_start_lf(s.as_bytes(), ix)
}
Assertion::StartLine { crlf: true } => {
look_matcher.is_start_crlf(s.as_bytes(), ix)
}
Assertion::EndLine { crlf: false } => {
look_matcher.is_end_lf(s.as_bytes(), ix)
}
Assertion::EndLine { crlf: true } => {
look_matcher.is_end_crlf(s.as_bytes(), ix)
}
Assertion::LeftWordBoundary => look_matcher
.is_word_start_unicode(s.as_bytes(), ix)
.unwrap(),
Assertion::RightWordBoundary => {
look_matcher.is_word_end_unicode(s.as_bytes(), ix).unwrap()
}
Assertion::LeftWordHalfBoundary => look_matcher
.is_word_start_half_unicode(s.as_bytes(), ix)
.unwrap(),
Assertion::RightWordHalfBoundary => look_matcher
.is_word_end_half_unicode(s.as_bytes(), ix)
.unwrap(),
Assertion::WordBoundary => {
look_matcher.is_word_unicode(s.as_bytes(), ix).unwrap()
}
Assertion::NotWordBoundary => look_matcher
.is_word_unicode_negate(s.as_bytes(), ix)
.unwrap(),
} {
break 'fail;
}
}
Insn::Split(x, y) => {
state.push(y, ix)?;
pc = x;
continue;
}
Insn::SplitUnanchored(x, y) => {
match_attempt_start = ix;
state.push(y, ix)?;
pc = x;
continue;
}
Insn::Jmp(target) => {
pc = target;
continue;
}
Insn::Save(slot) => state.save(slot, ix),
Insn::Save0(slot) => state.save(slot, 0),
Insn::SaveCaptureGroupStart(group) => {
let start_slot = group * 2;
if state.get(start_slot) == usize::MAX || state.get(start_slot + 1) <= ix {
state.save(start_slot, ix);
}
}
Insn::Restore(slot) => ix = state.get(slot),
Insn::RepeatGr {
lo,
hi,
next,
repeat,
} => {
let repcount = state.get(repeat);
if repcount == hi {
pc = next;
continue;
}
state.save(repeat, repcount + 1);
if repcount >= lo {
state.push(next, ix)?;
}
}
Insn::RepeatNg {
lo,
hi,
next,
repeat,
} => {
let repcount = state.get(repeat);
if repcount == hi {
pc = next;
continue;
}
state.save(repeat, repcount + 1);
if repcount >= lo {
state.push(pc + 1, ix)?;
pc = next;
continue;
}
}
Insn::RepeatEpsilonGr {
lo,
next,
repeat,
check,
} => {
let repcount = state.get(repeat);
if repcount > 0 && state.get(check) == ix {
pc = next;
continue;
}
state.save(repeat, repcount + 1);
if repcount >= lo {
state.save(check, ix);
state.push(next, ix)?;
}
}
Insn::RepeatEpsilonNg {
lo,
next,
repeat,
check,
} => {
let repcount = state.get(repeat);
if repcount > 0 && state.get(check) == ix {
pc = next;
continue;
}
state.save(repeat, repcount + 1);
if repcount >= lo {
state.save(check, ix);
state.push(pc + 1, ix)?;
pc = next;
continue;
}
}
Insn::GoBack(count) => {
for _ in 0..count {
if ix == 0 {
break 'fail;
}
ix = prev_codepoint_ix(s, ix);
}
}
Insn::FailNegativeLookAround => {
loop {
let (popped_pc, _) = state.pop();
if popped_pc == pc + 1 {
break;
}
}
break 'fail;
}
Insn::Backref { slot, casei } => {
let lo = state.get(slot);
if lo == usize::MAX {
break 'fail;
}
let hi = state.get(slot + 1);
if hi == usize::MAX {
break 'fail;
}
let ref_text = &s[lo..hi];
let ix_end = ix + ref_text.len();
if casei {
if !matches_literal_casei(s, ix, ix_end, ref_text) {
break 'fail;
}
} else if !matches_literal(s, ix, ix_end, ref_text) {
break 'fail;
}
ix = ix_end;
}
Insn::BackrefExistsCondition(group) => {
let lo = state.get(group * 2);
if lo == usize::MAX {
break 'fail;
}
}
Insn::Fail => {
break 'fail;
}
#[cfg(feature = "variable-lookbehinds")]
Insn::BackwardsDelegate(ReverseBackwardsDelegate {
ref dfa,
ref cache_pool,
pattern: _,
ref capture_group_extraction_inner,
capture_groups,
}) => {
let mut cache_guard = cache_pool.get();
let input = Input::new(s).anchored(Anchored::Yes).range(0..ix);
match dfa.try_search_rev(&mut cache_guard, &input) {
Ok(Some(match_result)) => {
let match_start = match_result.offset();
if let Some(inner) = capture_group_extraction_inner {
if let Some(range) = capture_groups {
let forward_input =
Input::new(s).span(match_start..ix).anchored(Anchored::Yes);
inner_slots.resize((range.end() - range.start() + 1) * 2, None);
if inner
.search_slots(&forward_input, &mut inner_slots)
.is_some()
{
store_capture_groups(&mut state, &inner_slots, range, true);
} else {
break 'fail;
}
} else {
ix = match_start;
}
} else {
ix = match_start;
}
}
_ => break 'fail,
}
}
Insn::BeginAtomic => {
let count = state.backtrack_count();
state.stack_push(count);
}
Insn::EndAtomic => {
let count = state.stack_pop();
state.backtrack_cut(count);
}
Insn::Delegate(Delegate {
ref inner,
pattern: _,
capture_groups,
}) => {
let input = Input::new(s).span(ix..s.len()).anchored(Anchored::Yes);
if let Some(range) = capture_groups {
inner_slots.resize((range.end() - range.start() + 1) * 2, None);
if inner.search_slots(&input, &mut inner_slots).is_some() {
store_capture_groups(&mut state, &inner_slots, range, false);
ix = inner_slots[1].unwrap().get();
} else {
break 'fail;
}
} else {
match inner.search_half(&input) {
Some(m) => ix = m.offset(),
_ => break 'fail,
}
}
}
Insn::AbsentRepeater(ref delegate) => {
let input = Input::new(s).span(ix..s.len()).anchored(Anchored::Yes);
let delegate_matches_here = delegate.inner.search_half(&input).is_some();
if delegate_matches_here {
} else if ix < s.len() {
state.push(pc + 1, ix)?;
ix += codepoint_len_at(s, ix);
continue;
} else {
}
}
Insn::ContinueFromPreviousMatchEnd { at_start } => {
if ix > pos || option_flags & OPTION_SKIPPED_EMPTY_MATCH != 0 {
if at_start && state.stack.len() == 1 {
return Ok(None);
}
break 'fail;
}
}
}
pc += 1;
}
#[cfg(feature = "std")]
if option_flags & OPTION_TRACE != 0 {
println!("fail");
}
if state.stack.is_empty() {
return Ok(None);
}
backtrack_count += 1;
if backtrack_count > options.backtrack_limit {
return Err(Error::RuntimeError(RuntimeError::BacktrackLimitExceeded));
}
let (newpc, newix) = state.pop();
pc = newpc;
ix = newix;
}
}
#[cfg(test)]
mod tests {
use super::*;
use quickcheck::{quickcheck, Arbitrary, Gen};
#[test]
fn state_push_pop() {
let mut state = State::new(1, MAX_STACK, 0);
state.push(0, 0).unwrap();
state.push(1, 1).unwrap();
assert_eq!(state.pop(), (1, 1));
assert_eq!(state.pop(), (0, 0));
assert!(state.stack.is_empty());
state.push(2, 2).unwrap();
assert_eq!(state.pop(), (2, 2));
assert!(state.stack.is_empty());
}
#[test]
fn state_save_override() {
let mut state = State::new(1, MAX_STACK, 0);
state.save(0, 10);
state.push(0, 0).unwrap();
state.save(0, 20);
assert_eq!(state.pop(), (0, 0));
assert_eq!(state.get(0), 10);
}
#[test]
fn state_save_override_twice() {
let mut state = State::new(1, MAX_STACK, 0);
state.save(0, 10);
state.push(0, 0).unwrap();
state.save(0, 20);
state.push(1, 1).unwrap();
state.save(0, 30);
assert_eq!(state.get(0), 30);
assert_eq!(state.pop(), (1, 1));
assert_eq!(state.get(0), 20);
assert_eq!(state.pop(), (0, 0));
assert_eq!(state.get(0), 10);
}
#[test]
fn state_explicit_stack() {
let mut state = State::new(1, MAX_STACK, 0);
state.stack_push(11);
state.stack_push(12);
state.push(100, 101).unwrap();
state.stack_push(13);
assert_eq!(state.stack_pop(), 13);
state.stack_push(14);
assert_eq!(state.pop(), (100, 101));
assert_eq!(state.stack_pop(), 12);
assert_eq!(state.stack_pop(), 11);
}
#[test]
fn state_backtrack_cut_simple() {
let mut state = State::new(2, MAX_STACK, 0);
state.save(0, 1);
state.save(1, 2);
let count = state.backtrack_count();
state.push(0, 0).unwrap();
state.save(0, 3);
assert_eq!(state.backtrack_count(), 1);
state.backtrack_cut(count);
assert_eq!(state.backtrack_count(), 0);
assert_eq!(state.get(0), 3);
assert_eq!(state.get(1), 2);
}
#[test]
fn state_backtrack_cut_complex() {
let mut state = State::new(2, MAX_STACK, 0);
state.save(0, 1);
state.save(1, 2);
state.push(0, 0).unwrap();
state.save(0, 3);
let count = state.backtrack_count();
state.push(1, 1).unwrap();
state.save(0, 4);
state.push(2, 2).unwrap();
state.save(1, 5);
assert_eq!(state.backtrack_count(), 3);
state.backtrack_cut(count);
assert_eq!(state.backtrack_count(), 1);
assert_eq!(state.get(0), 4);
assert_eq!(state.get(1), 5);
state.pop();
assert_eq!(state.backtrack_count(), 0);
assert_eq!(state.get(0), 1);
assert_eq!(state.get(1), 2);
}
#[derive(Clone, Debug)]
enum Operation {
Push,
Pop,
Save(usize, usize),
}
impl Arbitrary for Operation {
fn arbitrary(g: &mut Gen) -> Self {
match g.choose(&[0, 1, 2]) {
Some(0) => Operation::Push,
Some(1) => Operation::Pop,
_ => Operation::Save(
*g.choose(&[0usize, 1, 2, 3, 4]).unwrap(),
usize::arbitrary(g),
),
}
}
}
fn check_saves_for_operations(operations: Vec<Operation>) -> bool {
let slots = operations
.iter()
.map(|o| match o {
&Operation::Save(slot, _) => slot + 1,
_ => 0,
})
.max()
.unwrap_or(0);
if slots == 0 {
return true;
}
let mut stack = Vec::new();
let mut saves = vec![usize::MAX; slots];
let mut state = State::new(slots, MAX_STACK, 0);
let mut expected = Vec::new();
let mut actual = Vec::new();
for operation in operations {
match operation {
Operation::Push => {
stack.push((0, 0, saves.clone()));
state.push(0, 0).unwrap();
}
Operation::Pop => {
if let Some((_, _, previous_saves)) = stack.pop() {
saves = previous_saves;
state.pop();
}
}
Operation::Save(slot, value) => {
saves[slot] = value;
state.save(slot, value);
}
}
expected.push(saves.clone());
let mut actual_saves = vec![usize::MAX; slots];
for (i, item) in actual_saves.iter_mut().enumerate().take(slots) {
*item = state.get(i);
}
actual.push(actual_saves);
}
expected == actual
}
quickcheck! {
fn state_save_quickcheck(operations: Vec<Operation>) -> bool {
check_saves_for_operations(operations)
}
}
}