use core::panic::PanicInfo;
#[cfg(feature = "debug")]
use core::arch::asm;
#[cfg(feature = "debug")]
#[deprecated(note = "Use the logging macros from log module instead")]
pub fn debug_print(s: &str) {
let p = s.as_bytes().as_ptr();
for i in 0..s.len() {
let m = unsafe { p.add(i) };
unsafe {
asm!(
"svc #0xab",
in("r1") m,
inout("r0") 3 => _,
);
}
}
}
#[cfg(not(feature = "debug"))]
#[deprecated(note = "Use the logging macros from log module instead")]
pub fn debug_print(_s: &str) {}
pub fn to_hex(m: u32) -> [u8; 8] {
let mut hex = [0u8; 8];
let mut i = 0;
for c in m.to_be_bytes().iter() {
let c0 = char::from_digit((c >> 4).into(), 16).unwrap();
let c1 = char::from_digit((c & 0xf).into(), 16).unwrap();
hex[i] = c0 as u8;
hex[i + 1] = c1 as u8;
i += 2;
}
hex
}
fn to_dec(v: u32) -> [u8; 10] {
let mut dec = [0u8; 10];
let mut val = v;
let mut fact = 1_000_000_000;
let mut i = 0;
while fact != 0 {
let d = val / fact;
let c = char::from_digit(d.into(), 10).unwrap();
dec[i] = c as u8;
i += 1;
val -= d * fact;
fact /= 10;
}
dec
}
#[cfg_attr(test, panic_handler)]
pub fn test_panic(info: &PanicInfo) -> ! {
let loc = info.location().unwrap();
let bytes = to_dec(loc.line());
let s = core::str::from_utf8(&bytes)
.unwrap()
.trim_start_matches('0');
crate::log::error!(
"Panic in {} at line {}: {}",
loc.file(),
s,
info.message().as_str().unwrap()
);
ledger_secure_sdk_sys::exit_app(1);
}
#[cfg(feature = "unit_test")]
pub struct TestType {
pub modname: &'static str,
pub name: &'static str,
pub f: fn() -> Result<(), ()>,
}
#[cfg(feature = "unit_test")]
pub fn sdk_test_runner(tests: &[&TestType]) {
use core::ffi::c_void;
use ledger_secure_sdk_sys::{pic, pic_rs};
let mut failures = 0;
crate::log::info!("--- Tests ---");
for test_ in tests {
let test = pic_rs(*test_);
let modname;
let name;
unsafe {
let t = pic(test.modname.as_ptr() as *mut c_void) as *const u8;
let t = core::ptr::slice_from_raw_parts(t, test.modname.len());
let t: &[u8] = core::mem::transmute(t);
modname = core::str::from_utf8_unchecked(t);
let t = pic(test.name.as_ptr() as *mut c_void) as *const u8;
let t = core::ptr::slice_from_raw_parts(t, test.name.len());
let t: &[u8] = core::mem::transmute(t);
name = core::str::from_utf8_unchecked(t);
}
let fp = unsafe { pic(test.f as *mut c_void) };
let fp: fn() -> Result<(), ()> = unsafe { core::mem::transmute(fp) };
let res = fp();
let res_out = match res {
Ok(()) => "\x1b[1;32m ok \x1b[0m",
Err(()) => {
failures += 1;
"\x1b[1;31m fail \x1b[0m"
}
};
crate::log::info!("{} {}::{}", res_out, modname, name);
}
if failures > 0 {
ledger_secure_sdk_sys::exit_app(1);
}
ledger_secure_sdk_sys::exit_app(0);
}
#[cfg(feature = "unit_test")]
#[macro_export]
macro_rules! assert_eq_err {
($left:expr, $right:expr) => {{
match (&$left, &$right) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
crate::log::error!("assertion failed: `(left == right)`");
return Err(());
}
}
}
}};
}