stm32/rcc: port c0 to new api. Add c0 HSIKER/HSISYS support.
This commit is contained in:
		
							parent
							
								
									c8c4b0b701
								
							
						
					
					
						commit
						ae266f3bf5
					
				| @ -70,7 +70,7 @@ rand_core = "0.6.3" | |||||||
| sdio-host = "0.5.0" | sdio-host = "0.5.0" | ||||||
| critical-section = "1.1" | critical-section = "1.1" | ||||||
| #stm32-metapac = { version = "15" } | #stm32-metapac = { version = "15" } | ||||||
| stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-e853cf944b150898312984d092d63926970c340d" } | stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-e7f91751fbbf856e0cb30e50ae6db79f0409b085" } | ||||||
| vcell = "0.1.3" | vcell = "0.1.3" | ||||||
| bxcan = "0.7.0" | bxcan = "0.7.0" | ||||||
| nb = "1.0.0" | nb = "1.0.0" | ||||||
| @ -94,7 +94,7 @@ critical-section = { version = "1.1", features = ["std"] } | |||||||
| proc-macro2 = "1.0.36" | proc-macro2 = "1.0.36" | ||||||
| quote = "1.0.15" | quote = "1.0.15" | ||||||
| #stm32-metapac = { version = "15", default-features = false, features = ["metadata"]} | #stm32-metapac = { version = "15", default-features = false, features = ["metadata"]} | ||||||
| stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-e853cf944b150898312984d092d63926970c340d", default-features = false, features = ["metadata"]} | stm32-metapac = { git = "https://github.com/embassy-rs/stm32-data-generated", tag = "stm32-data-e7f91751fbbf856e0cb30e50ae6db79f0409b085", default-features = false, features = ["metadata"]} | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| [features] | [features] | ||||||
|  | |||||||
| @ -1,25 +1,56 @@ | |||||||
| use crate::pac::flash::vals::Latency; | use crate::pac::flash::vals::Latency; | ||||||
| use crate::pac::rcc::vals::Sw; | pub use crate::pac::rcc::vals::{ | ||||||
| pub use crate::pac::rcc::vals::{Hpre as AHBPrescaler, Hsidiv as HSIPrescaler, Ppre as APBPrescaler}; |     Hpre as AHBPrescaler, Hsidiv as HsiSysDiv, Hsikerdiv as HsiKerDiv, Ppre as APBPrescaler, Sw as Sysclk, | ||||||
|  | }; | ||||||
| use crate::pac::{FLASH, RCC}; | use crate::pac::{FLASH, RCC}; | ||||||
| use crate::time::Hertz; | use crate::time::Hertz; | ||||||
| 
 | 
 | ||||||
| /// HSI speed
 | /// HSI speed
 | ||||||
| pub const HSI_FREQ: Hertz = Hertz(48_000_000); | pub const HSI_FREQ: Hertz = Hertz(16_000_000); | ||||||
| 
 | 
 | ||||||
| /// System clock mux source
 | /// HSE Mode
 | ||||||
| #[derive(Clone, Copy)] | #[derive(Clone, Copy, Eq, PartialEq)] | ||||||
| pub enum Sysclk { | pub enum HseMode { | ||||||
|     HSE(Hertz), |     /// crystal/ceramic oscillator (HSEBYP=0)
 | ||||||
|     HSI(HSIPrescaler), |     Oscillator, | ||||||
|     LSI, |     /// external analog clock (low swing) (HSEBYP=1)
 | ||||||
|  |     Bypass, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// HSE Configuration
 | ||||||
|  | #[derive(Clone, Copy, Eq, PartialEq)] | ||||||
|  | pub struct Hse { | ||||||
|  |     /// HSE frequency.
 | ||||||
|  |     pub freq: Hertz, | ||||||
|  |     /// HSE mode.
 | ||||||
|  |     pub mode: HseMode, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// HSI Configuration
 | ||||||
|  | #[derive(Clone, Copy, Eq, PartialEq)] | ||||||
|  | pub struct Hsi { | ||||||
|  |     /// Division factor for HSISYS clock. Default is 4.
 | ||||||
|  |     pub sys_div: HsiSysDiv, | ||||||
|  |     /// Division factor for HSIKER clock. Default is 3.
 | ||||||
|  |     pub ker_div: HsiKerDiv, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Clocks configutation
 | /// Clocks configutation
 | ||||||
|  | #[non_exhaustive] | ||||||
| pub struct Config { | pub struct Config { | ||||||
|  |     /// HSI Configuration
 | ||||||
|  |     pub hsi: Option<Hsi>, | ||||||
|  | 
 | ||||||
|  |     /// HSE Configuration
 | ||||||
|  |     pub hse: Option<Hse>, | ||||||
|  | 
 | ||||||
|  |     /// System Clock Configuration
 | ||||||
|     pub sys: Sysclk, |     pub sys: Sysclk, | ||||||
|  | 
 | ||||||
|     pub ahb_pre: AHBPrescaler, |     pub ahb_pre: AHBPrescaler, | ||||||
|     pub apb_pre: APBPrescaler, |     pub apb1_pre: APBPrescaler, | ||||||
|  | 
 | ||||||
|  |     /// Low-Speed Clock Configuration
 | ||||||
|     pub ls: super::LsConfig, |     pub ls: super::LsConfig, | ||||||
| 
 | 
 | ||||||
|     /// Per-peripheral kernel clock selection muxes
 |     /// Per-peripheral kernel clock selection muxes
 | ||||||
| @ -30,9 +61,14 @@ impl Default for Config { | |||||||
|     #[inline] |     #[inline] | ||||||
|     fn default() -> Config { |     fn default() -> Config { | ||||||
|         Config { |         Config { | ||||||
|             sys: Sysclk::HSI(HSIPrescaler::DIV1), |             hsi: Some(Hsi { | ||||||
|  |                 sys_div: HsiSysDiv::DIV4, | ||||||
|  |                 ker_div: HsiKerDiv::DIV3, | ||||||
|  |             }), | ||||||
|  |             hse: None, | ||||||
|  |             sys: Sysclk::HSISYS, | ||||||
|             ahb_pre: AHBPrescaler::DIV1, |             ahb_pre: AHBPrescaler::DIV1, | ||||||
|             apb_pre: APBPrescaler::DIV1, |             apb1_pre: APBPrescaler::DIV1, | ||||||
|             ls: Default::default(), |             ls: Default::default(), | ||||||
|             mux: Default::default(), |             mux: Default::default(), | ||||||
|         } |         } | ||||||
| @ -40,111 +76,109 @@ impl Default for Config { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub(crate) unsafe fn init(config: Config) { | pub(crate) unsafe fn init(config: Config) { | ||||||
|     let (sys_clk, sw) = match config.sys { |     // Configure HSI
 | ||||||
|         Sysclk::HSI(div) => { |     let (hsi, hsisys, hsiker) = match config.hsi { | ||||||
|             // Enable HSI
 |         None => { | ||||||
|             RCC.cr().write(|w| { |             RCC.cr().modify(|w| w.set_hsion(false)); | ||||||
|                 w.set_hsidiv(div); |             (None, None, None) | ||||||
|                 w.set_hsion(true) |         } | ||||||
|  |         Some(hsi) => { | ||||||
|  |             RCC.cr().modify(|w| { | ||||||
|  |                 w.set_hsidiv(hsi.sys_div); | ||||||
|  |                 w.set_hsikerdiv(hsi.ker_div); | ||||||
|  |                 w.set_hsion(true); | ||||||
|             }); |             }); | ||||||
|             while !RCC.cr().read().hsirdy() {} |             while !RCC.cr().read().hsirdy() {} | ||||||
| 
 |             ( | ||||||
|             (HSI_FREQ / div, Sw::HSI) |                 Some(HSI_FREQ), | ||||||
|         } |                 Some(HSI_FREQ / hsi.sys_div), | ||||||
|         Sysclk::HSE(freq) => { |                 Some(HSI_FREQ / hsi.ker_div), | ||||||
|             // Enable HSE
 |             ) | ||||||
|             RCC.cr().write(|w| w.set_hseon(true)); |  | ||||||
|             while !RCC.cr().read().hserdy() {} |  | ||||||
| 
 |  | ||||||
|             (freq, Sw::HSE) |  | ||||||
|         } |  | ||||||
|         Sysclk::LSI => { |  | ||||||
|             // Enable LSI
 |  | ||||||
|             RCC.csr2().write(|w| w.set_lsion(true)); |  | ||||||
|             while !RCC.csr2().read().lsirdy() {} |  | ||||||
|             (super::LSI_FREQ, Sw::LSI) |  | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  |     // Configure HSE
 | ||||||
|  |     let hse = match config.hse { | ||||||
|  |         None => { | ||||||
|  |             RCC.cr().modify(|w| w.set_hseon(false)); | ||||||
|  |             None | ||||||
|  |         } | ||||||
|  |         Some(hse) => { | ||||||
|  |             match hse.mode { | ||||||
|  |                 HseMode::Bypass => assert!(max::HSE_BYP.contains(&hse.freq)), | ||||||
|  |                 HseMode::Oscillator => assert!(max::HSE_OSC.contains(&hse.freq)), | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             RCC.cr().modify(|w| w.set_hsebyp(hse.mode != HseMode::Oscillator)); | ||||||
|  |             RCC.cr().modify(|w| w.set_hseon(true)); | ||||||
|  |             while !RCC.cr().read().hserdy() {} | ||||||
|  |             Some(hse.freq) | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     let sys = match config.sys { | ||||||
|  |         Sysclk::HSISYS => unwrap!(hsisys), | ||||||
|  |         Sysclk::HSE => unwrap!(hse), | ||||||
|  |         _ => unreachable!(), | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     assert!(max::SYSCLK.contains(&sys)); | ||||||
|  | 
 | ||||||
|  |     // Calculate the AHB frequency (HCLK), among other things so we can calculate the correct flash read latency.
 | ||||||
|  |     let hclk = sys / config.ahb_pre; | ||||||
|  |     assert!(max::HCLK.contains(&hclk)); | ||||||
|  | 
 | ||||||
|  |     let (pclk1, pclk1_tim) = super::util::calc_pclk(hclk, config.apb1_pre); | ||||||
|  |     assert!(max::PCLK.contains(&pclk1)); | ||||||
|  | 
 | ||||||
|  |     let latency = match hclk.0 { | ||||||
|  |         ..=24_000_000 => Latency::WS0, | ||||||
|  |         _ => Latency::WS1, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     // Configure flash read access latency based on voltage scale and frequency
 | ||||||
|  |     FLASH.acr().modify(|w| { | ||||||
|  |         w.set_latency(latency); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // Spin until the effective flash latency is set.
 | ||||||
|  |     while FLASH.acr().read().latency() != latency {} | ||||||
|  | 
 | ||||||
|  |     // Now that boost mode and flash read access latency are configured, set up SYSCLK
 | ||||||
|  |     RCC.cfgr().modify(|w| { | ||||||
|  |         w.set_sw(config.sys); | ||||||
|  |         w.set_hpre(config.ahb_pre); | ||||||
|  |         w.set_ppre(config.apb1_pre); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|     let rtc = config.ls.init(); |     let rtc = config.ls.init(); | ||||||
| 
 | 
 | ||||||
|     // Determine the flash latency implied by the target clock speed
 |  | ||||||
|     // RM0454 § 3.3.4:
 |  | ||||||
|     let target_flash_latency = if sys_clk <= Hertz(24_000_000) { |  | ||||||
|         Latency::WS0 |  | ||||||
|     } else { |  | ||||||
|         Latency::WS1 |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     // Increase the number of cycles we wait for flash if the new value is higher
 |  | ||||||
|     // There's no harm in waiting a little too much before the clock change, but we'll
 |  | ||||||
|     // crash immediately if we don't wait enough after the clock change
 |  | ||||||
|     let mut set_flash_latency_after = false; |  | ||||||
|     FLASH.acr().modify(|w| { |  | ||||||
|         // Is the current flash latency less than what we need at the new SYSCLK?
 |  | ||||||
|         if w.latency().to_bits() <= target_flash_latency.to_bits() { |  | ||||||
|             // We must increase the number of wait states now
 |  | ||||||
|             w.set_latency(target_flash_latency) |  | ||||||
|         } else { |  | ||||||
|             // We may decrease the number of wait states later
 |  | ||||||
|             set_flash_latency_after = true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // RM0490 § 3.3.4:
 |  | ||||||
|         // > Prefetch is enabled by setting the PRFTEN bit of the FLASH access control register
 |  | ||||||
|         // > (FLASH_ACR). This feature is useful if at least one wait state is needed to access the
 |  | ||||||
|         // > Flash memory.
 |  | ||||||
|         //
 |  | ||||||
|         // Enable flash prefetching if we have at least one wait state, and disable it otherwise.
 |  | ||||||
|         w.set_prften(target_flash_latency.to_bits() > 0); |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if !set_flash_latency_after { |  | ||||||
|         // Spin until the effective flash latency is compatible with the clock change
 |  | ||||||
|         while FLASH.acr().read().latency() < target_flash_latency {} |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Configure SYSCLK source, HCLK divisor, and PCLK divisor all at once
 |  | ||||||
|     RCC.cfgr().modify(|w| { |  | ||||||
|         w.set_sw(sw); |  | ||||||
|         w.set_hpre(config.ahb_pre); |  | ||||||
|         w.set_ppre(config.apb_pre); |  | ||||||
|     }); |  | ||||||
|     // Spin until the SYSCLK changes have taken effect
 |  | ||||||
|     loop { |  | ||||||
|         let cfgr = RCC.cfgr().read(); |  | ||||||
|         if cfgr.sw() == sw && cfgr.hpre() == config.ahb_pre && cfgr.ppre() == config.apb_pre { |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Set the flash latency to require fewer wait states
 |  | ||||||
|     if set_flash_latency_after { |  | ||||||
|         FLASH.acr().modify(|w| w.set_latency(target_flash_latency)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let ahb_freq = sys_clk / config.ahb_pre; |  | ||||||
| 
 |  | ||||||
|     let (apb_freq, apb_tim_freq) = match config.apb_pre { |  | ||||||
|         APBPrescaler::DIV1 => (ahb_freq, ahb_freq), |  | ||||||
|         pre => { |  | ||||||
|             let freq = ahb_freq / pre; |  | ||||||
|             (freq, freq * 2u32) |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     config.mux.init(); |     config.mux.init(); | ||||||
| 
 | 
 | ||||||
|     // without this, the ringbuffered uart test fails.
 |  | ||||||
|     cortex_m::asm::dsb(); |  | ||||||
| 
 |  | ||||||
|     set_clocks!( |     set_clocks!( | ||||||
|         hsi: None, |         sys: Some(sys), | ||||||
|         lse: None, |         hclk1: Some(hclk), | ||||||
|         sys: Some(sys_clk), |         pclk1: Some(pclk1), | ||||||
|         hclk1: Some(ahb_freq), |         pclk1_tim: Some(pclk1_tim), | ||||||
|         pclk1: Some(apb_freq), |         hsi: hsi, | ||||||
|         pclk1_tim: Some(apb_tim_freq), |         hsiker: hsiker, | ||||||
|  |         hse: hse, | ||||||
|         rtc: rtc, |         rtc: rtc, | ||||||
|  | 
 | ||||||
|  |         // TODO
 | ||||||
|  |         lsi: None, | ||||||
|  |         lse: None, | ||||||
|     ); |     ); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | mod max { | ||||||
|  |     use core::ops::RangeInclusive; | ||||||
|  | 
 | ||||||
|  |     use crate::time::Hertz; | ||||||
|  | 
 | ||||||
|  |     pub(crate) const HSE_OSC: RangeInclusive<Hertz> = Hertz(4_000_000)..=Hertz(48_000_000); | ||||||
|  |     pub(crate) const HSE_BYP: RangeInclusive<Hertz> = Hertz(0)..=Hertz(48_000_000); | ||||||
|  |     pub(crate) const SYSCLK: RangeInclusive<Hertz> = Hertz(0)..=Hertz(48_000_000); | ||||||
|  |     pub(crate) const PCLK: RangeInclusive<Hertz> = Hertz(8)..=Hertz(48_000_000); | ||||||
|  |     pub(crate) const HCLK: RangeInclusive<Hertz> = Hertz(0)..=Hertz(48_000_000); | ||||||
|  | } | ||||||
|  | |||||||
| @ -260,6 +260,17 @@ pub fn config() -> Config { | |||||||
|     #[allow(unused_mut)] |     #[allow(unused_mut)] | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
| 
 | 
 | ||||||
|  |     #[cfg(feature = "stm32c031c6")] | ||||||
|  |     { | ||||||
|  |         config.rcc.hsi = Some(Hsi { | ||||||
|  |             sys_div: HsiSysDiv::DIV1, // 48Mhz
 | ||||||
|  |             ker_div: HsiKerDiv::DIV3, // 16Mhz
 | ||||||
|  |         }); | ||||||
|  |         config.rcc.sys = Sysclk::HSISYS; | ||||||
|  |         config.rcc.ahb_pre = AHBPrescaler::DIV1; | ||||||
|  |         config.rcc.apb1_pre = APBPrescaler::DIV1; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     #[cfg(feature = "stm32g071rb")] |     #[cfg(feature = "stm32g071rb")] | ||||||
|     { |     { | ||||||
|         config.rcc.hsi = true; |         config.rcc.hsi = true; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user