mirror of
https://github.com/italicsjenga/gba.git
synced 2025-01-22 23:56:32 +11:00
This commit is contained in:
parent
c4daa3e937
commit
c82c3bf7fe
2 changed files with 140 additions and 0 deletions
|
@ -97,6 +97,7 @@ pub mod keys;
|
|||
pub mod mgba;
|
||||
pub mod mmio;
|
||||
pub mod prelude;
|
||||
pub mod random;
|
||||
pub mod sound;
|
||||
pub mod timers;
|
||||
pub mod video;
|
||||
|
|
139
src/random.rs
Normal file
139
src/random.rs
Normal file
|
@ -0,0 +1,139 @@
|
|||
// Note(Lokathor): We have a generic LCG type below, but for now we can hide the
|
||||
// process of having to pick what multiplier and increment to use behind a
|
||||
// newtype that selects some default constants.
|
||||
|
||||
#[derive(Debug, Default, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||
#[repr(transparent)]
|
||||
pub struct Lcg32(GenericLcg32<32310901, 1>);
|
||||
impl Lcg32 {
|
||||
#[inline]
|
||||
pub const fn new(state: u32) -> Self {
|
||||
Self(GenericLcg32::new(state))
|
||||
}
|
||||
#[inline]
|
||||
pub fn next_u32(&mut self) -> u32 {
|
||||
self.0.next_u32()
|
||||
}
|
||||
#[inline]
|
||||
pub fn jump_state(&mut self, delta: u32) {
|
||||
self.0.jump_state(delta)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Default, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||
#[repr(transparent)]
|
||||
struct GenericLcg32<const MUL: u32, const ADD: u32>(u32);
|
||||
impl<const MUL: u32, const ADD: u32> GenericLcg32<MUL, ADD> {
|
||||
#[inline]
|
||||
pub const fn new(state: u32) -> Self {
|
||||
Self(state)
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn next_u32(&mut self) -> u32 {
|
||||
let next_state = self.0.wrapping_mul(MUL).wrapping_add(ADD);
|
||||
next_state
|
||||
}
|
||||
|
||||
#[inline]
|
||||
pub fn jump_state(&mut self, mut delta: u32) {
|
||||
let mut cur_mult: u32 = MUL;
|
||||
let mut cur_plus: u32 = ADD;
|
||||
let mut acc_mult: u32 = 1;
|
||||
let mut acc_plus: u32 = 0;
|
||||
while delta > 0 {
|
||||
if (delta & 1) > 0 {
|
||||
acc_mult = acc_mult.wrapping_mul(cur_mult);
|
||||
acc_plus = acc_plus.wrapping_mul(cur_mult).wrapping_add(cur_plus);
|
||||
}
|
||||
cur_plus = cur_mult.wrapping_add(1).wrapping_mul(cur_plus);
|
||||
cur_mult = cur_mult.wrapping_mul(cur_mult);
|
||||
delta /= 2;
|
||||
}
|
||||
self.0 = acc_mult.wrapping_mul(self.0).wrapping_add(acc_plus);
|
||||
}
|
||||
}
|
||||
|
||||
pub trait Gen32 {
|
||||
fn next_u32(&mut self) -> u32;
|
||||
|
||||
#[inline]
|
||||
fn next_u16(&mut self) -> u16 {
|
||||
(self.next_u32() >> 16) as u16
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn next_u8(&mut self) -> u8 {
|
||||
(self.next_u32() >> 24) as u8
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn next_bool(&mut self) -> bool {
|
||||
(self.next_u16() as i32) < 0
|
||||
}
|
||||
|
||||
#[inline]
|
||||
#[track_caller]
|
||||
fn next_bounded(&mut self, b: u16) -> u16 {
|
||||
assert!(b != 0, "Gen32::next_bounded> Bound must be non-zero.");
|
||||
let mut x: u32 = self.next_u16() as u32;
|
||||
let mut mul: u32 = (b as u32).wrapping_mul(x);
|
||||
let mut low: u16 = mul as u16;
|
||||
if low < b {
|
||||
let threshold = b.wrapping_neg() % b;
|
||||
while low < threshold {
|
||||
x = self.next_u32();
|
||||
mul = (b as u32).wrapping_mul(x);
|
||||
low = mul as u16;
|
||||
}
|
||||
}
|
||||
let high = (mul >> 16) as u16;
|
||||
high
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn pick<T: Copy>(&mut self, buf: &[T]) -> T {
|
||||
let len16: u16 = saturating_usize_as_u16(buf.len());
|
||||
buf[self.next_bounded(len16) as usize]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn pick_ref<'b, T>(&mut self, buf: &'b [T]) -> &'b T {
|
||||
let len16: u16 = saturating_usize_as_u16(buf.len());
|
||||
&buf[self.next_bounded(len16) as usize]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn pick_mut<'b, T>(&mut self, buf: &'b mut [T]) -> &'b mut T {
|
||||
let len16: u16 = saturating_usize_as_u16(buf.len());
|
||||
&mut buf[self.next_bounded(len16) as usize]
|
||||
}
|
||||
|
||||
#[inline]
|
||||
fn shuffle<T>(&mut self, buf: &mut [T]) {
|
||||
let mut possibility_count: u16 = saturating_usize_as_u16(buf.len());
|
||||
let mut this_index: usize = 0;
|
||||
let end_index = buf.len() - 1;
|
||||
while this_index < end_index {
|
||||
let offset = self.next_bounded(possibility_count) as usize;
|
||||
buf.swap(this_index, this_index + offset);
|
||||
possibility_count -= 1;
|
||||
this_index += 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Gen32 for Lcg32 {
|
||||
fn next_u32(&mut self) -> u32 {
|
||||
Lcg32::next_u32(self)
|
||||
}
|
||||
}
|
||||
|
||||
#[inline]
|
||||
const fn saturating_usize_as_u16(val: usize) -> u16 {
|
||||
if val <= u16::MAX as usize {
|
||||
val as u16
|
||||
} else {
|
||||
u16::MAX
|
||||
}
|
||||
}
|
Loading…
Add table
Reference in a new issue