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}