use std::fmt::{self, Write};
use std::ops::Range;
use tinyvec::TinyVec;
#[derive(Default, PartialEq, Eq)]
pub(crate) struct ArrayRangeSet(TinyVec<[Range<u64>; ARRAY_RANGE_SET_INLINE_CAPACITY]>);
impl fmt::Debug for ArrayRangeSet {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_char('[')?;
let mut first = true;
for range in self.0.iter() {
if !first {
f.write_char(',')?;
}
write!(f, "{range:?}")?;
first = false;
}
f.write_char(']')?;
Ok(())
}
}
const ARRAY_RANGE_SET_INLINE_CAPACITY: usize = 2;
impl Clone for ArrayRangeSet {
fn clone(&self) -> Self {
if self.0.is_inline() || self.0.len() > ARRAY_RANGE_SET_INLINE_CAPACITY {
return Self(self.0.clone());
}
let mut vec = TinyVec::new();
vec.extend_from_slice(self.0.as_slice());
Self(vec)
}
}
impl ArrayRangeSet {
pub(crate) fn new() -> Self {
Default::default()
}
pub(crate) fn iter(&self) -> impl DoubleEndedIterator<Item = Range<u64>> + '_ {
self.0.iter().cloned()
}
pub(crate) fn elts(&self) -> impl Iterator<Item = u64> + '_ {
self.iter().flatten()
}
pub(crate) fn len(&self) -> usize {
self.0.len()
}
#[cfg(test)]
pub(crate) fn contains(&self, x: u64) -> bool {
for range in self.0.iter() {
if range.start > x {
return false;
} else if range.contains(&x) {
return true;
}
}
false
}
pub(crate) fn iter_range(&self, range: Range<u64>) -> impl Iterator<Item = Range<u64>> + '_ {
self.iter().filter_map(move |r| {
if r.end > range.start && r.start < range.end {
Some(r.start.max(range.start)..r.end.min(range.end))
} else {
None
}
})
}
pub(crate) fn insert_one(&mut self, x: u64) -> bool {
self.insert(x..x + 1)
}
pub(crate) fn insert(&mut self, x: Range<u64>) -> bool {
let mut result = false;
if x.is_empty() {
return false;
}
let mut idx = 0;
while idx != self.0.len() {
let range = &mut self.0[idx];
if range.start > x.end {
self.0.insert(idx, x);
return true;
} else if range.start > x.start {
result = true;
range.start = x.start;
}
if x.end <= range.end {
return result;
} else if x.start <= range.end {
range.end = x.end;
while idx != self.0.len() - 1 {
let curr = self.0[idx].clone();
let next = self.0[idx + 1].clone();
if curr.end >= next.start {
self.0[idx].end = next.end.max(curr.end);
self.0.remove(idx + 1);
} else {
break;
}
}
return true;
}
idx += 1;
}
self.0.push(x);
true
}
pub(crate) fn remove(&mut self, x: Range<u64>) -> bool {
let mut result = false;
if x.is_empty() {
return false;
}
let mut idx = 0;
while idx != self.0.len() && x.start != x.end {
let range = self.0[idx].clone();
if x.end <= range.start {
return result;
} else if x.start >= range.end {
idx += 1;
continue;
}
result = true;
let left = range.start..x.start;
let right = x.end..range.end;
if left.is_empty() && right.is_empty() {
self.0.remove(idx);
} else if left.is_empty() {
self.0[idx] = right;
idx += 1;
} else if right.is_empty() {
self.0[idx] = left;
idx += 1;
} else {
self.0[idx] = right;
self.0.insert(idx, left);
idx += 2;
}
}
result
}
pub(crate) fn is_empty(&self) -> bool {
self.0.is_empty()
}
pub(crate) fn pop_min(&mut self) -> Option<Range<u64>> {
if !self.0.is_empty() {
Some(self.0.remove(0))
} else {
None
}
}
pub(crate) fn min(&self) -> Option<u64> {
self.iter().next().map(|x| x.start)
}
pub(crate) fn max(&self) -> Option<u64> {
self.iter().next_back().map(|x| x.end - 1)
}
}
#[cfg(test)]
impl proptest::arbitrary::Arbitrary for ArrayRangeSet {
type Parameters = ();
type Strategy = proptest::strategy::BoxedStrategy<Self>;
fn arbitrary_with(_: Self::Parameters) -> Self::Strategy {
use proptest::prelude::*;
prop::collection::vec((1u64..100, 1u64..50), 1..8)
.prop_map(|gaps_and_sizes| {
let mut ranges = Self::new();
let mut pos = 0u64;
for (gap, size) in gaps_and_sizes {
let start = pos + gap;
let end = start + size;
ranges.insert(start..end);
pos = end;
}
ranges
})
.boxed()
}
}