use crate::private::layout::CapTable;
use crate::private::layout::{
ListReader, PointerBuilder, PointerReader, StructBuilder, StructReader, StructSize,
};
use crate::Result;
use core::marker::PhantomData;
pub trait HasStructSize {
const STRUCT_SIZE: StructSize;
}
pub trait IntoInternalStructReader<'a> {
fn into_internal_struct_reader(self) -> StructReader<'a>;
}
pub trait IntoInternalListReader<'a> {
fn into_internal_list_reader(self) -> ListReader<'a>;
}
pub trait FromPointerReader<'a>: Sized {
fn get_from_pointer(
reader: &PointerReader<'a>,
default: Option<&'a [crate::Word]>,
) -> Result<Self>;
}
pub trait Owned: crate::introspect::Introspect {
type Reader<'a>: FromPointerReader<'a> + SetterInput<Self>;
type Builder<'a>: FromPointerBuilder<'a>;
}
pub trait OwnedStruct: crate::introspect::Introspect {
type Reader<'a>: From<StructReader<'a>> + SetterInput<Self> + IntoInternalStructReader<'a>;
type Builder<'a>: From<StructBuilder<'a>> + HasStructSize;
}
pub trait Pipelined {
type Pipeline;
}
pub trait FromPointerBuilder<'a>: Sized {
fn init_pointer(builder: PointerBuilder<'a>, length: u32) -> Self;
fn get_from_pointer(
builder: PointerBuilder<'a>,
default: Option<&'a [crate::Word]>,
) -> Result<Self>;
}
pub trait SetterInput<Receiver: ?Sized> {
fn set_pointer_builder(
builder: PointerBuilder<'_>,
input: Self,
canonicalize: bool,
) -> Result<()>;
}
pub trait Imbue<'a> {
fn imbue(&mut self, caps: &'a CapTable);
}
pub trait ImbueMut<'a> {
fn imbue_mut(&mut self, caps: &'a mut CapTable);
}
pub trait HasTypeId {
const TYPE_ID: u64;
}
pub trait IndexMove<I, T> {
fn index_move(&self, index: I) -> T;
}
pub struct ListIter<T, U> {
marker: PhantomData<U>,
list: T,
index: u32,
size: u32,
}
impl<T, U> ListIter<T, U> {
pub fn new(list: T, size: u32) -> Self {
Self {
list,
index: 0,
size,
marker: PhantomData,
}
}
}
impl<U, T: IndexMove<u32, U>> ::core::iter::Iterator for ListIter<T, U> {
type Item = U;
fn next(&mut self) -> ::core::option::Option<U> {
if self.index < self.size {
let result = self.list.index_move(self.index);
self.index += 1;
Some(result)
} else {
None
}
}
fn size_hint(&self) -> (usize, Option<usize>) {
let remaining = self.size as usize - self.index as usize;
(remaining, Some(remaining))
}
fn nth(&mut self, p: usize) -> Option<U> {
let Some(p) = p.try_into().ok() else {
self.index = self.size;
return None;
};
let Some(nth_index) = self.index.checked_add(p) else {
self.index = self.size;
return None;
};
if nth_index < self.size {
self.index = nth_index;
let result = self.list.index_move(self.index);
self.index += 1;
Some(result)
} else {
self.index = self.size;
None
}
}
}
impl<U, T: IndexMove<u32, U>> ::core::iter::ExactSizeIterator for ListIter<T, U> {
fn len(&self) -> usize {
self.size as usize - self.index as usize
}
}
impl<U, T: IndexMove<u32, U>> ::core::iter::DoubleEndedIterator for ListIter<T, U> {
fn next_back(&mut self) -> ::core::option::Option<U> {
if self.size > self.index {
self.size -= 1;
Some(self.list.index_move(self.size))
} else {
None
}
}
}
pub struct ShortListIter<T, U> {
marker: PhantomData<U>,
list: T,
index: u16,
size: u16,
}
impl<T, U> ShortListIter<T, U> {
pub fn new(list: T, size: u16) -> Self {
Self {
list,
index: 0,
size,
marker: PhantomData,
}
}
}
impl<U, T: IndexMove<u16, U>> ::core::iter::Iterator for ShortListIter<T, U> {
type Item = U;
fn next(&mut self) -> ::core::option::Option<U> {
if self.index < self.size {
let result = self.list.index_move(self.index);
self.index += 1;
Some(result)
} else {
None
}
}
fn size_hint(&self) -> (usize, Option<usize>) {
let remaining = self.size as usize - self.index as usize;
(remaining, Some(remaining))
}
fn nth(&mut self, p: usize) -> Option<U> {
let Some(p) = p.try_into().ok() else {
self.index = self.size;
return None;
};
let Some(nth_index) = self.index.checked_add(p) else {
self.index = self.size;
return None;
};
if nth_index < self.size {
self.index = nth_index;
let result = self.list.index_move(self.index);
self.index += 1;
Some(result)
} else {
self.index = self.size;
None
}
}
}
impl<U, T: IndexMove<u16, U>> ::core::iter::ExactSizeIterator for ShortListIter<T, U> {
fn len(&self) -> usize {
self.size as usize - self.index as usize
}
}
impl<U, T: IndexMove<u16, U>> ::core::iter::DoubleEndedIterator for ShortListIter<T, U> {
fn next_back(&mut self) -> ::core::option::Option<U> {
if self.size > self.index {
self.size -= 1;
Some(self.list.index_move(self.size))
} else {
None
}
}
}