pub struct MemFlags { /* private fields */ }Expand description
Flags for memory operations like load/store.
Each of these flags introduce a limited form of undefined behavior. The flags each enable certain optimizations that need to make additional assumptions. Generally, the semantics of a program does not change when a flag is removed, but adding a flag will.
In addition, the flags determine the endianness of the memory access. By default, any memory access uses the native endianness determined by the target ISA. This can be overridden for individual accesses by explicitly specifying little- or big-endian semantics via the flags.
Implementations§
Source§impl MemFlags
 
impl MemFlags
Sourcepub const fn trusted() -> Self
 
pub const fn trusted() -> Self
Create a set of flags representing an access from a “trusted” address, meaning it’s known to be aligned and non-trapping.
Sourcepub const fn alias_region(self) -> Option<AliasRegion>
 
pub const fn alias_region(self) -> Option<AliasRegion>
Reads the alias region that this memory operation works with.
Sourcepub const fn with_alias_region(self, region: Option<AliasRegion>) -> Self
 
pub const fn with_alias_region(self, region: Option<AliasRegion>) -> Self
Sets the alias region that this works on to the specified region.
Sourcepub fn set_alias_region(&mut self, region: Option<AliasRegion>)
 
pub fn set_alias_region(&mut self, region: Option<AliasRegion>)
Sets the alias region that this works on to the specified region.
Sourcepub fn set_by_name(&mut self, name: &str) -> Result<bool, &'static str>
 
pub fn set_by_name(&mut self, name: &str) -> Result<bool, &'static str>
Set a flag bit by name.
Returns true if the flag was found and set, false for an unknown flag name.
§Errors
Returns an error message if the name is known but couldn’t be applied
due to it being a semantic error.
Sourcepub const fn endianness(self, native_endianness: Endianness) -> Endianness
 
pub const fn endianness(self, native_endianness: Endianness) -> Endianness
Return endianness of the memory access. This will return the endianness explicitly specified by the flags if any, and will default to the native endianness otherwise. The native endianness has to be provided by the caller since it is not explicitly encoded in CLIF IR – this allows a front end to create IR without having to know the target endianness.
Sourcepub const fn explicit_endianness(self) -> Option<Endianness>
 
pub const fn explicit_endianness(self) -> Option<Endianness>
Return endianness of the memory access, if explicitly specified.
If the endianness is not explicitly specified, this will return None,
which means “native endianness”.
Sourcepub fn set_endianness(&mut self, endianness: Endianness)
 
pub fn set_endianness(&mut self, endianness: Endianness)
Set endianness of the memory access.
Sourcepub const fn with_endianness(self, endianness: Endianness) -> Self
 
pub const fn with_endianness(self, endianness: Endianness) -> Self
Set endianness of the memory access, returning new flags.
Sourcepub const fn notrap(self) -> bool
 
pub const fn notrap(self) -> bool
Test if this memory operation cannot trap.
By default MemFlags will assume that any load/store can trap and is
associated with a TrapCode::HeapOutOfBounds code. If the trap code is
configured to None though then this method will return true and
indicates that the memory operation will not trap.
If this returns true then the memory is accessible, which means
that accesses will not trap. This makes it possible to delete an unused
load or a dead store instruction.
This flag does not mean that the associated instruction can be
code-motioned to arbitrary places in the function so long as its data
dependencies are met. This only means that, given its current location
in the function, it will never trap. See the can_move method for more
details.
Sourcepub fn set_notrap(&mut self)
 
pub fn set_notrap(&mut self)
Sets the trap code for this MemFlags to None.
Sourcepub const fn with_notrap(self) -> Self
 
pub const fn with_notrap(self) -> Self
Sets the trap code for this MemFlags to None, returning the new
flags.
Sourcepub const fn can_move(self) -> bool
 
pub const fn can_move(self) -> bool
Is this memory operation safe to move so long as its data dependencies remain satisfied?
If this is true, then it is okay to code motion this instruction to
arbitrary locations, in the function, including across blocks and
conditional branches, so long as data dependencies (and trap ordering,
if any) are upheld.
If this is false, then this memory operation’s safety potentially
relies upon invariants that are not reflected in its data dependencies,
and therefore it is not safe to code motion this operation. For example,
this operation could be in a block that is dominated by a control-flow
bounds check, which is not reflected in its operands, and it would be
unsafe to code motion it above the bounds check, even if its data
dependencies would still be satisfied.
Sourcepub const fn set_can_move(&mut self)
 
pub const fn set_can_move(&mut self)
Set the can_move flag.
Sourcepub const fn with_can_move(self) -> Self
 
pub const fn with_can_move(self) -> Self
Set the can_move flag, returning new flags.
Sourcepub const fn aligned(self) -> bool
 
pub const fn aligned(self) -> bool
Test if the aligned flag is set.
By default, Cranelift memory instructions work with any unaligned effective address. If the
aligned flag is set, the instruction is permitted to trap or return a wrong result if the
effective address is misaligned.
Sourcepub fn set_aligned(&mut self)
 
pub fn set_aligned(&mut self)
Set the aligned flag.
Sourcepub const fn with_aligned(self) -> Self
 
pub const fn with_aligned(self) -> Self
Set the aligned flag, returning new flags.
Sourcepub const fn readonly(self) -> bool
 
pub const fn readonly(self) -> bool
Test if the readonly flag is set.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
Sourcepub fn set_readonly(&mut self)
 
pub fn set_readonly(&mut self)
Set the readonly flag.
Sourcepub const fn with_readonly(self) -> Self
 
pub const fn with_readonly(self) -> Self
Set the readonly flag, returning new flags.
Sourcepub const fn checked(self) -> bool
 
pub const fn checked(self) -> bool
Test if the checked bit is set.
Loads and stores with this flag are verified to access
pointers only with a validated PointsTo fact attached, and
with that fact validated, when using the proof-carrying-code
framework. If initial facts on program inputs are correct
(i.e., correctly denote the shape and types of data structures
in memory), and if PCC validates the compiled output, then all
checked-marked memory accesses are guaranteed (up to the
checker’s correctness) to access valid memory. This can be
used to ensure memory safety and sandboxing.
Sourcepub fn set_checked(&mut self)
 
pub fn set_checked(&mut self)
Set the checked bit.
Sourcepub const fn with_checked(self) -> Self
 
pub const fn with_checked(self) -> Self
Set the checked bit, returning new flags.
Sourcepub const fn trap_code(self) -> Option<TrapCode>
 
pub const fn trap_code(self) -> Option<TrapCode>
Get the trap code to report if this memory access traps.
A None trap code indicates that this memory access does not trap.
Sourcepub const fn with_trap_code(self, code: Option<TrapCode>) -> Self
 
pub const fn with_trap_code(self, code: Option<TrapCode>) -> Self
Configures these flags with the specified trap code code.
A trap code indicates that this memory operation cannot be optimized away and it must “stay where it is” in the programs. Traps are considered side effects, for example, and have meaning through the trap code that is communicated and which instruction trapped.
Trait Implementations§
impl Copy for MemFlags
impl Eq for MemFlags
impl StructuralPartialEq for MemFlags
Auto Trait Implementations§
impl Freeze for MemFlags
impl RefUnwindSafe for MemFlags
impl Send for MemFlags
impl Sync for MemFlags
impl Unpin for MemFlags
impl UnwindSafe for MemFlags
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
    T: ?Sized,
 
impl<T> BorrowMut<T> for Twhere
    T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
 
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> CloneToUninit for Twhere
    T: Clone,
 
impl<T> CloneToUninit for Twhere
    T: Clone,
Source§impl<Q, K> Equivalent<K> for Q
 
impl<Q, K> Equivalent<K> for Q
Source§impl<Q, K> Equivalent<K> for Q
 
impl<Q, K> Equivalent<K> for Q
Source§fn equivalent(&self, key: &K) -> bool
 
fn equivalent(&self, key: &K) -> bool
key and return true if they are equal.