opentitanlib/test_utils/
otp_ctrl.rsuse std::mem;
use std::time::Duration;
use anyhow::{bail, Context};
use thiserror::Error;
use top_earlgrey::top_earlgrey;
use crate::dif::otp_ctrl::{
DaiParam, DirectAccessCmd, Granularity, OtpCtrlReg, OtpCtrlStatus, OtpParamMmap, Partition,
SECRET_PARTITIONS,
};
use crate::io::jtag::Jtag;
use crate::test_utils::poll;
pub struct OtpParam;
impl OtpParam {
pub fn read_param(
jtag: &mut dyn Jtag,
param: DaiParam,
out_buf: &mut [u32],
) -> OtpDaiResult<()> {
let OtpParamMmap { byte_addr, size } = param.mmap();
if mem::size_of_val(out_buf) < size as usize {
let err = OtpDaiError::BufSize {
buf_size: mem::size_of_val(out_buf),
param_size: size,
};
return Err(err);
}
let Partition { access_granule, .. } = param.partition();
match access_granule {
Granularity::B32 => {
let out_iter = out_buf.iter_mut().take(size as usize);
for (idx, out_word) in out_iter.enumerate() {
let addr = byte_addr + (idx * mem::size_of::<u32>()) as u32;
let [lower, _] = OtpDai::read(jtag, addr, access_granule)?;
*out_word = lower;
}
}
Granularity::B64 => {
let out_iter = out_buf.chunks_mut(2).take(size as usize / 2);
for (idx, out_words) in out_iter.enumerate() {
let addr = byte_addr + (idx * mem::size_of::<u64>()) as u32;
let otp_words = OtpDai::read(jtag, addr, access_granule)?;
out_words[0] = otp_words[0];
out_words[1] = otp_words[1];
}
}
}
Ok(())
}
pub fn write_param(jtag: &mut dyn Jtag, param: DaiParam, data: &[u32]) -> OtpDaiResult<()> {
let OtpParamMmap { byte_addr, size } = param.mmap();
if !SECRET_PARTITIONS.contains(¶m.partition()) {
let mut curr_value = [0u32].repeat((size / 4).try_into().unwrap());
Self::read_param(jtag, param, curr_value.as_mut_slice())?;
for word in curr_value.iter() {
if *word != 0u32 {
return Err(OtpDaiError::WriteErrorAlreadyWritten { value: *word });
}
}
}
if mem::size_of_val(data) > size as usize {
let err = OtpDaiError::BufSize {
buf_size: mem::size_of_val(data),
param_size: size,
};
return Err(err);
}
let Partition { access_granule, .. } = param.partition();
match access_granule {
Granularity::B32 => {
let data_iter = data.iter().take(size as usize);
for (idx, data_word) in data_iter.enumerate() {
let addr = byte_addr + (idx * mem::size_of::<u32>()) as u32;
OtpDai::write(jtag, addr, access_granule, [*data_word, 0x00])?;
}
}
Granularity::B64 => {
let data_iter = data.chunks(2).take(size as usize / 2);
for (idx, data_words) in data_iter.enumerate() {
let addr = byte_addr + (idx * mem::size_of::<u64>()) as u32;
OtpDai::write(jtag, addr, access_granule, [data_words[0], data_words[1]])?;
}
}
}
Ok(())
}
}
pub struct OtpPartition;
impl OtpPartition {
pub fn lock(jtag: &mut dyn Jtag, partition: Partition) -> OtpDaiResult<()> {
OtpDai::lock(jtag, partition.byte_addr)
}
pub fn read_digest(jtag: &mut dyn Jtag, partition: Partition) -> OtpDaiResult<[u32; 2]> {
let OtpParamMmap { byte_addr, size } = partition.digest;
assert_eq!(size, 8, "OTP partition digests should be 2 words in size");
OtpDai::read(jtag, byte_addr, Granularity::B64)
}
}
struct OtpDai;
impl OtpDai {
const OTP_CTRL_BASE_ADDR: u32 = top_earlgrey::OTP_CTRL_CORE_BASE_ADDR as u32;
const POLL_TIMEOUT: Duration = Duration::from_millis(500);
const POLL_DELAY: Duration = Duration::from_millis(5);
pub fn read(
jtag: &mut dyn Jtag,
byte_addr: u32,
granule: Granularity,
) -> OtpDaiResult<[u32; 2]> {
Self::wait_for_idle(jtag)?;
let dai_address_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessAddress as u32;
jtag.write_memory32(dai_address_addr, &[byte_addr])
.map_err(|source| OtpDaiError::Jtag { source })?;
let dai_cmd_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessCmd as u32;
jtag.write_memory32(dai_cmd_addr, &[DirectAccessCmd::RD.bits()])
.map_err(|source| OtpDaiError::Jtag { source })?;
Self::wait_for_idle(jtag)?;
let dai_rdata0_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessRdata0 as u32;
let dai_rdata1_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessRdata1 as u32;
let mut rdata = [0x00; 2];
jtag.read_memory32(dai_rdata0_addr, &mut rdata[0..=0])
.map_err(|source| OtpDaiError::Jtag { source })?;
if granule == Granularity::B64 {
jtag.read_memory32(dai_rdata1_addr, &mut rdata[1..=1])
.map_err(|source| OtpDaiError::Jtag { source })?;
}
Ok(rdata)
}
pub fn write(
jtag: &mut dyn Jtag,
byte_addr: u32,
granule: Granularity,
values: [u32; 2],
) -> OtpDaiResult<()> {
Self::wait_for_idle(jtag)?;
let dai_address_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessAddress as u32;
jtag.write_memory32(dai_address_addr, &[byte_addr])
.map_err(|source| OtpDaiError::Jtag { source })?;
let dai_wdata0_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessWdata0 as u32;
let dai_wdata1_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessWdata1 as u32;
jtag.write_memory32(dai_wdata0_addr, &[values[0]])
.map_err(|source| OtpDaiError::Jtag { source })?;
if granule == Granularity::B64 {
jtag.write_memory32(dai_wdata1_addr, &[values[1]])
.map_err(|source| OtpDaiError::Jtag { source })?;
}
let dai_cmd_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessCmd as u32;
jtag.write_memory32(dai_cmd_addr, &[DirectAccessCmd::WR.bits()])
.map_err(|source| OtpDaiError::Jtag { source })?;
Self::wait_for_idle(jtag)?;
Ok(())
}
pub fn lock(jtag: &mut dyn Jtag, byte_addr: u32) -> OtpDaiResult<()> {
if byte_addr == Partition::CREATOR_SW_CFG.byte_addr
|| byte_addr == Partition::OWNER_SW_CFG.byte_addr
{
return Err(OtpDaiError::NotSupported);
}
Self::wait_for_idle(jtag)?;
let dai_address_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessAddress as u32;
jtag.write_memory32(dai_address_addr, &[byte_addr])
.map_err(|source| OtpDaiError::Jtag { source })?;
let dai_cmd_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::DirectAccessCmd as u32;
jtag.write_memory32(dai_cmd_addr, &[DirectAccessCmd::DIGEST.bits()])
.map_err(|source| OtpDaiError::Jtag { source })?;
Self::wait_for_idle(jtag)?;
Ok(())
}
pub fn wait_for_idle(jtag: &mut dyn Jtag) -> Result<(), OtpDaiError> {
let otp_status_addr = Self::OTP_CTRL_BASE_ADDR + OtpCtrlReg::Status as u32;
poll::poll_until(Self::POLL_TIMEOUT, Self::POLL_DELAY, || {
let mut status = [0];
jtag.read_memory32(otp_status_addr, &mut status)
.map_err(|source| OtpDaiError::Jtag { source })?;
let status =
OtpCtrlStatus::from_bits(status[0]).context("status has invalid bits set")?;
if status.intersects(OtpCtrlStatus::ERRORS) {
bail!("status {status:#b} has error bits set");
}
Ok(status.contains(OtpCtrlStatus::DAI_IDLE))
})
.map_err(|source| OtpDaiError::WaitForIdle { source })
}
}
pub type OtpDaiResult<T> = Result<T, OtpDaiError>;
#[derive(Debug, Error)]
pub enum OtpDaiError {
#[error("provided buffer has invalid size {buf_size} for parameter of size {param_size}")]
BufSize { buf_size: usize, param_size: u32 },
#[error("feature not supported for current partition")]
NotSupported,
#[error("failed to communicate over JTAG")]
Jtag { source: anyhow::Error },
#[error("failed to wait for otp_ctrl DAI to be idle")]
WaitForIdle { source: anyhow::Error },
#[error("writing to otp_ctrl direct access registers is disabled")]
WriteDisabled,
#[error(
"writing to OTP failed since field has already been written (with value: 0x{value:08X})"
)]
WriteErrorAlreadyWritten { value: u32 },
}