mod tee_client_api;
use self::tee_client_api::*;
use log::error;
use std::{mem, ptr};
use self::tee_client_api::{
TEEC_TempMemoryReference as TeecTempMemoryReference, TEEC_Value as TeecValue,
};
pub use self::tee_client_api::{
TEEC_Operation as TeecOperation, TEEC_Parameter as TeecParameter, TEEC_ERROR_NOT_SUPPORTED,
TEEC_ERROR_SHORT_BUFFER, TEEC_MEMREF_TEMP_INPUT, TEEC_MEMREF_TEMP_OUTPUT, TEEC_NONE,
TEEC_SUCCESS, TEEC_VALUE_INPUT,
};
pub static KEYSAFE_TA_UUID: TEEC_UUID = TEEC_UUID {
timeLow: 0x808032e0,
timeMid: 0xfd9e,
timeHiAndVersion: 0x4e6f,
clockSeqAndNode: [0x88, 0x96, 0x54, 0x47, 0x35, 0xc9, 0x84, 0x80],
};
pub fn teec_param_types(
param0_type: u32,
param1_type: u32,
param2_type: u32,
param3_type: u32,
) -> u32 {
((param0_type & 0xF) << 0)
| ((param1_type & 0xF) << 4)
| ((param2_type & 0xF) << 8)
| ((param3_type & 0xF) << 12)
}
pub fn get_zero_parameter() -> TeecParameter {
let zero_parameter: TeecParameter = unsafe { mem::zeroed() };
zero_parameter
}
pub fn get_value_parameter(a: u32, b: u32) -> TeecParameter {
TeecParameter { value: TeecValue { a, b } }
}
pub fn get_memref_output_parameter(data: &mut [u8]) -> TeecParameter {
TeecParameter {
tmpref: TeecTempMemoryReference {
buffer: data.as_mut_ptr() as *mut std::ffi::c_void,
size: data.len() as u64,
},
}
}
pub fn get_memref_input_parameter(data: &[u8]) -> TeecParameter {
TeecParameter {
tmpref: TeecTempMemoryReference {
buffer: data.as_ptr() as *mut std::ffi::c_void,
size: data.len() as u64,
},
}
}
pub fn create_operation(param_type: u32, params: [TeecParameter; 4]) -> TeecOperation {
TeecOperation {
started: 0,
paramTypes: param_type,
params,
imp: teec_operation_impl { reserved: 0 as ::std::os::raw::c_char },
}
}
pub fn call_command(op: &mut TeecOperation, command_id: u32) -> Result<(), u32> {
let mut tee_context = TeeContext::new()?;
let mut tee_session = tee_context.new_session()?;
let mut return_origin: u32 = 0;
tee_session.invoke_command(command_id, op, &mut return_origin)
}
pub fn call_command_on_device(
dev: &std::ffi::CStr,
op: &mut TeecOperation,
command_id: u32,
) -> Result<(), u32> {
let mut tee_context = TeeContext::new_with_device(dev)?;
let mut tee_session = tee_context.new_session()?;
let mut return_origin: u32 = 0;
tee_session.invoke_command(command_id, op, &mut return_origin)
}
struct TeeContext {
context: TEEC_Context,
}
impl TeeContext {
pub fn new() -> Result<Self, u32> {
Self::new_with_ptr(ptr::null())
}
pub fn new_with_device(dev: &std::ffi::CStr) -> Result<Self, u32> {
Self::new_with_ptr(dev.as_ptr())
}
fn new_with_ptr(dev: *const std::os::raw::c_char) -> Result<Self, u32> {
let mut context: TEEC_Context = unsafe { mem::zeroed() };
let result = unsafe { TEEC_InitializeContext(dev, &mut context) };
if result != TEEC_SUCCESS {
error!("Failed to initialize context: {:?}", result);
return Err(result);
}
Ok(TeeContext { context })
}
pub fn new_session(&mut self) -> Result<TeeSession, u32> {
let mut session: TEEC_Session = unsafe { mem::zeroed() };
let mut return_origin: u32 = 0;
let result = unsafe {
TEEC_OpenSession(
&mut self.context,
&mut session,
&KEYSAFE_TA_UUID,
TEEC_LOGIN_PUBLIC,
ptr::null_mut(),
ptr::null_mut(),
&mut return_origin,
)
};
if result != TEEC_SUCCESS {
error!("Failed to open session ({:?} {:?})\n", result, return_origin);
return Err(result);
}
Ok(TeeSession { session })
}
}
impl Drop for TeeContext {
fn drop(&mut self) {
unsafe { TEEC_FinalizeContext(&mut self.context) };
}
}
struct TeeSession {
session: TEEC_Session,
}
impl TeeSession {
pub fn invoke_command(
&mut self,
command_id: u32,
operation: *mut TEEC_Operation,
return_origin: *mut u32,
) -> Result<(), u32> {
let result =
unsafe { TEEC_InvokeCommand(&mut self.session, command_id, operation, return_origin) };
if result != TEEC_SUCCESS {
error!("TEEC_InvokeCommand failed with code {:?} origin {:?}", result, return_origin);
return Err(result);
}
Ok(())
}
}
impl Drop for TeeSession {
fn drop(&mut self) {
unsafe { TEEC_CloseSession(&mut self.session) };
}
}