#![allow(dead_code, reason = "Some of this is bound to be unused")]
#![allow(missing_docs, reason = "Experimental")]
use crate::{
peripherals::{I2C_ANA_MST, LP_CLKRST, PCR, PMU, UART0, UART1},
soc::regi2c,
};
define_clock_tree_types!();
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
#[allow(
clippy::enum_variant_names,
reason = "MHz suffix indicates physical unit."
)]
#[non_exhaustive]
pub enum CpuClock {
#[default]
_80MHz = 80,
_160MHz = 160,
_240MHz = 240,
}
impl CpuClock {
const PRESET_80: ClockConfig = ClockConfig {
xtal_clk: None,
hp_root_clk: Some(HpRootClkConfig::PllF160m),
cpu_clk: Some(CpuClkConfig::new(1)),
ahb_clk: Some(AhbClkConfig::new(3)), apb_clk: Some(ApbClkConfig::new(0)),
lp_fast_clk: Some(LpFastClkConfig::RcFast),
lp_slow_clk: Some(LpSlowClkConfig::RcSlow),
crypto_clk: Some(CryptoClkConfig::PllF480m),
timg_calibration_clock: None,
};
const PRESET_160: ClockConfig = ClockConfig {
xtal_clk: None,
hp_root_clk: Some(HpRootClkConfig::PllF160m),
cpu_clk: Some(CpuClkConfig::new(0)),
ahb_clk: Some(AhbClkConfig::new(3)), apb_clk: Some(ApbClkConfig::new(0)),
lp_fast_clk: Some(LpFastClkConfig::RcFast),
lp_slow_clk: Some(LpSlowClkConfig::RcSlow),
crypto_clk: Some(CryptoClkConfig::PllF480m),
timg_calibration_clock: None,
};
const PRESET_240: ClockConfig = ClockConfig {
xtal_clk: None,
hp_root_clk: Some(HpRootClkConfig::PllF240m),
cpu_clk: Some(CpuClkConfig::new(0)),
ahb_clk: Some(AhbClkConfig::new(5)), apb_clk: Some(ApbClkConfig::new(0)),
lp_fast_clk: Some(LpFastClkConfig::RcFast),
lp_slow_clk: Some(LpSlowClkConfig::RcSlow),
crypto_clk: Some(CryptoClkConfig::PllF480m),
timg_calibration_clock: None,
};
}
impl From<CpuClock> for ClockConfig {
fn from(value: CpuClock) -> ClockConfig {
match value {
CpuClock::_80MHz => CpuClock::PRESET_80,
CpuClock::_160MHz => CpuClock::PRESET_160,
CpuClock::_240MHz => CpuClock::PRESET_240,
}
}
}
impl Default for ClockConfig {
fn default() -> Self {
Self::from(CpuClock::default())
}
}
impl ClockConfig {
pub(crate) fn try_get_preset(self) -> Option<CpuClock> {
match self {
v if v == CpuClock::PRESET_80 => Some(CpuClock::_80MHz),
v if v == CpuClock::PRESET_160 => Some(CpuClock::_160MHz),
v if v == CpuClock::PRESET_240 => Some(CpuClock::_240MHz),
_ => None,
}
}
pub(crate) fn configure(mut self) {
self.xtal_clk = if PCR::regs().sysclk_conf().read().clk_xtal_freq().bits() == 40 {
Some(XtalClkConfig::_40)
} else {
Some(XtalClkConfig::_48)
};
self.apply();
}
}
fn configure_xtal_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<XtalClkConfig>,
_config: XtalClkConfig,
) {
}
fn enable_pll_clk_impl(clocks: &mut ClockTree, en: bool) {
if en {
PMU::regs().imm_hp_ck_power().write(|w| {
w.tie_high_xpd_bb_i2c().set_bit();
w.tie_high_xpd_bbpll().set_bit();
w.tie_high_xpd_bbpll_i2c().set_bit()
});
PMU::regs()
.imm_hp_ck_power()
.write(|w| w.tie_high_global_bbpll_icg().set_bit());
} else {
PMU::regs()
.imm_hp_ck_power()
.write(|w| w.tie_low_global_bbpll_icg().set_bit());
PMU::regs().imm_hp_ck_power().write(|w| {
w.tie_low_xpd_bb_i2c().set_bit();
w.tie_low_xpd_bbpll().set_bit();
w.tie_low_xpd_bbpll_i2c().set_bit()
});
return;
}
I2C_ANA_MST::regs().ana_conf0().modify(|_, w| {
w.bbpll_stop_force_high().clear_bit();
w.bbpll_stop_force_low().set_bit()
});
let div7_0: u8;
let dr1: u8;
let dr3: u8;
match unwrap!(clocks.xtal_clk()) {
XtalClkConfig::_40 => {
div7_0 = 12;
dr1 = 0;
dr3 = 0;
}
XtalClkConfig::_48 => {
div7_0 = 10;
dr1 = 1;
dr3 = 1;
}
}
const DIV_REF: u8 = 1; const DCHGP: u8 = 5;
const DBIAS: u8 = 3;
const HREF: u8 = 3;
const LREF: u8 = 1;
const I2C_BBPLL_OC_DCHGP_LSB: u32 = 4;
const I2C_BBPLL_OC_DHREF_SEL_LSB: u32 = 4;
const I2C_BBPLL_OC_DLREF_SEL_LSB: u32 = 6;
const I2C_BBPLL_LREF: u8 = (DCHGP << I2C_BBPLL_OC_DCHGP_LSB) | DIV_REF;
regi2c::I2C_BBPLL_OC_REF.write_reg(I2C_BBPLL_LREF);
regi2c::I2C_BBPLL_OC_DIV_REG.write_reg(div7_0);
regi2c::I2C_BBPLL_OC_DR1.write_field(dr1);
regi2c::I2C_BBPLL_OC_DR3.write_field(dr3);
regi2c::I2C_BBPLL_OC_DLREF_SEL.write_field(LREF);
regi2c::I2C_BBPLL_OC_DHREF_SEL.write_field(HREF);
regi2c::I2C_BBPLL_OC_VCO_DBIAS.write_field(DBIAS);
while I2C_ANA_MST::regs()
.ana_conf0()
.read()
.cal_done()
.bit_is_clear()
{}
crate::rom::ets_delay_us(10);
I2C_ANA_MST::regs().ana_conf0().modify(|_, w| {
w.bbpll_stop_force_high().set_bit();
w.bbpll_stop_force_low().clear_bit()
});
}
fn enable_rc_fast_clk_impl(_clocks: &mut ClockTree, en: bool) {
LP_CLKRST::regs()
.clk_to_hp()
.modify(|_, w| w.icg_hp_fosc().bit(en));
}
fn enable_xtal32k_clk_impl(_clocks: &mut ClockTree, en: bool) {
LP_CLKRST::regs()
.clk_to_hp()
.modify(|_, w| w.icg_hp_xtal32k().bit(en));
}
fn enable_osc_slow_clk_impl(_clocks: &mut ClockTree, en: bool) {
LP_CLKRST::regs()
.clk_to_hp()
.modify(|_, w| w.icg_hp_osc32k().bit(en));
}
fn enable_rc_slow_clk_impl(_clocks: &mut ClockTree, en: bool) {
LP_CLKRST::regs()
.clk_to_hp()
.modify(|_, w| w.icg_hp_sosc().bit(en));
}
fn enable_pll_f12m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_12m_clk_en().bit(en));
}
fn enable_pll_f20m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_20m_clk_en().bit(en));
}
fn enable_pll_f40m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_40m_clk_en().bit(en));
}
fn enable_pll_f48m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_48m_clk_en().bit(en));
}
fn enable_pll_f60m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_60m_clk_en().bit(en));
}
fn enable_pll_f80m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_80m_clk_en().bit(en));
}
fn enable_pll_f120m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_120m_clk_en().bit(en));
}
fn enable_pll_f160m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_160m_clk_en().bit(en));
}
fn enable_pll_f240m_impl(_clocks: &mut ClockTree, en: bool) {
PCR::regs()
.pll_div_clk_en()
.modify(|_, w| w.pll_240m_clk_en().bit(en));
}
fn enable_hp_root_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_hp_root_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<HpRootClkConfig>,
new_config: HpRootClkConfig,
) {
PCR::regs().sysclk_conf().modify(|_, w| unsafe {
w.soc_clk_sel().bits(match new_config {
HpRootClkConfig::Xtal => 0,
HpRootClkConfig::RcFast => 1,
HpRootClkConfig::PllF160m => 2,
HpRootClkConfig::PllF240m => 3,
})
});
}
fn enable_cpu_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_cpu_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<CpuClkConfig>,
new_config: CpuClkConfig,
) {
PCR::regs()
.cpu_freq_conf()
.modify(|_, w| unsafe { w.cpu_div_num().bits(new_config.divisor() as u8) });
PCR::regs()
.bus_clk_update()
.write(|w| w.bus_clock_update().set_bit());
}
fn enable_ahb_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_ahb_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<AhbClkConfig>,
new_config: AhbClkConfig,
) {
PCR::regs()
.ahb_freq_conf()
.modify(|_, w| unsafe { w.ahb_div_num().bits(new_config.divisor() as u8) });
PCR::regs()
.bus_clk_update()
.write(|w| w.bus_clock_update().set_bit());
}
fn enable_apb_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_apb_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<ApbClkConfig>,
new_config: ApbClkConfig,
) {
PCR::regs()
.apb_freq_conf()
.modify(|_, w| unsafe { w.apb_div_num().bits(new_config.divisor() as u8) });
}
fn enable_xtal_d2_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn enable_lp_fast_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_lp_fast_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<LpFastClkConfig>,
new_config: LpFastClkConfig,
) {
LP_CLKRST::regs().lp_clk_conf().modify(|_, w| unsafe {
w.fast_clk_sel().bits(match new_config {
LpFastClkConfig::RcFast => 0,
LpFastClkConfig::XtalD2 => 1,
LpFastClkConfig::Xtal => 2,
})
});
}
fn enable_lp_slow_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_lp_slow_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<LpSlowClkConfig>,
new_config: LpSlowClkConfig,
) {
LP_CLKRST::regs().lp_clk_conf().modify(|_, w| unsafe {
w.slow_clk_sel().bits(match new_config {
LpSlowClkConfig::RcSlow => 0,
LpSlowClkConfig::Xtal32k => 1,
LpSlowClkConfig::OscSlow => 3,
})
});
}
fn enable_crypto_clk_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_crypto_clk_impl(
_clocks: &mut ClockTree,
_old_config: Option<CryptoClkConfig>,
new_config: CryptoClkConfig,
) {
PCR::regs().sec_conf().modify(|_, w| unsafe {
w.sec_clk_sel().bits(match new_config {
CryptoClkConfig::Xtal => 0,
CryptoClkConfig::Fosc => 1,
CryptoClkConfig::PllF480m => 2,
})
});
}
fn enable_timg_calibration_clock_impl(_clocks: &mut ClockTree, _en: bool) {
}
fn configure_timg_calibration_clock_impl(
_clocks: &mut ClockTree,
_old_config: Option<TimgCalibrationClockConfig>,
new_config: TimgCalibrationClockConfig,
) {
PCR::regs().ctrl_32k_conf().modify(|_, w| unsafe {
w._32k_sel().bits(match new_config {
TimgCalibrationClockConfig::OscSlowClk => 0,
TimgCalibrationClockConfig::Xtal32kClk => 1,
TimgCalibrationClockConfig::RcSlowClk => 3,
TimgCalibrationClockConfig::RcFastDivClk => 4,
})
});
}
impl ParlIoInstance {
fn enable_rx_clock_impl(self, _clocks: &mut ClockTree, en: bool) {
PCR::regs()
.parl_clk_rx_conf()
.modify(|_, w| w.parl_clk_rx_en().bit(en));
}
fn configure_rx_clock_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<ParlIoRxClockConfig>,
new_config: ParlIoRxClockConfig,
) {
PCR::regs().parl_clk_rx_conf().modify(|_, w| unsafe {
w.parl_clk_rx_sel().bits(match new_config {
ParlIoRxClockConfig::XtalClk => 0,
ParlIoRxClockConfig::RcFastClk => 1,
ParlIoRxClockConfig::PllF240m => 2,
})
});
}
fn enable_tx_clock_impl(self, _clocks: &mut ClockTree, en: bool) {
PCR::regs()
.parl_clk_tx_conf()
.modify(|_, w| w.parl_clk_tx_en().bit(en));
}
fn configure_tx_clock_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<ParlIoTxClockConfig>,
new_config: ParlIoTxClockConfig,
) {
PCR::regs().parl_clk_tx_conf().modify(|_, w| unsafe {
w.parl_clk_tx_sel().bits(match new_config {
ParlIoTxClockConfig::XtalClk => 0,
ParlIoTxClockConfig::RcFastClk => 1,
ParlIoTxClockConfig::PllF240m => 2,
})
});
}
}
impl RmtInstance {
fn enable_sclk_impl(self, _clocks: &mut ClockTree, en: bool) {
PCR::regs().rmt_pd_ctrl().modify(|_, w| {
w.rmt_mem_force_pu().bit(en);
w.rmt_mem_force_pd().bit(!en)
});
PCR::regs()
.rmt_sclk_conf()
.modify(|_, w| w.sclk_en().bit(en));
}
fn configure_sclk_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<RmtSclkConfig>,
new_config: RmtSclkConfig,
) {
PCR::regs().rmt_sclk_conf().modify(|_, w| unsafe {
w.sclk_sel().bits(match new_config {
RmtSclkConfig::XtalClk => 0,
RmtSclkConfig::RcFastClk => 1,
RmtSclkConfig::PllF80m => 2,
})
});
}
}
impl TimgInstance {
fn enable_function_clock_impl(self, _clocks: &mut ClockTree, en: bool) {
let timg = match self {
TimgInstance::Timg0 => 0,
TimgInstance::Timg1 => 1,
};
PCR::regs()
.timergroup(timg)
.timer_clk_conf()
.modify(|_, w| w.timer_clk_en().bit(en));
}
fn configure_function_clock_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<TimgFunctionClockConfig>,
new_config: TimgFunctionClockConfig,
) {
let timg = match self {
TimgInstance::Timg0 => 0,
TimgInstance::Timg1 => 1,
};
PCR::regs()
.timergroup(timg)
.timer_clk_conf()
.modify(|_, w| unsafe {
w.timer_clk_sel().bits(match new_config {
TimgFunctionClockConfig::XtalClk => 0,
TimgFunctionClockConfig::RcFastClk => 1,
TimgFunctionClockConfig::PllF80m => 2,
})
});
}
fn enable_wdt_clock_impl(self, _clocks: &mut ClockTree, en: bool) {
let timg = match self {
TimgInstance::Timg0 => 0,
TimgInstance::Timg1 => 1,
};
PCR::regs()
.timergroup(timg)
.wdt_clk_conf()
.modify(|_, w| w.wdt_clk_en().bit(en));
}
fn configure_wdt_clock_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<TimgWdtClockConfig>,
new_config: TimgWdtClockConfig,
) {
let timg = match self {
TimgInstance::Timg0 => 0,
TimgInstance::Timg1 => 1,
};
PCR::regs()
.timergroup(timg)
.wdt_clk_conf()
.modify(|_, w| unsafe {
w.wdt_clk_sel().bits(match new_config {
TimgWdtClockConfig::XtalClk => 0,
TimgWdtClockConfig::RcFastClk => 1,
TimgWdtClockConfig::PllF80m => 2,
})
});
}
}
impl UartInstance {
fn enable_function_clock_impl(self, _clocks: &mut ClockTree, en: bool) {
let uart = match self {
UartInstance::Uart0 => {
return;
}
UartInstance::Uart1 => 1,
};
PCR::regs()
.uart(uart)
.clk_conf()
.modify(|_, w| w.sclk_en().bit(en));
}
fn configure_function_clock_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<UartFunctionClockConfig>,
new_config: UartFunctionClockConfig,
) {
PCR::regs()
.uart(match self {
UartInstance::Uart0 => 0,
UartInstance::Uart1 => 1,
})
.clk_conf()
.modify(|_, w| unsafe {
w.sclk_sel().bits(match new_config.sclk {
UartFunctionClockSclk::Xtal => 0,
UartFunctionClockSclk::RcFast => 1,
UartFunctionClockSclk::PllF80m => 2,
});
w.sclk_div_a().bits(0);
w.sclk_div_b().bits(0);
w.sclk_div_num().bits(new_config.div_num as _);
w
});
}
fn enable_baud_rate_generator_impl(self, _clocks: &mut ClockTree, _en: bool) {
}
fn configure_baud_rate_generator_impl(
self,
_clocks: &mut ClockTree,
_old_config: Option<UartBaudRateGeneratorConfig>,
new_config: UartBaudRateGeneratorConfig,
) {
let regs = match self {
UartInstance::Uart0 => UART0::regs(),
UartInstance::Uart1 => UART1::regs(),
};
regs.clkdiv().write(|w| unsafe {
w.clkdiv().bits(new_config.integral as _);
w.frag().bits(new_config.fractional as _)
});
}
}