// Licensed to the Apache Software Foundation (ASF) under one // or more contributor license agreements. See the NOTICE file // distributed with this work for additional information // regarding copyright ownership. The ASF licenses this file // to you under the Apache License, Version 2.0 (the // "License"); you may not use this file except in compliance // with the License. You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, // software distributed under the License is distributed on an // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the License for the // specific language governing permissions and limitations // under the License. #![no_std] #![no_main] #![feature(c_size_t)] extern crate alloc; use alloc::vec; use optee_utee::{ ta_close_session, ta_create, ta_destroy, ta_invoke_command, ta_open_session, trace_println, }; use optee_utee::{DataFlag, ObjectStorageConstants, PersistentObject}; use optee_utee::{Error, ErrorKind, Parameters, Result}; use proto::{Command}; #[ta_create] fn create() -> Result<()> { trace_println!("[+] TA create"); Ok(()) } #[ta_open_session] fn open_session(_params: &mut Parameters) -> Result<()> { trace_println!("[+] TA open session"); Ok(()) } #[ta_close_session] fn close_session() { trace_println!("[+] TA close session"); } #[ta_destroy] fn destroy() { trace_println!("[+] TA destroy"); } #[ta_invoke_command] fn invoke_command(cmd_id: u32, params: &mut Parameters) -> Result<()> { trace_println!("[+] TA invoke command"); match Command::from(cmd_id) { Command::Write => { return create_raw_object(params); } Command::Read => { return read_raw_object(params); } Command::Delete => { return delete_object(params); } _ => { return Err(Error::new(ErrorKind::NotSupported)); } } } pub fn delete_object(params: &mut Parameters) -> Result<()> { let mut p0 = unsafe { params.0.as_memref().unwrap() }; let mut obj_id = vec![0; p0.buffer().len() as usize]; obj_id.copy_from_slice(p0.buffer()); match PersistentObject::open( ObjectStorageConstants::Private, &mut obj_id, DataFlag::ACCESS_READ | DataFlag::ACCESS_WRITE_META, ) { Err(e) => { return Err(e); } Ok(mut object) => { object.close_and_delete()?; mem::forget(object); return Ok(()); } } } pub fn create_raw_object(params: &mut Parameters) -> Result<()> { let mut p0 = unsafe { params.0.as_memref().unwrap() }; let mut p1 = unsafe { params.1.as_memref().unwrap() }; let mut obj_id = vec![0; p0.buffer().len() as usize]; obj_id.copy_from_slice(p0.buffer()); let mut data_buffer = vec![0; p1.buffer().len() as usize]; data_buffer.copy_from_slice(p1.buffer()); let obj_data_flag = DataFlag::ACCESS_READ | DataFlag::ACCESS_WRITE | DataFlag::ACCESS_WRITE_META | DataFlag::OVERWRITE; let mut init_data: [u8; 0] = [0; 0]; match PersistentObject::create( ObjectStorageConstants::Private, &mut obj_id, obj_data_flag, None, &mut init_data, ) { Err(e) => { return Err(e); } Ok(mut object) => match object.write(&data_buffer) { Ok(()) => { return Ok(()); } Err(e_write) => { object.close_and_delete()?; mem::forget(object); return Err(e_write); } }, } } pub fn read_raw_object(params: &mut Parameters) -> Result<()> { let mut p0 = unsafe { params.0.as_memref().unwrap() }; let mut p1 = unsafe { params.1.as_memref().unwrap() }; let mut obj_id = vec![0; p0.buffer().len() as usize]; obj_id.copy_from_slice(p0.buffer()); let mut data_buffer = vec![0;p1.buffer().len() as usize]; data_buffer.copy_from_slice(p1.buffer()); match PersistentObject::open( ObjectStorageConstants::Private, &mut obj_id, DataFlag::ACCESS_READ | DataFlag::SHARE_READ, ) { Err(e) => return Err(e), Ok(object) => { let obj_info = object.info()?; if obj_info.data_size() > p1.buffer().len() { p1.set_updated_size(obj_info.data_size()); return Err(Error::new(ErrorKind::ShortBuffer)); } let read_bytes = object.read(&mut data_buffer).unwrap(); if read_bytes != obj_info.data_size() as u32 { return Err(Error::new(ErrorKind::ExcessData)); } p1.set_updated_size(read_bytes as usize); p1.buffer().copy_from_slice(&data_buffer); Ok(()) } } } // TA configurations const TA_FLAGS: u32 = 0; const TA_DATA_SIZE: u32 = 32 * 1024; const TA_STACK_SIZE: u32 = 2 * 1024; const TA_VERSION: &[u8] = b"0.1\0"; const TA_DESCRIPTION: &[u8] = b"This is a secure storage example.\0"; const EXT_PROP_VALUE_1: &[u8] = b"Secure Storage TA\0"; const EXT_PROP_VALUE_2: u32 = 0x0010; const TRACE_LEVEL: i32 = 4; const TRACE_EXT_PREFIX: &[u8] = b"TA\0"; const TA_FRAMEWORK_STACK_SIZE: u32 = 2048; include!(concat!(env!("OUT_DIR"), "/user_ta_header.rs"));