Files
RedBear-OS/src/iter.rs
T
Red Bear OS 1b3e94a20d Red Bear OS relibc baseline
From release 0.1.0 pre-patched archive.
This includes all Red Bear modifications previously maintained
as patches in local/patches/relibc/.
2026-06-27 09:19:26 +03:00

207 lines
7.3 KiB
Rust

//! Utilities to help use Rust iterators on C strings.
use core::{
iter::{FusedIterator, Iterator},
marker::PhantomData,
mem::MaybeUninit,
ptr::NonNull,
};
use crate::platform::types::*;
/// A minimal alternative to the `Zero` trait from num-traits, for use in
/// `NulTerminated`.
///
/// May be replaced with the one from num-traits at a later time if so
/// desired.
pub unsafe trait Zero {
fn is_zero(&self) -> bool;
}
unsafe impl Zero for c_char {
fn is_zero(&self) -> bool {
self == &0
}
}
unsafe impl Zero for wchar_t {
fn is_zero(&self) -> bool {
self == &0
}
}
/// An iterator over a nul-terminated buffer.
///
/// This is intended to allow safe, ergonomic iteration over C-style byte and
/// wide strings without first having to read through the string and construct
/// a slice. Assuming the safety requirements are upheld when constructing the
/// iterator, it allows for string iteration in safe Rust.
pub struct NulTerminated<'a, T: Zero> {
ptr: NonNull<T>,
phantom: PhantomData<&'a T>,
}
impl<'a, T: Zero> Iterator for NulTerminated<'a, T> {
type Item = &'a T;
fn next(&mut self) -> Option<Self::Item> {
// SAFETY: the caller is required to ensure a valid pointer to a
// 0-terminated buffer is provided, and the zero-check below ensures
// that iteration and pointer increments will stop in time.
let val_ref = unsafe { self.ptr.as_ref() };
if val_ref.is_zero() {
None
} else {
// SAFETY: the caller is required to provide a 0-terminated
// buffer, and this point will only be reached if the next element
// is at most the terminating 0.
self.ptr = unsafe { self.ptr.add(1) };
Some(val_ref)
}
}
}
impl<'a, T: Zero> NulTerminated<'a, T> {
/// Constructs a new iterator, starting at `ptr`, yielding elements of
/// type `&T` up to (but not including) the terminating nul.
///
/// The iterator returns `None` after the terminating nul has been
/// encountered.
///
/// # Safety
/// The provided pointer must be a valid pointer to a buffer of contiguous
/// elements of type `T`, and the value 0 must be present within the
/// buffer at or after `ptr` (not necessarily at the end). The buffer must
/// not be written to for the lifetime of the iterator.
pub unsafe fn new(ptr: *const T) -> Option<Self> {
Some(NulTerminated {
// NonNull can only wrap only *mut pointers...
ptr: NonNull::new(ptr.cast_mut())?,
phantom: PhantomData,
})
}
}
// Once the terminating nul has been encountered, the pointer will not advance
// further and the iterator will thus keep returning None.
impl<'a, T: Zero> FusedIterator for NulTerminated<'a, T> {}
/// An iterator over a nul-terminated buffer, including the terminating nul.
///
/// Similar to [`NulTerminated`], but includes the terminating nul.
pub struct NulTerminatedInclusive<'a, T: Zero> {
ptr_opt: Option<NonNull<T>>,
phantom: PhantomData<&'a T>,
}
impl<'a, T: Zero> Iterator for NulTerminatedInclusive<'a, T> {
type Item = &'a T;
fn next(&mut self) -> Option<Self::Item> {
if let Some(old_ptr) = self.ptr_opt {
// SAFETY: the caller is required to ensure a valid pointer to a
// 0-terminated buffer is provided, and the zero-check below
// ensures that iteration and pointer increments will stop in
// time.
let val_ref = unsafe { old_ptr.as_ref() };
self.ptr_opt = if val_ref.is_zero() {
None
} else {
// SAFETY: if a terminating nul value has been encountered,
// this will not be called
Some(unsafe { old_ptr.add(1) })
};
Some(val_ref)
} else {
None
}
}
}
impl<'a, T: Zero> NulTerminatedInclusive<'a, T> {
/// Constructs a new iterator, starting at `ptr`, yielding elements of
/// type `&T` up to and including the terminating nul.
///
/// The iterator returns `None` after the terminating nul has been
/// encountered.
///
/// # Safety
/// The provided pointer must be a valid pointer to a buffer of contiguous
/// elements of type `T`, and the value 0 must be present within the
/// buffer at or after `ptr` (not necessarily at the end). The buffer must
/// not be written to for the lifetime of the iterator.
pub unsafe fn new(ptr: *const T) -> Self {
NulTerminatedInclusive {
// NonNull can only wrap only *mut pointers...
ptr_opt: NonNull::new(ptr.cast_mut()),
phantom: PhantomData,
}
}
}
// Once the terminating nul has been encountered, the internal Option will be
// set to None, ensuring that we will keep returning None.
impl<'a, T: Zero> FusedIterator for NulTerminatedInclusive<'a, T> {}
/// A zipped iterator mapping an input iterator to an "out" pointer.
///
/// This is intended to allow safe, iterative writing to an "out pointer".
/// Special care needs to be taken to avoid creating references past the end
/// of the output buffer, thus the output is zipped with an "input" iterator
/// to ensure up-front control of the range of memory on which we create
/// references.
pub struct SrcDstPtrIter<'a, I: Iterator, U: Copy> {
src_iter: I,
dst_ptr: *mut U,
phantom: PhantomData<&'a mut U>,
}
impl<'a, I: Iterator, U: Copy> Iterator for SrcDstPtrIter<'a, I, U> {
type Item = (I::Item, &'a mut MaybeUninit<U>);
fn next(&mut self) -> Option<Self::Item> {
if let Some(src_item) = self.src_iter.next() {
let old_dst_ptr = self.dst_ptr;
// SAFETY: due to the caller requirements on `I` upon
// construction, the new pointer here may be either valid to turn
// into a reference or "one past the end". The latter is okay as
// long as it is only represented as a raw pointer.
self.dst_ptr = unsafe { self.dst_ptr.add(1) };
// SAFETY: self.dst_ptr may point "one past the end", but the
// caller is required upon construction to ensure that `I` does
// not over-iterate, and thus old_dst_ptr is always okay to
// dereference.
let out_mut_ref = unsafe { old_dst_ptr.as_uninit_mut() }.unwrap();
Some((src_item, out_mut_ref))
} else {
None
}
}
}
impl<'a, I: Iterator, U: Copy> SrcDstPtrIter<'a, I, U> {
/// Constructs a new iterator of "zipped" input and output.
///
/// The caller must provide an "input" iterator `I` and an "out pointer"
/// `ptr`. Assuming `I` has item type `T`, the new iterator will have
/// `type Item = (T, &mut MaybeUninit<U>)`.
///
/// # Safety
/// `ptr` must be a valid pointer to a writable buffer of contiguous (but
/// possibly uninitialized) elements of type `U`. The caller must ensure
/// that `I` does not return `Some` any more times than there are elements
/// in the output buffer. The caller must ensure that the iterator has
/// exclusive access to that buffer for the entire lifetime of the
/// iterator.
pub unsafe fn new(iter: I, ptr: *mut U) -> Self {
SrcDstPtrIter {
src_iter: iter,
dst_ptr: ptr,
phantom: PhantomData,
}
}
}