forked from stm32-rs/stm32h7xx-hal
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserial-dma.rs
175 lines (139 loc) · 5.27 KB
/
serial-dma.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
//! Example that transmits serial data using the DMA
//!
//! The first part of the example transmits 10 bytes over serial.
//!
//! The maximum transfer length for DMA1/DMA2 is limited to 65_535 items by
//! hardware. The second part of this example demonstrates splitting a transfer
//! into chunks and using the `next_transfer_with` method to start each part of
//! the transfer.
#![deny(warnings)]
#![no_main]
#![no_std]
use core::{mem, mem::MaybeUninit};
use cortex_m_rt::entry;
#[macro_use]
mod utilities;
use stm32h7xx_hal::{pac, prelude::*};
use stm32h7xx_hal::dma::{
dma::{DmaConfig, StreamsTuple},
MemoryToPeripheral, Transfer,
};
use log::info;
// DMA1/DMA2 cannot interact with our stack. Instead, buffers for use with the
// DMA must be placed somewhere that DMA1/DMA2 can access. In this case we use
// AXI SRAM.
//
// The runtime does not initialise these SRAM banks
#[link_section = ".axisram.buffers"]
static mut SHORT_BUFFER: MaybeUninit<[u8; 10]> = MaybeUninit::uninit();
#[link_section = ".axisram.buffers"]
static mut LONG_BUFFER: MaybeUninit<[u32; 0x8000]> = MaybeUninit::uninit();
#[entry]
fn main() -> ! {
utilities::logger::init();
let dp = pac::Peripherals::take().unwrap();
// Constrain and Freeze power
info!("Setup PWR... ");
let pwr = dp.PWR.constrain();
let pwrcfg = example_power!(pwr).freeze();
// Constrain and Freeze clock
info!("Setup RCC... ");
let rcc = dp.RCC.constrain();
let ccdr = rcc
.sys_ck(200.MHz())
.pll1_q_ck(200.MHz())
.freeze(pwrcfg, &dp.SYSCFG);
// Acquire the GPIOC peripheral. This also enables the clock for
// GPIOC in the RCC register.
let gpioc = dp.GPIOC.split(ccdr.peripheral.GPIOC);
let tx = gpioc.pc10.into_alternate();
let rx = gpioc.pc11.into_alternate();
info!("");
info!("stm32h7xx-hal example - Serial DMA");
info!("");
// Configure the serial peripheral.
let serial = dp
.USART3
.serial(
(tx, rx),
1_000_000.bps(),
ccdr.peripheral.USART3,
&ccdr.clocks,
)
.unwrap();
let (tx, _rx) = serial.split();
// Initialise the source buffer, without taking any references to
// uninitialised memory
let short_buffer: &'static mut [u8; 10] = {
let buf: &mut [MaybeUninit<u8>; 10] =
unsafe { &mut *(core::ptr::addr_of_mut!(SHORT_BUFFER) as *mut _) };
for (i, value) in buf.iter_mut().enumerate() {
unsafe {
value.as_mut_ptr().write(i as u8 + 96); // 0x60, 0x61, 0x62...
}
}
unsafe { SHORT_BUFFER.assume_init_mut() }
};
// view u32 buffer as u8. Endianess is undefined (little-endian on STM32H7)
let long_buffer: &'static mut [u8; 0x2_0010] = {
let buf: &mut [MaybeUninit<u32>; 0x8004] =
unsafe { &mut *(core::ptr::addr_of_mut!(LONG_BUFFER) as *mut _) };
for (i, value) in buf.iter_mut().enumerate() {
unsafe {
value.as_mut_ptr().write(i as u32);
}
}
unsafe {
&mut *(core::ptr::addr_of_mut!(LONG_BUFFER) as *mut [u8; 0x2_0010])
}
};
// Setup the DMA transfer on stream 0
//
// We need to specify the direction with a type annotation, since DMA
// transfers both to and from the UART are possible
let streams = StreamsTuple::new(dp.DMA1, ccdr.peripheral.DMA1);
let config = DmaConfig::default().memory_increment(true);
let mut transfer: Transfer<_, _, MemoryToPeripheral, _, _> =
Transfer::init(streams.0, tx, &mut short_buffer[..], None, config);
transfer.start(|serial| {
// This closure runs right after enabling the stream
// Enable DMA Tx buffer by setting the DMAT bit in the USART_CR3
// register
serial.enable_dma_tx();
});
// Wait for transfer to complete
while !transfer.get_transfer_complete_flag() {}
// Disable DMA Tx once complete (or before)
info!("Continuing with chunked transfer!");
// Split the long buffer into chunks: Hardware supports 65_535 max.
//
// The last chunk will be length 16, compared to all the others which will
// be length 32_768.
for mut chunk in &mut long_buffer.chunks_mut(32_768) {
// Using `next_transfer_with`
let _current = transfer
.next_transfer_with(|mut old, current, remaining| {
// Check that we really did complete the current transfer
assert_eq!(remaining, 0);
mem::swap(&mut old, &mut chunk);
(old, current)
})
.unwrap();
// Using `next_transfer`: this is equivalent to the above (except the
// assert) but less flexible
//transfer.next_transfer(chunk).unwrap();
// Wait for transfer to complete
while !transfer.get_transfer_complete_flag() {}
}
transfer.pause(|serial| {
// At this point, the DMA transfer is done, but the data is still in the
// UART output FIFO. Wait for it to complete
while !serial.is_txe() {}
});
info!("Chunked transfer complete!");
let (_stream, _serial, _, _) = transfer.free();
// We could re-use the stream or serial here
loop {
cortex_m::asm::nop()
}
}