pub struct VolatileRef<'a, T, A = ReadWrite>where
T: ?Sized,{ /* private fields */ }
Expand description
Volatile pointer type that respects Rust’s aliasing rules.
This pointer type behaves similar to Rust’s reference types:
- it requires exclusive
&mut self
access for mutability - only read-only types implement
Clone
andCopy
Send
andSync
are implemented ifT: Sync
However, trait implementations like fmt::Debug
and Eq
behave like they do on pointer
types and don’t access the referenced value.
To perform volatile operations on VolatileRef
types, use the as_ptr
or as_mut_ptr
methods to create a temporary
VolatilePtr
instance.
Since not all volatile resources (e.g. memory mapped device registers) are both readable
and writable, this type supports limiting the allowed access types through an optional second
generic parameter A
that can be one of ReadWrite
, ReadOnly
, or WriteOnly
. It defaults
to ReadWrite
, which allows all operations.
The size of this struct is the same as the size of the contained reference.
Implementations§
Source§impl<'a, T> VolatileRef<'a, T>where
T: ?Sized,
Constructor functions.
impl<'a, T> VolatileRef<'a, T>where
T: ?Sized,
Constructor functions.
These functions construct new VolatileRef
values. While the new
function creates a VolatileRef
instance with unrestricted access, there
are also functions for creating read-only or write-only instances.
Sourcepub unsafe fn new(pointer: NonNull<T>) -> Self
pub unsafe fn new(pointer: NonNull<T>) -> Self
Turns the given pointer into a VolatileRef
.
§Safety
- The pointer must be properly aligned.
- It must be “dereferenceable” in the sense defined in the
core::ptr
documentation. - The pointer must point to an initialized instance of T.
- You must enforce Rust’s aliasing rules, since the returned lifetime ’a is arbitrarily
chosen and does not necessarily reflect the actual lifetime of the data. In particular,
while this
VolatileRef
exists, the memory the pointer points to must not get accessed (read or written) through any other pointer.
Sourcepub const unsafe fn new_read_only(
pointer: NonNull<T>,
) -> VolatileRef<'a, T, ReadOnly>
pub const unsafe fn new_read_only( pointer: NonNull<T>, ) -> VolatileRef<'a, T, ReadOnly>
Turns the given pointer into a read-only VolatileRef
.
§Safety
- The pointer must be properly aligned.
- It must be “dereferenceable” in the sense defined in the
core::ptr
documentation. - The pointer must point to an initialized instance of T.
- You must enforce Rust’s aliasing rules, since the returned lifetime ’a is arbitrarily
chosen and does not necessarily reflect the actual lifetime of the data. In particular,
while this
VolatileRef
exists, the memory the pointer points to must not get mutated.
Sourcepub const unsafe fn new_restricted<A>(
access: A,
pointer: NonNull<T>,
) -> VolatileRef<'a, T, A>where
A: Access,
pub const unsafe fn new_restricted<A>(
access: A,
pointer: NonNull<T>,
) -> VolatileRef<'a, T, A>where
A: Access,
Turns the given pointer into a VolatileRef
instance with the given access.
§Safety
- The pointer must be properly aligned.
- It must be “dereferenceable” in the sense defined in the
core::ptr
documentation. - The pointer must point to an initialized instance of T.
- You must enforce Rust’s aliasing rules, since the returned lifetime ’a is arbitrarily
chosen and does not necessarily reflect the actual lifetime of the data. In particular,
while this
VolatileRef
exists, the memory the pointer points to must not get mutated. If the givenaccess
parameter allows write access, the pointer must not get read either while thisVolatileRef
exists.
Sourcepub fn from_ref(reference: &'a T) -> VolatileRef<'a, T, ReadOnly>where
T: 'a,
pub fn from_ref(reference: &'a T) -> VolatileRef<'a, T, ReadOnly>where
T: 'a,
Creates a VolatileRef
from the given shared reference.
Note: This function is only intended for testing, not for accessing real volatile
data. The reason is that the &mut T
argument is considered dereferenceable by Rust,
so the compiler is allowed to insert non-volatile reads. This might lead to undesired
(or even undefined?) behavior when accessing volatile data. So to be safe, only create
raw pointers to volatile data and use the Self::new
constructor instead.
Sourcepub fn from_mut_ref(reference: &'a mut T) -> Selfwhere
T: 'a,
pub fn from_mut_ref(reference: &'a mut T) -> Selfwhere
T: 'a,
Creates a VolatileRef
from the given mutable reference.
Note: This function is only intended for testing, not for accessing real volatile
data. The reason is that the &mut T
argument is considered dereferenceable by Rust,
so the compiler is allowed to insert non-volatile reads. This might lead to undesired
(or even undefined?) behavior when accessing volatile data. So to be safe, only create
raw pointers to volatile data and use the Self::new
constructor instead.
Source§impl<'a, T, A> VolatileRef<'a, T, A>where
T: ?Sized,
impl<'a, T, A> VolatileRef<'a, T, A>where
T: ?Sized,
Sourcepub fn borrow(&self) -> VolatileRef<'_, T, A::Restricted>where
A: RestrictAccess<ReadOnly>,
pub fn borrow(&self) -> VolatileRef<'_, T, A::Restricted>where
A: RestrictAccess<ReadOnly>,
Immutably borrows from this VolatileRef
.
This method creates a VolatileRef
tied to the lifetime of the &VolatileRef
it is created from.
This is useful for providing a volatile reference without moving the original VolatileRef
.
In comparison with creating a &VolatileRef<'a, T>
, this avoids the additional indirection and lifetime.
Sourcepub fn borrow_mut(&mut self) -> VolatileRef<'_, T, A>where
A: Access,
pub fn borrow_mut(&mut self) -> VolatileRef<'_, T, A>where
A: Access,
Mutably borrows from this VolatileRef
.
This method creates a VolatileRef
tied to the lifetime of the &mut VolatileRef
it is created from.
This is useful for providing a volatile reference without moving the original VolatileRef
.
In comparison with creating a &mut VolatileRef<'a, T>
, this avoids the additional indirection and lifetime.
Sourcepub fn as_ptr(&self) -> VolatilePtr<'_, T, A::Restricted>where
A: RestrictAccess<ReadOnly>,
pub fn as_ptr(&self) -> VolatilePtr<'_, T, A::Restricted>where
A: RestrictAccess<ReadOnly>,
Borrows this VolatileRef
as a read-only VolatilePtr
.
Use this method to do (partial) volatile reads of the referenced data.
Sourcepub fn as_mut_ptr(&mut self) -> VolatilePtr<'_, T, A>where
A: Access,
pub fn as_mut_ptr(&mut self) -> VolatilePtr<'_, T, A>where
A: Access,
Borrows this VolatileRef
as a mutable VolatilePtr
.
Use this method to do (partial) volatile reads or writes of the referenced data.
Sourcepub fn into_ptr(self) -> VolatilePtr<'a, T, A>where
A: Access,
pub fn into_ptr(self) -> VolatilePtr<'a, T, A>where
A: Access,
Converts this VolatileRef
into a VolatilePtr
with full access without shortening
the lifetime.
Use this method when you need a VolatilePtr
instance that lives for the full
lifetime 'a
.
This method consumes the VolatileRef
.
Source§impl<'a, T, A> VolatileRef<'a, T, A>where
T: ?Sized,
Methods for restricting access.
impl<'a, T, A> VolatileRef<'a, T, A>where
T: ?Sized,
Methods for restricting access.
Sourcepub fn restrict<To>(self) -> VolatileRef<'a, T, A::Restricted>where
A: RestrictAccess<To>,
pub fn restrict<To>(self) -> VolatileRef<'a, T, A::Restricted>where
A: RestrictAccess<To>,
Restricts access permissions to A
.
§Example
use volatile::access::{ReadOnly, WriteOnly};
use volatile::VolatileRef;
let mut value: i16 = -4;
let volatile = VolatileRef::from_mut_ref(&mut value);
let read_only = volatile.restrict::<ReadOnly>();
assert_eq!(read_only.as_ptr().read(), -4);
// read_only.as_ptr().write(10); // compile-time error
let no_access = read_only.restrict::<WriteOnly>();
// no_access.read(); // compile-time error
// no_access.write(10); // compile-time error
Source§impl<'a, T> VolatileRef<'a, T, ReadWrite>where
T: ?Sized,
Methods for restricting access.
impl<'a, T> VolatileRef<'a, T, ReadWrite>where
T: ?Sized,
Methods for restricting access.
Sourcepub fn read_only(self) -> VolatileRef<'a, T, ReadOnly>
pub fn read_only(self) -> VolatileRef<'a, T, ReadOnly>
Restricts access permissions to read-only.
§Example
use volatile::VolatileRef;
let mut value: i16 = -4;
let volatile = VolatileRef::from_mut_ref(&mut value);
let read_only = volatile.read_only();
assert_eq!(read_only.as_ptr().read(), -4);
// read_only.as_ptr().write(10); // compile-time error
Sourcepub fn write_only(self) -> VolatileRef<'a, T, WriteOnly>
pub fn write_only(self) -> VolatileRef<'a, T, WriteOnly>
Restricts access permissions to write-only.
§Example
Creating a write-only reference to a struct field:
use volatile::{VolatileRef};
#[derive(Clone, Copy)]
struct Example { field_1: u32, field_2: u8, }
let mut value = Example { field_1: 15, field_2: 255 };
let volatile = VolatileRef::from_mut_ref(&mut value);
let write_only = volatile.write_only();
// write_only.as_ptr().read(); // compile-time error