#![no_std]
pub extern crate typenum;
extern crate nodrop;
#[cfg(feature="serde")]
extern crate serde;
pub mod arr;
pub mod iter;
pub use iter::GenericArrayIter;
mod hex;
mod impls;
#[cfg(feature="serde")]
pub mod impl_serde;
use core::marker::PhantomData;
use core::mem;
pub use core::mem::transmute;
use core::ops::{Deref, DerefMut};
use core::slice;
use nodrop::NoDrop;
use typenum::bit::{B0, B1};
use typenum::uint::{UInt, UTerm, Unsigned};
pub unsafe trait ArrayLength<T>: Unsigned {
type ArrayType;
}
unsafe impl<T> ArrayLength<T> for UTerm {
type ArrayType = ();
}
#[allow(dead_code)]
#[repr(C)]
pub struct GenericArrayImplEven<T, U> {
parent1: U,
parent2: U,
_marker: PhantomData<T>,
}
impl<T: Clone, U: Clone> Clone for GenericArrayImplEven<T, U> {
fn clone(&self) -> GenericArrayImplEven<T, U> {
GenericArrayImplEven {
parent1: self.parent1.clone(),
parent2: self.parent2.clone(),
_marker: PhantomData,
}
}
}
impl<T: Copy, U: Copy> Copy for GenericArrayImplEven<T, U> {}
#[allow(dead_code)]
#[repr(C)]
pub struct GenericArrayImplOdd<T, U> {
parent1: U,
parent2: U,
data: T,
}
impl<T: Clone, U: Clone> Clone for GenericArrayImplOdd<T, U> {
fn clone(&self) -> GenericArrayImplOdd<T, U> {
GenericArrayImplOdd {
parent1: self.parent1.clone(),
parent2: self.parent2.clone(),
data: self.data.clone(),
}
}
}
impl<T: Copy, U: Copy> Copy for GenericArrayImplOdd<T, U> {}
unsafe impl<T, N: ArrayLength<T>> ArrayLength<T> for UInt<N, B0> {
type ArrayType = GenericArrayImplEven<T, N::ArrayType>;
}
unsafe impl<T, N: ArrayLength<T>> ArrayLength<T> for UInt<N, B1> {
type ArrayType = GenericArrayImplOdd<T, N::ArrayType>;
}
#[allow(dead_code)]
pub struct GenericArray<T, U: ArrayLength<T>> {
data: U::ArrayType,
}
impl<T, N> Deref for GenericArray<T, N>
where N: ArrayLength<T>
{
type Target = [T];
fn deref(&self) -> &[T] {
unsafe { slice::from_raw_parts(self as *const Self as *const T, N::to_usize()) }
}
}
impl<T, N> DerefMut for GenericArray<T, N>
where N: ArrayLength<T>
{
fn deref_mut(&mut self) -> &mut [T] {
unsafe { slice::from_raw_parts_mut(self as *mut Self as *mut T, N::to_usize()) }
}
}
impl<T, N> GenericArray<T, N>
where N: ArrayLength<T>
{
pub fn map_slice<S, F: Fn(&S) -> T>(s: &[S], f: F) -> GenericArray<T, N> {
assert_eq!(s.len(), N::to_usize());
map_inner(s, f)
}
pub fn map<U, F>(self, f: F) -> GenericArray<U, N>
where F: Fn(&T) -> U,
N: ArrayLength<U>
{
map_inner(&self, f)
}
pub fn as_slice(&self) -> &[T] {
self.deref()
}
pub fn as_mut_slice(&mut self) -> &mut [T] {
self.deref_mut()
}
#[inline]
pub fn from_slice(slice: &[T]) -> &GenericArray<T, N> {
assert_eq!(slice.len(), N::to_usize());
unsafe { &*(slice.as_ptr() as *const GenericArray<T, N>) }
}
#[inline]
pub fn from_mut_slice(slice: &mut [T]) -> &mut GenericArray<T, N> {
assert_eq!(slice.len(), N::to_usize());
unsafe { &mut *(slice.as_mut_ptr() as *mut GenericArray<T, N>) }
}
}
#[inline]
fn map_inner<S, F, T, N>(list: &[S], f: F) -> GenericArray<T, N>
where F: Fn(&S) -> T,
N: ArrayLength<T>
{
unsafe {
let mut res: NoDrop<GenericArray<T, N>> = NoDrop::new(mem::uninitialized());
for (s, r) in list.iter().zip(res.iter_mut()) {
core::ptr::write(r, f(s))
}
res.into_inner()
}
}
impl<T: Clone, N> GenericArray<T, N>
where N: ArrayLength<T>
{
pub fn clone_from_slice(list: &[T]) -> GenericArray<T, N> {
assert_eq!(list.len(), N::to_usize());
map_inner(list, |x: &T| x.clone())
}
}