A Rust library for accessing GPIO lines on Linux platforms using the GPIO character device.
This is the equivalent of libgpiod but in pure Rust.
The gpiocdev crate provides an API to access GPIOs from Rust applications.
The library is compatible with the Linux GPIO uAPI, both v1 and v2, including receiving line edge events.
The gpiocdev API provides a unified abstraction for both uAPI versions, but will return an error if v2 features are attempted to be used on a v1-only system.
uAPI v2 specific features include:
Compatibility with either uAPI version can be selected via features, with the default being uAPI v2. If built with both, the library can automatically detect and use the most current available version, so defaulting to v2 and falling back to v1 if that is unavailable.
The library makes no use of the deprecated sysfs GPIO API.
The majority of the GPIO uAPI is synchronous. The exceptions are waiting for edge events from Requests, and info change events from Chips. Support for asynchronous wrappers around these are provided through features for the following reactors:
|Reactor|Feature|Module| |---|---|---| |tokio|asynctokio|gpiocdev::tokio| |async-io|asyncio|gpiocdev::async_io|
Additionally, Chips and Requests also expose their underlying file descriptor, which may be used directly with an async reactor. An example of this is the gpiocdev-cli edges command, which can asynchronously wait on multiple lines spread across multiple chips using the mio reactor.
With respect to the synchronous uAPI functions, those can generally be considered non-blocking unless the GPIO line is provided by an expander connected to the host processor via a bus such as I2C or SPI. In such cases, and depending on the application requirements and the async reactor, calls to synchronous functions may need to be made from a separate thread so as not to stall a single-threaded reactor.
Requesting a line by name:
```rust let led0 = gpiocdev::findnamedline("LED0").unwrap(); let req = Request::builder() .withfoundline(&led0) .as_output(Value::Active) .request()?;
// change value later
req.set_value(led0.offset, Value::Inactive)
```
Getting a line value:
rust
// request the line
let req = Request::builder()
.on_chip("/dev/gpiochip0")
.with_line(23)
.as_input()
.request()?;
// get the value
let value = req.value(23)?;
Setting a line:
```rust // request the line and set its value let req = Request::builder() .onchip("/dev/gpiochip0") .withline(22) .as_output(Value::Active) .request()?;
// do something...
// change value later
req.set_value(22, Value::Inactive)
```
Waiting for events on a line:
```rust // request the line let req = Request::builder() .onchip("/dev/gpiochip0") .withline(23) .withedgedetection(EdgeDetection::BothEdges) .request()?;
// wait for line edge events
for event in req.edge_events() {
println!("{:?}", event?);
}
```
Multiple lines may be selected in a single request, and then be operated on as a unit.
Getting multiple lines:
rust
// request multiple input lines
let req = Request::builder()
.on_chip("/dev/gpiochip0")
.with_lines(&[18,23])
.as_input()
.request()?;
// get multiple line values at once
let mut values = Values::default();
req.values(&mut values)?;
Setting multiple lines:
rust
// request multiple output lines
let req = Request::builder()
.on_chip("/dev/gpiochip0")
.with_lines(&[17,22])
.as_output(Value::Active)
.request()?;
// set multiple line values at once
let mut values = Values::default();
values.set(17, Value::Inactive);
values.set(12, Value::Active);
req.set_values(&values)?;
All line attributes available via the kernel GPIO interface, such as pull-ups and debounce etc, can also be set:
rust
// request the line
let req = Request::builder()
.on_chip("/dev/gpiochip0")
.with_consumer("myapp")
.with_line(23)
.as_input()
.as_active_low()
.with_bias(Bias::PullUp)
.request()?;
// get the value
let value = req.value(23)?;
A good starting point to learn more is the Builder.
Licensed under either of
at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.