#[inline]
pub fn hide_mem<T>(ptr: &mut T) {
hide_mem_impl(ptr);
}
#[inline]
pub fn hide_ptr<P>(mut ptr: P) -> P {
hide_mem::<P>(&mut ptr);
ptr
}
#[cfg(feature = "nightly")]
use self::nightly::*;
#[cfg(not(feature = "no_cc"))]
use self::cc::*;
#[cfg(all(feature = "no_cc", not(feature = "nightly")))]
use self::fallback::*;
#[cfg(feature = "nightly")]
mod nightly {
#[inline]
pub fn hide_mem_impl<T>(ptr: *mut T) {
unsafe {
asm!("" : "=*m" (ptr) : "*0" (ptr));
}
}
}
#[cfg(not(feature = "no_cc"))]
mod cc {
use std::os::raw::c_void;
extern "C" {
fn clear_on_drop_hide(ptr: *mut c_void) -> *mut c_void;
}
#[inline]
pub fn hide_mem_impl<T>(ptr: *mut T) {
unsafe {
clear_on_drop_hide(ptr as *mut c_void);
}
}
}
#[cfg(all(feature = "no_cc", not(feature = "nightly")))]
mod fallback {
use std::sync::atomic::{ATOMIC_USIZE_INIT, AtomicUsize, Ordering};
#[inline]
pub fn hide_mem_impl<T>(ptr: *mut T) {
static DUMMY: AtomicUsize = ATOMIC_USIZE_INIT;
DUMMY.store(ptr as usize, Ordering::Release);
}
}
#[cfg(test)]
mod tests {
struct Place {
data: [u32; 4],
}
const DATA: [u32; 4] = [0x01234567, 0x89abcdef, 0xfedcba98, 0x76543210];
#[test]
fn hide_mem() {
let mut place = Place { data: DATA };
super::hide_mem(&mut place);
assert_eq!(place.data, DATA);
}
#[test]
fn hide_ptr() {
let mut place = Place { data: DATA };
let before = &mut place as *mut _;
let after = super::hide_ptr(&mut place);
assert_eq!(before, after as *mut _);
assert_eq!(after.data, DATA);
}
}