1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
use crate::types::ValueType; use std::sync::atomic::{ AtomicI16, AtomicI32, AtomicI64, AtomicI8, AtomicU16, AtomicU32, AtomicU64, AtomicU8, }; use std::{cell::Cell, marker::PhantomData, ops::Deref, slice}; pub trait Atomic { type Output; } impl Atomic for i8 { type Output = AtomicI8; } impl Atomic for i16 { type Output = AtomicI16; } impl Atomic for i32 { type Output = AtomicI32; } impl Atomic for i64 { type Output = AtomicI64; } impl Atomic for u8 { type Output = AtomicU8; } impl Atomic for u16 { type Output = AtomicU16; } impl Atomic for u32 { type Output = AtomicU32; } impl Atomic for u64 { type Output = AtomicU64; } impl Atomic for f32 { type Output = AtomicU32; } impl Atomic for f64 { type Output = AtomicU64; } pub trait Atomicity {} pub struct Atomically; impl Atomicity for Atomically {} pub struct NonAtomically; impl Atomicity for NonAtomically {} pub struct MemoryView<'a, T: 'a, A = NonAtomically> { ptr: *mut T, length: usize, _phantom: PhantomData<(&'a [Cell<T>], A)>, } impl<'a, T> MemoryView<'a, T, NonAtomically> where T: ValueType, { pub(super) unsafe fn new(ptr: *mut T, length: u32) -> Self { Self { ptr, length: length as usize, _phantom: PhantomData, } } } impl<'a, T: Atomic> MemoryView<'a, T> { pub fn atomically(&self) -> MemoryView<'a, T::Output, Atomically> { MemoryView { ptr: self.ptr as *mut T::Output, length: self.length, _phantom: PhantomData, } } } impl<'a, T> Deref for MemoryView<'a, T, NonAtomically> { type Target = [Cell<T>]; fn deref(&self) -> &[Cell<T>] { unsafe { slice::from_raw_parts(self.ptr as *const Cell<T>, self.length) } } } impl<'a, T> Deref for MemoryView<'a, T, Atomically> { type Target = [T]; fn deref(&self) -> &[T] { unsafe { slice::from_raw_parts(self.ptr as *const T, self.length) } } }