diff options
| author | Timo Kröger <[email protected]> | 2020-12-23 16:18:29 +0100 |
|---|---|---|
| committer | Timo Kröger <[email protected]> | 2020-12-30 19:24:12 +0100 |
| commit | 6695bf0f21065960d6e83f2248fc6ad9a8ffa528 (patch) | |
| tree | 8da853ea78101fa98a62986d1108d597a0777ddf /examples | |
| parent | c97d5262f56814639fa995c8426384efdf379c55 (diff) | |
Low power UART driver
Diffstat (limited to 'examples')
| -rw-r--r-- | examples/src/bin/buffered_uart.rs | 84 | ||||
| -rw-r--r-- | examples/src/bin/uart.rs | 121 |
2 files changed, 156 insertions, 49 deletions
diff --git a/examples/src/bin/buffered_uart.rs b/examples/src/bin/buffered_uart.rs new file mode 100644 index 000000000..6e15fbcfa --- /dev/null +++ b/examples/src/bin/buffered_uart.rs | |||
| @@ -0,0 +1,84 @@ | |||
| 1 | #![no_std] | ||
| 2 | #![no_main] | ||
| 3 | #![feature(type_alias_impl_trait)] | ||
| 4 | |||
| 5 | #[path = "../example_common.rs"] | ||
| 6 | mod example_common; | ||
| 7 | use example_common::*; | ||
| 8 | |||
| 9 | use cortex_m_rt::entry; | ||
| 10 | use defmt::panic; | ||
| 11 | use futures::pin_mut; | ||
| 12 | use nrf52840_hal::gpio; | ||
| 13 | |||
| 14 | use embassy::executor::{task, Executor}; | ||
| 15 | use embassy::io::{AsyncBufRead, AsyncBufReadExt, AsyncWrite, AsyncWriteExt}; | ||
| 16 | use embassy::util::Forever; | ||
| 17 | use embassy_nrf::buffered_uarte; | ||
| 18 | use embassy_nrf::interrupt; | ||
| 19 | |||
| 20 | #[task] | ||
| 21 | async fn run() { | ||
| 22 | let p = unwrap!(embassy_nrf::pac::Peripherals::take()); | ||
| 23 | |||
| 24 | let port0 = gpio::p0::Parts::new(p.P0); | ||
| 25 | |||
| 26 | let pins = buffered_uarte::Pins { | ||
| 27 | rxd: port0.p0_08.into_floating_input().degrade(), | ||
| 28 | txd: port0 | ||
| 29 | .p0_06 | ||
| 30 | .into_push_pull_output(gpio::Level::Low) | ||
| 31 | .degrade(), | ||
| 32 | cts: None, | ||
| 33 | rts: None, | ||
| 34 | }; | ||
| 35 | |||
| 36 | let irq = interrupt::take!(UARTE0_UART0); | ||
| 37 | let u = buffered_uarte::BufferedUarte::new( | ||
| 38 | p.UARTE0, | ||
| 39 | irq, | ||
| 40 | pins, | ||
| 41 | buffered_uarte::Parity::EXCLUDED, | ||
| 42 | buffered_uarte::Baudrate::BAUD115200, | ||
| 43 | ); | ||
| 44 | pin_mut!(u); | ||
| 45 | |||
| 46 | info!("uarte initialized!"); | ||
| 47 | |||
| 48 | unwrap!(u.write_all(b"Hello!\r\n").await); | ||
| 49 | info!("wrote hello in uart!"); | ||
| 50 | |||
| 51 | // Simple demo, reading 8-char chunks and echoing them back reversed. | ||
| 52 | loop { | ||
| 53 | info!("reading..."); | ||
| 54 | let mut buf = [0u8; 8]; | ||
| 55 | unwrap!(u.read_exact(&mut buf).await); | ||
| 56 | info!("read done, got {:[u8]}", buf); | ||
| 57 | |||
| 58 | // Reverse buf | ||
| 59 | for i in 0..4 { | ||
| 60 | let tmp = buf[i]; | ||
| 61 | buf[i] = buf[7 - i]; | ||
| 62 | buf[7 - i] = tmp; | ||
| 63 | } | ||
| 64 | |||
| 65 | info!("writing..."); | ||
| 66 | unwrap!(u.write_all(&buf).await); | ||
| 67 | info!("write done"); | ||
| 68 | } | ||
| 69 | } | ||
| 70 | |||
| 71 | static EXECUTOR: Forever<Executor> = Forever::new(); | ||
| 72 | |||
| 73 | #[entry] | ||
| 74 | fn main() -> ! { | ||
| 75 | info!("Hello World!"); | ||
| 76 | |||
| 77 | let executor = EXECUTOR.put(Executor::new(cortex_m::asm::sev)); | ||
| 78 | unwrap!(executor.spawn(run())); | ||
| 79 | |||
| 80 | loop { | ||
| 81 | executor.run(); | ||
| 82 | cortex_m::asm::wfe(); | ||
| 83 | } | ||
| 84 | } | ||
diff --git a/examples/src/bin/uart.rs b/examples/src/bin/uart.rs index 6e15fbcfa..107936686 100644 --- a/examples/src/bin/uart.rs +++ b/examples/src/bin/uart.rs | |||
| @@ -8,74 +8,97 @@ use example_common::*; | |||
| 8 | 8 | ||
| 9 | use cortex_m_rt::entry; | 9 | use cortex_m_rt::entry; |
| 10 | use defmt::panic; | 10 | use defmt::panic; |
| 11 | use futures::pin_mut; | ||
| 12 | use nrf52840_hal::gpio; | ||
| 13 | |||
| 14 | use embassy::executor::{task, Executor}; | 11 | use embassy::executor::{task, Executor}; |
| 15 | use embassy::io::{AsyncBufRead, AsyncBufReadExt, AsyncWrite, AsyncWriteExt}; | 12 | use embassy::time::{Duration, Timer}; |
| 16 | use embassy::util::Forever; | 13 | use embassy::util::Forever; |
| 17 | use embassy_nrf::buffered_uarte; | 14 | use embassy_nrf::{interrupt, pac, rtc, uarte}; |
| 18 | use embassy_nrf::interrupt; | 15 | use futures::future::{select, Either}; |
| 16 | use nrf52840_hal::clocks; | ||
| 17 | use nrf52840_hal::gpio; | ||
| 19 | 18 | ||
| 20 | #[task] | 19 | #[task] |
| 21 | async fn run() { | 20 | async fn run(mut uart: uarte::Uarte<pac::UARTE0>) { |
| 22 | let p = unwrap!(embassy_nrf::pac::Peripherals::take()); | ||
| 23 | |||
| 24 | let port0 = gpio::p0::Parts::new(p.P0); | ||
| 25 | |||
| 26 | let pins = buffered_uarte::Pins { | ||
| 27 | rxd: port0.p0_08.into_floating_input().degrade(), | ||
| 28 | txd: port0 | ||
| 29 | .p0_06 | ||
| 30 | .into_push_pull_output(gpio::Level::Low) | ||
| 31 | .degrade(), | ||
| 32 | cts: None, | ||
| 33 | rts: None, | ||
| 34 | }; | ||
| 35 | |||
| 36 | let irq = interrupt::take!(UARTE0_UART0); | ||
| 37 | let u = buffered_uarte::BufferedUarte::new( | ||
| 38 | p.UARTE0, | ||
| 39 | irq, | ||
| 40 | pins, | ||
| 41 | buffered_uarte::Parity::EXCLUDED, | ||
| 42 | buffered_uarte::Baudrate::BAUD115200, | ||
| 43 | ); | ||
| 44 | pin_mut!(u); | ||
| 45 | |||
| 46 | info!("uarte initialized!"); | 21 | info!("uarte initialized!"); |
| 47 | 22 | ||
| 48 | unwrap!(u.write_all(b"Hello!\r\n").await); | 23 | // Message must be in SRAM |
| 24 | let mut buf = [0; 8]; | ||
| 25 | buf.copy_from_slice(b"Hello!\r\n"); | ||
| 26 | |||
| 27 | uart.send(&buf).await; | ||
| 49 | info!("wrote hello in uart!"); | 28 | info!("wrote hello in uart!"); |
| 50 | 29 | ||
| 51 | // Simple demo, reading 8-char chunks and echoing them back reversed. | 30 | info!("reading..."); |
| 52 | loop { | 31 | loop { |
| 53 | info!("reading..."); | 32 | let received = match select( |
| 54 | let mut buf = [0u8; 8]; | 33 | uart.receive(&mut buf), |
| 55 | unwrap!(u.read_exact(&mut buf).await); | 34 | Timer::after(Duration::from_millis(10)), |
| 56 | info!("read done, got {:[u8]}", buf); | 35 | ) |
| 57 | 36 | .await | |
| 58 | // Reverse buf | 37 | { |
| 59 | for i in 0..4 { | 38 | Either::Left((buf, _)) => buf, |
| 60 | let tmp = buf[i]; | 39 | Either::Right((_, read)) => { |
| 61 | buf[i] = buf[7 - i]; | 40 | let (buf, n) = read.stop().await; |
| 62 | buf[7 - i] = tmp; | 41 | &buf[..n] |
| 42 | } | ||
| 43 | }; | ||
| 44 | |||
| 45 | if received.len() > 0 { | ||
| 46 | info!("read done, got {:[u8]}", received); | ||
| 47 | |||
| 48 | // Echo back received data | ||
| 49 | uart.send(received).await; | ||
| 63 | } | 50 | } |
| 64 | |||
| 65 | info!("writing..."); | ||
| 66 | unwrap!(u.write_all(&buf).await); | ||
| 67 | info!("write done"); | ||
| 68 | } | 51 | } |
| 69 | } | 52 | } |
| 70 | 53 | ||
| 54 | static RTC: Forever<rtc::RTC<pac::RTC1>> = Forever::new(); | ||
| 55 | static ALARM: Forever<rtc::Alarm<pac::RTC1>> = Forever::new(); | ||
| 71 | static EXECUTOR: Forever<Executor> = Forever::new(); | 56 | static EXECUTOR: Forever<Executor> = Forever::new(); |
| 72 | 57 | ||
| 73 | #[entry] | 58 | #[entry] |
| 74 | fn main() -> ! { | 59 | fn main() -> ! { |
| 75 | info!("Hello World!"); | 60 | info!("Hello World!"); |
| 76 | 61 | ||
| 77 | let executor = EXECUTOR.put(Executor::new(cortex_m::asm::sev)); | 62 | let p = unwrap!(embassy_nrf::pac::Peripherals::take()); |
| 78 | unwrap!(executor.spawn(run())); | 63 | |
| 64 | clocks::Clocks::new(p.CLOCK) | ||
| 65 | .enable_ext_hfosc() | ||
| 66 | .set_lfclk_src_external(clocks::LfOscConfiguration::NoExternalNoBypass) | ||
| 67 | .start_lfclk(); | ||
| 68 | |||
| 69 | let rtc = RTC.put(rtc::RTC::new(p.RTC1, interrupt::take!(RTC1))); | ||
| 70 | rtc.start(); | ||
| 71 | |||
| 72 | unsafe { embassy::time::set_clock(rtc) }; | ||
| 73 | |||
| 74 | let alarm = ALARM.put(rtc.alarm0()); | ||
| 75 | let executor = EXECUTOR.put(Executor::new_with_alarm(alarm, cortex_m::asm::sev)); | ||
| 76 | |||
| 77 | // Init UART | ||
| 78 | let port0 = gpio::p0::Parts::new(p.P0); | ||
| 79 | |||
| 80 | let pins = uarte::Pins { | ||
| 81 | rxd: port0.p0_08.into_floating_input().degrade(), | ||
| 82 | txd: port0 | ||
| 83 | .p0_06 | ||
| 84 | .into_push_pull_output(gpio::Level::Low) | ||
| 85 | .degrade(), | ||
| 86 | cts: None, | ||
| 87 | rts: None, | ||
| 88 | }; | ||
| 89 | |||
| 90 | // NOTE(unsafe): Safe becasue we do not use `mem::forget` anywhere. | ||
| 91 | let uart = unsafe { | ||
| 92 | uarte::Uarte::new( | ||
| 93 | p.UARTE0, | ||
| 94 | interrupt::take!(UARTE0_UART0), | ||
| 95 | pins, | ||
| 96 | uarte::Parity::EXCLUDED, | ||
| 97 | uarte::Baudrate::BAUD115200, | ||
| 98 | ) | ||
| 99 | }; | ||
| 100 | |||
| 101 | unwrap!(executor.spawn(run(uart))); | ||
| 79 | 102 | ||
| 80 | loop { | 103 | loop { |
| 81 | executor.run(); | 104 | executor.run(); |
