summaryrefslogtreecommitdiffstats
path: root/vendor/ena/tests/external_undo_log.rs
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/ena/tests/external_undo_log.rs')
-rw-r--r--vendor/ena/tests/external_undo_log.rs202
1 files changed, 202 insertions, 0 deletions
diff --git a/vendor/ena/tests/external_undo_log.rs b/vendor/ena/tests/external_undo_log.rs
new file mode 100644
index 000000000..2537826c9
--- /dev/null
+++ b/vendor/ena/tests/external_undo_log.rs
@@ -0,0 +1,202 @@
+#[macro_use]
+extern crate log;
+extern crate ena;
+
+use ena::{
+ snapshot_vec as sv,
+ undo_log::{Rollback, Snapshots, UndoLogs},
+ unify::{self as ut, EqUnifyValue, UnifyKey},
+};
+
+#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
+struct IntKey(u32);
+
+impl UnifyKey for IntKey {
+ type Value = Option<IntKey>;
+ fn index(&self) -> u32 {
+ self.0
+ }
+ fn from_index(u: u32) -> IntKey {
+ IntKey(u)
+ }
+ fn tag() -> &'static str {
+ "IntKey"
+ }
+}
+
+impl EqUnifyValue for IntKey {}
+
+enum UndoLog {
+ EqRelation(sv::UndoLog<ut::Delegate<IntKey>>),
+ Values(sv::UndoLog<i32>),
+}
+
+impl From<sv::UndoLog<ut::Delegate<IntKey>>> for UndoLog {
+ fn from(l: sv::UndoLog<ut::Delegate<IntKey>>) -> Self {
+ UndoLog::EqRelation(l)
+ }
+}
+
+impl From<sv::UndoLog<i32>> for UndoLog {
+ fn from(l: sv::UndoLog<i32>) -> Self {
+ UndoLog::Values(l)
+ }
+}
+
+impl Rollback<UndoLog> for TypeVariableStorage {
+ fn reverse(&mut self, undo: UndoLog) {
+ match undo {
+ UndoLog::EqRelation(undo) => self.eq_relations.reverse(undo),
+ UndoLog::Values(undo) => self.values.reverse(undo),
+ }
+ }
+}
+
+#[derive(Default)]
+struct TypeVariableStorage {
+ values: sv::SnapshotVecStorage<i32>,
+
+ eq_relations: ut::UnificationTableStorage<IntKey>,
+}
+
+impl TypeVariableStorage {
+ fn with_log<'a>(&'a mut self, undo_log: &'a mut TypeVariableUndoLogs) -> TypeVariableTable<'a> {
+ TypeVariableTable {
+ storage: self,
+ undo_log,
+ }
+ }
+
+ fn len(&mut self) -> usize {
+ assert_eq!(self.values.len(), self.eq_relations.len());
+ self.values.len()
+ }
+}
+
+struct TypeVariableTable<'a> {
+ storage: &'a mut TypeVariableStorage,
+
+ undo_log: &'a mut TypeVariableUndoLogs,
+}
+
+impl TypeVariableTable<'_> {
+ fn new(&mut self, i: i32) -> IntKey {
+ self.storage.values.with_log(&mut self.undo_log).push(i);
+ self.storage
+ .eq_relations
+ .with_log(&mut self.undo_log)
+ .new_key(None)
+ }
+}
+
+struct Snapshot {
+ undo_len: usize,
+}
+
+struct TypeVariableUndoLogs {
+ logs: Vec<UndoLog>,
+ num_open_snapshots: usize,
+}
+
+impl Default for TypeVariableUndoLogs {
+ fn default() -> Self {
+ Self {
+ logs: Default::default(),
+ num_open_snapshots: Default::default(),
+ }
+ }
+}
+
+impl<T> UndoLogs<T> for TypeVariableUndoLogs
+where
+ UndoLog: From<T>,
+{
+ fn num_open_snapshots(&self) -> usize {
+ self.num_open_snapshots
+ }
+ fn push(&mut self, undo: T) {
+ if self.in_snapshot() {
+ self.logs.push(undo.into())
+ }
+ }
+ fn clear(&mut self) {
+ self.logs.clear();
+ self.num_open_snapshots = 0;
+ }
+ fn extend<J>(&mut self, undos: J)
+ where
+ Self: Sized,
+ J: IntoIterator<Item = T>,
+ {
+ if self.in_snapshot() {
+ self.logs.extend(undos.into_iter().map(UndoLog::from))
+ }
+ }
+}
+
+impl Snapshots<UndoLog> for TypeVariableUndoLogs {
+ type Snapshot = Snapshot;
+ fn actions_since_snapshot(&self, snapshot: &Self::Snapshot) -> &[UndoLog] {
+ &self.logs[snapshot.undo_len..]
+ }
+
+ fn start_snapshot(&mut self) -> Self::Snapshot {
+ self.num_open_snapshots += 1;
+ Snapshot {
+ undo_len: self.logs.len(),
+ }
+ }
+
+ fn rollback_to<R>(&mut self, values: impl FnOnce() -> R, snapshot: Self::Snapshot)
+ where
+ R: Rollback<UndoLog>,
+ {
+ debug!("rollback_to({})", snapshot.undo_len);
+
+ if self.logs.len() > snapshot.undo_len {
+ let mut values = values();
+ while self.logs.len() > snapshot.undo_len {
+ values.reverse(self.logs.pop().unwrap());
+ }
+ }
+
+ if self.num_open_snapshots == 1 {
+ // The root snapshot. It's safe to clear the undo log because
+ // there's no snapshot further out that we might need to roll back
+ // to.
+ assert!(snapshot.undo_len == 0);
+ self.logs.clear();
+ }
+
+ self.num_open_snapshots -= 1;
+ }
+
+ fn commit(&mut self, snapshot: Self::Snapshot) {
+ debug!("commit({})", snapshot.undo_len);
+
+ if self.num_open_snapshots == 1 {
+ // The root snapshot. It's safe to clear the undo log because
+ // there's no snapshot further out that we might need to roll back
+ // to.
+ assert!(snapshot.undo_len == 0);
+ self.logs.clear();
+ }
+
+ self.num_open_snapshots -= 1;
+ }
+}
+
+/// Tests that a undo log stored externally can be used with TypeVariableTable
+#[test]
+fn external_undo_log() {
+ let mut storage = TypeVariableStorage::default();
+ let mut undo_log = TypeVariableUndoLogs::default();
+
+ let snapshot = undo_log.start_snapshot();
+ storage.with_log(&mut undo_log).new(1);
+ storage.with_log(&mut undo_log).new(2);
+ assert_eq!(storage.len(), 2);
+
+ undo_log.rollback_to(|| &mut storage, snapshot);
+ assert_eq!(storage.len(), 0);
+}