//! Phase-Locked Loops (PLL) // See [Chapter 2 Section 18](https://datasheets.raspberrypi.org/rp2040/rp2040_datasheet.pdf) for more details use core::{ convert::{Infallible, TryInto}, marker::PhantomData, ops::{Deref, Range, RangeInclusive}, }; use fugit::{HertzU32, RateExtU32}; use nb::Error::WouldBlock; use pac::RESETS; use crate::{clocks::ClocksManager, resets::SubsystemReset}; /// State of the PLL pub trait State {} /// PLL is disabled. pub struct Disabled { refdiv: u8, fbdiv: u16, post_div1: u8, post_div2: u8, frequency: HertzU32, } /// PLL is configured, started and locking into its designated frequency. pub struct Locking { post_div1: u8, post_div2: u8, frequency: HertzU32, } /// PLL is locked : it delivers a steady frequency. pub struct Locked { frequency: HertzU32, } impl State for Disabled {} impl State for Locked {} impl State for Locking {} /// Trait to handle both underlying devices from the PAC (PLL_SYS & PLL_USB) pub trait PhaseLockedLoopDevice: Deref + SubsystemReset { } impl PhaseLockedLoopDevice for rp2040_pac::PLL_SYS {} impl PhaseLockedLoopDevice for rp2040_pac::PLL_USB {} /// A PLL. pub struct PhaseLockedLoop { device: D, state: S, } impl PhaseLockedLoop { fn transition(self, state: To) -> PhaseLockedLoop { PhaseLockedLoop { device: self.device, state, } } /// Releases the underlying device. pub fn free(self) -> D { self.device } } /// Error type for the PLL module. /// See Chapter 2, Section 18 §2 for details on constraints triggering these errors. pub enum Error { /// Proposed VCO frequency is out of range. VcoFreqOutOfRange, /// Feedback Divider value is out of range. FeedbackDivOutOfRange, /// Post Divider value is out of range. PostDivOutOfRage, /// Reference Frequency is out of range. RefFreqOutOfRange, /// Bad argument : overflows, bad conversion, ... BadArgument, } /// Parameters for a PLL. pub struct PLLConfig { /// Voltage Controlled Oscillator frequency. pub vco_freq: HertzU32, /// Reference divider pub refdiv: u8, /// Post Divider 1 pub post_div1: u8, /// Post Divider 2 pub post_div2: u8, } /// Common configs for the two PLLs. Both assume the XOSC is cadenced at 12MHz ! /// See Chapter 2, Section 18, §2 pub mod common_configs { use super::PLLConfig; use fugit::HertzU32; /// Default, nominal configuration for PLL_SYS pub const PLL_SYS_125MHZ: PLLConfig = PLLConfig { vco_freq: HertzU32::MHz(1500), refdiv: 1, post_div1: 6, post_div2: 2, }; /// Default, nominal configuration for PLL_USB. pub const PLL_USB_48MHZ: PLLConfig = PLLConfig { vco_freq: HertzU32::MHz(480), refdiv: 1, post_div1: 5, post_div2: 2, }; } impl PhaseLockedLoop { /// Instantiates a new Phase-Locked-Loop device. pub fn new( dev: D, xosc_frequency: HertzU32, config: PLLConfig, ) -> Result, Error> { const VCO_FREQ_RANGE: RangeInclusive = HertzU32::MHz(400)..=HertzU32::MHz(1_600); const POSTDIV_RANGE: Range = 1..7; const FBDIV_RANGE: Range = 16..320; let vco_freq = config.vco_freq; if !VCO_FREQ_RANGE.contains(&vco_freq) { return Err(Error::VcoFreqOutOfRange); } if !POSTDIV_RANGE.contains(&config.post_div1) || !POSTDIV_RANGE.contains(&config.post_div2) { return Err(Error::PostDivOutOfRage); } let ref_freq_max_vco = (vco_freq.to_Hz() / 16).Hz(); let ref_freq_range: Range = HertzU32::MHz(5)..ref_freq_max_vco; let ref_freq_hz: HertzU32 = xosc_frequency .to_Hz() .checked_div(u32::from(config.refdiv)) .ok_or(Error::BadArgument)? .Hz(); if !ref_freq_range.contains(&ref_freq_hz) { return Err(Error::RefFreqOutOfRange); } let fbdiv = vco_freq .to_Hz() .checked_div(ref_freq_hz.to_Hz()) .ok_or(Error::BadArgument)?; let fbdiv: u16 = fbdiv.try_into().map_err(|_| Error::BadArgument)?; if !FBDIV_RANGE.contains(&fbdiv) { return Err(Error::FeedbackDivOutOfRange); } let refdiv = config.refdiv; let post_div1 = config.post_div1; let post_div2 = config.post_div2; let frequency: HertzU32 = ((ref_freq_hz / u32::from(refdiv)) * u32::from(fbdiv)) / (u32::from(post_div1) * u32::from(post_div2)); Ok(PhaseLockedLoop { state: Disabled { refdiv, fbdiv, post_div1, post_div2, frequency, }, device: dev, }) } /// Configures and starts the PLL : it switches to Locking state. pub fn initialize(self, resets: &mut rp2040_pac::RESETS) -> PhaseLockedLoop { self.device.reset_bring_up(resets); // Turn off PLL in case it is already running self.device.pwr.reset(); self.device.fbdiv_int.reset(); self.device.cs.write(|w| unsafe { w.refdiv().bits(self.state.refdiv); w }); self.device.fbdiv_int.write(|w| unsafe { w.fbdiv_int().bits(self.state.fbdiv); w }); // Turn on PLL self.device.pwr.modify(|_, w| { w.pd().clear_bit(); w.vcopd().clear_bit(); w }); let post_div1 = self.state.post_div1; let post_div2 = self.state.post_div2; let frequency = self.state.frequency; self.transition(Locking { post_div1, post_div2, frequency, }) } } /// A token that's given when the PLL is properly locked, so we can safely transition to the next state. pub struct LockedPLLToken { _private: PhantomData, } impl PhaseLockedLoop { /// Awaits locking of the PLL. pub fn await_lock(&self) -> nb::Result, Infallible> { if self.device.cs.read().lock().bit_is_clear() { return Err(WouldBlock); } Ok(LockedPLLToken { _private: PhantomData, }) } /// Exchanges a token for a Locked PLL. pub fn get_locked(self, _token: LockedPLLToken) -> PhaseLockedLoop { // Set up post dividers self.device.prim.write(|w| unsafe { w.postdiv1().bits(self.state.post_div1); w.postdiv2().bits(self.state.post_div2); w }); // Turn on post divider self.device.pwr.modify(|_, w| { w.postdivpd().clear_bit(); w }); let frequency = self.state.frequency; self.transition(Locked { frequency }) } } impl PhaseLockedLoop { /// Get the operating frequency for the PLL pub fn operating_frequency(&self) -> HertzU32 { self.state.frequency } } /// Blocking helper method to setup the PLL without going through all the steps. pub fn setup_pll_blocking( dev: D, xosc_frequency: HertzU32, config: PLLConfig, clocks: &mut ClocksManager, resets: &mut RESETS, ) -> Result, Error> { // Before we touch PLLs, switch sys and ref cleanly away from their aux sources. nb::block!(clocks.system_clock.reset_source_await()).unwrap(); nb::block!(clocks.reference_clock.reset_source_await()).unwrap(); let initialized_pll = PhaseLockedLoop::new(dev, xosc_frequency.convert(), config)?.initialize(resets); let locked_pll_token = nb::block!(initialized_pll.await_lock()).unwrap(); Ok(initialized_pll.get_locked(locked_pll_token)) }