1use std::marker::PhantomData;
2use std::ops::{Bound, Range, RangeBounds};
3use std::rc::Rc;
4use std::{fmt, iter, slice};
5
6use Chunk::*;
7#[cfg(feature = "nightly")]
8use rustc_macros::{Decodable_NoContext, Encodable_NoContext};
9
10use crate::{Idx, IndexVec};
11
12#[cfg(test)]
13mod tests;
14
15type Word = u64;
16const WORD_BYTES: usize = size_of::<Word>();
17const WORD_BITS: usize = WORD_BYTES * 8;
18
19const CHUNK_WORDS: usize = 32;
30const CHUNK_BITS: usize = CHUNK_WORDS * WORD_BITS; type ChunkSize = u16;
35const _: () = if !(CHUNK_BITS <= ChunkSize::MAX as usize) {
::core::panicking::panic("assertion failed: CHUNK_BITS <= ChunkSize::MAX as usize")
}assert!(CHUNK_BITS <= ChunkSize::MAX as usize);
36
37pub trait BitRelations<Rhs> {
38 fn union(&mut self, other: &Rhs) -> bool;
39 fn subtract(&mut self, other: &Rhs) -> bool;
40 fn intersect(&mut self, other: &Rhs) -> bool;
41}
42
43#[inline]
44fn inclusive_start_end<T: Idx>(
45 range: impl RangeBounds<T>,
46 domain: usize,
47) -> Option<(usize, usize)> {
48 let start = match range.start_bound().cloned() {
50 Bound::Included(start) => start.index(),
51 Bound::Excluded(start) => start.index() + 1,
52 Bound::Unbounded => 0,
53 };
54 let end = match range.end_bound().cloned() {
55 Bound::Included(end) => end.index(),
56 Bound::Excluded(end) => end.index().checked_sub(1)?,
57 Bound::Unbounded => domain - 1,
58 };
59 if !(end < domain) {
::core::panicking::panic("assertion failed: end < domain")
};assert!(end < domain);
60 if start > end {
61 return None;
62 }
63 Some((start, end))
64}
65
66macro_rules! bit_relations_inherent_impls {
67 () => {
68 pub fn union<Rhs>(&mut self, other: &Rhs) -> bool
71 where
72 Self: BitRelations<Rhs>,
73 {
74 <Self as BitRelations<Rhs>>::union(self, other)
75 }
76
77 pub fn subtract<Rhs>(&mut self, other: &Rhs) -> bool
80 where
81 Self: BitRelations<Rhs>,
82 {
83 <Self as BitRelations<Rhs>>::subtract(self, other)
84 }
85
86 pub fn intersect<Rhs>(&mut self, other: &Rhs) -> bool
89 where
90 Self: BitRelations<Rhs>,
91 {
92 <Self as BitRelations<Rhs>>::intersect(self, other)
93 }
94 };
95}
96
97#[cfg_attr(feature = "nightly", derive(const _: () =
{
impl<T, __D: ::rustc_serialize::Decoder>
::rustc_serialize::Decodable<__D> for DenseBitSet<T> where
PhantomData<T>: ::rustc_serialize::Decodable<__D> {
fn decode(__decoder: &mut __D) -> Self {
DenseBitSet {
domain_size: ::rustc_serialize::Decodable::decode(__decoder),
words: ::rustc_serialize::Decodable::decode(__decoder),
marker: ::rustc_serialize::Decodable::decode(__decoder),
}
}
}
};Decodable_NoContext, const _: () =
{
impl<T, __E: ::rustc_serialize::Encoder>
::rustc_serialize::Encodable<__E> for DenseBitSet<T> where
PhantomData<T>: ::rustc_serialize::Encodable<__E> {
fn encode(&self, __encoder: &mut __E) {
match *self {
DenseBitSet {
domain_size: ref __binding_0,
words: ref __binding_1,
marker: ref __binding_2 } => {
::rustc_serialize::Encodable::<__E>::encode(__binding_0,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_1,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_2,
__encoder);
}
}
}
}
};Encodable_NoContext))]
115#[derive(#[automatically_derived]
impl<T: ::core::cmp::Eq> ::core::cmp::Eq for DenseBitSet<T> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<usize>;
let _: ::core::cmp::AssertParamIsEq<Vec<Word>>;
let _: ::core::cmp::AssertParamIsEq<PhantomData<T>>;
}
}Eq, #[automatically_derived]
impl<T: ::core::cmp::PartialEq> ::core::cmp::PartialEq for DenseBitSet<T> {
#[inline]
fn eq(&self, other: &DenseBitSet<T>) -> bool {
self.domain_size == other.domain_size && self.words == other.words &&
self.marker == other.marker
}
}PartialEq, #[automatically_derived]
impl<T: ::core::hash::Hash> ::core::hash::Hash for DenseBitSet<T> {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&self.domain_size, state);
::core::hash::Hash::hash(&self.words, state);
::core::hash::Hash::hash(&self.marker, state)
}
}Hash)]
116pub struct DenseBitSet<T> {
117 domain_size: usize,
118 words: Vec<Word>,
119 marker: PhantomData<T>,
120}
121
122impl<T> DenseBitSet<T> {
123 pub fn domain_size(&self) -> usize {
125 self.domain_size
126 }
127}
128
129impl<T: Idx> DenseBitSet<T> {
130 #[inline]
132 pub fn new_empty(domain_size: usize) -> DenseBitSet<T> {
133 let num_words = num_words(domain_size);
134 DenseBitSet { domain_size, words: ::alloc::vec::from_elem(0, num_words)vec![0; num_words], marker: PhantomData }
135 }
136
137 #[inline]
139 pub fn new_filled(domain_size: usize) -> DenseBitSet<T> {
140 let num_words = num_words(domain_size);
141 let mut result =
142 DenseBitSet { domain_size, words: ::alloc::vec::from_elem(!0, num_words)vec![!0; num_words], marker: PhantomData };
143 result.clear_excess_bits();
144 result
145 }
146
147 #[inline]
149 pub fn clear(&mut self) {
150 self.words.fill(0);
151 }
152
153 fn clear_excess_bits(&mut self) {
155 clear_excess_bits_in_final_word(self.domain_size, &mut self.words);
156 }
157
158 pub fn count(&self) -> usize {
160 count_ones(&self.words)
161 }
162
163 #[inline]
165 pub fn contains(&self, elem: T) -> bool {
166 if !(elem.index() < self.domain_size) {
::core::panicking::panic("assertion failed: elem.index() < self.domain_size")
};assert!(elem.index() < self.domain_size);
167 let (word_index, mask) = word_index_and_mask(elem);
168 (self.words[word_index] & mask) != 0
169 }
170
171 #[inline]
173 pub fn superset(&self, other: &DenseBitSet<T>) -> bool {
174 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
175 self.words.iter().zip(&other.words).all(|(a, b)| (a & b) == *b)
176 }
177
178 #[inline]
180 pub fn is_empty(&self) -> bool {
181 self.words.iter().all(|a| *a == 0)
182 }
183
184 #[inline]
186 pub fn insert(&mut self, elem: T) -> bool {
187 if !(elem.index() < self.domain_size) {
{
::core::panicking::panic_fmt(format_args!("inserting element at index {0} but domain size is {1}",
elem.index(), self.domain_size));
}
};assert!(
188 elem.index() < self.domain_size,
189 "inserting element at index {} but domain size is {}",
190 elem.index(),
191 self.domain_size,
192 );
193 let (word_index, mask) = word_index_and_mask(elem);
194 let word_ref = &mut self.words[word_index];
195 let word = *word_ref;
196 let new_word = word | mask;
197 *word_ref = new_word;
198 new_word != word
199 }
200
201 #[inline]
202 pub fn insert_range(&mut self, elems: impl RangeBounds<T>) {
203 let Some((start, end)) = inclusive_start_end(elems, self.domain_size) else {
204 return;
205 };
206
207 let (start_word_index, start_mask) = word_index_and_mask(start);
208 let (end_word_index, end_mask) = word_index_and_mask(end);
209
210 for word_index in (start_word_index + 1)..end_word_index {
212 self.words[word_index] = !0;
213 }
214
215 if start_word_index != end_word_index {
216 self.words[start_word_index] |= !(start_mask - 1);
220 self.words[end_word_index] |= end_mask | (end_mask - 1);
223 } else {
224 self.words[start_word_index] |= end_mask | (end_mask - start_mask);
225 }
226 }
227
228 pub fn insert_all(&mut self) {
230 self.words.fill(!0);
231 self.clear_excess_bits();
232 }
233
234 #[inline]
236 pub fn contains_any(&self, elems: impl RangeBounds<T>) -> bool {
237 let Some((start, end)) = inclusive_start_end(elems, self.domain_size) else {
238 return false;
239 };
240 let (start_word_index, start_mask) = word_index_and_mask(start);
241 let (end_word_index, end_mask) = word_index_and_mask(end);
242
243 if start_word_index == end_word_index {
244 self.words[start_word_index] & (end_mask | (end_mask - start_mask)) != 0
245 } else {
246 if self.words[start_word_index] & !(start_mask - 1) != 0 {
247 return true;
248 }
249
250 let remaining = start_word_index + 1..end_word_index;
251 if remaining.start <= remaining.end {
252 self.words[remaining].iter().any(|&w| w != 0)
253 || self.words[end_word_index] & (end_mask | (end_mask - 1)) != 0
254 } else {
255 false
256 }
257 }
258 }
259
260 #[inline]
262 pub fn remove(&mut self, elem: T) -> bool {
263 if !(elem.index() < self.domain_size) {
::core::panicking::panic("assertion failed: elem.index() < self.domain_size")
};assert!(elem.index() < self.domain_size);
264 let (word_index, mask) = word_index_and_mask(elem);
265 let word_ref = &mut self.words[word_index];
266 let word = *word_ref;
267 let new_word = word & !mask;
268 *word_ref = new_word;
269 new_word != word
270 }
271
272 #[inline]
274 pub fn iter(&self) -> BitIter<'_, T> {
275 BitIter::new(&self.words)
276 }
277
278 pub fn last_set_in(&self, range: impl RangeBounds<T>) -> Option<T> {
279 let (start, end) = inclusive_start_end(range, self.domain_size)?;
280 let (start_word_index, _) = word_index_and_mask(start);
281 let (end_word_index, end_mask) = word_index_and_mask(end);
282
283 let end_word = self.words[end_word_index] & (end_mask | (end_mask - 1));
284 if end_word != 0 {
285 let pos = max_bit(end_word) + WORD_BITS * end_word_index;
286 if start <= pos {
287 return Some(T::new(pos));
288 }
289 }
290
291 if let Some(offset) =
295 self.words[start_word_index..end_word_index].iter().rposition(|&w| w != 0)
296 {
297 let word_idx = start_word_index + offset;
298 let start_word = self.words[word_idx];
299 let pos = max_bit(start_word) + WORD_BITS * word_idx;
300 if start <= pos {
301 return Some(T::new(pos));
302 }
303 }
304
305 None
306 }
307
308 self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::union(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::subtract(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::intersect(self, other);bit_relations_inherent_impls! {}
309
310 pub fn union_not(&mut self, other: &DenseBitSet<T>) {
315 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
316
317 bitwise(&mut self.words, &other.words, |a, b| a | !b);
323 self.clear_excess_bits();
326 }
327}
328
329impl<T: Idx> BitRelations<DenseBitSet<T>> for DenseBitSet<T> {
331 fn union(&mut self, other: &DenseBitSet<T>) -> bool {
332 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
333 bitwise(&mut self.words, &other.words, |a, b| a | b)
334 }
335
336 fn subtract(&mut self, other: &DenseBitSet<T>) -> bool {
337 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
338 bitwise(&mut self.words, &other.words, |a, b| a & !b)
339 }
340
341 fn intersect(&mut self, other: &DenseBitSet<T>) -> bool {
342 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
343 bitwise(&mut self.words, &other.words, |a, b| a & b)
344 }
345}
346
347impl<T: Idx> From<GrowableBitSet<T>> for DenseBitSet<T> {
348 fn from(bit_set: GrowableBitSet<T>) -> Self {
349 bit_set.bit_set
350 }
351}
352
353impl<T> Clone for DenseBitSet<T> {
354 fn clone(&self) -> Self {
355 DenseBitSet {
356 domain_size: self.domain_size,
357 words: self.words.clone(),
358 marker: PhantomData,
359 }
360 }
361
362 fn clone_from(&mut self, from: &Self) {
363 self.domain_size = from.domain_size;
364 self.words.clone_from(&from.words);
365 }
366}
367
368impl<T: Idx> fmt::Debug for DenseBitSet<T> {
369 fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
370 w.debug_list().entries(self.iter()).finish()
371 }
372}
373
374impl<T: Idx> ToString for DenseBitSet<T> {
375 fn to_string(&self) -> String {
376 let mut result = String::new();
377 let mut sep = '[';
378
379 let mut i = 0;
383 for word in &self.words {
384 let mut word = *word;
385 for _ in 0..WORD_BYTES {
386 let remain = self.domain_size - i;
388 let mask = if remain <= 8 { (1 << remain) - 1 } else { 0xFF };
390 if !(mask <= 0xFF) {
::core::panicking::panic("assertion failed: mask <= 0xFF")
};assert!(mask <= 0xFF);
391 let byte = word & mask;
392
393 result.push_str(&::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1:02x}", sep, byte))
})format!("{sep}{byte:02x}"));
394
395 if remain <= 8 {
396 break;
397 }
398 word >>= 8;
399 i += 8;
400 sep = '-';
401 }
402 sep = '|';
403 }
404 result.push(']');
405
406 result
407 }
408}
409
410pub struct BitIter<'a, T: Idx> {
411 word: Word,
415
416 offset: usize,
418
419 iter: slice::Iter<'a, Word>,
421
422 marker: PhantomData<T>,
423}
424
425impl<'a, T: Idx> BitIter<'a, T> {
426 #[inline]
427 fn new(words: &'a [Word]) -> BitIter<'a, T> {
428 BitIter {
434 word: 0,
435 offset: usize::MAX - (WORD_BITS - 1),
436 iter: words.iter(),
437 marker: PhantomData,
438 }
439 }
440}
441
442impl<'a, T: Idx> Iterator for BitIter<'a, T> {
443 type Item = T;
444 fn next(&mut self) -> Option<T> {
445 loop {
446 if self.word != 0 {
447 let bit_pos = self.word.trailing_zeros() as usize;
450 self.word ^= 1 << bit_pos;
451 return Some(T::new(bit_pos + self.offset));
452 }
453
454 self.word = *self.iter.next()?;
457 self.offset = self.offset.wrapping_add(WORD_BITS);
458 }
459 }
460}
461
462#[derive(#[automatically_derived]
impl<T: ::core::cmp::PartialEq> ::core::cmp::PartialEq for ChunkedBitSet<T> {
#[inline]
fn eq(&self, other: &ChunkedBitSet<T>) -> bool {
self.domain_size == other.domain_size && self.chunks == other.chunks
&& self.marker == other.marker
}
}PartialEq, #[automatically_derived]
impl<T: ::core::cmp::Eq> ::core::cmp::Eq for ChunkedBitSet<T> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<usize>;
let _: ::core::cmp::AssertParamIsEq<Box<[Chunk]>>;
let _: ::core::cmp::AssertParamIsEq<PhantomData<T>>;
}
}Eq)]
481pub struct ChunkedBitSet<T> {
482 domain_size: usize,
483
484 chunks: Box<[Chunk]>,
487
488 marker: PhantomData<T>,
489}
490
491#[derive(#[automatically_derived]
impl ::core::clone::Clone for Chunk {
#[inline]
fn clone(&self) -> Chunk {
match self {
Chunk::Zeros { chunk_domain_size: __self_0 } =>
Chunk::Zeros {
chunk_domain_size: ::core::clone::Clone::clone(__self_0),
},
Chunk::Ones { chunk_domain_size: __self_0 } =>
Chunk::Ones {
chunk_domain_size: ::core::clone::Clone::clone(__self_0),
},
Chunk::Mixed {
chunk_domain_size: __self_0,
ones_count: __self_1,
words: __self_2 } =>
Chunk::Mixed {
chunk_domain_size: ::core::clone::Clone::clone(__self_0),
ones_count: ::core::clone::Clone::clone(__self_1),
words: ::core::clone::Clone::clone(__self_2),
},
}
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Chunk {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
Chunk::Zeros { chunk_domain_size: __self_0 } =>
::core::fmt::Formatter::debug_struct_field1_finish(f, "Zeros",
"chunk_domain_size", &__self_0),
Chunk::Ones { chunk_domain_size: __self_0 } =>
::core::fmt::Formatter::debug_struct_field1_finish(f, "Ones",
"chunk_domain_size", &__self_0),
Chunk::Mixed {
chunk_domain_size: __self_0,
ones_count: __self_1,
words: __self_2 } =>
::core::fmt::Formatter::debug_struct_field3_finish(f, "Mixed",
"chunk_domain_size", __self_0, "ones_count", __self_1,
"words", &__self_2),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for Chunk {
#[inline]
fn eq(&self, other: &Chunk) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(Chunk::Zeros { chunk_domain_size: __self_0 }, Chunk::Zeros {
chunk_domain_size: __arg1_0 }) => __self_0 == __arg1_0,
(Chunk::Ones { chunk_domain_size: __self_0 }, Chunk::Ones {
chunk_domain_size: __arg1_0 }) => __self_0 == __arg1_0,
(Chunk::Mixed {
chunk_domain_size: __self_0,
ones_count: __self_1,
words: __self_2 }, Chunk::Mixed {
chunk_domain_size: __arg1_0,
ones_count: __arg1_1,
words: __arg1_2 }) =>
__self_0 == __arg1_0 && __self_1 == __arg1_1 &&
__self_2 == __arg1_2,
_ => unsafe { ::core::intrinsics::unreachable() }
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for Chunk {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<ChunkSize>;
let _: ::core::cmp::AssertParamIsEq<Rc<[Word; CHUNK_WORDS]>>;
}
}Eq)]
496enum Chunk {
497 Zeros { chunk_domain_size: ChunkSize },
499
500 Ones { chunk_domain_size: ChunkSize },
502
503 Mixed {
517 chunk_domain_size: ChunkSize,
518 ones_count: ChunkSize,
525 words: Rc<[Word; CHUNK_WORDS]>,
526 },
527}
528
529#[cfg(target_pointer_width = "64")]
531const _: [(); 16] = [(); ::std::mem::size_of::<Chunk>()];crate::static_assert_size!(Chunk, 16);
532
533impl<T> ChunkedBitSet<T> {
534 pub fn domain_size(&self) -> usize {
535 self.domain_size
536 }
537
538 #[cfg(test)]
539 fn assert_valid(&self) {
540 if self.domain_size == 0 {
541 assert!(self.chunks.is_empty());
542 return;
543 }
544
545 assert!((self.chunks.len() - 1) * CHUNK_BITS <= self.domain_size);
546 assert!(self.chunks.len() * CHUNK_BITS >= self.domain_size);
547 for chunk in self.chunks.iter() {
548 chunk.assert_valid();
549 }
550 }
551}
552
553impl<T: Idx> ChunkedBitSet<T> {
554 fn new(domain_size: usize, is_empty: bool) -> Self {
556 let chunks = if domain_size == 0 {
557 Box::new([])
558 } else {
559 let num_chunks = domain_size.index().div_ceil(CHUNK_BITS);
560 let mut last_chunk_domain_size = domain_size % CHUNK_BITS;
561 if last_chunk_domain_size == 0 {
562 last_chunk_domain_size = CHUNK_BITS;
563 };
564
565 let (normal_chunk, final_chunk) = if is_empty {
568 (
569 Zeros { chunk_domain_size: CHUNK_BITS as ChunkSize },
570 Zeros { chunk_domain_size: last_chunk_domain_size as ChunkSize },
571 )
572 } else {
573 (
574 Ones { chunk_domain_size: CHUNK_BITS as ChunkSize },
575 Ones { chunk_domain_size: last_chunk_domain_size as ChunkSize },
576 )
577 };
578 let mut chunks = ::alloc::vec::from_elem(normal_chunk, num_chunks)vec![normal_chunk; num_chunks].into_boxed_slice();
579 *chunks.as_mut().last_mut().unwrap() = final_chunk;
580 chunks
581 };
582 ChunkedBitSet { domain_size, chunks, marker: PhantomData }
583 }
584
585 #[inline]
587 pub fn new_empty(domain_size: usize) -> Self {
588 ChunkedBitSet::new(domain_size, true)
589 }
590
591 #[inline]
593 pub fn new_filled(domain_size: usize) -> Self {
594 ChunkedBitSet::new(domain_size, false)
595 }
596
597 pub fn clear(&mut self) {
598 *self = ChunkedBitSet::new_empty(self.domain_size);
600 }
601
602 #[cfg(test)]
603 fn chunks(&self) -> &[Chunk] {
604 &self.chunks
605 }
606
607 pub fn count(&self) -> usize {
609 self.chunks.iter().map(|chunk| chunk.count()).sum()
610 }
611
612 pub fn is_empty(&self) -> bool {
613 self.chunks.iter().all(|chunk| #[allow(non_exhaustive_omitted_patterns)] match chunk {
Zeros { .. } => true,
_ => false,
}matches!(chunk, Zeros { .. }))
614 }
615
616 #[inline]
618 pub fn contains(&self, elem: T) -> bool {
619 if !(elem.index() < self.domain_size) {
::core::panicking::panic("assertion failed: elem.index() < self.domain_size")
};assert!(elem.index() < self.domain_size);
620 let chunk = &self.chunks[chunk_index(elem)];
621 match &chunk {
622 Zeros { .. } => false,
623 Ones { .. } => true,
624 Mixed { words, .. } => {
625 let (word_index, mask) = chunk_word_index_and_mask(elem);
626 (words[word_index] & mask) != 0
627 }
628 }
629 }
630
631 #[inline]
632 pub fn iter(&self) -> ChunkedBitIter<'_, T> {
633 ChunkedBitIter::new(self)
634 }
635
636 pub fn insert(&mut self, elem: T) -> bool {
638 if !(elem.index() < self.domain_size) {
::core::panicking::panic("assertion failed: elem.index() < self.domain_size")
};assert!(elem.index() < self.domain_size);
639 let chunk_index = chunk_index(elem);
640 let chunk = &mut self.chunks[chunk_index];
641 match *chunk {
642 Zeros { chunk_domain_size } => {
643 if chunk_domain_size > 1 {
644 let mut words = {
645 let words = Rc::<[Word; CHUNK_WORDS]>::new_zeroed();
647 unsafe { words.assume_init() }
649 };
650 let words_ref = Rc::get_mut(&mut words).unwrap();
651
652 let (word_index, mask) = chunk_word_index_and_mask(elem);
653 words_ref[word_index] |= mask;
654 *chunk = Mixed { chunk_domain_size, ones_count: 1, words };
655 } else {
656 *chunk = Ones { chunk_domain_size };
657 }
658 true
659 }
660 Ones { .. } => false,
661 Mixed { chunk_domain_size, ref mut ones_count, ref mut words } => {
662 let (word_index, mask) = chunk_word_index_and_mask(elem);
664 if (words[word_index] & mask) == 0 {
665 *ones_count += 1;
666 if *ones_count < chunk_domain_size {
667 let words = Rc::make_mut(words);
668 words[word_index] |= mask;
669 } else {
670 *chunk = Ones { chunk_domain_size };
671 }
672 true
673 } else {
674 false
675 }
676 }
677 }
678 }
679
680 pub fn insert_all(&mut self) {
682 *self = ChunkedBitSet::new_filled(self.domain_size);
684 }
685
686 pub fn remove(&mut self, elem: T) -> bool {
688 if !(elem.index() < self.domain_size) {
::core::panicking::panic("assertion failed: elem.index() < self.domain_size")
};assert!(elem.index() < self.domain_size);
689 let chunk_index = chunk_index(elem);
690 let chunk = &mut self.chunks[chunk_index];
691 match *chunk {
692 Zeros { .. } => false,
693 Ones { chunk_domain_size } => {
694 if chunk_domain_size > 1 {
695 let mut words = {
696 let words = Rc::<[Word; CHUNK_WORDS]>::new_zeroed();
698 unsafe { words.assume_init() }
700 };
701 let words_ref = Rc::get_mut(&mut words).unwrap();
702
703 let num_words = num_words(chunk_domain_size as usize);
705 words_ref[..num_words].fill(!0);
706 clear_excess_bits_in_final_word(
707 chunk_domain_size as usize,
708 &mut words_ref[..num_words],
709 );
710 let (word_index, mask) = chunk_word_index_and_mask(elem);
711 words_ref[word_index] &= !mask;
712 *chunk = Mixed { chunk_domain_size, ones_count: chunk_domain_size - 1, words };
713 } else {
714 *chunk = Zeros { chunk_domain_size };
715 }
716 true
717 }
718 Mixed { chunk_domain_size, ref mut ones_count, ref mut words } => {
719 let (word_index, mask) = chunk_word_index_and_mask(elem);
721 if (words[word_index] & mask) != 0 {
722 *ones_count -= 1;
723 if *ones_count > 0 {
724 let words = Rc::make_mut(words);
725 words[word_index] &= !mask;
726 } else {
727 *chunk = Zeros { chunk_domain_size }
728 }
729 true
730 } else {
731 false
732 }
733 }
734 }
735 }
736
737 fn chunk_iter(&self, chunk_index: usize) -> ChunkIter<'_> {
738 match self.chunks.get(chunk_index) {
739 Some(Zeros { .. }) => ChunkIter::Zeros,
740 Some(Ones { chunk_domain_size }) => ChunkIter::Ones(0..*chunk_domain_size as usize),
741 Some(Mixed { chunk_domain_size, words, .. }) => {
742 let num_words = num_words(*chunk_domain_size as usize);
743 ChunkIter::Mixed(BitIter::new(&words[0..num_words]))
744 }
745 None => ChunkIter::Finished,
746 }
747 }
748
749 self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::union(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::subtract(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::intersect(self, other);bit_relations_inherent_impls! {}
750}
751
752impl<T: Idx> BitRelations<ChunkedBitSet<T>> for ChunkedBitSet<T> {
753 fn union(&mut self, other: &ChunkedBitSet<T>) -> bool {
754 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
755
756 let mut changed = false;
757 for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
758 match (&mut self_chunk, &other_chunk) {
759 (_, Zeros { .. }) | (Ones { .. }, _) => {}
760 (Zeros { .. }, _) | (Mixed { .. }, Ones { .. }) => {
761 *self_chunk = other_chunk.clone();
763 changed = true;
764 }
765 (
766 Mixed {
767 chunk_domain_size,
768 ones_count: self_chunk_ones_count,
769 words: self_chunk_words,
770 },
771 Mixed { words: other_chunk_words, .. },
772 ) => {
773 let num_words = num_words(*chunk_domain_size as usize);
779
780 if self_chunk_words[0..num_words] == other_chunk_words[0..num_words] {
784 continue;
785 }
786
787 let op = |a, b| a | b;
790 if !bitwise_changes(
791 &self_chunk_words[0..num_words],
792 &other_chunk_words[0..num_words],
793 op,
794 ) {
795 continue;
796 }
797
798 let self_chunk_words = Rc::make_mut(self_chunk_words);
800 let has_changed = bitwise(
801 &mut self_chunk_words[0..num_words],
802 &other_chunk_words[0..num_words],
803 op,
804 );
805 if true {
if !has_changed {
::core::panicking::panic("assertion failed: has_changed")
};
};debug_assert!(has_changed);
806 *self_chunk_ones_count =
807 count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
808 if *self_chunk_ones_count == *chunk_domain_size {
809 *self_chunk = Ones { chunk_domain_size: *chunk_domain_size };
810 }
811 changed = true;
812 }
813 }
814 }
815 changed
816 }
817
818 fn subtract(&mut self, other: &ChunkedBitSet<T>) -> bool {
819 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
820
821 let mut changed = false;
822 for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
823 match (&mut self_chunk, &other_chunk) {
824 (Zeros { .. }, _) | (_, Zeros { .. }) => {}
825 (Ones { chunk_domain_size } | Mixed { chunk_domain_size, .. }, Ones { .. }) => {
826 changed = true;
827 *self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
828 }
829 (
830 Ones { chunk_domain_size },
831 Mixed { ones_count: other_chunk_ones_count, words: other_chunk_words, .. },
832 ) => {
833 changed = true;
834 let num_words = num_words(*chunk_domain_size as usize);
835 if true {
if !(num_words > 0 && num_words <= CHUNK_WORDS) {
::core::panicking::panic("assertion failed: num_words > 0 && num_words <= CHUNK_WORDS")
};
};debug_assert!(num_words > 0 && num_words <= CHUNK_WORDS);
836 let mut tail_mask =
837 1 << (*chunk_domain_size - ((num_words - 1) * WORD_BITS) as u16) - 1;
838 let mut self_chunk_words = **other_chunk_words;
839 for word in self_chunk_words[0..num_words].iter_mut().rev() {
840 *word = !*word & tail_mask;
841 tail_mask = Word::MAX;
842 }
843 let self_chunk_ones_count = *chunk_domain_size - *other_chunk_ones_count;
844 if true {
match (&self_chunk_ones_count,
&(count_ones(&self_chunk_words[0..num_words]) as ChunkSize)) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_eq!(
845 self_chunk_ones_count,
846 count_ones(&self_chunk_words[0..num_words]) as ChunkSize
847 );
848 *self_chunk = Mixed {
849 chunk_domain_size: *chunk_domain_size,
850 ones_count: self_chunk_ones_count,
851 words: Rc::new(self_chunk_words),
852 };
853 }
854 (
855 Mixed {
856 chunk_domain_size,
857 ones_count: self_chunk_ones_count,
858 words: self_chunk_words,
859 },
860 Mixed { words: other_chunk_words, .. },
861 ) => {
862 let num_words = num_words(*chunk_domain_size as usize);
864 let op = |a: Word, b: Word| a & !b;
865 if !bitwise_changes(
866 &self_chunk_words[0..num_words],
867 &other_chunk_words[0..num_words],
868 op,
869 ) {
870 continue;
871 }
872
873 let self_chunk_words = Rc::make_mut(self_chunk_words);
874 let has_changed = bitwise(
875 &mut self_chunk_words[0..num_words],
876 &other_chunk_words[0..num_words],
877 op,
878 );
879 if true {
if !has_changed {
::core::panicking::panic("assertion failed: has_changed")
};
};debug_assert!(has_changed);
880 *self_chunk_ones_count =
881 count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
882 if *self_chunk_ones_count == 0 {
883 *self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
884 }
885 changed = true;
886 }
887 }
888 }
889 changed
890 }
891
892 fn intersect(&mut self, other: &ChunkedBitSet<T>) -> bool {
893 match (&self.domain_size, &other.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, other.domain_size);
894
895 let mut changed = false;
896 for (mut self_chunk, other_chunk) in self.chunks.iter_mut().zip(other.chunks.iter()) {
897 match (&mut self_chunk, &other_chunk) {
898 (Zeros { .. }, _) | (_, Ones { .. }) => {}
899 (Ones { .. }, Zeros { .. } | Mixed { .. }) | (Mixed { .. }, Zeros { .. }) => {
900 changed = true;
901 *self_chunk = other_chunk.clone();
902 }
903 (
904 Mixed {
905 chunk_domain_size,
906 ones_count: self_chunk_ones_count,
907 words: self_chunk_words,
908 },
909 Mixed { words: other_chunk_words, .. },
910 ) => {
911 let num_words = num_words(*chunk_domain_size as usize);
913 let op = |a, b| a & b;
914 if !bitwise_changes(
915 &self_chunk_words[0..num_words],
916 &other_chunk_words[0..num_words],
917 op,
918 ) {
919 continue;
920 }
921
922 let self_chunk_words = Rc::make_mut(self_chunk_words);
923 let has_changed = bitwise(
924 &mut self_chunk_words[0..num_words],
925 &other_chunk_words[0..num_words],
926 op,
927 );
928 if true {
if !has_changed {
::core::panicking::panic("assertion failed: has_changed")
};
};debug_assert!(has_changed);
929 *self_chunk_ones_count =
930 count_ones(&self_chunk_words[0..num_words]) as ChunkSize;
931 if *self_chunk_ones_count == 0 {
932 *self_chunk = Zeros { chunk_domain_size: *chunk_domain_size };
933 }
934 changed = true;
935 }
936 }
937 }
938
939 changed
940 }
941}
942
943impl<T> Clone for ChunkedBitSet<T> {
944 fn clone(&self) -> Self {
945 ChunkedBitSet {
946 domain_size: self.domain_size,
947 chunks: self.chunks.clone(),
948 marker: PhantomData,
949 }
950 }
951
952 fn clone_from(&mut self, from: &Self) {
957 match (&self.domain_size, &from.domain_size) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(self.domain_size, from.domain_size);
958 if true {
match (&self.chunks.len(), &from.chunks.len()) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_eq!(self.chunks.len(), from.chunks.len());
959
960 self.chunks.clone_from(&from.chunks)
961 }
962}
963
964pub struct ChunkedBitIter<'a, T: Idx> {
965 bit_set: &'a ChunkedBitSet<T>,
966
967 chunk_index: usize,
969
970 chunk_iter: ChunkIter<'a>,
972}
973
974impl<'a, T: Idx> ChunkedBitIter<'a, T> {
975 #[inline]
976 fn new(bit_set: &'a ChunkedBitSet<T>) -> ChunkedBitIter<'a, T> {
977 ChunkedBitIter { bit_set, chunk_index: 0, chunk_iter: bit_set.chunk_iter(0) }
978 }
979}
980
981impl<'a, T: Idx> Iterator for ChunkedBitIter<'a, T> {
982 type Item = T;
983
984 fn next(&mut self) -> Option<T> {
985 loop {
986 match &mut self.chunk_iter {
987 ChunkIter::Zeros => {}
988 ChunkIter::Ones(iter) => {
989 if let Some(next) = iter.next() {
990 return Some(T::new(next + self.chunk_index * CHUNK_BITS));
991 }
992 }
993 ChunkIter::Mixed(iter) => {
994 if let Some(next) = iter.next() {
995 return Some(T::new(next + self.chunk_index * CHUNK_BITS));
996 }
997 }
998 ChunkIter::Finished => return None,
999 }
1000 self.chunk_index += 1;
1001 self.chunk_iter = self.bit_set.chunk_iter(self.chunk_index);
1002 }
1003 }
1004}
1005
1006impl Chunk {
1007 #[cfg(test)]
1008 fn assert_valid(&self) {
1009 match *self {
1010 Zeros { chunk_domain_size } | Ones { chunk_domain_size } => {
1011 assert!(chunk_domain_size as usize <= CHUNK_BITS);
1012 }
1013 Mixed { chunk_domain_size, ones_count, ref words } => {
1014 assert!(chunk_domain_size as usize <= CHUNK_BITS);
1015 assert!(0 < ones_count && ones_count < chunk_domain_size);
1016
1017 assert_eq!(count_ones(words.as_slice()) as ChunkSize, ones_count);
1019
1020 let num_words = num_words(chunk_domain_size as usize);
1022 if num_words < CHUNK_WORDS {
1023 assert_eq!(count_ones(&words[num_words..]) as ChunkSize, 0);
1024 }
1025 }
1026 }
1027 }
1028
1029 fn count(&self) -> usize {
1031 match *self {
1032 Zeros { .. } => 0,
1033 Ones { chunk_domain_size } => chunk_domain_size as usize,
1034 Mixed { ones_count, .. } => usize::from(ones_count),
1035 }
1036 }
1037}
1038
1039enum ChunkIter<'a> {
1040 Zeros,
1041 Ones(Range<usize>),
1042 Mixed(BitIter<'a, usize>),
1043 Finished,
1044}
1045
1046impl<T: Idx> fmt::Debug for ChunkedBitSet<T> {
1047 fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
1048 w.debug_list().entries(self.iter()).finish()
1049 }
1050}
1051
1052#[inline]
1065fn bitwise<Op>(out_vec: &mut [Word], in_vec: &[Word], op: Op) -> bool
1066where
1067 Op: Fn(Word, Word) -> Word,
1068{
1069 match (&out_vec.len(), &in_vec.len()) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(out_vec.len(), in_vec.len());
1070 let mut changed = 0;
1071 for (out_elem, in_elem) in iter::zip(out_vec, in_vec) {
1072 let old_val = *out_elem;
1073 let new_val = op(old_val, *in_elem);
1074 *out_elem = new_val;
1075 changed |= old_val ^ new_val;
1080 }
1081 changed != 0
1082}
1083
1084#[inline]
1086fn bitwise_changes<Op>(out_vec: &[Word], in_vec: &[Word], op: Op) -> bool
1087where
1088 Op: Fn(Word, Word) -> Word,
1089{
1090 match (&out_vec.len(), &in_vec.len()) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(out_vec.len(), in_vec.len());
1091 for (out_elem, in_elem) in iter::zip(out_vec, in_vec) {
1092 let old_val = *out_elem;
1093 let new_val = op(old_val, *in_elem);
1094 if old_val != new_val {
1095 return true;
1096 }
1097 }
1098 false
1099}
1100
1101#[derive(#[automatically_derived]
impl<T: ::core::cmp::PartialEq> ::core::cmp::PartialEq for MixedBitSet<T> {
#[inline]
fn eq(&self, other: &MixedBitSet<T>) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(MixedBitSet::Small(__self_0), MixedBitSet::Small(__arg1_0))
=> __self_0 == __arg1_0,
(MixedBitSet::Large(__self_0), MixedBitSet::Large(__arg1_0))
=> __self_0 == __arg1_0,
_ => unsafe { ::core::intrinsics::unreachable() }
}
}
}PartialEq, #[automatically_derived]
impl<T: ::core::cmp::Eq> ::core::cmp::Eq for MixedBitSet<T> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<DenseBitSet<T>>;
let _: ::core::cmp::AssertParamIsEq<ChunkedBitSet<T>>;
}
}Eq)]
1113pub enum MixedBitSet<T> {
1114 Small(DenseBitSet<T>),
1115 Large(ChunkedBitSet<T>),
1116}
1117
1118impl<T> MixedBitSet<T> {
1119 pub fn domain_size(&self) -> usize {
1120 match self {
1121 MixedBitSet::Small(set) => set.domain_size(),
1122 MixedBitSet::Large(set) => set.domain_size(),
1123 }
1124 }
1125}
1126
1127impl<T: Idx> MixedBitSet<T> {
1128 #[inline]
1129 pub fn new_empty(domain_size: usize) -> MixedBitSet<T> {
1130 if domain_size <= CHUNK_BITS {
1131 MixedBitSet::Small(DenseBitSet::new_empty(domain_size))
1132 } else {
1133 MixedBitSet::Large(ChunkedBitSet::new_empty(domain_size))
1134 }
1135 }
1136
1137 #[inline]
1138 pub fn is_empty(&self) -> bool {
1139 match self {
1140 MixedBitSet::Small(set) => set.is_empty(),
1141 MixedBitSet::Large(set) => set.is_empty(),
1142 }
1143 }
1144
1145 #[inline]
1146 pub fn contains(&self, elem: T) -> bool {
1147 match self {
1148 MixedBitSet::Small(set) => set.contains(elem),
1149 MixedBitSet::Large(set) => set.contains(elem),
1150 }
1151 }
1152
1153 #[inline]
1154 pub fn insert(&mut self, elem: T) -> bool {
1155 match self {
1156 MixedBitSet::Small(set) => set.insert(elem),
1157 MixedBitSet::Large(set) => set.insert(elem),
1158 }
1159 }
1160
1161 pub fn insert_all(&mut self) {
1162 match self {
1163 MixedBitSet::Small(set) => set.insert_all(),
1164 MixedBitSet::Large(set) => set.insert_all(),
1165 }
1166 }
1167
1168 #[inline]
1169 pub fn remove(&mut self, elem: T) -> bool {
1170 match self {
1171 MixedBitSet::Small(set) => set.remove(elem),
1172 MixedBitSet::Large(set) => set.remove(elem),
1173 }
1174 }
1175
1176 pub fn iter(&self) -> MixedBitIter<'_, T> {
1177 match self {
1178 MixedBitSet::Small(set) => MixedBitIter::Small(set.iter()),
1179 MixedBitSet::Large(set) => MixedBitIter::Large(set.iter()),
1180 }
1181 }
1182
1183 #[inline]
1184 pub fn clear(&mut self) {
1185 match self {
1186 MixedBitSet::Small(set) => set.clear(),
1187 MixedBitSet::Large(set) => set.clear(),
1188 }
1189 }
1190
1191 self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::union(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::subtract(self, other);
Self
Rhs
&mut Self
self
&Rhs
other
bool
<Self as BitRelations<Rhs>>::intersect(self, other);bit_relations_inherent_impls! {}
1192}
1193
1194impl<T> Clone for MixedBitSet<T> {
1195 fn clone(&self) -> Self {
1196 match self {
1197 MixedBitSet::Small(set) => MixedBitSet::Small(set.clone()),
1198 MixedBitSet::Large(set) => MixedBitSet::Large(set.clone()),
1199 }
1200 }
1201
1202 fn clone_from(&mut self, from: &Self) {
1207 match (self, from) {
1208 (MixedBitSet::Small(set), MixedBitSet::Small(from)) => set.clone_from(from),
1209 (MixedBitSet::Large(set), MixedBitSet::Large(from)) => set.clone_from(from),
1210 _ => { ::core::panicking::panic_fmt(format_args!("MixedBitSet size mismatch")); }panic!("MixedBitSet size mismatch"),
1211 }
1212 }
1213}
1214
1215impl<T: Idx> BitRelations<MixedBitSet<T>> for MixedBitSet<T> {
1216 fn union(&mut self, other: &MixedBitSet<T>) -> bool {
1217 match (self, other) {
1218 (MixedBitSet::Small(set), MixedBitSet::Small(other)) => set.union(other),
1219 (MixedBitSet::Large(set), MixedBitSet::Large(other)) => set.union(other),
1220 _ => { ::core::panicking::panic_fmt(format_args!("MixedBitSet size mismatch")); }panic!("MixedBitSet size mismatch"),
1221 }
1222 }
1223
1224 fn subtract(&mut self, other: &MixedBitSet<T>) -> bool {
1225 match (self, other) {
1226 (MixedBitSet::Small(set), MixedBitSet::Small(other)) => set.subtract(other),
1227 (MixedBitSet::Large(set), MixedBitSet::Large(other)) => set.subtract(other),
1228 _ => { ::core::panicking::panic_fmt(format_args!("MixedBitSet size mismatch")); }panic!("MixedBitSet size mismatch"),
1229 }
1230 }
1231
1232 fn intersect(&mut self, _other: &MixedBitSet<T>) -> bool {
1233 {
::core::panicking::panic_fmt(format_args!("not implemented: {0}",
format_args!("implement if/when necessary")));
};unimplemented!("implement if/when necessary");
1234 }
1235}
1236
1237impl<T: Idx> fmt::Debug for MixedBitSet<T> {
1238 fn fmt(&self, w: &mut fmt::Formatter<'_>) -> fmt::Result {
1239 match self {
1240 MixedBitSet::Small(set) => set.fmt(w),
1241 MixedBitSet::Large(set) => set.fmt(w),
1242 }
1243 }
1244}
1245
1246pub enum MixedBitIter<'a, T: Idx> {
1247 Small(BitIter<'a, T>),
1248 Large(ChunkedBitIter<'a, T>),
1249}
1250
1251impl<'a, T: Idx> Iterator for MixedBitIter<'a, T> {
1252 type Item = T;
1253 fn next(&mut self) -> Option<T> {
1254 match self {
1255 MixedBitIter::Small(iter) => iter.next(),
1256 MixedBitIter::Large(iter) => iter.next(),
1257 }
1258 }
1259}
1260
1261#[derive(#[automatically_derived]
impl<T: ::core::clone::Clone + Idx> ::core::clone::Clone for GrowableBitSet<T>
{
#[inline]
fn clone(&self) -> GrowableBitSet<T> {
GrowableBitSet { bit_set: ::core::clone::Clone::clone(&self.bit_set) }
}
}Clone, #[automatically_derived]
impl<T: ::core::fmt::Debug + Idx> ::core::fmt::Debug for GrowableBitSet<T> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field1_finish(f,
"GrowableBitSet", "bit_set", &&self.bit_set)
}
}Debug, #[automatically_derived]
impl<T: ::core::cmp::PartialEq + Idx> ::core::cmp::PartialEq for
GrowableBitSet<T> {
#[inline]
fn eq(&self, other: &GrowableBitSet<T>) -> bool {
self.bit_set == other.bit_set
}
}PartialEq)]
1269pub struct GrowableBitSet<T: Idx> {
1270 bit_set: DenseBitSet<T>,
1271}
1272
1273impl<T: Idx> Default for GrowableBitSet<T> {
1274 fn default() -> Self {
1275 GrowableBitSet::new_empty()
1276 }
1277}
1278
1279impl<T: Idx> GrowableBitSet<T> {
1280 pub fn ensure(&mut self, min_domain_size: usize) {
1282 if self.bit_set.domain_size < min_domain_size {
1283 self.bit_set.domain_size = min_domain_size;
1284 }
1285
1286 let min_num_words = num_words(min_domain_size);
1287 if self.bit_set.words.len() < min_num_words {
1288 self.bit_set.words.resize(min_num_words, 0)
1289 }
1290 }
1291
1292 pub fn new_empty() -> GrowableBitSet<T> {
1293 GrowableBitSet { bit_set: DenseBitSet::new_empty(0) }
1294 }
1295
1296 pub fn with_capacity(capacity: usize) -> GrowableBitSet<T> {
1297 GrowableBitSet { bit_set: DenseBitSet::new_empty(capacity) }
1298 }
1299
1300 #[inline]
1302 pub fn insert(&mut self, elem: T) -> bool {
1303 self.ensure(elem.index() + 1);
1304 self.bit_set.insert(elem)
1305 }
1306
1307 #[inline]
1308 pub fn insert_range(&mut self, elems: Range<T>) {
1309 self.ensure(elems.end.index());
1310 self.bit_set.insert_range(elems);
1311 }
1312
1313 #[inline]
1315 pub fn remove(&mut self, elem: T) -> bool {
1316 self.ensure(elem.index() + 1);
1317 self.bit_set.remove(elem)
1318 }
1319
1320 #[inline]
1321 pub fn clear(&mut self) {
1322 self.bit_set.clear();
1323 }
1324
1325 #[inline]
1326 pub fn count(&self) -> usize {
1327 self.bit_set.count()
1328 }
1329
1330 #[inline]
1331 pub fn is_empty(&self) -> bool {
1332 self.bit_set.is_empty()
1333 }
1334
1335 #[inline]
1336 pub fn contains(&self, elem: T) -> bool {
1337 let (word_index, mask) = word_index_and_mask(elem);
1338 self.bit_set.words.get(word_index).is_some_and(|word| (word & mask) != 0)
1339 }
1340
1341 #[inline]
1342 pub fn contains_any(&self, elems: Range<T>) -> bool {
1343 elems.start.index() < self.bit_set.domain_size
1344 && self
1345 .bit_set
1346 .contains_any(elems.start..T::new(elems.end.index().min(self.bit_set.domain_size)))
1347 }
1348
1349 #[inline]
1350 pub fn iter(&self) -> BitIter<'_, T> {
1351 self.bit_set.iter()
1352 }
1353
1354 #[inline]
1355 pub fn len(&self) -> usize {
1356 self.bit_set.count()
1357 }
1358}
1359
1360impl<T: Idx> From<DenseBitSet<T>> for GrowableBitSet<T> {
1361 fn from(bit_set: DenseBitSet<T>) -> Self {
1362 Self { bit_set }
1363 }
1364}
1365
1366#[cfg_attr(feature = "nightly", derive(const _: () =
{
impl<R: Idx, C: Idx, __D: ::rustc_serialize::Decoder>
::rustc_serialize::Decodable<__D> for BitMatrix<R, C> where
PhantomData<(R, C)>: ::rustc_serialize::Decodable<__D> {
fn decode(__decoder: &mut __D) -> Self {
BitMatrix {
num_rows: ::rustc_serialize::Decodable::decode(__decoder),
num_columns: ::rustc_serialize::Decodable::decode(__decoder),
words: ::rustc_serialize::Decodable::decode(__decoder),
marker: ::rustc_serialize::Decodable::decode(__decoder),
}
}
}
};Decodable_NoContext, const _: () =
{
impl<R: Idx, C: Idx, __E: ::rustc_serialize::Encoder>
::rustc_serialize::Encodable<__E> for BitMatrix<R, C> where
PhantomData<(R, C)>: ::rustc_serialize::Encodable<__E> {
fn encode(&self, __encoder: &mut __E) {
match *self {
BitMatrix {
num_rows: ref __binding_0,
num_columns: ref __binding_1,
words: ref __binding_2,
marker: ref __binding_3 } => {
::rustc_serialize::Encodable::<__E>::encode(__binding_0,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_1,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_2,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_3,
__encoder);
}
}
}
}
};Encodable_NoContext))]
1374#[derive(#[automatically_derived]
impl<R: ::core::clone::Clone + Idx, C: ::core::clone::Clone + Idx>
::core::clone::Clone for BitMatrix<R, C> {
#[inline]
fn clone(&self) -> BitMatrix<R, C> {
BitMatrix {
num_rows: ::core::clone::Clone::clone(&self.num_rows),
num_columns: ::core::clone::Clone::clone(&self.num_columns),
words: ::core::clone::Clone::clone(&self.words),
marker: ::core::clone::Clone::clone(&self.marker),
}
}
}Clone, #[automatically_derived]
impl<R: ::core::cmp::Eq + Idx, C: ::core::cmp::Eq + Idx> ::core::cmp::Eq for
BitMatrix<R, C> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<usize>;
let _: ::core::cmp::AssertParamIsEq<Vec<Word>>;
let _: ::core::cmp::AssertParamIsEq<PhantomData<(R, C)>>;
}
}Eq, #[automatically_derived]
impl<R: ::core::cmp::PartialEq + Idx, C: ::core::cmp::PartialEq + Idx>
::core::cmp::PartialEq for BitMatrix<R, C> {
#[inline]
fn eq(&self, other: &BitMatrix<R, C>) -> bool {
self.num_rows == other.num_rows &&
self.num_columns == other.num_columns &&
self.words == other.words && self.marker == other.marker
}
}PartialEq, #[automatically_derived]
impl<R: ::core::hash::Hash + Idx, C: ::core::hash::Hash + Idx>
::core::hash::Hash for BitMatrix<R, C> {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&self.num_rows, state);
::core::hash::Hash::hash(&self.num_columns, state);
::core::hash::Hash::hash(&self.words, state);
::core::hash::Hash::hash(&self.marker, state)
}
}Hash)]
1375pub struct BitMatrix<R: Idx, C: Idx> {
1376 num_rows: usize,
1377 num_columns: usize,
1378 words: Vec<Word>,
1379 marker: PhantomData<(R, C)>,
1380}
1381
1382impl<R: Idx, C: Idx> BitMatrix<R, C> {
1383 pub fn new(num_rows: usize, num_columns: usize) -> BitMatrix<R, C> {
1385 let words_per_row = num_words(num_columns);
1388 BitMatrix {
1389 num_rows,
1390 num_columns,
1391 words: ::alloc::vec::from_elem(0, num_rows * words_per_row)vec![0; num_rows * words_per_row],
1392 marker: PhantomData,
1393 }
1394 }
1395
1396 pub fn from_row_n(row: &DenseBitSet<C>, num_rows: usize) -> BitMatrix<R, C> {
1398 let num_columns = row.domain_size();
1399 let words_per_row = num_words(num_columns);
1400 match (&words_per_row, &row.words.len()) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(words_per_row, row.words.len());
1401 BitMatrix {
1402 num_rows,
1403 num_columns,
1404 words: iter::repeat_n(&row.words, num_rows).flatten().cloned().collect(),
1405 marker: PhantomData,
1406 }
1407 }
1408
1409 pub fn rows(&self) -> impl Iterator<Item = R> {
1410 (0..self.num_rows).map(R::new)
1411 }
1412
1413 fn range(&self, row: R) -> (usize, usize) {
1415 let words_per_row = num_words(self.num_columns);
1416 let start = row.index() * words_per_row;
1417 (start, start + words_per_row)
1418 }
1419
1420 pub fn insert(&mut self, row: R, column: C) -> bool {
1425 if !(row.index() < self.num_rows && column.index() < self.num_columns) {
::core::panicking::panic("assertion failed: row.index() < self.num_rows && column.index() < self.num_columns")
};assert!(row.index() < self.num_rows && column.index() < self.num_columns);
1426 let (start, _) = self.range(row);
1427 let (word_index, mask) = word_index_and_mask(column);
1428 let words = &mut self.words[..];
1429 let word = words[start + word_index];
1430 let new_word = word | mask;
1431 words[start + word_index] = new_word;
1432 word != new_word
1433 }
1434
1435 pub fn contains(&self, row: R, column: C) -> bool {
1440 if !(row.index() < self.num_rows && column.index() < self.num_columns) {
::core::panicking::panic("assertion failed: row.index() < self.num_rows && column.index() < self.num_columns")
};assert!(row.index() < self.num_rows && column.index() < self.num_columns);
1441 let (start, _) = self.range(row);
1442 let (word_index, mask) = word_index_and_mask(column);
1443 (self.words[start + word_index] & mask) != 0
1444 }
1445
1446 pub fn intersect_rows(&self, row1: R, row2: R) -> Vec<C> {
1451 if !(row1.index() < self.num_rows && row2.index() < self.num_rows) {
::core::panicking::panic("assertion failed: row1.index() < self.num_rows && row2.index() < self.num_rows")
};assert!(row1.index() < self.num_rows && row2.index() < self.num_rows);
1452 let (row1_start, row1_end) = self.range(row1);
1453 let (row2_start, row2_end) = self.range(row2);
1454 let mut result = Vec::with_capacity(self.num_columns);
1455 for (base, (i, j)) in (row1_start..row1_end).zip(row2_start..row2_end).enumerate() {
1456 let mut v = self.words[i] & self.words[j];
1457 for bit in 0..WORD_BITS {
1458 if v == 0 {
1459 break;
1460 }
1461 if v & 0x1 != 0 {
1462 result.push(C::new(base * WORD_BITS + bit));
1463 }
1464 v >>= 1;
1465 }
1466 }
1467 result
1468 }
1469
1470 pub fn union_rows(&mut self, read: R, write: R) -> bool {
1478 if !(read.index() < self.num_rows && write.index() < self.num_rows) {
::core::panicking::panic("assertion failed: read.index() < self.num_rows && write.index() < self.num_rows")
};assert!(read.index() < self.num_rows && write.index() < self.num_rows);
1479 let (read_start, read_end) = self.range(read);
1480 let (write_start, write_end) = self.range(write);
1481 let words = &mut self.words[..];
1482 let mut changed = 0;
1483 for (read_index, write_index) in iter::zip(read_start..read_end, write_start..write_end) {
1484 let word = words[write_index];
1485 let new_word = word | words[read_index];
1486 words[write_index] = new_word;
1487 changed |= word ^ new_word;
1489 }
1490 changed != 0
1491 }
1492
1493 pub fn union_row_with(&mut self, with: &DenseBitSet<C>, write: R) -> bool {
1496 if !(write.index() < self.num_rows) {
::core::panicking::panic("assertion failed: write.index() < self.num_rows")
};assert!(write.index() < self.num_rows);
1497 match (&with.domain_size(), &self.num_columns) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(with.domain_size(), self.num_columns);
1498 let (write_start, write_end) = self.range(write);
1499 bitwise(&mut self.words[write_start..write_end], &with.words, |a, b| a | b)
1500 }
1501
1502 pub fn insert_all_into_row(&mut self, row: R) {
1504 if !(row.index() < self.num_rows) {
::core::panicking::panic("assertion failed: row.index() < self.num_rows")
};assert!(row.index() < self.num_rows);
1505 let (start, end) = self.range(row);
1506 let words = &mut self.words[..];
1507 for index in start..end {
1508 words[index] = !0;
1509 }
1510 clear_excess_bits_in_final_word(self.num_columns, &mut self.words[..end]);
1511 }
1512
1513 pub fn words(&self) -> &[Word] {
1515 &self.words
1516 }
1517
1518 pub fn iter(&self, row: R) -> BitIter<'_, C> {
1521 if !(row.index() < self.num_rows) {
::core::panicking::panic("assertion failed: row.index() < self.num_rows")
};assert!(row.index() < self.num_rows);
1522 let (start, end) = self.range(row);
1523 BitIter::new(&self.words[start..end])
1524 }
1525
1526 pub fn count(&self, row: R) -> usize {
1528 let (start, end) = self.range(row);
1529 count_ones(&self.words[start..end])
1530 }
1531}
1532
1533impl<R: Idx, C: Idx> fmt::Debug for BitMatrix<R, C> {
1534 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
1535 struct OneLinePrinter<T>(T);
1537 impl<T: fmt::Debug> fmt::Debug for OneLinePrinter<T> {
1538 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
1539 fmt.write_fmt(format_args!("{0:?}", self.0))write!(fmt, "{:?}", self.0)
1540 }
1541 }
1542
1543 fmt.write_fmt(format_args!("BitMatrix({0}x{1}) ", self.num_rows,
self.num_columns))write!(fmt, "BitMatrix({}x{}) ", self.num_rows, self.num_columns)?;
1544 let items = self.rows().flat_map(|r| self.iter(r).map(move |c| (r, c)));
1545 fmt.debug_set().entries(items.map(OneLinePrinter)).finish()
1546 }
1547}
1548
1549#[derive(#[automatically_derived]
impl<R: ::core::clone::Clone, C: ::core::clone::Clone> ::core::clone::Clone
for SparseBitMatrix<R, C> where R: Idx, C: Idx {
#[inline]
fn clone(&self) -> SparseBitMatrix<R, C> {
SparseBitMatrix {
num_columns: ::core::clone::Clone::clone(&self.num_columns),
rows: ::core::clone::Clone::clone(&self.rows),
}
}
}Clone, #[automatically_derived]
impl<R: ::core::fmt::Debug, C: ::core::fmt::Debug> ::core::fmt::Debug for
SparseBitMatrix<R, C> where R: Idx, C: Idx {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field2_finish(f,
"SparseBitMatrix", "num_columns", &self.num_columns, "rows",
&&self.rows)
}
}Debug)]
1561pub struct SparseBitMatrix<R, C>
1562where
1563 R: Idx,
1564 C: Idx,
1565{
1566 num_columns: usize,
1567 rows: IndexVec<R, Option<DenseBitSet<C>>>,
1568}
1569
1570impl<R: Idx, C: Idx> SparseBitMatrix<R, C> {
1571 pub fn new(num_columns: usize) -> Self {
1573 Self { num_columns, rows: IndexVec::new() }
1574 }
1575
1576 fn ensure_row(&mut self, row: R) -> &mut DenseBitSet<C> {
1577 self.rows.get_or_insert_with(row, || DenseBitSet::new_empty(self.num_columns))
1580 }
1581
1582 pub fn insert(&mut self, row: R, column: C) -> bool {
1587 self.ensure_row(row).insert(column)
1588 }
1589
1590 pub fn remove(&mut self, row: R, column: C) -> bool {
1596 match self.rows.get_mut(row) {
1597 Some(Some(row)) => row.remove(column),
1598 _ => false,
1599 }
1600 }
1601
1602 pub fn clear(&mut self, row: R) {
1605 if let Some(Some(row)) = self.rows.get_mut(row) {
1606 row.clear();
1607 }
1608 }
1609
1610 pub fn contains(&self, row: R, column: C) -> bool {
1615 self.row(row).is_some_and(|r| r.contains(column))
1616 }
1617
1618 pub fn union_rows(&mut self, read: R, write: R) -> bool {
1626 if read == write || self.row(read).is_none() {
1627 return false;
1628 }
1629
1630 self.ensure_row(write);
1631 if let (Some(read_row), Some(write_row)) = self.rows.pick2_mut(read, write) {
1632 write_row.union(read_row)
1633 } else {
1634 ::core::panicking::panic("internal error: entered unreachable code")unreachable!()
1635 }
1636 }
1637
1638 pub fn insert_all_into_row(&mut self, row: R) {
1640 self.ensure_row(row).insert_all();
1641 }
1642
1643 pub fn rows(&self) -> impl Iterator<Item = R> {
1644 self.rows.indices()
1645 }
1646
1647 pub fn iter(&self, row: R) -> impl Iterator<Item = C> {
1650 self.row(row).into_iter().flat_map(|r| r.iter())
1651 }
1652
1653 pub fn row(&self, row: R) -> Option<&DenseBitSet<C>> {
1654 self.rows.get(row)?.as_ref()
1655 }
1656
1657 pub fn intersect_row<Set>(&mut self, row: R, set: &Set) -> bool
1662 where
1663 DenseBitSet<C>: BitRelations<Set>,
1664 {
1665 match self.rows.get_mut(row) {
1666 Some(Some(row)) => row.intersect(set),
1667 _ => false,
1668 }
1669 }
1670
1671 pub fn subtract_row<Set>(&mut self, row: R, set: &Set) -> bool
1676 where
1677 DenseBitSet<C>: BitRelations<Set>,
1678 {
1679 match self.rows.get_mut(row) {
1680 Some(Some(row)) => row.subtract(set),
1681 _ => false,
1682 }
1683 }
1684
1685 pub fn union_row<Set>(&mut self, row: R, set: &Set) -> bool
1690 where
1691 DenseBitSet<C>: BitRelations<Set>,
1692 {
1693 self.ensure_row(row).union(set)
1694 }
1695}
1696
1697#[inline]
1698fn num_words<T: Idx>(domain_size: T) -> usize {
1699 domain_size.index().div_ceil(WORD_BITS)
1700}
1701
1702#[inline]
1703fn word_index_and_mask<T: Idx>(elem: T) -> (usize, Word) {
1704 let elem = elem.index();
1705 let word_index = elem / WORD_BITS;
1706 let mask = 1 << (elem % WORD_BITS);
1707 (word_index, mask)
1708}
1709
1710#[inline]
1711fn chunk_index<T: Idx>(elem: T) -> usize {
1712 elem.index() / CHUNK_BITS
1713}
1714
1715#[inline]
1716fn chunk_word_index_and_mask<T: Idx>(elem: T) -> (usize, Word) {
1717 let chunk_elem = elem.index() % CHUNK_BITS;
1718 word_index_and_mask(chunk_elem)
1719}
1720
1721fn clear_excess_bits_in_final_word(domain_size: usize, words: &mut [Word]) {
1722 let num_bits_in_final_word = domain_size % WORD_BITS;
1723 if num_bits_in_final_word > 0 {
1724 let mask = (1 << num_bits_in_final_word) - 1;
1725 words[words.len() - 1] &= mask;
1726 }
1727}
1728
1729#[inline]
1730fn max_bit(word: Word) -> usize {
1731 WORD_BITS - 1 - word.leading_zeros() as usize
1732}
1733
1734#[inline]
1735fn count_ones(words: &[Word]) -> usize {
1736 words.iter().map(|word| word.count_ones() as usize).sum()
1737}