use std::marker::PhantomData;
use std::ops::{Bound, Range, RangeBounds};
use std::rc::Rc;
use std::{fmt, iter, slice};
use Chunk::*;
#[cfg(feature = "nightly")]
use rustc_macros::{Decodable_NoContext, Encodable_NoContext};
use crate::{Idx, IndexVec};
#[cfg(test)]
mod tests;
type Word = u64;
const WORD_BYTES: usize = size_of::<Word>();
const WORD_BITS: usize = WORD_BYTES * 8;
const CHUNK_WORDS: usize = 32;
const CHUNK_BITS: usize = CHUNK_WORDS * WORD_BITS;
type ChunkSize = u16;
const _: () = assert!(CHUNK_BITS <= ChunkSize::MAX as usize);
pub trait BitRelations<Rhs> {
fn union(&mut self, other: &Rhs) -> bool;
fn subtract(&mut self, other: &Rhs) -> bool;
fn intersect(&mut self, other: &Rhs) -> bool;
}
#[inline]
fn inclusive_start_end<T: Idx>(
range: impl RangeBounds<T>,
domain: usize,
) -> Option<(usize, usize)> {
let start = match range.start_bound().cloned() {
Bound::Included(start) => start.index(),
Bound::Excluded(start) => start.index() + 1,
Bound::Unbounded => 0,
};
let end = match range.end_bound().cloned() {
Bound::Included(end) => end.index(),
Bound::Excluded(end) => end.index().checked_sub(1)?,
Bound::Unbounded => domain - 1,
};
assert!(end < domain);
if start > end {
return None;
}
Some((start, end))
}
macro_rules! bit_relations_inherent_impls {
() => {
pub fn union<Rhs>(&mut self, other: &Rhs) -> bool
where
Self: BitRelations<Rhs>,
{
<Self as BitRelations<Rhs>>::union(self, other)
}
pub fn subtract<Rhs>(&mut self, other: &Rhs) -> bool
where
Self: BitRelations<Rhs>,
{
<Self as BitRelations<Rhs>>::subtract(self, other)
}
pub fn intersect<Rhs>(&mut self, other: &Rhs) -> bool
where
Self: BitRelations<Rhs>,
{
<Self as BitRelations<Rhs>>::intersect(self, other)
}
};
}
#[cfg_attr(feature = "nightly", derive(Decodable_NoContext, Encodable_NoContext))]
#[derive(Eq, PartialEq, Hash)]
pub struct DenseBitSet<T> {
domain_size: usize,
words: Vec<Word>,
marker: PhantomData<T>,
}
impl<T> DenseBitSet<T> {
pub fn domain_size(&self) -> usize {
self.domain_size
}
}
impl<T: Idx> DenseBitSet<T> {
#[inline]
pub fn new_empty(domain_size: usize) -> DenseBitSet<T> {
let num_words = num_words(domain_size);
DenseBitSet { domain_size, words: vec![0; num_words], marker: PhantomData }
}
#[inline]
pub fn new_filled(domain_size: usize) -> DenseBitSet<T> {
let num_words = num_words(domain_size);
let mut result =
DenseBitSet { domain_size, words: vec![!0; num_words], marker: PhantomData };
result.clear_excess_bits();
result
}
#[inline]
pub fn clear(&mut self) {
self.words.fill(0);
}
fn clear_excess_bits(&mut self) {
clear_excess_bits_in_final_word(self.domain_size, &mut self.words);
}
pub fn count(&self) -> usize {
count_ones(&self.words)
}
#[inline]
pub fn contains(&self, elem: T) -> bool {
assert!(elem.index() < self.domain_size);
let (word_index, mask) = word_index_and_mask(elem);
(self.words[word_index] & mask) != 0
}
#[inline]
pub fn superset(&self, other: &DenseBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
self.words.iter().zip(&other.words).all(|(a, b)| (a & b) == *b)
}
#[inline]
pub fn is_empty(&self) -> bool {
self.words.iter().all(|a| *a == 0)
}
#[inline]
pub fn insert(&mut self, elem: T) -> bool {
assert!(
elem.index() < self.domain_size,
"inserting element at index {} but domain size is {}",
elem.index(),
self.domain_size,
);
let (word_index, mask) = word_index_and_mask(elem);
let word_ref = &mut self.words[word_index];
let word = *word_ref;
let new_word = word | mask;
*word_ref = new_word;
new_word != word
}
#[inline]
pub fn insert_range(&mut self, elems: impl RangeBounds<T>) {
let Some((start, end)) = inclusive_start_end(elems, self.domain_size) else {
return;
};
let (start_word_index, start_mask) = word_index_and_mask(start);
let (end_word_index, end_mask) = word_index_and_mask(end);
for word_index in (start_word_index + 1)..end_word_index {
self.words[word_index] = !0;
}
if start_word_index != end_word_index {
self.words[start_word_index] |= !(start_mask - 1);
self.words[end_word_index] |= end_mask | (end_mask - 1);
} else {
self.words[start_word_index] |= end_mask | (end_mask - start_mask);
}
}
pub fn insert_all(&mut self) {
self.words.fill(!0);
self.clear_excess_bits();
}
#[inline]
pub fn contains_any(&self, elems: impl RangeBounds<T>) -> bool {
let Some((start, end)) = inclusive_start_end(elems, self.domain_size) else {
return false;
};
let (start_word_index, start_mask) = word_index_and_mask(start);
let (end_word_index, end_mask) = word_index_and_mask(end);
if start_word_index == end_word_index {
self.words[start_word_index] & (end_mask | (end_mask - start_mask)) != 0
} else {
if self.words[start_word_index] & !(start_mask - 1) != 0 {
return true;
}
let remaining = start_word_index + 1..end_word_index;
if remaining.start <= remaining.end {
self.words[remaining].iter().any(|&w| w != 0)
|| self.words[end_word_index] & (end_mask | (end_mask - 1)) != 0
} else {
false
}
}
}
#[inline]
pub fn remove(&mut self, elem: T) -> bool {
assert!(elem.index() < self.domain_size);
let (word_index, mask) = word_index_and_mask(elem);
let word_ref = &mut self.words[word_index];
let word = *word_ref;
let new_word = word & !mask;
*word_ref = new_word;
new_word != word
}
#[inline]
pub fn iter(&self) -> BitIter<'_, T> {
BitIter::new(&self.words)
}
pub fn last_set_in(&self, range: impl RangeBounds<T>) -> Option<T> {
let (start, end) = inclusive_start_end(range, self.domain_size)?;
let (start_word_index, _) = word_index_and_mask(start);
let (end_word_index, end_mask) = word_index_and_mask(end);
let end_word = self.words[end_word_index] & (end_mask | (end_mask - 1));
if end_word != 0 {
let pos = max_bit(end_word) + WORD_BITS * end_word_index;
if start <= pos {
return Some(T::new(pos));
}
}
if let Some(offset) =
self.words[start_word_index..end_word_index].iter().rposition(|&w| w != 0)
{
let word_idx = start_word_index + offset;
let start_word = self.words[word_idx];
let pos = max_bit(start_word) + WORD_BITS * word_idx;
if start <= pos {
return Some(T::new(pos));
}
}
None
}
bit_relations_inherent_impls! {}
pub fn union_not(&mut self, other: &DenseBitSet<T>) {
assert_eq!(self.domain_size, other.domain_size);
bitwise(&mut self.words, &other.words, |a, b| a | !b);
self.clear_excess_bits();
}
}
impl<T: Idx> BitRelations<DenseBitSet<T>> for DenseBitSet<T> {
fn union(&mut self, other: &DenseBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
bitwise(&mut self.words, &other.words, |a, b| a | b)
}
fn subtract(&mut self, other: &DenseBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
bitwise(&mut self.words, &other.words, |a, b| a & !b)
}
fn intersect(&mut self, other: &DenseBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
bitwise(&mut self.words, &other.words, |a, b| a & b)
}
}
impl<T: Idx> From<GrowableBitSet<T>> for DenseBitSet<T> {
fn from(bit_set: GrowableBitSet<T>) -> Self {
bit_set.bit_set
}
}
impl<T> Clone for DenseBitSet<T> {
fn clone(&self) -> Self {
DenseBitSet {
domain_size: self.domain_size,
words: self.words.clone(),
marker: PhantomData,
}
}
fn clone_from(&mut self, from: &Self) {
self.domain_size = from.domain_size;
self.words.clone_from(&from.words);
}
}
impl<T: Idx> fmt::Debug for DenseBitSet<T> {
fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
w.debug_list().entries(self.iter()).finish()
}
}
impl<T: Idx> ToString for DenseBitSet<T> {
fn to_string(&self) -> String {
let mut result = String::new();
let mut sep = '[';
let mut i = 0;
for word in &self.words {
let mut word = *word;
for _ in 0..WORD_BYTES {
let remain = self.domain_size - i;
let mask = if remain <= 8 { (1 << remain) - 1 } else { 0xFF };
assert!(mask <= 0xFF);
let byte = word & mask;
result.push_str(&format!("{sep}{byte:02x}"));
if remain <= 8 {
break;
}
word >>= 8;
i += 8;
sep = '-';
}
sep = '|';
}
result.push(']');
result
}
}
pub struct BitIter<'a, T: Idx> {
word: Word,
offset: usize,
iter: slice::Iter<'a, Word>,
marker: PhantomData<T>,
}
impl<'a, T: Idx> BitIter<'a, T> {
#[inline]
fn new(words: &'a [Word]) -> BitIter<'a, T> {
BitIter {
word: 0,
offset: usize::MAX - (WORD_BITS - 1),
iter: words.iter(),
marker: PhantomData,
}
}
}
impl<'a, T: Idx> Iterator for BitIter<'a, T> {
type Item = T;
fn next(&mut self) -> Option<T> {
loop {
if self.word != 0 {
let bit_pos = self.word.trailing_zeros() as usize;
self.word ^= 1 << bit_pos;
return Some(T::new(bit_pos + self.offset));
}
self.word = *self.iter.next()?;
self.offset = self.offset.wrapping_add(WORD_BITS);
}
}
}
#[derive(PartialEq, Eq)]
pub struct ChunkedBitSet<T> {
domain_size: usize,
chunks: Box<[Chunk]>,
marker: PhantomData<T>,
}
#[derive(Clone, Debug, PartialEq, Eq)]
enum Chunk {
Zeros { chunk_domain_size: ChunkSize },
Ones { chunk_domain_size: ChunkSize },
Mixed {
chunk_domain_size: ChunkSize,
ones_count: ChunkSize,
words: Rc<[Word; CHUNK_WORDS]>,
},
}
#[cfg(target_pointer_width = "64")]
crate::static_assert_size!(Chunk, 16);
impl<T> ChunkedBitSet<T> {
pub fn domain_size(&self) -> usize {
self.domain_size
}
#[cfg(test)]
fn assert_valid(&self) {
if self.domain_size == 0 {
assert!(self.chunks.is_empty());
return;
}
assert!((self.chunks.len() - 1) * CHUNK_BITS <= self.domain_size);
assert!(self.chunks.len() * CHUNK_BITS >= self.domain_size);
for chunk in self.chunks.iter() {
chunk.assert_valid();
}
}
}
impl<T: Idx> ChunkedBitSet<T> {
fn new(domain_size: usize, is_empty: bool) -> Self {
let chunks = if domain_size == 0 {
Box::new([])
} else {
let num_chunks = domain_size.index().div_ceil(CHUNK_BITS);
let mut last_chunk_domain_size = domain_size % CHUNK_BITS;
if last_chunk_domain_size == 0 {
last_chunk_domain_size = CHUNK_BITS;
};
let (normal_chunk, final_chunk) = if is_empty {
(
Zeros { chunk_domain_size: CHUNK_BITS as ChunkSize },
Zeros { chunk_domain_size: last_chunk_domain_size as ChunkSize },
)
} else {
(
Ones { chunk_domain_size: CHUNK_BITS as ChunkSize },
Ones { chunk_domain_size: last_chunk_domain_size as ChunkSize },
)
};
let mut chunks = vec![normal_chunk; num_chunks].into_boxed_slice();
*chunks.as_mut().last_mut().unwrap() = final_chunk;
chunks
};
ChunkedBitSet { domain_size, chunks, marker: PhantomData }
}
#[inline]
pub fn new_empty(domain_size: usize) -> Self {
ChunkedBitSet::new(domain_size, true)
}
#[inline]
pub fn new_filled(domain_size: usize) -> Self {
ChunkedBitSet::new(domain_size, false)
}
pub fn clear(&mut self) {
*self = ChunkedBitSet::new_empty(self.domain_size);
}
#[cfg(test)]
fn chunks(&self) -> &[Chunk] {
&self.chunks
}
pub fn count(&self) -> usize {
self.chunks.iter().map(|chunk| chunk.count()).sum()
}
pub fn is_empty(&self) -> bool {
self.chunks.iter().all(|chunk| matches!(chunk, Zeros { .. }))
}
#[inline]
pub fn contains(&self, elem: T) -> bool {
assert!(elem.index() < self.domain_size);
let chunk = &self.chunks[chunk_index(elem)];
match &chunk {
Zeros { .. } => false,
Ones { .. } => true,
Mixed { words, .. } => {
let (word_index, mask) = chunk_word_index_and_mask(elem);
(words[word_index] & mask) != 0
}
}
}
#[inline]
pub fn iter(&self) -> ChunkedBitIter<'_, T> {
ChunkedBitIter::new(self)
}
pub fn insert(&mut self, elem: T) -> bool {
assert!(elem.index() < self.domain_size);
let chunk_index = chunk_index(elem);
let chunk = &mut self.chunks[chunk_index];
match *chunk {
Zeros { chunk_domain_size } => {
if chunk_domain_size > 1 {
let mut words = {
let words = Rc::<[Word; CHUNK_WORDS]>::new_zeroed();
unsafe { words.assume_init() }
};
let words_ref = Rc::get_mut(&mut words).unwrap();
let (word_index, mask) = chunk_word_index_and_mask(elem);
words_ref[word_index] |= mask;
*chunk = Mixed { chunk_domain_size, ones_count: 1, words };
} else {
*chunk = Ones { chunk_domain_size };
}
true
}
Ones { .. } => false,
Mixed { chunk_domain_size, ref mut ones_count, ref mut words } => {
let (word_index, mask) = chunk_word_index_and_mask(elem);
if (words[word_index] & mask) == 0 {
*ones_count += 1;
if *ones_count < chunk_domain_size {
let words = Rc::make_mut(words);
words[word_index] |= mask;
} else {
*chunk = Ones { chunk_domain_size };
}
true
} else {
false
}
}
}
}
pub fn insert_all(&mut self) {
*self = ChunkedBitSet::new_filled(self.domain_size);
}
pub fn remove(&mut self, elem: T) -> bool {
assert!(elem.index() < self.domain_size);
let chunk_index = chunk_index(elem);
let chunk = &mut self.chunks[chunk_index];
match *chunk {
Zeros { .. } => false,
Ones { chunk_domain_size } => {
if chunk_domain_size > 1 {
let mut words = {
let words = Rc::<[Word; CHUNK_WORDS]>::new_zeroed();
unsafe { words.assume_init() }
};
let words_ref = Rc::get_mut(&mut words).unwrap();
let num_words = num_words(chunk_domain_size as usize);
words_ref[..num_words].fill(!0);
clear_excess_bits_in_final_word(
chunk_domain_size as usize,
&mut words_ref[..num_words],
);
let (word_index, mask) = chunk_word_index_and_mask(elem);
words_ref[word_index] &= !mask;
*chunk = Mixed { chunk_domain_size, ones_count: chunk_domain_size - 1, words };
} else {
*chunk = Zeros { chunk_domain_size };
}
true
}
Mixed { chunk_domain_size, ref mut ones_count, ref mut words } => {
let (word_index, mask) = chunk_word_index_and_mask(elem);
if (words[word_index] & mask) != 0 {
*ones_count -= 1;
if *ones_count > 0 {
let words = Rc::make_mut(words);
words[word_index] &= !mask;
} else {
*chunk = Zeros { chunk_domain_size }
}
true
} else {
false
}
}
}
}
fn chunk_iter(&self, chunk_index: usize) -> ChunkIter<'_> {
match self.chunks.get(chunk_index) {
Some(Zeros { .. }) => ChunkIter::Zeros,
Some(Ones { chunk_domain_size }) => ChunkIter::Ones(0..*chunk_domain_size as usize),
Some(Mixed { chunk_domain_size, words, .. }) => {
let num_words = num_words(*chunk_domain_size as usize);
ChunkIter::Mixed(BitIter::new(&words[0..num_words]))
}
None => ChunkIter::Finished,
}
}
bit_relations_inherent_impls! {}
}
impl<T: Idx> BitRelations<ChunkedBitSet<T>> for ChunkedBitSet<T> {
fn union(&mut self, other: &ChunkedBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
let mut changed = false;
for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
match (&mut self_chunk, &other_chunk) {
(_, Zeros { .. }) | (Ones { .. }, _) => {}
(Zeros { .. }, _) | (Mixed { .. }, Ones { .. }) => {
*self_chunk = other_chunk.clone();
changed = true;
}
(
Mixed {
chunk_domain_size,
ones_count: self_chunk_ones_count,
words: self_chunk_words,
},
Mixed { words: other_chunk_words, .. },
) => {
let num_words = num_words(*chunk_domain_size as usize);
if self_chunk_words[0..num_words] == other_chunk_words[0..num_words] {
continue;
}
let op = |a, b| a | b;
if !bitwise_changes(
&self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
) {
continue;
}
let self_chunk_words = Rc::make_mut(self_chunk_words);
let has_changed = bitwise(
&mut self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
);
debug_assert!(has_changed);
*self_chunk_ones_count =
count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
if *self_chunk_ones_count == *chunk_domain_size {
*self_chunk = Ones { chunk_domain_size: *chunk_domain_size };
}
changed = true;
}
}
}
changed
}
fn subtract(&mut self, other: &ChunkedBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
let mut changed = false;
for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
match (&mut self_chunk, &other_chunk) {
(Zeros { .. }, _) | (_, Zeros { .. }) => {}
(Ones { chunk_domain_size } | Mixed { chunk_domain_size, .. }, Ones { .. }) => {
changed = true;
*self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
}
(
Ones { chunk_domain_size },
Mixed { ones_count: other_chunk_ones_count, words: other_chunk_words, .. },
) => {
changed = true;
let num_words = num_words(*chunk_domain_size as usize);
debug_assert!(num_words > 0 && num_words <= CHUNK_WORDS);
let mut self_chunk_words = **other_chunk_words;
for word in self_chunk_words[0..num_words].iter_mut() {
*word = !*word;
}
clear_excess_bits_in_final_word(
*chunk_domain_size as usize,
&mut self_chunk_words[..num_words],
);
let self_chunk_ones_count = *chunk_domain_size - *other_chunk_ones_count;
debug_assert_eq!(
self_chunk_ones_count,
count_ones(&self_chunk_words[0..num_words]) as ChunkSize
);
*self_chunk = Mixed {
chunk_domain_size: *chunk_domain_size,
ones_count: self_chunk_ones_count,
words: Rc::new(self_chunk_words),
};
}
(
Mixed {
chunk_domain_size,
ones_count: self_chunk_ones_count,
words: self_chunk_words,
},
Mixed { words: other_chunk_words, .. },
) => {
let num_words = num_words(*chunk_domain_size as usize);
let op = |a: Word, b: Word| a & !b;
if !bitwise_changes(
&self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
) {
continue;
}
let self_chunk_words = Rc::make_mut(self_chunk_words);
let has_changed = bitwise(
&mut self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
);
debug_assert!(has_changed);
*self_chunk_ones_count =
count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
if *self_chunk_ones_count == 0 {
*self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
}
changed = true;
}
}
}
changed
}
fn intersect(&mut self, other: &ChunkedBitSet<T>) -> bool {
assert_eq!(self.domain_size, other.domain_size);
let mut changed = false;
for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
match (&mut self_chunk, &other_chunk) {
(Zeros { .. }, _) | (_, Ones { .. }) => {}
(Ones { .. }, Zeros { .. } | Mixed { .. }) | (Mixed { .. }, Zeros { .. }) => {
changed = true;
*self_chunk = other_chunk.clone();
}
(
Mixed {
chunk_domain_size,
ones_count: self_chunk_ones_count,
words: self_chunk_words,
},
Mixed { words: other_chunk_words, .. },
) => {
let num_words = num_words(*chunk_domain_size as usize);
let op = |a, b| a & b;
if !bitwise_changes(
&self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
) {
continue;
}
let self_chunk_words = Rc::make_mut(self_chunk_words);
let has_changed = bitwise(
&mut self_chunk_words[0..num_words],
&other_chunk_words[0..num_words],
op,
);
debug_assert!(has_changed);
*self_chunk_ones_count =
count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
if *self_chunk_ones_count == 0 {
*self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
}
changed = true;
}
}
}
changed
}
}
impl<T> Clone for ChunkedBitSet<T> {
fn clone(&self) -> Self {
ChunkedBitSet {
domain_size: self.domain_size,
chunks: self.chunks.clone(),
marker: PhantomData,
}
}
fn clone_from(&mut self, from: &Self) {
assert_eq!(self.domain_size, from.domain_size);
debug_assert_eq!(self.chunks.len(), from.chunks.len());
self.chunks.clone_from(&from.chunks)
}
}
pub struct ChunkedBitIter<'a, T: Idx> {
bit_set: &'a ChunkedBitSet<T>,
chunk_index: usize,
chunk_iter: ChunkIter<'a>,
}
impl<'a, T: Idx> ChunkedBitIter<'a, T> {
#[inline]
fn new(bit_set: &'a ChunkedBitSet<T>) -> ChunkedBitIter<'a, T> {
ChunkedBitIter { bit_set, chunk_index: 0, chunk_iter: bit_set.chunk_iter(0) }
}
}
impl<'a, T: Idx> Iterator for ChunkedBitIter<'a, T> {
type Item = T;
fn next(&mut self) -> Option<T> {
loop {
match &mut self.chunk_iter {
ChunkIter::Zeros => {}
ChunkIter::Ones(iter) => {
if let Some(next) = iter.next() {
return Some(T::new(next + self.chunk_index * CHUNK_BITS));
}
}
ChunkIter::Mixed(iter) => {
if let Some(next) = iter.next() {
return Some(T::new(next + self.chunk_index * CHUNK_BITS));
}
}
ChunkIter::Finished => return None,
}
self.chunk_index += 1;
self.chunk_iter = self.bit_set.chunk_iter(self.chunk_index);
}
}
}
impl Chunk {
#[cfg(test)]
fn assert_valid(&self) {
match *self {
Zeros { chunk_domain_size } | Ones { chunk_domain_size } => {
assert!(chunk_domain_size as usize <= CHUNK_BITS);
}
Mixed { chunk_domain_size, ones_count, ref words } => {
assert!(chunk_domain_size as usize <= CHUNK_BITS);
assert!(0 < ones_count && ones_count < chunk_domain_size);
assert_eq!(count_ones(words.as_slice()) as ChunkSize, ones_count);
let num_words = num_words(chunk_domain_size as usize);
if num_words < CHUNK_WORDS {
assert_eq!(count_ones(&words[num_words..]) as ChunkSize, 0);
}
}
}
}
fn count(&self) -> usize {
match *self {
Zeros { .. } => 0,
Ones { chunk_domain_size } => chunk_domain_size as usize,
Mixed { ones_count, .. } => usize::from(ones_count),
}
}
}
enum ChunkIter<'a> {
Zeros,
Ones(Range<usize>),
Mixed(BitIter<'a, usize>),
Finished,
}
impl<T: Idx> fmt::Debug for ChunkedBitSet<T> {
fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
w.debug_list().entries(self.iter()).finish()
}
}
#[inline]
fn bitwise<Op>(out_vec: &mut [Word], in_vec: &[Word], op: Op) -> bool
where
Op: Fn(Word, Word) -> Word,
{
assert_eq!(out_vec.len(), in_vec.len());
let mut changed = 0;
for (out_elem, in_elem) in iter::zip(out_vec, in_vec) {
let old_val = *out_elem;
let new_val = op(old_val, *in_elem);
*out_elem = new_val;
changed |= old_val ^ new_val;
}
changed != 0
}
#[inline]
fn bitwise_changes<Op>(out_vec: &[Word], in_vec: &[Word], op: Op) -> bool
where
Op: Fn(Word, Word) -> Word,
{
assert_eq!(out_vec.len(), in_vec.len());
for (out_elem, in_elem) in iter::zip(out_vec, in_vec) {
let old_val = *out_elem;
let new_val = op(old_val, *in_elem);
if old_val != new_val {
return true;
}
}
false
}
#[derive(PartialEq, Eq)]
pub enum MixedBitSet<T> {
Small(DenseBitSet<T>),
Large(ChunkedBitSet<T>),
}
impl<T> MixedBitSet<T> {
pub fn domain_size(&self) -> usize {
match self {
MixedBitSet::Small(set) => set.domain_size(),
MixedBitSet::Large(set) => set.domain_size(),
}
}
}
impl<T: Idx> MixedBitSet<T> {
#[inline]
pub fn new_empty(domain_size: usize) -> MixedBitSet<T> {
if domain_size <= CHUNK_BITS {
MixedBitSet::Small(DenseBitSet::new_empty(domain_size))
} else {
MixedBitSet::Large(ChunkedBitSet::new_empty(domain_size))
}
}
#[inline]
pub fn is_empty(&self) -> bool {
match self {
MixedBitSet::Small(set) => set.is_empty(),
MixedBitSet::Large(set) => set.is_empty(),
}
}
#[inline]
pub fn contains(&self, elem: T) -> bool {
match self {
MixedBitSet::Small(set) => set.contains(elem),
MixedBitSet::Large(set) => set.contains(elem),
}
}
#[inline]
pub fn insert(&mut self, elem: T) -> bool {
match self {
MixedBitSet::Small(set) => set.insert(elem),
MixedBitSet::Large(set) => set.insert(elem),
}
}
pub fn insert_all(&mut self) {
match self {
MixedBitSet::Small(set) => set.insert_all(),
MixedBitSet::Large(set) => set.insert_all(),
}
}
#[inline]
pub fn remove(&mut self, elem: T) -> bool {
match self {
MixedBitSet::Small(set) => set.remove(elem),
MixedBitSet::Large(set) => set.remove(elem),
}
}
pub fn iter(&self) -> MixedBitIter<'_, T> {
match self {
MixedBitSet::Small(set) => MixedBitIter::Small(set.iter()),
MixedBitSet::Large(set) => MixedBitIter::Large(set.iter()),
}
}
#[inline]
pub fn clear(&mut self) {
match self {
MixedBitSet::Small(set) => set.clear(),
MixedBitSet::Large(set) => set.clear(),
}
}
bit_relations_inherent_impls! {}
}
impl<T> Clone for MixedBitSet<T> {
fn clone(&self) -> Self {
match self {
MixedBitSet::Small(set) => MixedBitSet::Small(set.clone()),
MixedBitSet::Large(set) => MixedBitSet::Large(set.clone()),
}
}
fn clone_from(&mut self, from: &Self) {
match (self, from) {
(MixedBitSet::Small(set), MixedBitSet::Small(from)) => set.clone_from(from),
(MixedBitSet::Large(set), MixedBitSet::Large(from)) => set.clone_from(from),
_ => panic!("MixedBitSet size mismatch"),
}
}
}
impl<T: Idx> BitRelations<MixedBitSet<T>> for MixedBitSet<T> {
fn union(&mut self, other: &MixedBitSet<T>) -> bool {
match (self, other) {
(MixedBitSet::Small(set), MixedBitSet::Small(other)) => set.union(other),
(MixedBitSet::Large(set), MixedBitSet::Large(other)) => set.union(other),
_ => panic!("MixedBitSet size mismatch"),
}
}
fn subtract(&mut self, other: &MixedBitSet<T>) -> bool {
match (self, other) {
(MixedBitSet::Small(set), MixedBitSet::Small(other)) => set.subtract(other),
(MixedBitSet::Large(set), MixedBitSet::Large(other)) => set.subtract(other),
_ => panic!("MixedBitSet size mismatch"),
}
}
fn intersect(&mut self, _other: &MixedBitSet<T>) -> bool {
unimplemented!("implement if/when necessary");
}
}
impl<T: Idx> fmt::Debug for MixedBitSet<T> {
fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
MixedBitSet::Small(set) => set.fmt(w),
MixedBitSet::Large(set) => set.fmt(w),
}
}
}
pub enum MixedBitIter<'a, T: Idx> {
Small(BitIter<'a, T>),
Large(ChunkedBitIter<'a, T>),
}
impl<'a, T: Idx> Iterator for MixedBitIter<'a, T> {
type Item = T;
fn next(&mut self) -> Option<T> {
match self {
MixedBitIter::Small(iter) => iter.next(),
MixedBitIter::Large(iter) => iter.next(),
}
}
}
#[derive(Clone, Debug, PartialEq)]
pub struct GrowableBitSet<T: Idx> {
bit_set: DenseBitSet<T>,
}
impl<T: Idx> Default for GrowableBitSet<T> {
fn default() -> Self {
GrowableBitSet::new_empty()
}
}
impl<T: Idx> GrowableBitSet<T> {
pub fn ensure(&mut self, min_domain_size: usize) {
if self.bit_set.domain_size < min_domain_size {
self.bit_set.domain_size = min_domain_size;
}
let min_num_words = num_words(min_domain_size);
if self.bit_set.words.len() < min_num_words {
self.bit_set.words.resize(min_num_words, 0)
}
}
pub fn new_empty() -> GrowableBitSet<T> {
GrowableBitSet { bit_set: DenseBitSet::new_empty(0) }
}
pub fn with_capacity(capacity: usize) -> GrowableBitSet<T> {
GrowableBitSet { bit_set: DenseBitSet::new_empty(capacity) }
}
#[inline]
pub fn insert(&mut self, elem: T) -> bool {
self.ensure(elem.index() + 1);
self.bit_set.insert(elem)
}
#[inline]
pub fn insert_range(&mut self, elems: Range<T>) {
self.ensure(elems.end.index());
self.bit_set.insert_range(elems);
}
#[inline]
pub fn remove(&mut self, elem: T) -> bool {
self.ensure(elem.index() + 1);
self.bit_set.remove(elem)
}
#[inline]
pub fn clear(&mut self) {
self.bit_set.clear();
}
#[inline]
pub fn count(&self) -> usize {
self.bit_set.count()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.bit_set.is_empty()
}
#[inline]
pub fn contains(&self, elem: T) -> bool {
let (word_index, mask) = word_index_and_mask(elem);
self.bit_set.words.get(word_index).is_some_and(|word| (word & mask) != 0)
}
#[inline]
pub fn contains_any(&self, elems: Range<T>) -> bool {
elems.start.index() < self.bit_set.domain_size
&& self
.bit_set
.contains_any(elems.start..T::new(elems.end.index().min(self.bit_set.domain_size)))
}
#[inline]
pub fn iter(&self) -> BitIter<'_, T> {
self.bit_set.iter()
}
#[inline]
pub fn len(&self) -> usize {
self.bit_set.count()
}
}
impl<T: Idx> From<DenseBitSet<T>> for GrowableBitSet<T> {
fn from(bit_set: DenseBitSet<T>) -> Self {
Self { bit_set }
}
}
#[cfg_attr(feature = "nightly", derive(Decodable_NoContext, Encodable_NoContext))]
#[derive(Clone, Eq, PartialEq, Hash)]
pub struct BitMatrix<R: Idx, C: Idx> {
num_rows: usize,
num_columns: usize,
words: Vec<Word>,
marker: PhantomData<(R, C)>,
}
impl<R: Idx, C: Idx> BitMatrix<R, C> {
pub fn new(num_rows: usize, num_columns: usize) -> BitMatrix<R, C> {
let words_per_row = num_words(num_columns);
BitMatrix {
num_rows,
num_columns,
words: vec![0; num_rows * words_per_row],
marker: PhantomData,
}
}
pub fn from_row_n(row: &DenseBitSet<C>, num_rows: usize) -> BitMatrix<R, C> {
let num_columns = row.domain_size();
let words_per_row = num_words(num_columns);
assert_eq!(words_per_row, row.words.len());
BitMatrix {
num_rows,
num_columns,
words: iter::repeat_n(&row.words, num_rows).flatten().cloned().collect(),
marker: PhantomData,
}
}
pub fn rows(&self) -> impl Iterator<Item = R> {
(0..self.num_rows).map(R::new)
}
fn range(&self, row: R) -> (usize, usize) {
let words_per_row = num_words(self.num_columns);
let start = row.index() * words_per_row;
(start, start + words_per_row)
}
pub fn insert(&mut self, row: R, column: C) -> bool {
assert!(row.index() < self.num_rows && column.index() < self.num_columns);
let (start, _) = self.range(row);
let (word_index, mask) = word_index_and_mask(column);
let words = &mut self.words[..];
let word = words[start + word_index];
let new_word = word | mask;
words[start + word_index] = new_word;
word != new_word
}
pub fn contains(&self, row: R, column: C) -> bool {
assert!(row.index() < self.num_rows && column.index() < self.num_columns);
let (start, _) = self.range(row);
let (word_index, mask) = word_index_and_mask(column);
(self.words[start + word_index] & mask) != 0
}
pub fn intersect_rows(&self, row1: R, row2: R) -> Vec<C> {
assert!(row1.index() < self.num_rows && row2.index() < self.num_rows);
let (row1_start, row1_end) = self.range(row1);
let (row2_start, row2_end) = self.range(row2);
let mut result = Vec::with_capacity(self.num_columns);
for (base, (i, j)) in (row1_start..row1_end).zip(row2_start..row2_end).enumerate() {
let mut v = self.words[i] & self.words[j];
for bit in 0..WORD_BITS {
if v == 0 {
break;
}
if v & 0x1 != 0 {
result.push(C::new(base * WORD_BITS + bit));
}
v >>= 1;
}
}
result
}
pub fn union_rows(&mut self, read: R, write: R) -> bool {
assert!(read.index() < self.num_rows && write.index() < self.num_rows);
let (read_start, read_end) = self.range(read);
let (write_start, write_end) = self.range(write);
let words = &mut self.words[..];
let mut changed = 0;
for (read_index, write_index) in iter::zip(read_start..read_end, write_start..write_end) {
let word = words[write_index];
let new_word = word | words[read_index];
words[write_index] = new_word;
changed |= word ^ new_word;
}
changed != 0
}
pub fn union_row_with(&mut self, with: &DenseBitSet<C>, write: R) -> bool {
assert!(write.index() < self.num_rows);
assert_eq!(with.domain_size(), self.num_columns);
let (write_start, write_end) = self.range(write);
bitwise(&mut self.words[write_start..write_end], &with.words, |a, b| a | b)
}
pub fn insert_all_into_row(&mut self, row: R) {
assert!(row.index() < self.num_rows);
let (start, end) = self.range(row);
let words = &mut self.words[..];
for index in start..end {
words[index] = !0;
}
clear_excess_bits_in_final_word(self.num_columns, &mut self.words[..end]);
}
pub fn words(&self) -> &[Word] {
&self.words
}
pub fn iter(&self, row: R) -> BitIter<'_, C> {
assert!(row.index() < self.num_rows);
let (start, end) = self.range(row);
BitIter::new(&self.words[start..end])
}
pub fn count(&self, row: R) -> usize {
let (start, end) = self.range(row);
count_ones(&self.words[start..end])
}
}
impl<R: Idx, C: Idx> fmt::Debug for BitMatrix<R, C> {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
struct OneLinePrinter<T>(T);
impl<T: fmt::Debug> fmt::Debug for OneLinePrinter<T> {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(fmt, "{:?}", self.0)
}
}
write!(fmt, "BitMatrix({}x{}) ", self.num_rows, self.num_columns)?;
let items = self.rows().flat_map(|r| self.iter(r).map(move |c| (r, c)));
fmt.debug_set().entries(items.map(OneLinePrinter)).finish()
}
}
#[derive(Clone, Debug)]
pub struct SparseBitMatrix<R, C>
where
R: Idx,
C: Idx,
{
num_columns: usize,
rows: IndexVec<R, Option<DenseBitSet<C>>>,
}
impl<R: Idx, C: Idx> SparseBitMatrix<R, C> {
pub fn new(num_columns: usize) -> Self {
Self { num_columns, rows: IndexVec::new() }
}
fn ensure_row(&mut self, row: R) -> &mut DenseBitSet<C> {
self.rows.get_or_insert_with(row, || DenseBitSet::new_empty(self.num_columns))
}
pub fn insert(&mut self, row: R, column: C) -> bool {
self.ensure_row(row).insert(column)
}
pub fn remove(&mut self, row: R, column: C) -> bool {
match self.rows.get_mut(row) {
Some(Some(row)) => row.remove(column),
_ => false,
}
}
pub fn clear(&mut self, row: R) {
if let Some(Some(row)) = self.rows.get_mut(row) {
row.clear();
}
}
pub fn contains(&self, row: R, column: C) -> bool {
self.row(row).is_some_and(|r| r.contains(column))
}
pub fn union_rows(&mut self, read: R, write: R) -> bool {
if read == write || self.row(read).is_none() {
return false;
}
self.ensure_row(write);
if let (Some(read_row), Some(write_row)) = self.rows.pick2_mut(read, write) {
write_row.union(read_row)
} else {
unreachable!()
}
}
pub fn insert_all_into_row(&mut self, row: R) {
self.ensure_row(row).insert_all();
}
pub fn rows(&self) -> impl Iterator<Item = R> {
self.rows.indices()
}
pub fn iter(&self, row: R) -> impl Iterator<Item = C> {
self.row(row).into_iter().flat_map(|r| r.iter())
}
pub fn row(&self, row: R) -> Option<&DenseBitSet<C>> {
self.rows.get(row)?.as_ref()
}
pub fn intersect_row<Set>(&mut self, row: R, set: &Set) -> bool
where
DenseBitSet<C>: BitRelations<Set>,
{
match self.rows.get_mut(row) {
Some(Some(row)) => row.intersect(set),
_ => false,
}
}
pub fn subtract_row<Set>(&mut self, row: R, set: &Set) -> bool
where
DenseBitSet<C>: BitRelations<Set>,
{
match self.rows.get_mut(row) {
Some(Some(row)) => row.subtract(set),
_ => false,
}
}
pub fn union_row<Set>(&mut self, row: R, set: &Set) -> bool
where
DenseBitSet<C>: BitRelations<Set>,
{
self.ensure_row(row).union(set)
}
}
#[inline]
fn num_words<T: Idx>(domain_size: T) -> usize {
domain_size.index().div_ceil(WORD_BITS)
}
#[inline]
fn word_index_and_mask<T: Idx>(elem: T) -> (usize, Word) {
let elem = elem.index();
let word_index = elem / WORD_BITS;
let mask = 1 << (elem % WORD_BITS);
(word_index, mask)
}
#[inline]
fn chunk_index<T: Idx>(elem: T) -> usize {
elem.index() / CHUNK_BITS
}
#[inline]
fn chunk_word_index_and_mask<T: Idx>(elem: T) -> (usize, Word) {
let chunk_elem = elem.index() % CHUNK_BITS;
word_index_and_mask(chunk_elem)
}
fn clear_excess_bits_in_final_word(domain_size: usize, words: &mut [Word]) {
let num_bits_in_final_word = domain_size % WORD_BITS;
if num_bits_in_final_word > 0 {
let mask = (1 << num_bits_in_final_word) - 1;
words[words.len() - 1] &= mask;
}
}
#[inline]
fn max_bit(word: Word) -> usize {
WORD_BITS - 1 - word.leading_zeros() as usize
}
#[inline]
fn count_ones(words: &[Word]) -> usize {
words.iter().map(|word| word.count_ones() as usize).sum()
}