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
use endbasic_std::gpio::{Pin, PinMode, Pins};
use rppal::gpio;
use std::collections::HashMap;
use std::io;
#[derive(Default)]
pub struct RppalPins {
chip: Option<gpio::Gpio>,
inputs: HashMap<Pin, gpio::InputPin>,
outputs: HashMap<Pin, gpio::OutputPin>,
}
fn gpio_error_to_io_error(e: gpio::Error) -> io::Error {
match e {
gpio::Error::Io(e) => e,
gpio::Error::PermissionDenied(e) => io::Error::new(io::ErrorKind::PermissionDenied, e),
gpio::Error::PinNotAvailable(pin) => {
io::Error::new(io::ErrorKind::InvalidInput, format!("Unknown pin number {}", pin))
}
e => io::Error::new(io::ErrorKind::Other, e.to_string()),
}
}
impl RppalPins {
fn get_chip(&mut self) -> io::Result<&mut gpio::Gpio> {
if self.chip.is_none() {
self.chip = Some(gpio::Gpio::new().map_err(gpio_error_to_io_error)?);
}
Ok(self.chip.as_mut().unwrap())
}
}
impl Pins for RppalPins {
fn setup(&mut self, pin: Pin, mode: PinMode) -> io::Result<()> {
self.clear(pin)?;
let chip = self.get_chip()?;
let gpio_pin = chip.get(pin.0).map_err(gpio_error_to_io_error)?;
match mode {
PinMode::In => {
self.inputs.insert(pin, gpio_pin.into_input());
}
PinMode::InPullDown => {
self.inputs.insert(pin, gpio_pin.into_input_pulldown());
}
PinMode::InPullUp => {
self.inputs.insert(pin, gpio_pin.into_input_pullup());
}
PinMode::Out => {
self.outputs.insert(pin, gpio_pin.into_output());
}
};
Ok(())
}
fn clear(&mut self, pin: Pin) -> io::Result<()> {
self.inputs.remove(&pin);
self.outputs.remove(&pin);
Ok(())
}
fn clear_all(&mut self) -> io::Result<()> {
self.inputs.clear();
self.outputs.clear();
Ok(())
}
fn read(&mut self, pin: Pin) -> io::Result<bool> {
if !self.inputs.contains_key(&pin) || self.outputs.contains_key(&pin) {
return Err(io::Error::new(
io::ErrorKind::AlreadyExists,
"Pin not configured for read; use GPIO_SETUP first",
));
}
let pin = self.inputs.get(&pin).unwrap();
match pin.read() {
gpio::Level::High => Ok(true),
gpio::Level::Low => Ok(false),
}
}
fn write(&mut self, pin: Pin, v: bool) -> io::Result<()> {
if self.inputs.contains_key(&pin) || !self.outputs.contains_key(&pin) {
return Err(io::Error::new(
io::ErrorKind::AlreadyExists,
"Pin not configured for write; use GPIO_SETUP first",
));
}
let pin = self.outputs.get_mut(&pin).unwrap();
if v {
pin.write(gpio::Level::High);
} else {
pin.write(gpio::Level::Low);
}
Ok(())
}
}