embassy/tests/rp/src/bin/gpio.rs

217 lines
4.9 KiB
Rust
Raw Normal View History

2022-07-09 02:13:43 +02:00
#![no_std]
#![no_main]
#![feature(type_alias_impl_trait)]
2023-05-30 00:10:36 +02:00
#[path = "../common.rs"]
mod common;
2022-07-09 02:13:43 +02:00
use defmt::{assert, *};
use embassy_executor::Spawner;
2022-07-09 02:13:43 +02:00
use embassy_rp::gpio::{Flex, Input, Level, Output, OutputOpenDrain, Pull};
use {defmt_rtt as _, panic_probe as _};
#[embassy_executor::main]
async fn main(_spawner: Spawner) {
let p = embassy_rp::init(Default::default());
2022-07-09 02:13:43 +02:00
info!("Hello World!");
let (mut a, mut b) = (p.PIN_0, p.PIN_1);
// Test initial output
{
let b = Input::new(&mut b, Pull::None);
{
let _a = Output::new(&mut a, Level::Low);
delay();
assert!(b.is_low());
}
{
let _a = Output::new(&mut a, Level::High);
delay();
assert!(b.is_high());
}
}
// Test input no pull
{
let b = Input::new(&mut b, Pull::None);
// no pull, the status is undefined
let mut a = Output::new(&mut a, Level::Low);
delay();
assert!(b.is_low());
a.set_high();
delay();
assert!(b.is_high());
}
// Test input pulldown
{
let b = Input::new(&mut b, Pull::Down);
delay();
assert!(b.is_low());
let mut a = Output::new(&mut a, Level::Low);
delay();
assert!(b.is_low());
a.set_high();
delay();
assert!(b.is_high());
}
// Test input pullup
{
let b = Input::new(&mut b, Pull::Up);
delay();
assert!(b.is_high());
let mut a = Output::new(&mut a, Level::Low);
delay();
assert!(b.is_low());
a.set_high();
delay();
assert!(b.is_high());
}
// OUTPUT OPEN DRAIN
{
let mut b = OutputOpenDrain::new(&mut b, Level::High);
let mut a = Flex::new(&mut a);
a.set_as_input();
// When an OutputOpenDrain is high, it doesn't drive the pin.
2022-12-06 21:09:27 +01:00
b.set_high();
2022-07-09 02:13:43 +02:00
a.set_pull(Pull::Up);
delay();
assert!(a.is_high());
a.set_pull(Pull::Down);
delay();
assert!(a.is_low());
// When an OutputOpenDrain is low, it drives the pin low.
2022-12-06 21:09:27 +01:00
b.set_low();
2022-07-09 02:13:43 +02:00
a.set_pull(Pull::Up);
delay();
assert!(a.is_low());
a.set_pull(Pull::Down);
delay();
assert!(a.is_low());
2022-12-06 21:09:27 +01:00
// Check high again
2022-07-09 02:13:43 +02:00
b.set_high();
a.set_pull(Pull::Up);
delay();
assert!(a.is_high());
a.set_pull(Pull::Down);
delay();
assert!(a.is_low());
2022-12-06 21:09:27 +01:00
// When an OutputOpenDrain is high, it reads the input value in the pin.
b.set_high();
a.set_as_input();
a.set_pull(Pull::Up);
delay();
assert!(b.is_high());
a.set_as_output();
a.set_low();
delay();
assert!(b.is_low());
// When an OutputOpenDrain is low, it always reads low.
b.set_low();
a.set_as_input();
a.set_pull(Pull::Up);
delay();
assert!(b.is_low());
a.set_as_output();
a.set_low();
delay();
assert!(b.is_low());
2022-07-09 02:13:43 +02:00
}
// FLEX
// Test initial output
{
//Flex pin configured as input
let mut b = Flex::new(&mut b);
b.set_as_input();
{
//Flex pin configured as output
let mut a = Flex::new(&mut a); //Flex pin configured as output
a.set_low(); // Pin state must be set before configuring the pin, thus we avoid unknown state
a.set_as_output();
delay();
assert!(b.is_low());
}
{
//Flex pin configured as output
let mut a = Flex::new(&mut a);
a.set_high();
a.set_as_output();
delay();
assert!(b.is_high());
}
}
// Test input no pull
{
let mut b = Flex::new(&mut b);
b.set_as_input(); // no pull by default.
let mut a = Flex::new(&mut a);
a.set_low();
a.set_as_output();
delay();
assert!(b.is_low());
a.set_high();
delay();
assert!(b.is_high());
}
// Test input pulldown
{
let mut b = Flex::new(&mut b);
b.set_as_input();
b.set_pull(Pull::Down);
delay();
assert!(b.is_low());
let mut a = Flex::new(&mut a);
a.set_low();
a.set_as_output();
delay();
assert!(b.is_low());
a.set_high();
delay();
assert!(b.is_high());
}
// Test input pullup
{
let mut b = Flex::new(&mut b);
b.set_as_input();
b.set_pull(Pull::Up);
delay();
assert!(b.is_high());
let mut a = Flex::new(&mut a);
a.set_high();
a.set_as_output();
delay();
assert!(b.is_high());
a.set_low();
delay();
assert!(b.is_low());
}
info!("Test OK");
cortex_m::asm::bkpt();
}
fn delay() {
cortex_m::asm::delay(10000);
}