1use core::{fmt, ptr, slice};
6
7use crate::peripheral::itm::Stim;
8
9unsafe fn write_words(stim: &mut Stim, bytes: &[u32]) {
11    let mut p = bytes.as_ptr();
12    for _ in 0..bytes.len() {
13        while !stim.is_fifo_ready() {}
14        stim.write_u32(ptr::read(p));
15        p = p.offset(1);
16    }
17}
18
19unsafe fn write_aligned_impl(port: &mut Stim, buffer: &[u8]) {
23    let len = buffer.len();
24
25    if len == 0 {
26        return;
27    }
28
29    let split = len & !0b11;
30    #[allow(clippy::cast_ptr_alignment)]
31    write_words(
32        port,
33        slice::from_raw_parts(buffer.as_ptr() as *const u32, split >> 2),
34    );
35
36    let mut left = len & 0b11;
38    let mut ptr = buffer.as_ptr().add(split);
39
40    if left > 1 {
42        while !port.is_fifo_ready() {}
43
44        #[allow(clippy::cast_ptr_alignment)]
45        port.write_u16(ptr::read(ptr as *const u16));
46
47        ptr = ptr.offset(2);
48        left -= 2;
49    }
50
51    if left == 1 {
53        while !port.is_fifo_ready() {}
54        port.write_u8(*ptr);
55    }
56}
57
58struct Port<'p>(&'p mut Stim);
59
60impl<'p> fmt::Write for Port<'p> {
61    #[inline]
62    fn write_str(&mut self, s: &str) -> fmt::Result {
63        write_all(self.0, s.as_bytes());
64        Ok(())
65    }
66}
67
68#[repr(align(4))]
73pub struct Aligned<T: ?Sized>(pub T);
74
75#[allow(clippy::missing_inline_in_public_items)]
77pub fn write_all(port: &mut Stim, buffer: &[u8]) {
78    unsafe {
79        let mut len = buffer.len();
80        let mut ptr = buffer.as_ptr();
81
82        if len == 0 {
83            return;
84        }
85
86        if ptr as usize % 2 == 1 {
88            while !port.is_fifo_ready() {}
89            port.write_u8(*ptr);
90
91            ptr = ptr.offset(1);
93            len -= 1;
94        }
95
96        if ptr as usize % 4 == 2 {
98            if len > 1 {
99                while !port.is_fifo_ready() {}
101
102                #[allow(clippy::cast_ptr_alignment)]
104                port.write_u16(ptr::read(ptr as *const u16));
105
106                ptr = ptr.offset(2);
108                len -= 2;
109            } else {
110                if len == 1 {
111                    while !port.is_fifo_ready() {}
113                    port.write_u8(*ptr);
114                }
115
116                return;
117            }
118        }
119
120        write_aligned_impl(port, slice::from_raw_parts(ptr, len));
122    }
123}
124
125#[allow(clippy::missing_inline_in_public_items)]
142pub fn write_aligned(port: &mut Stim, buffer: &Aligned<[u8]>) {
143    unsafe { write_aligned_impl(port, &buffer.0) }
144}
145
146#[inline]
148pub fn write_fmt(port: &mut Stim, args: fmt::Arguments) {
149    use core::fmt::Write;
150
151    Port(port).write_fmt(args).ok();
152}
153
154#[inline]
156pub fn write_str(port: &mut Stim, string: &str) {
157    write_all(port, string.as_bytes())
158}