#![feature(allocator_api)]
#![no_main]
#![no_std]
extern crate alloc;
use defmt_semihosting as _;
use alloc::collections::LinkedList;
use core::{mem::MaybeUninit, panic::PanicInfo};
use cortex_m as _;
use cortex_m_rt::entry;
use embedded_alloc::TlsfHeap as Heap;
#[global_allocator]
static HEAP: Heap = Heap::empty();
const HEAP_SIZE: usize = 30 * 1024;
pub type TestTable<'a> = &'a [(fn() -> (), &'static str)];
fn test_global_heap() {
const ELEMS: usize = 250;
let mut allocated = LinkedList::new();
for _ in 0..ELEMS {
allocated.push_back(0);
}
for i in 0..ELEMS {
allocated.push_back(i as i32);
}
assert_eq!(allocated.len(), 2 * ELEMS);
for _ in 0..ELEMS {
allocated.pop_front();
}
for i in 0..ELEMS {
assert_eq!(allocated.pop_front().unwrap(), i as i32);
}
}
fn test_allocator_api() {
const HEAP_SIZE: usize = 256;
let mut heap_mem: [MaybeUninit<u8>; HEAP_SIZE] = [MaybeUninit::uninit(); HEAP_SIZE];
let local_heap: Heap = Heap::empty();
unsafe { local_heap.init(heap_mem.as_mut_ptr() as usize, HEAP_SIZE) }
const ELEMS: usize = 2;
let mut allocated = LinkedList::new_in(local_heap);
for _ in 0..ELEMS {
allocated.push_back(0);
}
for i in 0..ELEMS {
allocated.push_back(i as i32);
}
assert_eq!(allocated.len(), 2 * ELEMS);
for _ in 0..ELEMS {
allocated.pop_front();
}
for i in 0..ELEMS {
assert_eq!(allocated.pop_front().unwrap(), i as i32);
}
}
#[entry]
fn main() -> ! {
unsafe {
embedded_alloc::init!(HEAP, HEAP_SIZE);
}
let tests: TestTable = &[
(test_global_heap, "test_global_heap"),
(test_allocator_api, "test_allocator_api"),
];
for (test_fn, test_name) in tests {
defmt::info!("{}: start", test_name);
test_fn();
defmt::info!("{}: pass", test_name);
}
semihosting::process::exit(0);
}
#[panic_handler]
fn panic(info: &PanicInfo) -> ! {
defmt::error!("{}", info);
semihosting::process::exit(-1);
}