use crate::storage::SparseSetIndex; use fixedbitset::FixedBitSet; use std::marker::PhantomData; #[derive(Debug, Eq, PartialEq, Clone)] pub struct Access { reads_all: bool, /// A combined set of T read and write accesses. reads_and_writes: FixedBitSet, writes: FixedBitSet, marker: PhantomData, } impl Default for Access { fn default() -> Self { Self { reads_all: false, reads_and_writes: Default::default(), writes: Default::default(), marker: PhantomData, } } } impl Access { pub fn grow(&mut self, bits: usize) { self.reads_and_writes.grow(bits); self.writes.grow(bits); } pub fn add_read(&mut self, index: T) { self.reads_and_writes.grow(index.sparse_set_index() + 1); self.reads_and_writes.insert(index.sparse_set_index()); } pub fn add_write(&mut self, index: T) { self.reads_and_writes.grow(index.sparse_set_index() + 1); self.writes.grow(index.sparse_set_index() + 1); self.reads_and_writes.insert(index.sparse_set_index()); self.writes.insert(index.sparse_set_index()); } pub fn has_read(&self, index: T) -> bool { if self.reads_all { true } else { self.reads_and_writes.contains(index.sparse_set_index()) } } pub fn has_write(&self, index: T) -> bool { self.writes.contains(index.sparse_set_index()) } pub fn read_all(&mut self) { self.reads_all = true; } pub fn reads_all(&self) -> bool { self.reads_all } pub fn clear(&mut self) { self.reads_all = false; self.reads_and_writes.clear(); self.writes.clear(); } pub fn extend(&mut self, other: &Access) { self.reads_all = self.reads_all || other.reads_all; self.reads_and_writes.union_with(&other.reads_and_writes); self.writes.union_with(&other.writes); } pub fn is_compatible(&self, other: &Access) -> bool { if self.reads_all { 0 == other.writes.count_ones(..) } else if other.reads_all { 0 == self.writes.count_ones(..) } else { self.writes.is_disjoint(&other.reads_and_writes) && self.reads_and_writes.is_disjoint(&other.writes) } } pub fn get_conflicts(&self, other: &Access) -> Vec { let mut conflicts = FixedBitSet::default(); if self.reads_all { conflicts.extend(other.writes.ones()); } if other.reads_all { conflicts.extend(self.writes.ones()); } conflicts.extend(self.writes.intersection(&other.reads_and_writes)); conflicts.extend(self.reads_and_writes.intersection(&other.writes)); conflicts .ones() .map(SparseSetIndex::get_sparse_set_index) .collect() } } #[derive(Clone)] pub struct FilteredAccess { access: Access, with: FixedBitSet, without: FixedBitSet, } impl Default for FilteredAccess { fn default() -> Self { Self { access: Access::default(), with: Default::default(), without: Default::default(), } } } impl FilteredAccess { #[inline] pub fn access(&self) -> &Access { &self.access } pub fn add_read(&mut self, index: T) { self.access.add_read(index.clone()); self.add_with(index); } pub fn add_write(&mut self, index: T) { self.access.add_write(index.clone()); self.add_with(index); } pub fn add_with(&mut self, index: T) { self.with.grow(index.sparse_set_index() + 1); self.with.insert(index.sparse_set_index()); } pub fn add_without(&mut self, index: T) { self.without.grow(index.sparse_set_index() + 1); self.without.insert(index.sparse_set_index()); } pub fn is_compatible(&self, other: &FilteredAccess) -> bool { if self.access.is_compatible(&other.access) { true } else { self.with.intersection(&other.without).next().is_some() || self.without.intersection(&other.with).next().is_some() } } } pub struct FilteredAccessSet { combined_access: Access, filtered_accesses: Vec>, } impl FilteredAccessSet { #[inline] pub fn combined_access(&self) -> &Access { &self.combined_access } #[inline] pub fn combined_access_mut(&mut self) -> &mut Access { &mut self.combined_access } pub fn get_conflicts(&self, filtered_access: &FilteredAccess) -> Vec { // if combined unfiltered access is incompatible, check each filtered access for // compatibility if !filtered_access.access.is_compatible(&self.combined_access) { for current_filtered_access in self.filtered_accesses.iter() { if !current_filtered_access.is_compatible(&filtered_access) { return current_filtered_access .access .get_conflicts(&filtered_access.access); } } } Vec::new() } pub fn add(&mut self, filtered_access: FilteredAccess) { self.combined_access.extend(&filtered_access.access); self.filtered_accesses.push(filtered_access); } } impl Default for FilteredAccessSet { fn default() -> Self { Self { combined_access: Default::default(), filtered_accesses: Vec::new(), } } } #[cfg(test)] mod tests { use crate::query::Access; #[test] fn access_get_conflicts() { let mut access_a = Access::::default(); access_a.add_read(0); access_a.add_read(1); let mut access_b = Access::::default(); access_b.add_read(0); access_b.add_write(1); assert_eq!(access_a.get_conflicts(&access_b), vec![1]); let mut access_c = Access::::default(); access_c.add_write(0); access_c.add_write(1); assert_eq!(access_a.get_conflicts(&access_c), vec![0, 1]); assert_eq!(access_b.get_conflicts(&access_c), vec![0, 1]); let mut access_d = Access::::default(); access_d.add_read(0); assert_eq!(access_d.get_conflicts(&access_a), vec![]); assert_eq!(access_d.get_conflicts(&access_b), vec![]); assert_eq!(access_d.get_conflicts(&access_c), vec![0]); } }