2022-09-12 20:46:03 +00:00
|
|
|
// Copyright © 2022, Microsoft Corporation
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
//
|
|
|
|
|
|
|
|
use anyhow::anyhow;
|
|
|
|
#[cfg(target_arch = "aarch64")]
|
|
|
|
use arch::aarch64::layout::{TPM_SIZE, TPM_START};
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
|
|
|
use arch::x86_64::layout::{TPM_SIZE, TPM_START};
|
|
|
|
use std::cmp;
|
|
|
|
use std::sync::{Arc, Barrier};
|
|
|
|
use thiserror::Error;
|
|
|
|
use tpm::emulator::{BackendCmd, Emulator};
|
|
|
|
use tpm::TPM_CRB_BUFFER_MAX;
|
|
|
|
use tpm::TPM_SUCCESS;
|
|
|
|
use vm_device::BusDevice;
|
|
|
|
|
|
|
|
#[derive(Error, Debug)]
|
|
|
|
pub enum Error {
|
|
|
|
#[error("Emulator doesn't implement min required capabilities: {0}")]
|
|
|
|
CheckCaps(#[source] anyhow::Error),
|
|
|
|
#[error("Failed to initialize tpm: {0}")]
|
|
|
|
Init(#[source] anyhow::Error),
|
|
|
|
}
|
|
|
|
type Result<T> = anyhow::Result<T, Error>;
|
|
|
|
|
2023-01-25 17:00:57 +00:00
|
|
|
#[allow(dead_code)]
|
|
|
|
enum LocStateFields {
|
|
|
|
TpmEstablished,
|
|
|
|
LocAssigned,
|
|
|
|
ActiveLocality,
|
|
|
|
Reserved,
|
|
|
|
TpmRegValidSts,
|
|
|
|
}
|
|
|
|
|
|
|
|
enum LocStsFields {
|
|
|
|
Granted,
|
|
|
|
BeenSeized,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
enum IntfIdFields {
|
|
|
|
InterfaceType,
|
|
|
|
InterfaceVersion,
|
|
|
|
CapLocality,
|
|
|
|
CapCRBIdleBypass,
|
|
|
|
Reserved1,
|
|
|
|
CapDataXferSizeSupport,
|
|
|
|
CapFIFO,
|
|
|
|
CapCRB,
|
|
|
|
CapIFRes,
|
|
|
|
InterfaceSelector,
|
|
|
|
IntfSelLock,
|
|
|
|
Reserved2,
|
|
|
|
Rid,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
enum IntfId2Fields {
|
|
|
|
Vid,
|
|
|
|
Did,
|
|
|
|
}
|
|
|
|
|
|
|
|
enum CtrlStsFields {
|
|
|
|
TpmSts,
|
|
|
|
TpmIdle,
|
|
|
|
}
|
|
|
|
|
|
|
|
enum CrbRegister {
|
|
|
|
LocState(LocStateFields),
|
|
|
|
LocSts(LocStsFields),
|
|
|
|
IntfId(IntfIdFields),
|
|
|
|
IntfId2(IntfId2Fields),
|
|
|
|
CtrlSts(CtrlStsFields),
|
|
|
|
}
|
|
|
|
|
2022-09-12 20:46:03 +00:00
|
|
|
/* crb 32-bit registers */
|
|
|
|
const CRB_LOC_STATE: u32 = 0x0;
|
|
|
|
//Register Fields
|
2023-01-25 17:00:57 +00:00
|
|
|
// Field => (base, offset, length)
|
|
|
|
// base: starting position of the register
|
|
|
|
// offset: lowest bit in the bit field numbered from 0
|
2022-09-12 20:46:03 +00:00
|
|
|
// length: length of the bit field
|
2023-01-25 17:00:57 +00:00
|
|
|
const fn get_crb_loc_state_field(f: LocStateFields) -> (u32, u32, u32) {
|
|
|
|
let (offset, len) = match f {
|
|
|
|
LocStateFields::TpmEstablished => (0, 1),
|
|
|
|
LocStateFields::LocAssigned => (1, 1),
|
|
|
|
LocStateFields::ActiveLocality => (2, 3),
|
|
|
|
LocStateFields::Reserved => (5, 2),
|
|
|
|
LocStateFields::TpmRegValidSts => (7, 1),
|
|
|
|
};
|
|
|
|
|
|
|
|
(CRB_LOC_STATE, offset, len)
|
|
|
|
}
|
|
|
|
|
2022-09-12 20:46:03 +00:00
|
|
|
const CRB_LOC_CTRL: u32 = 0x08;
|
|
|
|
const CRB_LOC_CTRL_REQUEST_ACCESS: u32 = 1 << 0;
|
|
|
|
const CRB_LOC_CTRL_RELINQUISH: u32 = 1 << 1;
|
|
|
|
const CRB_LOC_CTRL_RESET_ESTABLISHMENT_BIT: u32 = 1 << 3;
|
|
|
|
const CRB_LOC_STS: u32 = 0x0C;
|
2023-01-25 17:00:57 +00:00
|
|
|
const fn get_crb_loc_sts_field(f: LocStsFields) -> (u32, u32, u32) {
|
|
|
|
let (offset, len) = match f {
|
|
|
|
LocStsFields::Granted => (0, 1),
|
|
|
|
LocStsFields::BeenSeized => (1, 1),
|
|
|
|
};
|
|
|
|
|
|
|
|
(CRB_LOC_STS, offset, len)
|
|
|
|
}
|
|
|
|
|
2022-09-12 20:46:03 +00:00
|
|
|
const CRB_INTF_ID: u32 = 0x30;
|
2023-01-25 17:00:57 +00:00
|
|
|
const fn get_crb_intf_id_field(f: IntfIdFields) -> (u32, u32, u32) {
|
|
|
|
let (offset, len) = match f {
|
|
|
|
IntfIdFields::InterfaceType => (0, 4),
|
|
|
|
IntfIdFields::InterfaceVersion => (4, 4),
|
|
|
|
IntfIdFields::CapLocality => (8, 1),
|
|
|
|
IntfIdFields::CapCRBIdleBypass => (9, 1),
|
|
|
|
IntfIdFields::Reserved1 => (10, 1),
|
|
|
|
IntfIdFields::CapDataXferSizeSupport => (11, 2),
|
|
|
|
IntfIdFields::CapFIFO => (13, 1),
|
|
|
|
IntfIdFields::CapCRB => (14, 1),
|
|
|
|
IntfIdFields::CapIFRes => (15, 2),
|
|
|
|
IntfIdFields::InterfaceSelector => (17, 2),
|
|
|
|
IntfIdFields::IntfSelLock => (19, 1),
|
|
|
|
IntfIdFields::Reserved2 => (20, 4),
|
|
|
|
IntfIdFields::Rid => (24, 8),
|
|
|
|
};
|
|
|
|
|
|
|
|
(CRB_INTF_ID, offset, len)
|
|
|
|
}
|
|
|
|
|
2022-09-12 20:46:03 +00:00
|
|
|
const CRB_INTF_ID2: u32 = 0x34;
|
2023-01-25 17:00:57 +00:00
|
|
|
const fn get_crb_intf_id2_field(f: IntfId2Fields) -> (u32, u32, u32) {
|
|
|
|
let (offset, len) = match f {
|
|
|
|
IntfId2Fields::Vid => (0, 16),
|
|
|
|
IntfId2Fields::Did => (16, 16),
|
|
|
|
};
|
|
|
|
|
|
|
|
(CRB_INTF_ID2, offset, len)
|
|
|
|
}
|
|
|
|
|
2022-09-12 20:46:03 +00:00
|
|
|
const CRB_CTRL_REQ: u32 = 0x40;
|
|
|
|
const CRB_CTRL_REQ_CMD_READY: u32 = 1 << 0;
|
|
|
|
const CRB_CTRL_REQ_GO_IDLE: u32 = 1 << 1;
|
|
|
|
const CRB_CTRL_STS: u32 = 0x44;
|
2023-01-25 17:00:57 +00:00
|
|
|
const fn get_crb_ctrl_sts_field(f: CtrlStsFields) -> (u32, u32, u32) {
|
|
|
|
let (offset, len) = match f {
|
|
|
|
CtrlStsFields::TpmSts => (0, 1),
|
|
|
|
CtrlStsFields::TpmIdle => (1, 1),
|
|
|
|
};
|
|
|
|
|
|
|
|
(CRB_CTRL_STS, offset, len)
|
|
|
|
}
|
2022-09-12 20:46:03 +00:00
|
|
|
const CRB_CTRL_CANCEL: u32 = 0x48;
|
|
|
|
const CRB_CANCEL_INVOKE: u32 = 1 << 0;
|
|
|
|
const CRB_CTRL_START: u32 = 0x4C;
|
|
|
|
const CRB_START_INVOKE: u32 = 1 << 0;
|
|
|
|
const CRB_CTRL_CMD_LADDR: u32 = 0x5C;
|
|
|
|
const CRB_CTRL_CMD_HADDR: u32 = 0x60;
|
|
|
|
const CRB_CTRL_RSP_SIZE: u32 = 0x64;
|
|
|
|
const CRB_CTRL_RSP_ADDR: u32 = 0x68;
|
|
|
|
const CRB_DATA_BUFFER: u32 = 0x80;
|
|
|
|
|
|
|
|
const TPM_CRB_NO_LOCALITY: u32 = 0xff;
|
|
|
|
|
|
|
|
const TPM_CRB_ADDR_BASE: u32 = TPM_START.0 as u32;
|
|
|
|
const TPM_CRB_ADDR_SIZE: usize = TPM_SIZE as usize;
|
|
|
|
|
2023-01-24 23:45:10 +00:00
|
|
|
const TPM_CRB_R_MAX: usize = CRB_DATA_BUFFER as usize;
|
2022-09-12 20:46:03 +00:00
|
|
|
|
|
|
|
// CRB Protocol details
|
|
|
|
const CRB_INTF_TYPE_CRB_ACTIVE: u32 = 0b1;
|
|
|
|
const CRB_INTF_VERSION_CRB: u32 = 0b1;
|
|
|
|
const CRB_INTF_CAP_LOCALITY_0_ONLY: u32 = 0b0;
|
|
|
|
const CRB_INTF_CAP_IDLE_FAST: u32 = 0b0;
|
|
|
|
const CRB_INTF_CAP_XFER_SIZE_64: u32 = 0b11;
|
|
|
|
const CRB_INTF_CAP_FIFO_NOT_SUPPORTED: u32 = 0b0;
|
|
|
|
const CRB_INTF_CAP_CRB_SUPPORTED: u32 = 0b1;
|
|
|
|
const CRB_INTF_IF_SELECTOR_CRB: u32 = 0b1;
|
|
|
|
const PCI_VENDOR_ID_IBM: u32 = 0x1014;
|
|
|
|
const CRB_CTRL_CMD_SIZE_REG: u32 = 0x58;
|
|
|
|
const CRB_CTRL_CMD_SIZE: usize = TPM_CRB_ADDR_SIZE - CRB_DATA_BUFFER as usize;
|
|
|
|
|
2023-01-25 17:00:57 +00:00
|
|
|
// Returns (register base, offset, len)
|
|
|
|
const fn get_field(reg: CrbRegister) -> (u32, u32, u32) {
|
2022-09-12 20:46:03 +00:00
|
|
|
match reg {
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::LocState(f) => get_crb_loc_state_field(f),
|
|
|
|
CrbRegister::LocSts(f) => get_crb_loc_sts_field(f),
|
|
|
|
CrbRegister::IntfId(f) => get_crb_intf_id_field(f),
|
|
|
|
CrbRegister::IntfId2(f) => get_crb_intf_id2_field(f),
|
|
|
|
CrbRegister::CtrlSts(f) => get_crb_ctrl_sts_field(f),
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-25 17:00:57 +00:00
|
|
|
// Set a particular field in a Register
|
|
|
|
fn set_reg_field(regs: &mut [u32; TPM_CRB_R_MAX], reg: CrbRegister, value: u32) {
|
|
|
|
let (base, offset, len) = get_field(reg);
|
|
|
|
let mask = (!(0_u32) >> (32 - len)) << offset;
|
|
|
|
regs[base as usize] = (regs[base as usize] & !mask) | ((value << offset) & mask);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
|
2023-01-25 17:00:57 +00:00
|
|
|
// Get the value of a particular field in a Register
|
|
|
|
const fn get_reg_field(regs: &[u32; TPM_CRB_R_MAX], reg: CrbRegister) -> u32 {
|
|
|
|
let (base, offset, len) = get_field(reg);
|
|
|
|
let mask = (!(0_u32) >> (32 - len)) << offset;
|
|
|
|
(regs[base as usize] & mask) >> offset
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn locality_from_addr(addr: u32) -> u8 {
|
|
|
|
(addr >> 12) as u8
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Tpm {
|
|
|
|
emulator: Emulator,
|
2023-01-24 23:45:10 +00:00
|
|
|
regs: [u32; TPM_CRB_R_MAX],
|
2022-09-12 20:46:03 +00:00
|
|
|
backend_buff_size: usize,
|
|
|
|
data_buff: [u8; TPM_CRB_BUFFER_MAX],
|
|
|
|
data_buff_len: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Tpm {
|
|
|
|
pub fn new(path: String) -> Result<Self> {
|
|
|
|
let emulator = Emulator::new(path)
|
|
|
|
.map_err(|e| Error::Init(anyhow!("Failed while initializing tpm Emulator: {:?}", e)))?;
|
|
|
|
let mut tpm = Tpm {
|
|
|
|
emulator,
|
2023-01-24 23:45:10 +00:00
|
|
|
regs: [0; TPM_CRB_R_MAX],
|
2022-09-12 20:46:03 +00:00
|
|
|
backend_buff_size: TPM_CRB_BUFFER_MAX,
|
|
|
|
data_buff: [0; TPM_CRB_BUFFER_MAX],
|
|
|
|
data_buff_len: 0,
|
|
|
|
};
|
|
|
|
tpm.reset()?;
|
|
|
|
Ok(tpm)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_active_locality(&mut self) -> u32 {
|
2023-01-25 17:00:57 +00:00
|
|
|
if get_reg_field(
|
|
|
|
&self.regs,
|
|
|
|
CrbRegister::LocState(LocStateFields::LocAssigned),
|
|
|
|
) == 0
|
|
|
|
{
|
2022-09-12 20:46:03 +00:00
|
|
|
return TPM_CRB_NO_LOCALITY;
|
|
|
|
}
|
2023-01-25 17:00:57 +00:00
|
|
|
get_reg_field(
|
|
|
|
&self.regs,
|
|
|
|
CrbRegister::LocState(LocStateFields::ActiveLocality),
|
|
|
|
)
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn request_completed(&mut self, result: isize) {
|
|
|
|
self.regs[CRB_CTRL_START as usize] = !CRB_START_INVOKE;
|
|
|
|
if result != 0 {
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::CtrlSts(CtrlStsFields::TpmSts),
|
|
|
|
1,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn reset(&mut self) -> Result<()> {
|
2023-01-25 17:42:17 +00:00
|
|
|
let cur_buff_size = self.emulator.get_buffer_size();
|
2023-01-24 23:45:10 +00:00
|
|
|
self.regs = [0; TPM_CRB_R_MAX];
|
2022-09-12 20:46:03 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::LocState(LocStateFields::TpmRegValidSts),
|
|
|
|
1,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::CtrlSts(CtrlStsFields::TpmIdle),
|
|
|
|
1,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::IntfId(IntfIdFields::InterfaceType),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_TYPE_CRB_ACTIVE,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::InterfaceVersion),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_VERSION_CRB,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::CapLocality),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_CAP_LOCALITY_0_ONLY,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::CapCRBIdleBypass),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_CAP_IDLE_FAST,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::CapDataXferSizeSupport),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_CAP_XFER_SIZE_64,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::CapFIFO),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_CAP_FIFO_NOT_SUPPORTED,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::CapCRB),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_CAP_CRB_SUPPORTED,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
2023-01-25 17:00:57 +00:00
|
|
|
CrbRegister::IntfId(IntfIdFields::InterfaceSelector),
|
2022-09-12 20:46:03 +00:00
|
|
|
CRB_INTF_IF_SELECTOR_CRB,
|
|
|
|
);
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::IntfId(IntfIdFields::Rid),
|
|
|
|
0b0000,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::IntfId2(IntfId2Fields::Vid),
|
|
|
|
PCI_VENDOR_ID_IBM,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
|
|
|
|
self.regs[CRB_CTRL_CMD_SIZE_REG as usize] = CRB_CTRL_CMD_SIZE as u32;
|
|
|
|
self.regs[CRB_CTRL_CMD_LADDR as usize] = TPM_CRB_ADDR_BASE + CRB_DATA_BUFFER;
|
|
|
|
self.regs[CRB_CTRL_RSP_SIZE as usize] = CRB_CTRL_CMD_SIZE as u32;
|
|
|
|
self.regs[CRB_CTRL_RSP_ADDR as usize] = TPM_CRB_ADDR_BASE + CRB_DATA_BUFFER;
|
|
|
|
|
|
|
|
self.backend_buff_size = cmp::min(cur_buff_size, TPM_CRB_BUFFER_MAX);
|
|
|
|
|
|
|
|
if let Err(e) = self.emulator.startup_tpm(self.backend_buff_size) {
|
|
|
|
return Err(Error::Init(anyhow!(
|
|
|
|
"Failed while running Startup TPM. Error: {:?}",
|
|
|
|
e
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BusDevice for Tpm {
|
|
|
|
fn read(&mut self, _base: u64, offset: u64, data: &mut [u8]) {
|
|
|
|
let mut offset: u32 = offset as u32;
|
|
|
|
let read_len: usize = data.len();
|
|
|
|
|
|
|
|
if offset >= CRB_DATA_BUFFER
|
|
|
|
&& (offset + read_len as u32) < (CRB_DATA_BUFFER + self.data_buff.len() as u32)
|
|
|
|
{
|
|
|
|
// Read from Data Buffer
|
|
|
|
let start: usize = (offset as usize) - (CRB_DATA_BUFFER as usize);
|
|
|
|
let end: usize = start + read_len;
|
|
|
|
data[..].clone_from_slice(&self.data_buff[start..end]);
|
|
|
|
} else {
|
|
|
|
offset &= 0xff;
|
|
|
|
let mut val = self.regs[offset as usize];
|
|
|
|
|
|
|
|
if offset == CRB_LOC_STATE && !self.emulator.get_established_flag() {
|
|
|
|
val |= 0x1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if data.len() <= 4 {
|
|
|
|
data.clone_from_slice(val.to_ne_bytes()[0..read_len].as_ref());
|
|
|
|
} else {
|
|
|
|
error!(
|
|
|
|
"Invalid tpm read: offset {:#X}, data length {:?}",
|
|
|
|
offset,
|
|
|
|
data.len()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
debug!(
|
|
|
|
"MMIO Read: offset {:#X} len {:?} val = {:02X?} ",
|
|
|
|
offset,
|
|
|
|
data.len(),
|
|
|
|
data
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write(&mut self, _base: u64, offset: u64, data: &[u8]) -> Option<Arc<Barrier>> {
|
|
|
|
debug!(
|
|
|
|
"MMIO Write: offset {:#X} len {:?} input data {:02X?}",
|
|
|
|
offset,
|
|
|
|
data.len(),
|
|
|
|
data
|
|
|
|
);
|
|
|
|
let mut offset: u32 = offset as u32;
|
|
|
|
if offset < CRB_DATA_BUFFER {
|
|
|
|
offset &= 0xff;
|
|
|
|
}
|
|
|
|
let locality = locality_from_addr(offset) as u32;
|
|
|
|
let write_len = data.len();
|
|
|
|
|
|
|
|
if offset >= CRB_DATA_BUFFER
|
|
|
|
&& (offset + write_len as u32) < (CRB_DATA_BUFFER + self.data_buff.len() as u32)
|
|
|
|
{
|
|
|
|
let start: usize = (offset as usize) - (CRB_DATA_BUFFER as usize);
|
|
|
|
if start == 0 {
|
|
|
|
// If filling data_buff at index 0, reset length to 0
|
|
|
|
self.data_buff_len = 0;
|
|
|
|
self.data_buff.fill(0);
|
|
|
|
}
|
|
|
|
let end: usize = start + data.len();
|
|
|
|
self.data_buff[start..end].clone_from_slice(data);
|
|
|
|
self.data_buff_len += data.len();
|
|
|
|
} else {
|
|
|
|
// Ctrl Commands that take more than 4 bytes as input are not yet supported
|
|
|
|
// CTRL_RSP_ADDR usually gets 8 byte write request. Last 4 bytes are zeros.
|
|
|
|
if write_len > 4 && offset != CRB_CTRL_RSP_ADDR {
|
|
|
|
error!(
|
|
|
|
"Invalid tpm write: offset {:#X}, data length {}",
|
|
|
|
offset,
|
|
|
|
data.len()
|
|
|
|
);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut input: [u8; 4] = [0; 4];
|
|
|
|
input.copy_from_slice(&data[0..4]);
|
|
|
|
let v = u32::from_le_bytes(input);
|
|
|
|
|
|
|
|
match offset {
|
|
|
|
CRB_CTRL_CMD_SIZE_REG => {
|
|
|
|
self.regs[CRB_CTRL_CMD_SIZE_REG as usize] = v;
|
|
|
|
}
|
|
|
|
CRB_CTRL_CMD_LADDR => {
|
|
|
|
self.regs[CRB_CTRL_CMD_LADDR as usize] = v;
|
|
|
|
}
|
|
|
|
CRB_CTRL_CMD_HADDR => {
|
|
|
|
self.regs[CRB_CTRL_CMD_HADDR as usize] = v;
|
|
|
|
}
|
|
|
|
CRB_CTRL_RSP_SIZE => {
|
|
|
|
self.regs[CRB_CTRL_RSP_SIZE as usize] = v;
|
|
|
|
}
|
|
|
|
CRB_CTRL_RSP_ADDR => {
|
|
|
|
self.regs[CRB_CTRL_RSP_ADDR as usize] = v;
|
|
|
|
}
|
|
|
|
CRB_CTRL_REQ => match v {
|
|
|
|
CRB_CTRL_REQ_CMD_READY => {
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::CtrlSts(CtrlStsFields::TpmIdle),
|
|
|
|
0,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
CRB_CTRL_REQ_GO_IDLE => {
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::CtrlSts(CtrlStsFields::TpmIdle),
|
|
|
|
1,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
error!("Invalid value passed to CRTL_REQ register");
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
CRB_CTRL_CANCEL => {
|
|
|
|
if v == CRB_CANCEL_INVOKE
|
|
|
|
&& (self.regs[CRB_CTRL_START as usize] & CRB_START_INVOKE != 0)
|
|
|
|
{
|
|
|
|
if let Err(e) = self.emulator.cancel_cmd() {
|
|
|
|
error!("Failed to run cancel command. Error: {:?}", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CRB_CTRL_START => {
|
|
|
|
if v == CRB_START_INVOKE
|
|
|
|
&& ((self.regs[CRB_CTRL_START as usize] & CRB_START_INVOKE) == 0)
|
|
|
|
&& self.get_active_locality() == locality
|
|
|
|
{
|
|
|
|
self.regs[CRB_CTRL_START as usize] |= CRB_START_INVOKE;
|
|
|
|
|
2023-01-25 21:38:13 +00:00
|
|
|
let mut cmd = BackendCmd {
|
|
|
|
buffer: &mut self.data_buff,
|
2022-09-12 20:46:03 +00:00
|
|
|
input_len: cmp::min(self.data_buff_len, TPM_CRB_BUFFER_MAX),
|
2023-01-25 17:10:05 +00:00
|
|
|
};
|
2022-09-12 20:46:03 +00:00
|
|
|
|
2023-01-25 21:38:13 +00:00
|
|
|
let _ = self.emulator.deliver_request(&mut cmd);
|
2022-09-12 20:46:03 +00:00
|
|
|
|
|
|
|
self.request_completed(TPM_SUCCESS as isize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CRB_LOC_CTRL => {
|
|
|
|
warn!(
|
|
|
|
"CRB_LOC_CTRL locality to write = {:?} val = {:?}",
|
|
|
|
locality, v
|
|
|
|
);
|
|
|
|
match v {
|
|
|
|
CRB_LOC_CTRL_RESET_ESTABLISHMENT_BIT => {}
|
|
|
|
CRB_LOC_CTRL_RELINQUISH => {
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::LocState(LocStateFields::LocAssigned),
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::LocSts(LocStsFields::Granted),
|
|
|
|
0,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
CRB_LOC_CTRL_REQUEST_ACCESS => {
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::LocSts(LocStsFields::Granted),
|
|
|
|
1,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::LocSts(LocStsFields::BeenSeized),
|
|
|
|
0,
|
|
|
|
);
|
|
|
|
set_reg_field(
|
|
|
|
&mut self.regs,
|
|
|
|
CrbRegister::LocState(LocStateFields::LocAssigned),
|
|
|
|
1,
|
|
|
|
);
|
2022-09-12 20:46:03 +00:00
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
error!("Invalid value to write in CRB_LOC_CTRL {:#X} ", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
error!(
|
|
|
|
"Invalid tpm write: offset {:#X}, data length {:?}",
|
|
|
|
offset,
|
|
|
|
data.len()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_set_get_reg_field() {
|
2023-01-24 23:45:10 +00:00
|
|
|
let mut regs: [u32; TPM_CRB_R_MAX] = [0; TPM_CRB_R_MAX];
|
2023-01-25 17:00:57 +00:00
|
|
|
set_reg_field(&mut regs, CrbRegister::IntfId(IntfIdFields::Rid), 0xAC);
|
2022-09-12 20:46:03 +00:00
|
|
|
assert_eq!(
|
2023-01-25 17:00:57 +00:00
|
|
|
get_reg_field(®s, CrbRegister::IntfId(IntfIdFields::Rid)),
|
2022-09-12 20:46:03 +00:00
|
|
|
0xAC,
|
|
|
|
concat!("Test: ", stringify!(set_get_reg_field))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|