mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-12-27 08:38:07 +00:00
feat(core/rust): implement storage wrapper layer
[no changelog]
This commit is contained in:
parent
0f549c02da
commit
cac98ad7d4
@ -228,11 +228,15 @@ fn generate_trezorhal_bindings() {
|
||||
|
||||
let bindings = prepare_bindings()
|
||||
.header("trezorhal.h")
|
||||
// common
|
||||
.allowlist_var("HW_ENTROPY_DATA")
|
||||
// secbool
|
||||
.allowlist_type("secbool")
|
||||
.must_use_type("secbool")
|
||||
.allowlist_var("sectrue")
|
||||
.allowlist_var("secfalse")
|
||||
// flash
|
||||
.allowlist_function("flash_init")
|
||||
// storage
|
||||
.allowlist_var("EXTERNAL_SALT_SIZE")
|
||||
.allowlist_var("FLAG_PUBLIC")
|
||||
@ -247,6 +251,7 @@ fn generate_trezorhal_bindings() {
|
||||
.allowlist_function("storage_has_pin")
|
||||
.allowlist_function("storage_get_pin_rem")
|
||||
.allowlist_function("storage_change_pin")
|
||||
.allowlist_function("storage_ensure_not_wipe_code")
|
||||
.allowlist_function("storage_has")
|
||||
.allowlist_function("storage_get")
|
||||
.allowlist_function("storage_set")
|
||||
|
@ -12,6 +12,7 @@ pub mod random;
|
||||
#[cfg(feature = "model_tr")]
|
||||
pub mod rgb_led;
|
||||
pub mod slip39;
|
||||
pub mod storage;
|
||||
pub mod uzlib;
|
||||
|
||||
pub mod buffers;
|
||||
|
333
core/embed/rust/src/trezorhal/storage.rs
Normal file
333
core/embed/rust/src/trezorhal/storage.rs
Normal file
@ -0,0 +1,333 @@
|
||||
use super::ffi;
|
||||
use crate::error::Error;
|
||||
use core::ptr;
|
||||
use cstr_core::{cstr, CStr};
|
||||
|
||||
/// Result of PIN delay callback.
|
||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||
pub enum PinCallbackResult {
|
||||
/// Continue waiting for PIN unlock.
|
||||
Continue,
|
||||
/// Abort the unlock attempt before time is up.
|
||||
Abort,
|
||||
}
|
||||
|
||||
/// PIN delay callback function type.
|
||||
/// The storage layer will call this function while the PIN timeout is in
|
||||
/// progress. This is useful for showing UI progress bar.
|
||||
/// `wait` is the total number of seconds waiting.
|
||||
/// `progress` is a value between 0 and 1000, where 1000 indicates 100%.
|
||||
/// `message` is a message to show to the user.
|
||||
pub type PinDelayCallback = fn(wait: u32, progress: u32, message: &str) -> PinCallbackResult;
|
||||
|
||||
pub type ExternalSalt = [u8; ffi::EXTERNAL_SALT_SIZE as usize];
|
||||
|
||||
/// Static reference to the currently set PIN callback function.
|
||||
static mut PIN_UI_CALLBACK: Option<PinDelayCallback> = None;
|
||||
|
||||
/// C-compatible wrapper for the Rust callback.
|
||||
unsafe extern "C" fn callback_wrapper(
|
||||
wait: u32,
|
||||
progress: u32,
|
||||
message: *const cty::c_char,
|
||||
) -> ffi::secbool {
|
||||
let message = unsafe { CStr::from_ptr(message as _) };
|
||||
let result = unsafe {
|
||||
PIN_UI_CALLBACK
|
||||
.map(|c| c(wait, progress, message.to_str().unwrap_or("")))
|
||||
.unwrap_or(PinCallbackResult::Continue)
|
||||
};
|
||||
if matches!(result, PinCallbackResult::Abort) {
|
||||
ffi::sectrue
|
||||
} else {
|
||||
ffi::secfalse
|
||||
}
|
||||
}
|
||||
|
||||
/// Storage error type.
|
||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||
pub enum StorageError {
|
||||
/// Data is too long or not appropriate for the operation.
|
||||
InvalidData,
|
||||
/// Write failed.
|
||||
WriteFailed,
|
||||
/// Read failed.
|
||||
ReadFailed,
|
||||
/// Delete failed.
|
||||
DeleteFailed,
|
||||
/// Failed to perform the get-and-increment operation on a counter.
|
||||
CounterFailed,
|
||||
}
|
||||
|
||||
impl From<StorageError> for Error {
|
||||
fn from(err: StorageError) -> Self {
|
||||
match err {
|
||||
StorageError::InvalidData => Error::ValueError(cstr!("Invalid data for storage")),
|
||||
StorageError::WriteFailed => Error::ValueError(cstr!("Storage write failed")),
|
||||
StorageError::ReadFailed => Error::ValueError(cstr!("Storage read failed")),
|
||||
StorageError::DeleteFailed => Error::ValueError(cstr!("Storage delete failed")),
|
||||
StorageError::CounterFailed => {
|
||||
Error::ValueError(cstr!("Retrieving counter value failed"))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub type StorageResult<T> = Result<T, StorageError>;
|
||||
|
||||
/// Initialize the storage layer.
|
||||
/// This function must be called before any other storage function.
|
||||
pub fn init() {
|
||||
unsafe {
|
||||
ffi::storage_init(
|
||||
Some(callback_wrapper),
|
||||
ffi::HW_ENTROPY_DATA.as_ptr(),
|
||||
ffi::HW_ENTROPY_DATA.len() as u16,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/// Set the PIN callback function.
|
||||
/// See documentation for `PinDelayCallback` for more information.
|
||||
pub fn set_pin_delay_callback(callback: PinDelayCallback) {
|
||||
unsafe {
|
||||
PIN_UI_CALLBACK = Some(callback);
|
||||
}
|
||||
}
|
||||
|
||||
/// Wipe storage.
|
||||
pub fn wipe() {
|
||||
unsafe { ffi::storage_wipe() }
|
||||
}
|
||||
|
||||
/// Check if storage is unlocked.
|
||||
pub fn is_unlocked() -> bool {
|
||||
ffi::sectrue == unsafe { ffi::storage_is_unlocked() }
|
||||
}
|
||||
|
||||
/// Lock storage.
|
||||
pub fn lock() {
|
||||
unsafe { ffi::storage_lock() }
|
||||
}
|
||||
|
||||
/// Unlock storage with PIN and optional external salt.
|
||||
/// Returns true if the PIN + salt combination is correct.
|
||||
pub fn unlock(pin: &str, salt: Option<&ExternalSalt>) -> bool {
|
||||
let salt = salt.map(|s| s.as_ptr()).unwrap_or(ptr::null());
|
||||
ffi::sectrue == unsafe { ffi::storage_unlock(pin.as_ptr() as *const _, pin.len(), salt) }
|
||||
}
|
||||
|
||||
/// Change PIN and/or external salt.
|
||||
/// Returns true if the PIN + salt combination is correct and the change was
|
||||
/// successful.
|
||||
pub fn change_pin(
|
||||
old_pin: &str,
|
||||
new_pin: &str,
|
||||
old_salt: Option<&ExternalSalt>,
|
||||
new_salt: Option<&ExternalSalt>,
|
||||
) -> bool {
|
||||
ffi::sectrue
|
||||
== unsafe {
|
||||
ffi::storage_change_pin(
|
||||
old_pin.as_ptr() as *const _,
|
||||
old_pin.len(),
|
||||
new_pin.as_ptr() as *const _,
|
||||
new_pin.len(),
|
||||
old_salt.map(|s| s.as_ptr()).unwrap_or(ptr::null()),
|
||||
new_salt.map(|s| s.as_ptr()).unwrap_or(ptr::null()),
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/// Check if storage has PIN set.
|
||||
pub fn has_pin() -> bool {
|
||||
ffi::sectrue == unsafe { ffi::storage_has_pin() }
|
||||
}
|
||||
|
||||
/// Get remaining PIN attempts.
|
||||
pub fn get_pin_remaining() -> u32 {
|
||||
unsafe { ffi::storage_get_pin_rem() }
|
||||
}
|
||||
|
||||
pub fn ensure_not_wipe_pin(pin: &str) {
|
||||
unsafe {
|
||||
ffi::storage_ensure_not_wipe_code(pin.as_ptr(), pin.len());
|
||||
}
|
||||
}
|
||||
|
||||
/// Check if value for `appkey` exists.
|
||||
pub fn has(appkey: u16) -> bool {
|
||||
ffi::sectrue == unsafe { ffi::storage_has(appkey) }
|
||||
}
|
||||
|
||||
/// Get length of value stored for `appkey`.
|
||||
/// Returns an error if the value does not exist.
|
||||
pub fn get_length(appkey: u16) -> StorageResult<usize> {
|
||||
let mut len = 0;
|
||||
if ffi::sectrue == unsafe { ffi::storage_get(appkey, ptr::null_mut() as _, 0, &mut len) } {
|
||||
Ok(len as usize)
|
||||
} else {
|
||||
Err(StorageError::ReadFailed)
|
||||
}
|
||||
}
|
||||
|
||||
/// Get value for `appkey` from storage.
|
||||
/// The provided `data` buffer must be long enough to hold the value. If the
|
||||
/// buffer is shorter than the stored value, an error is returned. Use
|
||||
/// `get_length` to find the size of the value beforehand. On success, the data
|
||||
/// is copied to `data`, and subslice is returned. Data previously
|
||||
/// in the buffer beyond the copied data is not modified.
|
||||
/// If `appkey` is not present in storage, or the value is private and storage
|
||||
/// is locked, returns an error.
|
||||
pub fn get(appkey: u16, data: &mut [u8]) -> StorageResult<&[u8]> {
|
||||
let max_len = data.len().min(u16::MAX as usize) as u16;
|
||||
let mut len = 0u16;
|
||||
if ffi::sectrue
|
||||
== unsafe { ffi::storage_get(appkey, data.as_mut_ptr() as _, max_len, &mut len) }
|
||||
{
|
||||
Ok(&data[0..len as usize])
|
||||
} else {
|
||||
Err(StorageError::ReadFailed)
|
||||
}
|
||||
}
|
||||
|
||||
/// Set value for `appkey` in storage.
|
||||
/// The maximum length of the value is `u16::MAX`. Passing longer buffers will
|
||||
/// result in an error.
|
||||
/// If storage is locked and the value is not public-writable, returns an error.
|
||||
pub fn set(appkey: u16, data: &[u8]) -> StorageResult<()> {
|
||||
if data.len() > u16::MAX as usize {
|
||||
Err(StorageError::InvalidData)
|
||||
} else if ffi::sectrue
|
||||
== unsafe { ffi::storage_set(appkey, data.as_ptr() as _, data.len() as u16) }
|
||||
{
|
||||
Ok(())
|
||||
} else {
|
||||
Err(StorageError::WriteFailed)
|
||||
}
|
||||
}
|
||||
|
||||
/// Set value for monotonic counter `appkey` in storage.
|
||||
pub fn set_counter(appkey: u16, counter: u32) -> StorageResult<()> {
|
||||
if ffi::sectrue == unsafe { ffi::storage_set_counter(appkey, counter) } {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(StorageError::WriteFailed)
|
||||
}
|
||||
}
|
||||
|
||||
/// Atomically get and increment the value for monotonic counter `appkey` in
|
||||
/// storage.
|
||||
pub fn get_next_counter(appkey: u16) -> StorageResult<u32> {
|
||||
let mut counter = 0u32;
|
||||
if ffi::sectrue == unsafe { ffi::storage_next_counter(appkey, &mut counter) } {
|
||||
Ok(counter)
|
||||
} else {
|
||||
Err(StorageError::CounterFailed)
|
||||
}
|
||||
}
|
||||
|
||||
/// Delete value for `appkey` from storage.
|
||||
/// Returns Ok if the value was successfully deleted.
|
||||
/// If the storage is locked, or the value does not exist in storage, returns
|
||||
/// an error.
|
||||
pub fn delete(appkey: u16) -> StorageResult<()> {
|
||||
if ffi::sectrue == unsafe { ffi::storage_delete(appkey) } {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(StorageError::DeleteFailed)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
|
||||
const APPKEY: u16 = 0x0101;
|
||||
|
||||
fn init_storage(unlock: bool) {
|
||||
unsafe {
|
||||
ffi::flash_init();
|
||||
}
|
||||
init();
|
||||
wipe();
|
||||
lock();
|
||||
if unlock {
|
||||
super::unlock("", None);
|
||||
}
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_init() {
|
||||
init();
|
||||
}
|
||||
|
||||
static mut PIN_CALLBACK_CALLED: bool = false;
|
||||
|
||||
fn pin_callback(_wait: u32, _progress: u32, _message: &str) -> PinCallbackResult {
|
||||
unsafe {
|
||||
PIN_CALLBACK_CALLED = true;
|
||||
}
|
||||
PinCallbackResult::Continue
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_callback() {
|
||||
set_pin_delay_callback(pin_callback);
|
||||
init_storage(false);
|
||||
unsafe {
|
||||
PIN_CALLBACK_CALLED = false;
|
||||
}
|
||||
unlock("", None);
|
||||
assert!(unsafe { PIN_CALLBACK_CALLED });
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_get_set() {
|
||||
init_storage(true);
|
||||
let mut data = [0u8; 42];
|
||||
for i in 0..data.len() {
|
||||
data[i] = i as u8;
|
||||
}
|
||||
assert!(set(APPKEY, &data).is_ok());
|
||||
let mut data2 = [0u8; 42];
|
||||
assert_eq!(get(APPKEY, &mut data2).unwrap().len(), data.len());
|
||||
assert_eq!(get(APPKEY, &mut data2).unwrap(), data);
|
||||
assert_eq!(data, data2);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_unlock() {
|
||||
init_storage(false);
|
||||
assert!(!is_unlocked());
|
||||
let buf = [0u8; 32];
|
||||
assert_eq!(set(APPKEY, &buf), Err(StorageError::WriteFailed));
|
||||
assert!(unlock("", None));
|
||||
assert!(is_unlocked());
|
||||
assert_eq!(set(APPKEY, &buf), Ok(()));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_pin() {
|
||||
init_storage(false);
|
||||
assert!(!is_unlocked());
|
||||
assert!(!has_pin());
|
||||
assert!(unlock("", None));
|
||||
assert!(is_unlocked());
|
||||
assert!(!has_pin());
|
||||
|
||||
lock();
|
||||
assert!(!is_unlocked());
|
||||
assert!(!has_pin());
|
||||
|
||||
assert!(!unlock("1234", None));
|
||||
|
||||
//unlock("", None);
|
||||
assert!(change_pin("", "1234", None, None));
|
||||
assert!(has_pin());
|
||||
|
||||
lock();
|
||||
assert!(!unlock("", None));
|
||||
assert!(unlock("1234", None));
|
||||
}
|
||||
}
|
@ -3,6 +3,7 @@
|
||||
#include "display.h"
|
||||
#include "display_interface.h"
|
||||
#include "dma2d.h"
|
||||
#include "flash.h"
|
||||
#include "fonts/fonts.h"
|
||||
#include "rgb_led.h"
|
||||
#include "secbool.h"
|
||||
|
Loading…
Reference in New Issue
Block a user