Rob Bradford adb318f4cd misc: Remove redundant "use" imports
With the nightly toolchain (2024-02-18) cargo check will flag up
redundant imports either because they are pulled in by the prelude on
earlier match.

Remove those redundant imports.

Signed-off-by: Rob Bradford <rbradford@rivosinc.com>
2024-02-19 17:54:30 +00:00

441 lines
11 KiB
Rust

// Copyright © 2022, Microsoft Corporation
//
// SPDX-License-Identifier: Apache-2.0
//
#[macro_use]
extern crate log;
pub mod emulator;
pub mod socket;
use anyhow::anyhow;
use thiserror::Error;
pub const TPM_CRB_BUFFER_MAX: usize = 3968; // 0x1_000 - 0x80
pub const TPM_SUCCESS: u32 = 0x0;
/*
* Structures required to process Request and Responses of Control commands
* used by control channel over UNIX socket interface
*
* All messages contain big-endian data.
*
* Reference: https://github.com/stefanberger/swtpm/blob/master/man/man3/swtpm_ioctls.pod
*/
#[derive(Debug, Clone, Copy)]
pub enum Commands {
CmdGetCapability = 1,
CmdInit,
CmdShutdown,
CmdGetTpmEstablished,
CmdSetLocality,
CmdHashStart,
CmdHashData,
CmdHashEnd,
CmdCancelTpmCmd,
CmdStoreVolatile,
CmdResetTpmEstablished,
CmdGetStateBlob,
CmdSetStateBlob,
CmdStop,
CmdGetConfig,
CmdSetDatafd,
CmdSetBufferSize,
}
#[derive(Error, Debug)]
pub enum Error {
#[error("Failed converting buf to PTM : {0}")]
ConvertToPtm(#[source] anyhow::Error),
}
type Result<T> = anyhow::Result<T, Error>;
#[derive(PartialEq, Eq, Copy, Clone, Debug)]
pub enum MemberType {
Request,
Response,
Error,
Cap,
}
pub trait Ptm {
// Get Member Type
fn get_member_type(&self) -> MemberType;
// Set Member Type
fn set_member_type(&mut self, mem: MemberType);
// Convert PTM Request to bytes to be sent to tpm
fn ptm_to_request(&self) -> Vec<u8>;
// Update PTM from tpm's response
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()>;
// Update tpm result
fn set_result_code(&mut self, res: u32);
fn get_result_code(&self) -> u32;
}
/*
* Every response for a tpm Control Command execution must hold tpm return
* code (PtmResult) as its first element.
* Based on the type of input Control Command additional data could be
* appended to the response.
*/
pub type PtmResult = u32;
impl Ptm for PtmResult {
fn ptm_to_request(&self) -> Vec<u8> {
Vec::new()
}
fn get_member_type(&self) -> MemberType {
MemberType::Response
}
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()> {
let expected_len = 4;
let len = buf.len();
if len != expected_len {
return Err(Error::ConvertToPtm(anyhow!(
"PtmRes buffer is of incorrect length. Got {len} expected {expected_len}."
)));
}
*self = u32::from_be_bytes(buf[..].try_into().unwrap());
Ok(())
}
fn set_member_type(&mut self, _mem: MemberType) {}
fn set_result_code(&mut self, res: u32) {
*self = res;
}
fn get_result_code(&self) -> u32 {
*self
}
}
/* GET_CAPABILITY Response */
pub type PtmCap = u64;
impl Ptm for PtmCap {
fn ptm_to_request(&self) -> Vec<u8> {
// tpm's GetCapability call doesn't need any supporting message
// return an empty Buffer
Vec::new()
}
fn get_member_type(&self) -> MemberType {
MemberType::Cap
}
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()> {
let expected_len = 8;
let len = buf.len();
if len != expected_len {
return Err(Error::ConvertToPtm(anyhow!(
"Response for GetCapability cmd is of incorrect length. Got {len} expected {expected_len}."
)));
}
*self = u64::from_be_bytes(buf[..].try_into().unwrap());
Ok(())
}
fn set_member_type(&mut self, _mem: MemberType) {}
fn set_result_code(&mut self, _res: u32) {}
fn get_result_code(&self) -> u32 {
((*self) >> 32) as u32
}
}
/* GET_TPMESTABLISHED Response */
#[derive(Debug)]
pub struct PtmEstResp {
pub bit: u8,
}
#[derive(Debug)]
pub struct PtmEst {
member: MemberType,
pub resp: PtmEstResp,
pub result_code: PtmResult,
}
impl PtmEst {
pub fn new() -> Self {
Self {
member: MemberType::Response,
result_code: 0,
resp: PtmEstResp { bit: 0 },
}
}
}
impl Default for PtmEst {
fn default() -> Self {
Self::new()
}
}
impl Ptm for PtmEst {
fn ptm_to_request(&self) -> Vec<u8> {
// tpm's GetTpmEstablished call doesn't need any supporting message
// return an empty Buffer
Vec::new()
}
fn get_member_type(&self) -> MemberType {
self.member
}
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()> {
let expected_len = 8;
let len = buf.len();
if len != expected_len {
return Err(Error::ConvertToPtm(anyhow!(
"Response for GetTpmEstablished cmd is of incorrect length. Got {len} expected {expected_len}."
)));
}
self.set_result_code(u32::from_be_bytes(buf[..4].try_into().unwrap()));
self.resp.bit = buf[4];
Ok(())
}
fn set_member_type(&mut self, _mem: MemberType) {}
fn set_result_code(&mut self, res: u32) {
self.result_code = res
}
fn get_result_code(&self) -> u32 {
self.result_code
}
}
/* INIT Response */
#[derive(Debug)]
pub struct PtmInit {
pub member: MemberType,
/* request */
pub init_flags: u32,
/* response */
pub result_code: PtmResult,
}
impl Default for PtmInit {
fn default() -> Self {
Self::new()
}
}
impl PtmInit {
pub fn new() -> Self {
Self {
member: MemberType::Request,
init_flags: 0,
result_code: 0,
}
}
}
impl Ptm for PtmInit {
fn ptm_to_request(&self) -> Vec<u8> {
let mut buf: Vec<u8> = Vec::<u8>::new();
buf.extend_from_slice(&self.init_flags.to_be_bytes());
buf
}
fn get_member_type(&self) -> MemberType {
self.member
}
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()> {
let expected_len = 4;
let len = buf.len();
if len != expected_len {
return Err(Error::ConvertToPtm(anyhow!(
"Response for Init cmd is of incorrect length. Got {len} expected {expected_len}."
)));
}
self.set_member_type(MemberType::Response);
self.set_result_code(u32::from_be_bytes(buf[..].try_into().unwrap()));
Ok(())
}
fn set_member_type(&mut self, mem: MemberType) {
self.member = mem
}
fn set_result_code(&mut self, res: u32) {
self.result_code = res
}
fn get_result_code(&self) -> u32 {
self.result_code
}
}
/*
* PTM_SET_BUFFERSIZE: Set the buffer size to be used by the tpm.
* A 0 on input queries for the current buffer size. Any other
* number will try to set the buffer size. The returned number is
* the buffer size that will be used, which can be larger than the
* requested one, if it was below the minimum, or smaller than the
* requested one, if it was above the maximum.
*
* SET_BUFFERSIZE Response
*/
#[derive(Debug)]
pub struct PtmSBSReq {
buffersize: u32,
}
#[derive(Debug)]
pub struct PtmSBSResp {
bufsize: u32,
minsize: u32,
maxsize: u32,
}
#[derive(Debug)]
pub struct PtmSetBufferSize {
pub mem: MemberType,
/* request */
pub req: PtmSBSReq,
/* response */
pub resp: PtmSBSResp,
pub result_code: PtmResult,
}
impl PtmSetBufferSize {
pub fn new(req_buffsize: u32) -> Self {
Self {
mem: MemberType::Request,
req: PtmSBSReq {
buffersize: req_buffsize,
},
resp: PtmSBSResp {
bufsize: 0,
minsize: 0,
maxsize: 0,
},
result_code: 0,
}
}
pub fn get_bufsize(&self) -> u32 {
self.resp.bufsize
}
}
impl Ptm for PtmSetBufferSize {
fn ptm_to_request(&self) -> Vec<u8> {
let mut buf: Vec<u8> = Vec::<u8>::new();
buf.extend_from_slice(&self.req.buffersize.to_be_bytes());
buf
}
fn get_member_type(&self) -> MemberType {
self.mem
}
fn update_ptm_with_response(&mut self, buf: &[u8]) -> Result<()> {
let expected_len = 16;
let len = buf.len();
if len != expected_len {
return Err(Error::ConvertToPtm(anyhow!(
"Response for CmdSetBufferSize cmd is of incorrect length. Got {len} expected {expected_len}."
)));
}
self.set_member_type(MemberType::Response);
self.set_result_code(u32::from_be_bytes(buf[0..4].try_into().unwrap()));
let bufsize = &buf[4..8];
self.resp.bufsize = u32::from_be_bytes(bufsize.try_into().unwrap());
let minsize = &buf[8..12];
self.resp.minsize = u32::from_be_bytes(minsize.try_into().unwrap());
let maxsize = &buf[12..16];
self.resp.maxsize = u32::from_be_bytes(maxsize.try_into().unwrap());
Ok(())
}
fn set_member_type(&mut self, mem: MemberType) {
self.mem = mem
}
fn set_result_code(&mut self, res: u32) {
self.result_code = res
}
fn get_result_code(&self) -> u32 {
self.result_code
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_ptmresult() -> Result<()> {
let buf: &[u8] = &[0, 0, 0, 1];
let mut result_code: PtmResult = 0;
result_code.update_ptm_with_response(buf)?;
assert_eq!(result_code.get_result_code(), 0x1);
Ok(())
}
#[test]
fn test_ptmcap() -> Result<()> {
let mut cap: PtmCap = 0x0;
let buf: &[u8] = &[0, 0, 0, 0xE, 0, 0, 0xFF, 0xFF];
cap.update_ptm_with_response(buf)?;
assert_eq!(cap.get_result_code(), 0xE);
Ok(())
}
#[test]
fn test_ptmest() -> Result<()> {
let mut est: PtmEst = PtmEst::new();
let buf: &[u8] = &[0, 0, 0xE, 0, 0xC, 0, 1, 1];
est.update_ptm_with_response(buf)?;
assert_eq!(est.get_result_code(), 0xE00);
assert_eq!(est.resp.bit, 0xC);
Ok(())
}
#[test]
/*PtmInit Testing */
fn test_ptminit() -> Result<()> {
let mut init: PtmInit = PtmInit::new();
init.init_flags = 0x1;
let buf = init.ptm_to_request();
assert_eq!(buf, [0x0, 0x0, 0x0, 0x1]);
let response_buf: &[u8] = &[0, 0, 0xE, 0];
init.update_ptm_with_response(response_buf)?;
assert_eq!(init.get_result_code(), 0xE00);
Ok(())
}
#[test]
/* PtmSetBufferSize Testing */
fn test_ptmsetbuffersize() -> Result<()> {
let mut psbs: PtmSetBufferSize = PtmSetBufferSize::new(1024);
// Member type should be Request after initialization
assert_eq!(psbs.get_member_type(), MemberType::Request);
let buf: &[u8] = &[
0, 0x12, 0x34, 0x56, 0, 0, 0, 0xA, 0, 0, 0, 0xB, 0, 0, 0, 0xC,
];
psbs.update_ptm_with_response(buf)?;
assert_eq!(psbs.get_member_type(), MemberType::Response);
assert_eq!(psbs.get_result_code(), 0x123456);
assert_eq!(psbs.resp.bufsize, 0xA);
assert_eq!(psbs.resp.minsize, 0xB);
assert_eq!(psbs.resp.maxsize, 0xC);
Ok(())
}
}