mirror of
https://github.com/italicsjenga/rp-hal-boards.git
synced 2025-01-10 20:41:31 +11:00
100 lines
2.7 KiB
Rust
100 lines
2.7 KiB
Rust
|
//! Ring Oscillator (ROSC)
|
||
|
// See [Chapter 2 Section 17](https://datasheets.raspberrypi.org/rp2040/rp2040_datasheet.pdf) for more details
|
||
|
|
||
|
use embedded_time::rate::Extensions;
|
||
|
use embedded_time::rate::Hertz;
|
||
|
|
||
|
/// State of the Ring Oscillator (typestate trait)
|
||
|
pub trait State {}
|
||
|
|
||
|
/// ROSC is disabled (typestate)
|
||
|
pub struct Disabled;
|
||
|
|
||
|
/// ROSC is initialized, ie we've given parameters (typestate)
|
||
|
pub struct Enabled {
|
||
|
freq_hz: Hertz,
|
||
|
}
|
||
|
|
||
|
/// ROSC is in dormant mode (see Chapter 2, Section 17, §7)
|
||
|
pub struct Dormant;
|
||
|
|
||
|
impl State for Disabled {}
|
||
|
impl State for Enabled {}
|
||
|
impl State for Dormant {}
|
||
|
|
||
|
/// A Ring Oscillator.
|
||
|
pub struct RingOscillator<S: State> {
|
||
|
device: rp2040_pac::ROSC,
|
||
|
state: S,
|
||
|
}
|
||
|
|
||
|
impl<S: State> RingOscillator<S> {
|
||
|
/// Transitions the oscillator to another state.
|
||
|
fn transition<To: State>(self, state: To) -> RingOscillator<To> {
|
||
|
RingOscillator {
|
||
|
device: self.device,
|
||
|
state,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Releases the underlying device.
|
||
|
pub fn free(self) -> rp2040_pac::ROSC {
|
||
|
self.device
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl RingOscillator<Disabled> {
|
||
|
/// Creates a new RingOscillator from the underlying device.
|
||
|
pub fn new(dev: rp2040_pac::ROSC) -> Self {
|
||
|
RingOscillator {
|
||
|
device: dev,
|
||
|
state: Disabled,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Initializes the ROSC : frequency range is set, startup delay is calculated and set.
|
||
|
pub fn initialize(self) -> RingOscillator<Enabled> {
|
||
|
self.device.ctrl.write(|w| w.enable().enable());
|
||
|
|
||
|
self.transition(Enabled {
|
||
|
freq_hz: 6_500_000u32.Hz(),
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl RingOscillator<Enabled> {
|
||
|
/// Approx operating frequency of the ROSC in hertz
|
||
|
pub fn operating_frequency(&self) -> Hertz {
|
||
|
self.state.freq_hz
|
||
|
}
|
||
|
|
||
|
/// Disables the ROSC
|
||
|
pub fn disable(self) -> RingOscillator<Disabled> {
|
||
|
self.device.ctrl.modify(|_r, w| w.enable().disable());
|
||
|
|
||
|
self.transition(Disabled)
|
||
|
}
|
||
|
|
||
|
/// Generate random bit based on the Ring oscillator
|
||
|
/// This is not suited for security purposes
|
||
|
pub fn get_random_bit(&self) -> bool {
|
||
|
self.device.randombit.read().randombit().bit()
|
||
|
}
|
||
|
|
||
|
/// Put the ROSC in DORMANT state.
|
||
|
///
|
||
|
/// # Safety
|
||
|
/// This method is marked unsafe because prior to switch the ROSC into DORMANT state,
|
||
|
/// PLLs must be stopped and IRQs have to be properly configured.
|
||
|
/// This method does not do any of that, it merely switches the ROSC to DORMANT state.
|
||
|
/// See Chapter 2, Section 16, §5) for details.
|
||
|
pub unsafe fn dormant(self) -> RingOscillator<Dormant> {
|
||
|
//taken from the C SDK
|
||
|
const ROSC_DORMANT_VALUE: u32 = 0x636f6d61;
|
||
|
|
||
|
self.device.dormant.write(|w| w.bits(ROSC_DORMANT_VALUE));
|
||
|
|
||
|
self.transition(Dormant)
|
||
|
}
|
||
|
}
|