pub use self::fields::*;
use crate::{analog::adc::Attenuation, peripherals::EFUSE};
mod fields;
pub struct Efuse;
impl Efuse {
pub fn read_base_mac_address() -> [u8; 6] {
Self::read_field_be(MAC)
}
pub fn get_flash_encryption() -> bool {
(Self::read_field_le::<u8>(SPI_BOOT_CRYPT_CNT).count_ones() % 2) != 0
}
pub fn get_rwdt_multiplier() -> u8 {
Self::read_field_le::<u8>(WDT_DELAY_SEL)
}
pub fn get_block_version() -> (u8, u8) {
(
Self::read_field_le::<u8>(BLK_VERSION_MAJOR),
Self::read_field_le::<u8>(BLK_VERSION_MINOR),
)
}
pub fn get_rtc_calib_version() -> u8 {
let (major, _minor) = Self::get_block_version();
if major == 1 {
1
} else {
0
}
}
pub fn get_rtc_calib_init_code(unit: u8, atten: Attenuation) -> Option<u16> {
let version = Self::get_rtc_calib_version();
if version != 1 {
return None;
}
let adc_icode_diff: [u16; 4] = if unit == 0 {
[
Self::read_field_le(ADC1_INIT_CODE_ATTEN0),
Self::read_field_le(ADC1_INIT_CODE_ATTEN1),
Self::read_field_le(ADC1_INIT_CODE_ATTEN2),
Self::read_field_le(ADC1_INIT_CODE_ATTEN3),
]
} else {
[
Self::read_field_le(ADC2_INIT_CODE_ATTEN0),
Self::read_field_le(ADC2_INIT_CODE_ATTEN1),
Self::read_field_le(ADC2_INIT_CODE_ATTEN2),
Self::read_field_le(ADC2_INIT_CODE_ATTEN3),
]
};
let mut adc_icode = [0; 4];
if unit == 0 {
adc_icode[0] = adc_icode_diff[0] + 1850;
adc_icode[1] = adc_icode_diff[1] + adc_icode[0] + 90;
adc_icode[2] = adc_icode_diff[2] + adc_icode[1];
adc_icode[3] = adc_icode_diff[3] + adc_icode[2] + 70;
} else {
adc_icode[0] = adc_icode_diff[0] + 2020;
adc_icode[1] = adc_icode_diff[1] + adc_icode[0];
adc_icode[2] = adc_icode_diff[2] + adc_icode[1];
adc_icode[3] = adc_icode_diff[3] + adc_icode[2];
}
Some(
adc_icode[match atten {
Attenuation::Attenuation0dB => 0,
Attenuation::Attenuation2p5dB => 1,
Attenuation::Attenuation6dB => 2,
Attenuation::Attenuation11dB => 3,
}],
)
}
pub fn get_rtc_calib_cal_mv(_unit: u8, _atten: Attenuation) -> u16 {
850
}
pub fn get_rtc_calib_cal_code(unit: u8, atten: Attenuation) -> Option<u16> {
let version = Self::get_rtc_calib_version();
if version != 1 {
return None;
}
let adc_vol_diff: [u16; 8] = [
Self::read_field_le(ADC1_CAL_VOL_ATTEN0),
Self::read_field_le(ADC1_CAL_VOL_ATTEN1),
Self::read_field_le(ADC1_CAL_VOL_ATTEN2),
Self::read_field_le(ADC1_CAL_VOL_ATTEN3),
Self::read_field_le(ADC2_CAL_VOL_ATTEN0),
Self::read_field_le(ADC2_CAL_VOL_ATTEN1),
Self::read_field_le(ADC2_CAL_VOL_ATTEN2),
Self::read_field_le(ADC2_CAL_VOL_ATTEN3),
];
let mut adc1_vol = [0; 4];
let mut adc2_vol = [0; 4];
adc1_vol[3] = adc_vol_diff[3] + 900;
adc1_vol[2] = adc_vol_diff[2] + adc1_vol[3] + 800;
adc1_vol[1] = adc_vol_diff[1] + adc1_vol[2] + 700;
adc1_vol[0] = adc_vol_diff[0] + adc1_vol[1] + 800;
adc2_vol[3] = adc1_vol[3] - adc_vol_diff[7] + 15;
adc2_vol[2] = adc1_vol[2] - adc_vol_diff[6] + 20;
adc2_vol[1] = adc1_vol[1] - adc_vol_diff[5] + 10;
adc2_vol[0] = adc1_vol[0] - adc_vol_diff[4] + 40;
let atten = match atten {
Attenuation::Attenuation0dB => 0,
Attenuation::Attenuation2p5dB => 1,
Attenuation::Attenuation6dB => 2,
Attenuation::Attenuation11dB => 3,
};
Some(if unit == 0 {
adc1_vol[atten]
} else {
adc2_vol[atten]
})
}
}
#[derive(Copy, Clone)]
pub(crate) enum EfuseBlock {
Block0,
Block1,
Block2,
Block3,
Block4,
Block5,
Block6,
Block7,
Block8,
Block9,
Block10,
}
impl EfuseBlock {
pub(crate) fn address(self) -> *const u32 {
use EfuseBlock::*;
let efuse = unsafe { &*EFUSE::ptr() };
match self {
Block0 => efuse.rd_wr_dis().as_ptr(),
Block1 => efuse.rd_mac_spi_sys_0().as_ptr(),
Block2 => efuse.rd_sys_part1_data0().as_ptr(),
Block3 => efuse.rd_usr_data0().as_ptr(),
Block4 => efuse.rd_key0_data0().as_ptr(),
Block5 => efuse.rd_key1_data0().as_ptr(),
Block6 => efuse.rd_key2_data0().as_ptr(),
Block7 => efuse.rd_key3_data0().as_ptr(),
Block8 => efuse.rd_key4_data0().as_ptr(),
Block9 => efuse.rd_key5_data0().as_ptr(),
Block10 => efuse.rd_sys_part2_data0().as_ptr(),
}
}
}