2018-11-01 00:06:06 +00:00
|
|
|
use salsa::{Database, ParallelDatabase, Snapshot};
|
2018-10-23 04:59:12 +00:00
|
|
|
use std::panic::{self, AssertUnwindSafe};
|
2019-06-01 20:15:45 +00:00
|
|
|
use std::sync::atomic::{AtomicU32, Ordering::SeqCst};
|
2018-10-23 04:59:12 +00:00
|
|
|
|
2019-01-25 15:25:17 +00:00
|
|
|
#[salsa::query_group(PanicSafelyStruct)]
|
2019-01-12 10:11:59 +00:00
|
|
|
trait PanicSafelyDatabase: salsa::Database {
|
|
|
|
#[salsa::input]
|
|
|
|
fn one(&self) -> usize;
|
2018-10-23 04:59:12 +00:00
|
|
|
|
2019-01-12 10:11:59 +00:00
|
|
|
fn panic_safely(&self) -> ();
|
2019-06-01 20:15:45 +00:00
|
|
|
|
|
|
|
fn outer(&self) -> ();
|
2018-10-23 04:59:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn panic_safely(db: &impl PanicSafelyDatabase) -> () {
|
|
|
|
assert_eq!(db.one(), 1);
|
|
|
|
}
|
|
|
|
|
2019-06-01 20:15:45 +00:00
|
|
|
static OUTER_CALLS: AtomicU32 = AtomicU32::new(0);
|
|
|
|
|
|
|
|
fn outer(db: &impl PanicSafelyDatabase) -> () {
|
|
|
|
OUTER_CALLS.fetch_add(1, SeqCst);
|
|
|
|
db.panic_safely();
|
|
|
|
}
|
|
|
|
|
2019-01-25 15:25:17 +00:00
|
|
|
#[salsa::database(PanicSafelyStruct)]
|
2018-10-23 04:59:12 +00:00
|
|
|
#[derive(Default)]
|
|
|
|
struct DatabaseStruct {
|
|
|
|
runtime: salsa::Runtime<DatabaseStruct>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl salsa::Database for DatabaseStruct {
|
2019-09-27 09:49:15 +00:00
|
|
|
fn salsa_runtime(&self) -> &salsa::Runtime<Self> {
|
2018-10-23 04:59:12 +00:00
|
|
|
&self.runtime
|
|
|
|
}
|
2019-09-27 09:49:15 +00:00
|
|
|
|
|
|
|
fn salsa_runtime_mut(&mut self) -> &mut salsa::Runtime<Self> {
|
|
|
|
&mut self.runtime
|
|
|
|
}
|
2018-10-23 04:59:12 +00:00
|
|
|
}
|
|
|
|
|
2018-10-31 19:59:00 +00:00
|
|
|
impl salsa::ParallelDatabase for DatabaseStruct {
|
2018-11-01 00:06:06 +00:00
|
|
|
fn snapshot(&self) -> Snapshot<Self> {
|
|
|
|
Snapshot::new(DatabaseStruct {
|
2018-11-01 00:05:31 +00:00
|
|
|
runtime: self.runtime.snapshot(self),
|
2018-10-31 19:59:00 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-23 04:59:12 +00:00
|
|
|
#[test]
|
|
|
|
fn should_panic_safely() {
|
2018-11-01 08:30:54 +00:00
|
|
|
let mut db = DatabaseStruct::default();
|
2019-06-01 20:15:45 +00:00
|
|
|
db.set_one(0);
|
2018-10-23 04:59:12 +00:00
|
|
|
|
|
|
|
// Invoke `db.panic_safely() without having set `db.one`. `db.one` will
|
2019-06-01 20:15:45 +00:00
|
|
|
// return 0 and we should catch the panic.
|
2018-10-31 19:59:00 +00:00
|
|
|
let result = panic::catch_unwind(AssertUnwindSafe({
|
2018-11-01 00:05:31 +00:00
|
|
|
let db = db.snapshot();
|
2018-10-31 19:59:00 +00:00
|
|
|
move || db.panic_safely()
|
|
|
|
}));
|
2018-10-23 04:59:12 +00:00
|
|
|
assert!(result.is_err());
|
|
|
|
|
|
|
|
// Set `db.one` to 1 and assert ok
|
2019-01-27 21:06:14 +00:00
|
|
|
db.set_one(1);
|
2018-10-23 04:59:12 +00:00
|
|
|
let result = panic::catch_unwind(AssertUnwindSafe(|| db.panic_safely()));
|
2019-06-01 20:15:45 +00:00
|
|
|
assert!(result.is_ok());
|
|
|
|
|
|
|
|
// Check, that memoized outer is not invalidated by a panic
|
|
|
|
{
|
|
|
|
assert_eq!(OUTER_CALLS.load(SeqCst), 0);
|
|
|
|
db.outer();
|
|
|
|
assert_eq!(OUTER_CALLS.load(SeqCst), 1);
|
|
|
|
|
|
|
|
db.set_one(0);
|
|
|
|
let result = panic::catch_unwind(AssertUnwindSafe(|| db.outer()));
|
|
|
|
assert!(result.is_err());
|
|
|
|
assert_eq!(OUTER_CALLS.load(SeqCst), 1);
|
|
|
|
|
|
|
|
db.set_one(1);
|
|
|
|
db.outer();
|
|
|
|
assert_eq!(OUTER_CALLS.load(SeqCst), 1);
|
|
|
|
}
|
2018-10-23 04:59:12 +00:00
|
|
|
}
|
2019-01-10 09:48:07 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn storages_are_unwind_safe() {
|
|
|
|
fn check_unwind_safe<T: std::panic::UnwindSafe>() {}
|
|
|
|
check_unwind_safe::<&DatabaseStruct>();
|
|
|
|
}
|
2019-01-18 10:52:02 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn panics_clear_query_stack() {
|
|
|
|
let db = DatabaseStruct::default();
|
|
|
|
|
|
|
|
// Invoke `db.panic_if_not_one() without having set `db.input`. `db.input`
|
|
|
|
// will default to 0 and we should catch the panic.
|
|
|
|
let result = panic::catch_unwind(AssertUnwindSafe(|| db.panic_safely()));
|
|
|
|
assert!(result.is_err());
|
|
|
|
|
|
|
|
// The database has been poisoned and any attempt to increment the
|
|
|
|
// revision should panic.
|
|
|
|
assert_eq!(db.salsa_runtime().active_query(), None);
|
|
|
|
}
|