use super::Cursor;
#[derive(Debug)]
pub struct CursorList<C> {
cursors: Vec<C>,
min_key: Vec<usize>,
min_val: Vec<usize>,
}
impl<C: Cursor> CursorList<C> {
pub fn new(cursors: Vec<C>, storage: &[C::Storage]) -> Self {
let mut result = CursorList {
cursors,
min_key: Vec::new(),
min_val: Vec::new(),
};
result.minimize_keys(storage);
result
}
fn minimize_keys(&mut self, storage: &[C::Storage]) {
self.min_key.clear();
let mut iter = self.cursors.iter().enumerate().flat_map(|(idx, cur)| cur.get_key(&storage[idx]).map(|key| (idx, key)));
if let Some((idx, key)) = iter.next() {
let mut min_key = key;
self.min_key.push(idx);
for (idx, key) in iter {
match key.cmp(&min_key) {
std::cmp::Ordering::Less => {
self.min_key.clear();
self.min_key.push(idx);
min_key = key;
}
std::cmp::Ordering::Equal => {
self.min_key.push(idx);
}
std::cmp::Ordering::Greater => { }
}
}
}
self.minimize_vals(storage);
}
fn minimize_vals(&mut self, storage: &[C::Storage]) {
self.min_val.clear();
let mut iter = self.min_key.iter().cloned().flat_map(|idx| self.cursors[idx].get_val(&storage[idx]).map(|val| (idx, val)));
if let Some((idx, val)) = iter.next() {
let mut min_val = val;
self.min_val.push(idx);
for (idx, val) in iter {
match val.cmp(&min_val) {
std::cmp::Ordering::Less => {
self.min_val.clear();
self.min_val.push(idx);
min_val = val;
}
std::cmp::Ordering::Equal => {
self.min_val.push(idx);
}
std::cmp::Ordering::Greater => { }
}
}
}
}
}
use crate::trace::implementations::WithLayout;
impl<C: Cursor> WithLayout for CursorList<C> {
type Layout = C::Layout;
}
impl<C: Cursor> Cursor for CursorList<C> {
type Storage = Vec<C::Storage>;
#[inline]
fn key_valid(&self, _storage: &Vec<C::Storage>) -> bool { !self.min_key.is_empty() }
#[inline]
fn val_valid(&self, _storage: &Vec<C::Storage>) -> bool { !self.min_val.is_empty() }
#[inline]
fn key<'a>(&self, storage: &'a Vec<C::Storage>) -> Self::Key<'a> {
debug_assert!(self.key_valid(storage));
debug_assert!(self.cursors[self.min_key[0]].key_valid(&storage[self.min_key[0]]));
self.cursors[self.min_key[0]].key(&storage[self.min_key[0]])
}
#[inline]
fn val<'a>(&self, storage: &'a Vec<C::Storage>) -> Self::Val<'a> {
debug_assert!(self.key_valid(storage));
debug_assert!(self.val_valid(storage));
debug_assert!(self.cursors[self.min_val[0]].val_valid(&storage[self.min_val[0]]));
self.cursors[self.min_val[0]].val(&storage[self.min_val[0]])
}
#[inline]
fn get_key<'a>(&self, storage: &'a Vec<C::Storage>) -> Option<Self::Key<'a>> {
self.min_key.get(0).map(|idx| self.cursors[*idx].key(&storage[*idx]))
}
#[inline]
fn get_val<'a>(&self, storage: &'a Vec<C::Storage>) -> Option<Self::Val<'a>> {
self.min_val.get(0).map(|idx| self.cursors[*idx].val(&storage[*idx]))
}
#[inline]
fn map_times<L: FnMut(Self::TimeGat<'_>, Self::DiffGat<'_>)>(&mut self, storage: &Vec<C::Storage>, mut logic: L) {
for &index in self.min_val.iter() {
self.cursors[index].map_times(&storage[index], |t,d| logic(t,d));
}
}
#[inline]
fn step_key(&mut self, storage: &Vec<C::Storage>) {
for &index in self.min_key.iter() {
self.cursors[index].step_key(&storage[index]);
}
self.minimize_keys(storage);
}
#[inline]
fn seek_key(&mut self, storage: &Vec<C::Storage>, key: Self::Key<'_>) {
for (cursor, storage) in self.cursors.iter_mut().zip(storage) {
cursor.seek_key(storage, key);
}
self.minimize_keys(storage);
}
#[inline]
fn step_val(&mut self, storage: &Vec<C::Storage>) {
for &index in self.min_val.iter() {
self.cursors[index].step_val(&storage[index]);
}
self.minimize_vals(storage);
}
#[inline]
fn seek_val(&mut self, storage: &Vec<C::Storage>, val: Self::Val<'_>) {
for (cursor, storage) in self.cursors.iter_mut().zip(storage) {
cursor.seek_val(storage, val);
}
self.minimize_vals(storage);
}
#[inline]
fn rewind_keys(&mut self, storage: &Vec<C::Storage>) {
for (cursor, storage) in self.cursors.iter_mut().zip(storage) {
cursor.rewind_keys(storage);
}
self.minimize_keys(storage);
}
#[inline]
fn rewind_vals(&mut self, storage: &Vec<C::Storage>) {
for &index in self.min_key.iter() {
self.cursors[index].rewind_vals(&storage[index]);
}
self.minimize_vals(storage);
}
}