opentitanlib/io/gpio.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 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
// Copyright lowRISC contributors (OpenTitan project).
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
use anyhow::Result;
use clap::ValueEnum;
use serde::{Deserialize, Serialize};
use std::time::Duration;
use thiserror::Error;
use crate::impl_serializable_error;
use crate::transport::TransportError;
/// Errors related to the GPIO interface.
#[derive(Debug, Error, Serialize, Deserialize)]
pub enum GpioError {
#[error("Invalid pin name {0}")]
InvalidPinName(String),
#[error("Invalid pin number {0}")]
InvalidPinNumber(u8),
/// The current mode of the pin (input) does not support the requested operation (set
/// level).
#[error("Invalid mode for pin {0}")]
InvalidPinMode(u8),
/// The hardware does not support the requested mode (open drain, pull down input, etc.)
#[error("Unsupported mode {0} requested")]
UnsupportedPinMode(PinMode),
/// The hardware does not support the requested mode (open drain, pull down input, etc.)
#[error("Unsupported pull mode {0} requested")]
UnsupportedPullMode(PullMode),
#[error("Conflicting pin configurations for pin {0}: host:{1}, target:{2}")]
PinModeConflict(String, String, String),
#[error("Conflicting pin logic values for pin {0}: host:{1}, target:{2}")]
PinValueConflict(String, String, String),
#[error("Undefined pin logic value for pin {0}")]
PinValueUndefined(String),
#[error("Unsupported voltage {0}V requested")]
UnsupportedPinVoltage(f32),
#[error("Unsupported number of pins: {0}")]
UnsupportedNumberOfPins(usize),
#[error("Mismatched bitbang data length: {0} != {1}")]
MismatchedDataLength(usize, usize),
#[error("Bitbang data beyond the {0} least significant bits")]
InvalidBitbangData(usize),
#[error("Bitbang delay of zero, immediately preceding `await()`, or at end of sequence, not permitted")]
InvalidBitbangDelay,
#[error("Dac-bang samples not a multiple of number of pins")]
InvalidDacBangData,
#[error("Dac-bang delay of zero or immediately adjacent to linear(), or at end of sequence, not permitted")]
InvalidDacBangDelay,
#[error("Generic error: {0}")]
Generic(String),
}
impl_serializable_error!(GpioError);
/// Mode of I/O pins.
#[derive(Clone, Copy, Debug, Serialize, Deserialize, PartialEq, Eq, ValueEnum)]
#[value(rename_all = "verbatim")]
pub enum PinMode {
Input,
PushPull,
OpenDrain,
AnalogInput,
AnalogOutput,
Alternate, // Pin used for UART/SPI/I2C or something else
}
impl std::fmt::Display for PinMode {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
std::fmt::Debug::fmt(self, f)
}
}
/// Mode of weak pull (relevant in Input and OpenDrain modes).
#[derive(Clone, Copy, Debug, Serialize, Deserialize, PartialEq, Eq, ValueEnum)]
#[value(rename_all = "verbatim")]
pub enum PullMode {
None,
PullUp,
PullDown,
}
impl std::fmt::Display for PullMode {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
std::fmt::Debug::fmt(self, f)
}
}
/// A trait which represents a single GPIO pin.
pub trait GpioPin {
/// Reads the value of the GPIO pin.
fn read(&self) -> Result<bool>;
/// Sets the value of the GPIO pin to `value`.
fn write(&self, value: bool) -> Result<()>;
/// Sets the mode of the GPIO pin as input, output, or open drain I/O.
fn set_mode(&self, mode: PinMode) -> Result<()>;
/// Sets the weak pull resistors of the GPIO pin.
fn set_pull_mode(&self, mode: PullMode) -> Result<()>;
/// Reads the analog value of the GPIO pin in Volts. `AnalogInput` mode disables digital
/// circuitry for better results, but this method may also work in other modes.
fn analog_read(&self) -> Result<f32> {
Err(TransportError::UnsupportedOperation.into())
}
/// Sets the analog value of the GPIO pin to `value` Volts, must be in `AnalogOutput` mode.
fn analog_write(&self, _volts: f32) -> Result<()> {
Err(TransportError::UnsupportedOperation.into())
}
/// Simultaneously sets mode, value, and weak pull, some transports may guarantee atomicity.
fn set(
&self,
mode: Option<PinMode>,
value: Option<bool>,
pull: Option<PullMode>,
analog_value: Option<f32>,
) -> Result<()> {
// Transports must override this function for truly atomic behavior. Default
// implementation below applies each setting separately.
if let Some(mode) = mode {
self.set_mode(mode)?;
}
if let Some(pull) = pull {
self.set_pull_mode(pull)?;
}
if let Some(value) = value {
self.write(value)?;
}
if let Some(analog_value) = analog_value {
self.analog_write(analog_value)?;
}
Ok(())
}
/// Not meant for API clients, this method returns the pin name as it is known to the
/// transport (which may have been through one or more alias mappings from the name provided
/// by the API client.) This method is used by implementations of `GpioMonitoring`.
fn get_internal_pin_name(&self) -> Option<&str> {
None
}
}
#[derive(Clone, Copy, Debug, Serialize, Deserialize, PartialEq, Eq)]
pub enum Edge {
Rising,
Falling,
}
#[derive(Clone, Copy, Debug, Serialize, Deserialize, PartialEq, Eq)]
pub enum ClockNature {
/// Unix time can be computed as (t + offset) / resolution, where t is a 64-bit timestamp
/// value from `MonitoringEvent`.
Wallclock {
/// If resolution is microseconds, `resolution` will be 1_000_000.
resolution: u64,
/// Offset relative to Unix epoch, measured according to above resolution.
offset: Option<u64>,
},
/// The 64-bit timestamp values could be emulator clock counts, or some other measure that
/// increases monotonically, but not necessarily uniformly in relation to wall clock time.
Unspecified,
}
/// Represents an edge detected on the GPIO pin.
#[derive(Clone, Copy, Debug, Serialize, Deserialize)]
pub struct MonitoringEvent {
/// Identification of the signal that had an event, in the form of an index into the array
/// originally passed to `monitoring_read()`.
pub signal_index: u8,
/// Rising or falling edge
pub edge: Edge,
/// Timestamp of the edge, resolution and epoch is transport-specific, more information in
/// `ClockNature`.
pub timestamp: u64,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct MonitoringStartResponse {
/// Transport timestamp at the time monitoring started.
pub timestamp: u64,
/// Initial logic level for each of the given pins.
pub initial_levels: Vec<bool>,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct MonitoringReadResponse {
/// List of events having occurred since the start or the last read.
pub events: Vec<MonitoringEvent>,
/// All events at or before this timestamp are guaranteed to be included.
pub timestamp: u64,
}
/// A trait implemented by transports which support advanced edge-detection on GPIO pins. This
/// trait allows monitoring a set of pins, and getting a stream of "events" (rising and falling
/// edges with timestamps) for any change among the set.
pub trait GpioMonitoring {
fn get_clock_nature(&self) -> Result<ClockNature>;
/// Set up edge trigger detection on the given set of pins, transport will buffer the list
/// internally, return the initial level of each of the given pins.
fn monitoring_start(&self, pins: &[&dyn GpioPin]) -> Result<MonitoringStartResponse>;
/// Retrieve list of events detected thus far, optionally stopping the possibly expensive edge
/// detection. Buffer overrun will be reported as an `Err`, and result in the stopping of the
/// edge detection irrespective of the parameter value.
fn monitoring_read(
&self,
pins: &[&dyn GpioPin],
continue_monitoring: bool,
) -> Result<MonitoringReadResponse>;
}
/// Represents one entry in the specification of a bitbanging waveform. Pins must have been
/// configured as either `PushPull`, `OpenDrain` or `Input` prior to requesting a bitbang
/// operation.
///
/// `Write` and `Both` are somewhat equivalent to their namesakes in `spi::Transfer`, that is
/// `Write` requests using the given data for generating waveforms on any of the pins configured
/// as output, while disregarding the actual levels of the pins. `Both` requests generating a
/// waveform, and also sampling every pin at each clock tick.
pub enum BitbangEntry<'rd, 'wr> {
/// Represents a sequence of pin values. Bit 0 in each byte controls the first `GpioPin` in
/// the slice given to `GpioBitbanging::run()`, bit 1 controls the second `GpioPin`, and so
/// on. Each byte is applied to the pins in sequence, with a particular delay between each
/// given by the `clock_tick` argument to `run()`.
Write(&'wr [u8]),
/// Same as `Write`, but this `BitbangEntry` owns the data.
WriteOwned(Box<[u8]>),
/// Represents a sequence of pin values as above, but additionally captures the value of any
/// pins in `Input` or `OpenDrain` mode. At each clock tick, the input levels are sampled
/// just before the given output levels are applied, meaning that the data coming back will be
/// offset by one sample.
Both(&'wr [u8], &'rd mut [u8]),
/// Same as `Both`, but this `BitbangEntry` owns the data, which will be overwritten.
BothOwned(Box<[u8]>),
/// Represents a delay of the given number of clock ticks in which the output levels are held
/// as indicated by the last byte of the preceding `Write`/`Both` entry.
///
/// A delay of zero is invalid. A delay of one tick is equivalent to not specifying any
/// `Delay` between two `Write` blocks, which is also equivalent to concatenating the two into
/// a single `Write` block.
Delay(u32),
/// Similar to `Delay`, but waits until `(pin_values ^ pattern) & mask` equals zero, that is,
/// until the set of pins indicated by ones in `mask` all have the the value indicated in
/// `pattern`.
Await { mask: u8, pattern: u8 },
}
pub enum DacBangEntry<'wr> {
/// Represents a sequence of volt values. If `N` pin is being driven, then samples are
/// interleaved such that at first, the first `N` values are applied, one to each of the pins,
/// at next tick the next `N` values are applied, and so on, with a particular delay between
/// each tick given by the `clock_tick` argument to `run()`.
Write(&'wr [f32]),
/// Same as `Write`, but this `DacBangEntry` owns the data.
WriteOwned(Box<[f32]>),
/// Represents a delay of the given number of clock ticks in which the output levels are held
/// as indicated by the last values of the preceding `Write` entry.
///
/// A delay of zero is invalid. A delay of one tick is equivalent to not specifying any
/// `Delay` between two `Write` blocks, which is also equivalent to concatenating the two into
/// a single `Write` block.
Delay(u32),
/// Represents a time span of the given number of clock ticks during which the voltage
/// linearly transitions from the previous to the subsequent value.
///
/// A value of zero is invalid. A value of one tick is equivalent to not specifying any delay
/// between two `Write` blocks. A value of two will result in a single intermediate sample
/// halfway between the two voltages. In general, a value of N will result in N-1
/// intermediate samples being inserted at this point.
Linear(u32),
}
/// A trait implemented by transports which support synchronous bit-banging on GPIO pins, similar
/// to FTDI devices. This trait allows generation of arbitrary waveforms on a set of pins, and
/// optionally getting back samples from same or other pins, taken at precise times.
pub trait GpioBitbanging {
/// Apply the given sequence of values to the given set of GPIO pins, by each tick of a clock
/// with the given period. This function does not change the mode of any pins, they must
/// already be put into `PushPull`, `OpenDrain` or `Input` mode as appropriate. (In the
/// latter case, the specified waveform data does not matter, as the pin is not driving, and
/// would be included in the set of pins only in order to have it sampled at each clock tick.)
/// Returns a `GpioBitbangOperation` which must be continuously polled, to know when the
/// waveform is complete.
fn start<'a>(
&self,
pins: &[&dyn GpioPin],
clock_tick: Duration,
waveform: Box<[BitbangEntry<'a, 'a>]>,
) -> Result<Box<dyn GpioBitbangOperation<'a, 'a> + 'a>>;
/// Convenience method which starts the bitbanging operation, and blocks until it is complete.
fn run<'a>(
&self,
pins: &[&dyn GpioPin],
clock_tick: Duration,
waveform: Box<[BitbangEntry<'a, 'a>]>,
) -> Result<Box<[BitbangEntry<'a, 'a>]>> {
let mut bitbang_op = self.start(pins, clock_tick, waveform)?;
while !bitbang_op.query()? {}
bitbang_op.get_result()
}
/// Apply given sequence of voltage values to the given set of pins assumed to already be in
/// AnalogOutput mode.
fn dac_start(
&self,
pins: &[&dyn GpioPin],
clock_tick: Duration,
waveform: Box<[DacBangEntry]>,
) -> Result<Box<dyn GpioDacBangOperation>>;
/// Convenience method which starts the DAC-banging operation, and blocks until it is
/// complete.
fn dac_run(
&self,
pins: &[&dyn GpioPin],
clock_tick: Duration,
waveform: Box<[DacBangEntry]>,
) -> Result<()> {
let mut dacbang_op = self.dac_start(pins, clock_tick, waveform)?;
while !dacbang_op.query()? {}
Ok(())
}
}
/// Object representing an ongoing operation of generating a prescribed waveform on a number of
/// pins. Must be frequently polled via the `query()` function, in order to guerantee completion
/// of the operation.
pub trait GpioBitbangOperation<'rd, 'wr> {
/// Returns `true` when done, `false` means to call it again.
fn query(&mut self) -> Result<bool>;
/// After `query()` has returned `true`, this method can be called to get back the array of
/// `BitbangEntry` originally given, this is particularly useful if there are `WriteOwned` or
/// `BothOwned` among the entries of the array.
fn get_result(self: Box<Self>) -> Result<Box<[BitbangEntry<'rd, 'wr>]>>;
}
pub trait GpioDacBangOperation {
/// Returns `true` when done, `false` means to call it again.
fn query(&mut self) -> Result<bool>;
}