blob: ad360e4ef734590dd19585faa4a6d90a3e5d4256 [file] [log] [blame]
//! Virtualize the HMAC interface to enable multiple users of an underlying
//! HMAC hardware peripheral.
use core::cell::Cell;
use core::marker::PhantomData;
use kernel::common::cells::OptionalCell;
use kernel::common::leasable_buffer::LeasableBuffer;
use kernel::common::{ListLink, ListNode};
use kernel::hil::digest;
use kernel::hil::digest::DigestType;
use kernel::ReturnCode;
pub struct VirtualMuxHmac<'a, A: digest::Digest<'a, T>, T: DigestType> {
mux: &'a MuxHmac<'a, A, T>,
next: ListLink<'a, VirtualMuxHmac<'a, A, T>>,
client: OptionalCell<&'a dyn digest::Client<'a, T>>,
id: u32,
}
impl<'a, A: digest::Digest<'a, T>, T: DigestType> ListNode<'a, VirtualMuxHmac<'a, A, T>>
for VirtualMuxHmac<'a, A, T>
{
fn next(&self) -> &'a ListLink<VirtualMuxHmac<'a, A, T>> {
&self.next
}
}
impl<'a, A: digest::Digest<'a, T>, T: DigestType> VirtualMuxHmac<'a, A, T> {
pub fn new(mux_hmac: &'a MuxHmac<'a, A, T>) -> VirtualMuxHmac<'a, A, T> {
let id = mux_hmac.next_id.get();
mux_hmac.next_id.set(id + 1);
VirtualMuxHmac {
mux: mux_hmac,
next: ListLink::empty(),
client: OptionalCell::empty(),
id: id,
}
}
}
impl<'a, A: digest::Digest<'a, T>, T: DigestType> digest::Digest<'a, T>
for VirtualMuxHmac<'a, A, T>
{
/// Set the client instance which will receive `add_data_done()` and
/// `hash_done()` callbacks
fn set_client(&'a self, client: &'a dyn digest::Client<'a, T>) {
self.mux.hmac.set_client(client);
}
/// Add data to the hmac IP.
/// All data passed in is fed to the HMAC hardware block.
/// Returns the number of bytes written on success
fn add_data(
&self,
data: LeasableBuffer<'static, u8>,
) -> Result<usize, (ReturnCode, &'static mut [u8])> {
// Check if any mux is enabled. If it isn't we enable it for us.
if self.mux.running.get() == false {
self.mux.running.set(true);
self.mux.running_id.set(self.id);
self.mux.hmac.add_data(data)
} else if self.mux.running_id.get() == self.id {
self.mux.hmac.add_data(data)
} else {
Err((ReturnCode::EBUSY, data.take()))
}
}
/// Request the hardware block to generate a HMAC
/// This doesn't return anything, instead the client needs to have
/// set a `hash_done` handler.
fn run(&'a self, digest: &'static mut T) -> Result<(), (ReturnCode, &'static mut T)> {
// Check if any mux is enabled. If it isn't we enable it for us.
if self.mux.running.get() == false {
self.mux.running.set(true);
self.mux.running_id.set(self.id);
self.mux.hmac.run(digest)
} else if self.mux.running_id.get() == self.id {
self.mux.hmac.run(digest)
} else {
Err((ReturnCode::EBUSY, digest))
}
}
/// Disable the HMAC hardware and clear the keys and any other sensitive
/// data
fn clear_data(&self) {
if self.mux.running_id.get() == self.id {
self.mux.running.set(false);
self.mux.hmac.clear_data()
}
}
}
impl<'a, A: digest::Digest<'a, T>, T: DigestType> digest::Client<'a, T>
for VirtualMuxHmac<'a, A, T>
{
fn add_data_done(&'a self, result: Result<(), ReturnCode>, data: &'static mut [u8]) {
self.client
.map(move |client| client.add_data_done(result, data));
}
fn hash_done(&'a self, result: Result<(), ReturnCode>, digest: &'static mut T) {
self.client
.map(move |client| client.hash_done(result, digest));
}
}
impl<'a, A: digest::Digest<'a, T> + digest::HMACSha256, T: DigestType> digest::HMACSha256
for VirtualMuxHmac<'a, A, T>
{
fn set_mode_hmacsha256(&self, key: &[u8; 32]) -> Result<(), ReturnCode> {
// Check if any mux is enabled. If it isn't we enable it for us.
if self.mux.running.get() == false {
self.mux.running.set(true);
self.mux.running_id.set(self.id);
self.mux.hmac.set_mode_hmacsha256(key)
} else if self.mux.running_id.get() == self.id {
self.mux.hmac.set_mode_hmacsha256(key)
} else {
Err(ReturnCode::EBUSY)
}
}
}
pub struct MuxHmac<'a, A: digest::Digest<'a, T>, T: DigestType> {
hmac: &'a A,
running: Cell<bool>,
running_id: Cell<u32>,
next_id: Cell<u32>,
phantom: PhantomData<&'a T>,
}
impl<'a, A: digest::Digest<'a, T>, T: DigestType> MuxHmac<'a, A, T> {
pub const fn new(hmac: &'a A) -> MuxHmac<'a, A, T> {
MuxHmac {
hmac,
running: Cell::new(false),
running_id: Cell::new(0),
next_id: Cell::new(0),
phantom: PhantomData,
}
}
}