rhai/src/config/hashing.rs

225 lines
5.9 KiB
Rust
Raw Normal View History

2022-10-31 16:59:07 +01:00
//! Fixed hashing seeds for stable hashing.
//!
//! Set to [`None`] to disable stable hashing.
//!
2022-11-04 14:22:31 +01:00
//! See [`set_rhai_ahash_seed`].
2022-10-31 16:59:07 +01:00
//!
2022-11-04 14:22:31 +01:00
//! # Example
2022-10-31 16:59:07 +01:00
//!
2022-11-04 14:22:31 +01:00
//! ```rust
//! // Set the hashing seed to [1, 2, 3, 4]
//! rhai::config::hashing::set_ahash_seed(Some([1, 2, 3, 4])).unwrap();
//! ```
//! Alternatively, set this at compile time via the `RHAI_AHASH_SEED` environment variable.
//!
//! # Example
//!
//! ```sh
//! env RHAI_AHASH_SEED ="[236,800,954,213]"
//! ```
2022-10-31 16:59:07 +01:00
// [236,800,954,213], haha funny yume nikki reference epic uboachan face numberworld nexus moment 100
use crate::config::hashing_env;
2022-11-05 03:49:34 +01:00
use core::panic::{RefUnwindSafe, UnwindSafe};
2022-11-04 14:22:31 +01:00
#[cfg(feature = "no_std")]
use std::prelude::v1::*;
use std::{
2022-11-01 13:25:45 +01:00
cell::UnsafeCell,
marker::PhantomData,
2022-11-04 14:22:31 +01:00
mem,
2022-11-01 13:25:45 +01:00
mem::MaybeUninit,
sync::atomic::{AtomicBool, AtomicUsize, Ordering},
2022-11-01 13:25:45 +01:00
};
// omg its hokma from record team here to record our locks
// what does this do?
// so what this does is keep track of a global address in memory that acts as a global lock
// i stole this from crossbeam so read their docs for more
2022-11-04 14:22:31 +01:00
#[must_use]
struct HokmaLock {
lock: AtomicUsize,
}
impl HokmaLock {
2022-11-04 14:22:31 +01:00
#[inline(always)]
pub const fn new() -> Self {
Self {
lock: AtomicUsize::new(0),
}
}
2022-11-04 14:22:31 +01:00
pub fn write(&'static self) -> WhenTheHokmaSuppression {
loop {
let previous = self.lock.swap(1, Ordering::SeqCst);
if previous != 1 {
2022-11-04 14:22:31 +01:00
return WhenTheHokmaSuppression {
hokma: self,
state: previous,
};
}
}
}
}
2022-11-04 14:22:31 +01:00
struct WhenTheHokmaSuppression {
hokma: &'static HokmaLock,
state: usize,
}
2022-11-04 14:22:31 +01:00
impl WhenTheHokmaSuppression {
#[inline]
pub fn the_price_of_silence(self) {
self.hokma.lock.store(self.state, Ordering::SeqCst);
2022-11-04 14:22:31 +01:00
mem::forget(self)
}
}
2022-11-04 14:22:31 +01:00
impl Drop for WhenTheHokmaSuppression {
#[inline]
fn drop(&mut self) {
self.hokma
.lock
.store(self.state.wrapping_add(2), Ordering::SeqCst)
}
}
2022-11-04 14:22:31 +01:00
#[inline(always)]
#[must_use]
fn hokmalock(address: usize) -> &'static HokmaLock {
const LEN: usize = 787;
const LCK: HokmaLock = HokmaLock::new();
static RECORDS: [HokmaLock; LEN] = [LCK; LEN];
&RECORDS[address % LEN]
}
// Safety: lol, there is a reason its called "SusLock<T>"
2022-11-04 14:22:31 +01:00
#[must_use]
2022-11-01 13:25:45 +01:00
struct SusLock<T>
where
2022-11-05 14:46:08 +01:00
T: 'static,
2022-11-01 13:25:45 +01:00
{
2022-11-04 14:22:31 +01:00
initialized: AtomicBool,
2022-11-01 13:25:45 +01:00
data: UnsafeCell<MaybeUninit<T>>,
_marker: PhantomData<T>,
}
2022-10-31 16:59:07 +01:00
impl<T> SusLock<T>
where
2022-11-05 14:46:08 +01:00
T: 'static,
{
2022-11-04 14:22:31 +01:00
#[inline]
2022-11-01 13:25:45 +01:00
pub const fn new() -> SusLock<T> {
SusLock {
2022-11-04 14:22:31 +01:00
initialized: AtomicBool::new(false),
2022-11-01 13:25:45 +01:00
data: UnsafeCell::new(MaybeUninit::uninit()),
_marker: PhantomData,
}
2022-10-31 16:59:07 +01:00
}
2022-11-04 14:22:31 +01:00
#[must_use]
2022-11-05 14:46:08 +01:00
pub fn get(&self) -> Option<&'static T> {
2022-11-04 14:22:31 +01:00
if self.initialized.load(Ordering::SeqCst) {
let hokma = hokmalock(unsafe { mem::transmute(self.data.get()) });
// we forgo the optimistic read, because we don't really care
let guard = hokma.write();
let val = {
let cast: *const T = self.data.get().cast();
2022-11-05 14:46:08 +01:00
unsafe { mem::transmute::<*const T, &'static T>(cast) }
};
guard.the_price_of_silence();
Some(val)
2022-11-01 13:25:45 +01:00
} else {
return None;
2022-10-31 16:59:07 +01:00
}
2022-11-01 13:25:45 +01:00
}
2022-10-31 16:59:07 +01:00
2022-11-04 14:22:31 +01:00
#[must_use]
2022-11-05 14:46:08 +01:00
pub fn get_or_init(&self, f: impl FnOnce() -> T) -> Option<&'static T> {
2022-11-04 14:22:31 +01:00
if !self.initialized.load(Ordering::SeqCst) {
2022-11-05 14:46:08 +01:00
let value = f();
2022-11-04 14:22:31 +01:00
self.initialized.store(true, Ordering::SeqCst);
let hokma = hokmalock(unsafe { mem::transmute(self.data.get()) });
hokma.write();
2022-11-01 13:25:45 +01:00
unsafe {
self.data.get().write(MaybeUninit::new(value));
2022-10-31 16:59:07 +01:00
}
}
2022-11-01 13:25:45 +01:00
self.get()
2022-10-31 16:59:07 +01:00
}
2022-11-01 13:25:45 +01:00
pub fn set(&self, value: T) -> Result<(), T> {
2022-11-04 14:22:31 +01:00
if self.initialized.load(Ordering::SeqCst) {
2022-11-01 13:25:45 +01:00
Err(value)
} else {
let _ = self.get_or_init(|| value);
Ok(())
2022-10-31 16:59:07 +01:00
}
}
2022-11-01 13:25:45 +01:00
}
2022-10-31 16:59:07 +01:00
unsafe impl<T: Sync + Send> Sync for SusLock<T> where T: 'static + Copy {}
unsafe impl<T: Send> Send for SusLock<T> where T: 'static + Copy {}
impl<T: RefUnwindSafe + UnwindSafe> RefUnwindSafe for SusLock<T> where T: 'static + Copy {}
2022-10-31 16:59:07 +01:00
impl<T> Drop for SusLock<T>
where
T: 'static + Copy,
{
2022-11-04 14:22:31 +01:00
#[inline]
2022-11-01 13:25:45 +01:00
fn drop(&mut self) {
2022-11-04 14:22:31 +01:00
if self.initialized.load(Ordering::SeqCst) {
2022-11-01 13:25:45 +01:00
unsafe { (&mut *self.data.get()).assume_init_drop() };
}
2022-10-31 16:59:07 +01:00
}
}
2022-11-01 13:25:45 +01:00
static AHASH_SEED: SusLock<Option<[u64; 4]>> = SusLock::new();
2022-11-04 14:22:31 +01:00
/// Set the hashing seed. This is used to hash functions etc.
2022-11-01 13:25:45 +01:00
///
2022-11-04 14:22:31 +01:00
/// This is a static global value and affects every Rhai instance.
2022-11-01 13:25:45 +01:00
/// This should not be used _unless_ you know you need it.
///
2022-11-04 14:22:31 +01:00
/// # Warning
///
/// * You can only call this function **ONCE** for the entire duration of program execution.
/// * You **MUST** call this before performing **ANY** Rhai operation (e.g. creating an [`Engine`]).
///
/// # Error
2022-11-01 13:25:45 +01:00
///
2022-11-04 14:22:31 +01:00
/// Returns an error containing the existing hashing seed if already set.
///
/// # Example
///
/// ```rust
/// # use rhai::Engine;
/// // Set the hashing seed to [1, 2, 3, 4]
/// rhai::config::hashing::set_ahash_seed(Some([1, 2, 3, 4])).unwrap();
///
/// // Use Rhai AFTER setting the hashing seed
/// let engine = Engine::new();
/// ```
#[inline(always)]
2022-11-01 13:25:45 +01:00
pub fn set_ahash_seed(new_seed: Option<[u64; 4]>) -> Result<(), Option<[u64; 4]>> {
AHASH_SEED.set(new_seed)
}
2022-10-31 16:59:07 +01:00
2022-11-04 14:22:31 +01:00
/// Get the current hashing Seed.
///
/// If the seed is not yet defined, the `RHAI_AHASH_SEED` environment variable (if any) is used.
///
/// Otherwise, the hashing seed is randomized to protect against DOS attacks.
2022-11-01 13:25:45 +01:00
///
/// See [`set_rhai_ahash_seed`] for more.
2022-11-04 14:22:31 +01:00
#[inline]
#[must_use]
2022-11-05 14:46:08 +01:00
pub fn get_ahash_seed() -> &Option<[u64; 4]> {
match AHASH_SEED.get_or_init(|| hashing_env::AHASH_SEED) {
Some(ash) => ash,
None => None,
}
2022-11-01 13:25:45 +01:00
}