use core::cell::Cell;
use kernel::deferred_call::{DeferredCall, DeferredCallClient};
use kernel::hil::digest::Sha256;
use kernel::hil::digest::{Client, ClientData, ClientHash, ClientVerify};
use kernel::hil::digest::{ClientDataHash, ClientDataVerify, DigestDataHash, DigestDataVerify};
use kernel::hil::digest::{Digest, DigestData, DigestHash, DigestVerify};
use kernel::utilities::cells::{MapCell, OptionalCell};
use kernel::utilities::leasable_buffer::SubSlice;
use kernel::utilities::leasable_buffer::SubSliceMut;
use kernel::utilities::leasable_buffer::SubSliceMutImmut;
use kernel::ErrorCode;
#[derive(Clone, Copy, PartialEq)]
pub enum State {
Idle,
Data,
Hash,
Verify,
CancelData,
CancelHash,
CancelVerify,
}
const SHA_BLOCK_LEN_BYTES: usize = 64;
const SHA_256_OUTPUT_LEN_BYTES: usize = 32;
const NUM_ROUND_CONSTANTS: usize = 64;
const ROUND_CONSTANTS: [u32; NUM_ROUND_CONSTANTS] = [
0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5,
0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174,
0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967,
0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,
0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3,
0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2,
];
pub struct Sha256Software<'a> {
state: Cell<State>,
client: OptionalCell<&'a dyn Client<SHA_256_OUTPUT_LEN_BYTES>>,
input_data: OptionalCell<SubSliceMutImmut<'static, u8>>,
data_buffer: MapCell<[u8; SHA_BLOCK_LEN_BYTES]>,
buffered_length: Cell<usize>,
total_length: Cell<usize>,
output_data: Cell<Option<&'static mut [u8; SHA_256_OUTPUT_LEN_BYTES]>>,
hash_values: Cell<[u32; 8]>,
deferred_call: DeferredCall,
}
impl Sha256Software<'_> {
pub fn new() -> Self {
let s = Self {
state: Cell::new(State::Idle),
client: OptionalCell::empty(),
input_data: OptionalCell::empty(),
data_buffer: MapCell::new([0; SHA_BLOCK_LEN_BYTES]),
buffered_length: Cell::new(0),
total_length: Cell::new(0),
output_data: Cell::new(None),
hash_values: Cell::new([0; 8]),
deferred_call: DeferredCall::new(),
};
s.initialize();
s
}
pub fn busy(&self) -> bool {
match self.state.get() {
State::Idle => false,
_ => true,
}
}
fn initialize(&self) {
let new_state = match self.state.get() {
State::Idle => State::Idle,
State::Data | State::CancelData => State::CancelData,
State::Hash | State::CancelHash => State::CancelHash,
State::Verify | State::CancelVerify => State::CancelVerify,
};
self.state.set(new_state);
self.buffered_length.set(0);
self.total_length.set(0);
self.data_buffer.map(|b| {
for i in 0..SHA_BLOCK_LEN_BYTES {
b[i] = 0;
}
});
self.hash_values.set([
0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, 0x510e527f, 0x9b05688c, 0x1f83d9ab,
0x5be0cd19,
]);
}
fn complete_sha256(&self) {
let mut buffered_length = self.buffered_length.get();
if buffered_length == 64 {
self.data_buffer.map(|b| {
self.compute_block(b);
for i in 0..SHA_BLOCK_LEN_BYTES {
b[i] = 0;
}
});
buffered_length -= 64;
}
if buffered_length < 64 {
self.data_buffer.map(|b| {
for i in buffered_length..SHA_BLOCK_LEN_BYTES {
b[i] = 0;
}
});
}
self.data_buffer.map(|b| {
b.get_mut(buffered_length).map(|d| *d = 0x80);
buffered_length += 1;
if buffered_length > 56 {
for i in buffered_length..SHA_BLOCK_LEN_BYTES {
b[i] = 0;
}
self.compute_block(b);
for i in 0..SHA_BLOCK_LEN_BYTES {
b[i] = 0;
}
buffered_length = 0;
}
let total_length = self.total_length.get();
let length64 = (total_length * 8) as u64;
let len_high: u32 = (length64 >> 32) as u32;
let len_low: u32 = (length64 & 0xffffffff) as u32;
b[56] = (len_high >> 24 & 0xff) as u8;
b[57] = (len_high >> 16 & 0xff) as u8;
b[58] = (len_high >> 8 & 0xff) as u8;
b[59] = (len_high >> 0 & 0xff) as u8;
b[60] = (len_low >> 24 & 0xff) as u8;
b[61] = (len_low >> 16 & 0xff) as u8;
b[62] = (len_low >> 8 & 0xff) as u8;
b[63] = (len_low >> 0 & 0xff) as u8;
self.compute_block(b);
});
}
fn compute_sha256(&self) {
if let Some(mut data) = self.input_data.take() {
let data_length = data.len();
self.total_length.set(self.total_length.get() + data_length);
let mut buffered_length = self.buffered_length.get();
if buffered_length != 0 {
self.data_buffer.map(|b| {
let copy_len = if data_length + buffered_length >= SHA_BLOCK_LEN_BYTES {
SHA_BLOCK_LEN_BYTES - buffered_length
} else {
data_length
};
for i in 0..copy_len {
b[i + buffered_length] = data[i];
}
data.slice(copy_len..data.len());
buffered_length += copy_len;
if buffered_length == SHA_BLOCK_LEN_BYTES {
self.compute_block(b);
buffered_length = 0;
}
});
}
while data.len() >= 64 {
self.compute_buffer(&data[0..64]);
data.slice(64..data.len());
}
if data.len() != 0 {
self.data_buffer.map(|b| {
for i in 0..data.len() {
b[i] = data[i];
}
buffered_length = data.len();
data.slice(data.len()..data.len());
});
}
self.input_data.set(data);
self.buffered_length.set(buffered_length);
} else { }
}
fn right_rotate(&self, x: u32, rotate: u32) -> u32 {
(x >> rotate) | (x << (32 - rotate))
}
fn compute_buffer(&self, buffer: &[u8]) {
let mut message_schedule: [u32; 64] = [0; 64];
for i in 0..16 {
let val: u32 = (buffer[i * 4 + 0] as u32) << 24
| (buffer[i * 4 + 1] as u32) << 16
| (buffer[i * 4 + 2] as u32) << 8
| (buffer[i * 4 + 3] as u32);
message_schedule[i] = val;
}
self.perform_sha(&mut message_schedule);
}
fn compute_block(&self, data: &[u8; 64]) {
self.compute_buffer(data);
}
fn perform_sha(&self, message_schedule: &mut [u32; 64]) {
for i in 16..64 {
let mut s0 = self.right_rotate(message_schedule[i - 15], 7);
s0 ^= self.right_rotate(message_schedule[i - 15], 18);
s0 ^= message_schedule[i - 15] >> 3;
let mut s1 = self.right_rotate(message_schedule[i - 2], 17);
s1 ^= self.right_rotate(message_schedule[i - 2], 19);
s1 ^= message_schedule[i - 2] >> 10;
message_schedule[i] = message_schedule[i - 16] + s0 + message_schedule[i - 7] + s1;
}
let mut hashes = self.hash_values.get();
for i in 0..64 {
let s1 = self.right_rotate(hashes[4], 6)
^ self.right_rotate(hashes[4], 11)
^ self.right_rotate(hashes[4], 25);
let ch = (hashes[4] & hashes[5]) ^ ((!hashes[4]) & hashes[6]);
let constant = ROUND_CONSTANTS[i];
let temp1 = hashes[7] + s1 + ch + constant + message_schedule[i];
let s0 = self.right_rotate(hashes[0], 2)
^ self.right_rotate(hashes[0], 13)
^ self.right_rotate(hashes[0], 22);
let maj = (hashes[0] & hashes[1]) ^ (hashes[0] & hashes[2]) ^ (hashes[1] & hashes[2]);
let temp2 = s0 + maj;
hashes[7] = hashes[6];
hashes[6] = hashes[5];
hashes[5] = hashes[4];
hashes[4] = hashes[3].wrapping_add(temp1);
hashes[3] = hashes[2];
hashes[2] = hashes[1];
hashes[1] = hashes[0];
hashes[0] = temp1.wrapping_add(temp2);
}
let mut new_hashes = self.hash_values.get();
for i in 0..8 {
new_hashes[i] = new_hashes[i].wrapping_add(hashes[i]);
}
self.hash_values.set(new_hashes);
}
}
impl<'a> DigestData<'a, 32> for Sha256Software<'a> {
fn add_data(
&self,
data: SubSlice<'static, u8>,
) -> Result<(), (ErrorCode, SubSlice<'static, u8>)> {
if self.busy() {
Err((ErrorCode::BUSY, data))
} else {
self.state.set(State::Data);
self.deferred_call.set();
self.input_data.set(SubSliceMutImmut::Immutable(data));
self.compute_sha256();
Ok(())
}
}
fn add_mut_data(
&self,
data: SubSliceMut<'static, u8>,
) -> Result<(), (ErrorCode, SubSliceMut<'static, u8>)> {
if self.busy() {
Err((ErrorCode::BUSY, data))
} else {
self.state.set(State::Data);
self.deferred_call.set();
self.input_data.set(SubSliceMutImmut::Mutable(data));
self.compute_sha256();
Ok(())
}
}
fn clear_data(&self) {
self.initialize();
}
fn set_data_client(&'a self, _client: &'a (dyn ClientData<32> + 'a)) {
unimplemented!()
}
}
impl<'a> DigestHash<'a, 32> for Sha256Software<'a> {
fn run(
&'a self,
digest: &'static mut [u8; 32],
) -> Result<(), (ErrorCode, &'static mut [u8; 32])> {
if self.busy() {
Err((ErrorCode::BUSY, digest))
} else {
self.state.set(State::Hash);
self.complete_sha256();
for i in 0..8 {
let val = self.hash_values.get()[i];
digest[4 * i + 3] = (val >> 0 & 0xff) as u8;
digest[4 * i + 2] = (val >> 8 & 0xff) as u8;
digest[4 * i + 1] = (val >> 16 & 0xff) as u8;
digest[4 * i + 0] = (val >> 24 & 0xff) as u8;
}
self.output_data.set(Some(digest));
self.deferred_call.set();
Ok(())
}
}
fn set_hash_client(&'a self, _client: &'a (dyn ClientHash<32> + 'a)) {
unimplemented!()
}
}
impl<'a> DigestVerify<'a, 32> for Sha256Software<'a> {
fn verify(
&'a self,
compare: &'static mut [u8; 32],
) -> Result<(), (ErrorCode, &'static mut [u8; 32])> {
if self.busy() {
Err((ErrorCode::BUSY, compare))
} else {
self.state.set(State::Verify);
self.complete_sha256();
self.output_data.set(Some(compare));
self.deferred_call.set();
Ok(())
}
}
fn set_verify_client(&'a self, _client: &'a (dyn ClientVerify<32> + 'a)) {
unimplemented!()
}
}
impl<'a> Digest<'a, 32> for Sha256Software<'a> {
fn set_client(&'a self, client: &'a dyn Client<32>) {
self.client.set(client);
}
}
impl DeferredCallClient for Sha256Software<'_> {
fn handle_deferred_call(&self) {
let prior = self.state.get();
self.state.set(State::Idle);
match prior {
State::Idle => {}
State::Verify => {
let output = self.output_data.replace(None).unwrap();
let mut pass = true;
for i in 0..8 {
let hashval = self.hash_values.get()[i];
if output[4 * i + 3] != (hashval >> 0 & 0xff) as u8
|| output[4 * i + 2] != (hashval >> 8 & 0xff) as u8
|| output[4 * i + 1] != (hashval >> 16 & 0xff) as u8
|| output[4 * i + 0] != (hashval >> 24 & 0xff) as u8
{
pass = false;
break;
}
}
self.state.set(State::Idle);
self.clear_data();
self.client.map(|c| {
c.verification_done(Ok(pass), output);
});
}
State::Data => {
let data = self.input_data.take().unwrap();
self.state.set(State::Idle);
match data {
SubSliceMutImmut::Mutable(buffer) => {
self.client.map(|client| {
client.add_mut_data_done(Ok(()), buffer);
});
}
SubSliceMutImmut::Immutable(buffer) => {
self.client.map(|client| {
client.add_data_done(Ok(()), buffer);
});
}
}
}
State::Hash => {
let output = self.output_data.replace(None).unwrap();
self.state.set(State::Idle);
self.clear_data();
self.client.map(|c| {
c.hash_done(Ok(()), output);
});
}
State::CancelData => {
self.state.set(State::Idle);
self.clear_data();
let data = self.input_data.take().unwrap();
match data {
SubSliceMutImmut::Mutable(buffer) => {
self.client.map(|client| {
client.add_mut_data_done(Err(ErrorCode::CANCEL), buffer);
});
}
SubSliceMutImmut::Immutable(buffer) => {
self.client.map(|client| {
client.add_data_done(Err(ErrorCode::CANCEL), buffer);
});
}
}
}
State::CancelVerify => {
self.state.set(State::Idle);
self.clear_data();
let output = self.output_data.replace(None).unwrap();
self.client.map(|client| {
client.verification_done(Err(ErrorCode::CANCEL), output);
});
}
State::CancelHash => {
self.state.set(State::Idle);
self.clear_data();
let output = self.output_data.replace(None).unwrap();
self.client.map(|client| {
client.hash_done(Err(ErrorCode::CANCEL), output);
});
}
}
}
fn register(&'static self) {
self.deferred_call.register(self);
}
}
impl Sha256 for Sha256Software<'_> {
fn set_mode_sha256(&self) -> Result<(), ErrorCode> {
Ok(())
}
}
impl<'a> DigestDataHash<'a, 32> for Sha256Software<'a> {
fn set_client(&'a self, _client: &'a dyn ClientDataHash<32>) {
unimplemented!()
}
}
impl<'a> DigestDataVerify<'a, 32> for Sha256Software<'a> {
fn set_client(&'a self, _client: &'a dyn ClientDataVerify<32>) {
unimplemented!()
}
}