2021-10-18 16:23:39 +02:00
|
|
|
#![macro_use]
|
|
|
|
|
2023-02-01 00:48:33 +01:00
|
|
|
//! Programmable Peripheral Interconnect (PPI/DPPI) driver.
|
2021-10-18 16:23:39 +02:00
|
|
|
//!
|
|
|
|
//! The (Distributed) Programmable Peripheral Interconnect interface allows for an autonomous interoperability
|
|
|
|
//! between peripherals through their events and tasks. There are fixed PPI channels and fully
|
|
|
|
//! configurable ones. Fixed channels can only connect specific events to specific tasks. For fully
|
|
|
|
//! configurable channels, it is possible to choose, via software, the event and the task that it
|
|
|
|
//! will triggered by the event.
|
|
|
|
//!
|
|
|
|
//! On nRF52 devices, there is also a fork task endpoint, where the user can configure one more task
|
|
|
|
//! to be triggered by the same event, even fixed PPI channels have a configurable fork task.
|
2021-10-18 16:26:27 +02:00
|
|
|
//!
|
2021-10-18 16:23:39 +02:00
|
|
|
//! The DPPI for nRF53 and nRF91 devices works in a different way. Every channel can support infinitely
|
|
|
|
//! many tasks and events, but any single task or event can only be coupled with one channel.
|
2021-10-18 16:26:27 +02:00
|
|
|
//!
|
2021-10-18 16:23:39 +02:00
|
|
|
|
|
|
|
use core::ptr::NonNull;
|
2022-06-12 22:15:44 +02:00
|
|
|
|
2022-07-23 14:00:19 +02:00
|
|
|
use embassy_hal_common::{impl_peripheral, PeripheralRef};
|
2021-10-18 16:23:39 +02:00
|
|
|
|
2022-07-23 14:00:19 +02:00
|
|
|
use crate::{peripherals, Peripheral};
|
2022-06-12 22:15:44 +02:00
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
#[cfg(feature = "_dppi")]
|
|
|
|
mod dppi;
|
|
|
|
#[cfg(feature = "_ppi")]
|
|
|
|
mod ppi;
|
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// An instance of the Programmable peripheral interconnect on nRF devices.
|
2021-10-18 16:23:39 +02:00
|
|
|
pub struct Ppi<'d, C: Channel, const EVENT_COUNT: usize, const TASK_COUNT: usize> {
|
2022-07-23 14:00:19 +02:00
|
|
|
ch: PeripheralRef<'d, C>,
|
2021-10-19 10:13:08 +02:00
|
|
|
#[cfg(feature = "_dppi")]
|
2021-10-18 16:23:39 +02:00
|
|
|
events: [Event; EVENT_COUNT],
|
2021-10-19 10:13:08 +02:00
|
|
|
#[cfg(feature = "_dppi")]
|
2021-10-18 16:23:39 +02:00
|
|
|
tasks: [Task; TASK_COUNT],
|
|
|
|
}
|
|
|
|
|
2022-08-24 01:52:15 +02:00
|
|
|
#[cfg(feature = "_dppi")]
|
2021-10-19 10:13:08 +02:00
|
|
|
const REGISTER_DPPI_CONFIG_OFFSET: usize = 0x80 / core::mem::size_of::<u32>();
|
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
/// Represents a task that a peripheral can do.
|
|
|
|
///
|
2022-08-24 01:54:27 +02:00
|
|
|
/// When a task is subscribed to a PPI channel, it will run when the channel is triggered by
|
|
|
|
/// a published event.
|
2021-10-18 16:23:39 +02:00
|
|
|
#[derive(PartialEq, Eq, Clone, Copy)]
|
2022-08-24 01:54:27 +02:00
|
|
|
pub struct Task(NonNull<u32>);
|
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
impl Task {
|
2022-08-24 01:54:27 +02:00
|
|
|
/// Create a new `Task` from a task register pointer
|
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// `ptr` must be a pointer to a valid `TASKS_*` register from an nRF peripheral.
|
|
|
|
pub unsafe fn new_unchecked(ptr: NonNull<u32>) -> Self {
|
|
|
|
Self(ptr)
|
|
|
|
}
|
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
pub(crate) fn from_reg<T>(reg: &T) -> Self {
|
|
|
|
Self(unsafe { NonNull::new_unchecked(reg as *const _ as *mut _) })
|
|
|
|
}
|
2021-10-19 10:13:08 +02:00
|
|
|
|
2022-08-24 01:52:15 +02:00
|
|
|
/// Address of subscription register for this task.
|
|
|
|
#[cfg(feature = "_dppi")]
|
2021-10-19 10:13:08 +02:00
|
|
|
pub fn subscribe_reg(&self) -> *mut u32 {
|
|
|
|
unsafe { self.0.as_ptr().add(REGISTER_DPPI_CONFIG_OFFSET) }
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// NonNull is not send, but this event is only allowed to point at registers and those exist in any context on the same core.
|
|
|
|
unsafe impl Send for Task {}
|
|
|
|
|
|
|
|
/// Represents an event that a peripheral can publish.
|
|
|
|
///
|
2022-08-24 01:54:27 +02:00
|
|
|
/// An event can be set to publish on a PPI channel when the event happens.
|
2021-10-18 16:23:39 +02:00
|
|
|
#[derive(PartialEq, Eq, Clone, Copy)]
|
2022-08-24 01:54:27 +02:00
|
|
|
pub struct Event(NonNull<u32>);
|
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
impl Event {
|
2022-08-24 01:54:27 +02:00
|
|
|
/// Create a new `Event` from an event register pointer
|
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// `ptr` must be a pointer to a valid `EVENTS_*` register from an nRF peripheral.
|
|
|
|
pub unsafe fn new_unchecked(ptr: NonNull<u32>) -> Self {
|
|
|
|
Self(ptr)
|
|
|
|
}
|
|
|
|
|
2021-10-18 16:23:39 +02:00
|
|
|
pub(crate) fn from_reg<T>(reg: &T) -> Self {
|
|
|
|
Self(unsafe { NonNull::new_unchecked(reg as *const _ as *mut _) })
|
|
|
|
}
|
2021-10-19 10:13:08 +02:00
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Address of publish register for this event.
|
2022-08-24 01:52:15 +02:00
|
|
|
#[cfg(feature = "_dppi")]
|
2021-10-19 10:13:08 +02:00
|
|
|
pub fn publish_reg(&self) -> *mut u32 {
|
|
|
|
unsafe { self.0.as_ptr().add(REGISTER_DPPI_CONFIG_OFFSET) }
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// NonNull is not send, but this event is only allowed to point at registers and those exist in any context on the same core.
|
|
|
|
unsafe impl Send for Event {}
|
|
|
|
|
|
|
|
// ======================
|
|
|
|
// traits
|
|
|
|
|
|
|
|
pub(crate) mod sealed {
|
|
|
|
pub trait Channel {}
|
|
|
|
pub trait Group {}
|
|
|
|
}
|
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Interface for PPI channels.
|
2022-07-23 14:00:19 +02:00
|
|
|
pub trait Channel: sealed::Channel + Peripheral<P = Self> + Sized {
|
2021-10-18 16:23:39 +02:00
|
|
|
/// Returns the number of the channel
|
|
|
|
fn number(&self) -> usize;
|
2021-10-26 09:45:29 +02:00
|
|
|
}
|
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Interface for PPI channels that can be configured.
|
2022-07-23 14:27:45 +02:00
|
|
|
pub trait ConfigurableChannel: Channel + Into<AnyConfigurableChannel> {
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Convert into a type erased configurable channel.
|
2021-10-26 09:45:29 +02:00
|
|
|
fn degrade(self) -> AnyConfigurableChannel;
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Interface for PPI channels that cannot be configured.
|
2022-07-23 14:27:45 +02:00
|
|
|
pub trait StaticChannel: Channel + Into<AnyStaticChannel> {
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Convert into a type erased static channel.
|
2021-10-26 09:45:29 +02:00
|
|
|
fn degrade(self) -> AnyStaticChannel;
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Interface for a group of PPI channels.
|
2021-10-18 16:23:39 +02:00
|
|
|
pub trait Group: sealed::Group + Sized {
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Returns the number of the group.
|
2021-10-18 16:23:39 +02:00
|
|
|
fn number(&self) -> usize;
|
2022-08-22 10:36:33 +02:00
|
|
|
/// Convert into a type erased group.
|
2021-10-26 09:45:29 +02:00
|
|
|
fn degrade(self) -> AnyGroup {
|
|
|
|
AnyGroup {
|
|
|
|
number: self.number() as u8,
|
|
|
|
}
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ======================
|
|
|
|
// channels
|
|
|
|
|
2021-10-26 09:45:29 +02:00
|
|
|
/// The any channel can represent any static channel at runtime.
|
|
|
|
/// This can be used to have fewer generic parameters in some places.
|
|
|
|
pub struct AnyStaticChannel {
|
|
|
|
pub(crate) number: u8,
|
|
|
|
}
|
2022-07-23 14:00:19 +02:00
|
|
|
impl_peripheral!(AnyStaticChannel);
|
2021-10-26 09:45:29 +02:00
|
|
|
impl sealed::Channel for AnyStaticChannel {}
|
|
|
|
impl Channel for AnyStaticChannel {
|
|
|
|
fn number(&self) -> usize {
|
|
|
|
self.number as usize
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl StaticChannel for AnyStaticChannel {
|
|
|
|
fn degrade(self) -> AnyStaticChannel {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The any configurable channel can represent any configurable channel at runtime.
|
|
|
|
/// This can be used to have fewer generic parameters in some places.
|
|
|
|
pub struct AnyConfigurableChannel {
|
|
|
|
pub(crate) number: u8,
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
2022-07-23 14:00:19 +02:00
|
|
|
impl_peripheral!(AnyConfigurableChannel);
|
2021-10-26 09:45:29 +02:00
|
|
|
impl sealed::Channel for AnyConfigurableChannel {}
|
|
|
|
impl Channel for AnyConfigurableChannel {
|
2021-10-18 16:23:39 +02:00
|
|
|
fn number(&self) -> usize {
|
|
|
|
self.number as usize
|
|
|
|
}
|
2021-10-26 09:45:29 +02:00
|
|
|
}
|
|
|
|
impl ConfigurableChannel for AnyConfigurableChannel {
|
|
|
|
fn degrade(self) -> AnyConfigurableChannel {
|
|
|
|
self
|
2021-10-18 16:23:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! impl_ppi_channel {
|
2021-10-26 16:10:34 +02:00
|
|
|
($type:ident, $number:expr) => {
|
2021-10-19 10:13:08 +02:00
|
|
|
impl crate::ppi::sealed::Channel for peripherals::$type {}
|
|
|
|
impl crate::ppi::Channel for peripherals::$type {
|
2021-10-18 16:23:39 +02:00
|
|
|
fn number(&self) -> usize {
|
|
|
|
$number
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2021-10-26 09:45:29 +02:00
|
|
|
($type:ident, $number:expr => static) => {
|
2021-10-26 16:10:34 +02:00
|
|
|
impl_ppi_channel!($type, $number);
|
2021-10-26 09:45:29 +02:00
|
|
|
impl crate::ppi::StaticChannel for peripherals::$type {
|
|
|
|
fn degrade(self) -> crate::ppi::AnyStaticChannel {
|
|
|
|
use crate::ppi::Channel;
|
|
|
|
crate::ppi::AnyStaticChannel {
|
|
|
|
number: self.number() as u8,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-07-23 14:27:45 +02:00
|
|
|
|
|
|
|
impl From<peripherals::$type> for crate::ppi::AnyStaticChannel {
|
|
|
|
fn from(val: peripherals::$type) -> Self {
|
|
|
|
crate::ppi::StaticChannel::degrade(val)
|
|
|
|
}
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
};
|
2021-10-26 09:45:29 +02:00
|
|
|
($type:ident, $number:expr => configurable) => {
|
2021-10-26 16:10:34 +02:00
|
|
|
impl_ppi_channel!($type, $number);
|
2021-10-26 09:45:29 +02:00
|
|
|
impl crate::ppi::ConfigurableChannel for peripherals::$type {
|
|
|
|
fn degrade(self) -> crate::ppi::AnyConfigurableChannel {
|
|
|
|
use crate::ppi::Channel;
|
|
|
|
crate::ppi::AnyConfigurableChannel {
|
|
|
|
number: self.number() as u8,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-07-23 14:27:45 +02:00
|
|
|
|
|
|
|
impl From<peripherals::$type> for crate::ppi::AnyConfigurableChannel {
|
|
|
|
fn from(val: peripherals::$type) -> Self {
|
|
|
|
crate::ppi::ConfigurableChannel::degrade(val)
|
|
|
|
}
|
|
|
|
}
|
2021-10-18 16:23:39 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// ======================
|
|
|
|
// groups
|
|
|
|
|
2022-08-22 10:36:33 +02:00
|
|
|
/// A type erased PPI group.
|
2021-10-18 16:23:39 +02:00
|
|
|
pub struct AnyGroup {
|
|
|
|
number: u8,
|
|
|
|
}
|
2022-07-23 14:00:19 +02:00
|
|
|
impl_peripheral!(AnyGroup);
|
2021-10-18 16:23:39 +02:00
|
|
|
impl sealed::Group for AnyGroup {}
|
|
|
|
impl Group for AnyGroup {
|
|
|
|
fn number(&self) -> usize {
|
|
|
|
self.number as usize
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! impl_group {
|
|
|
|
($type:ident, $number:expr) => {
|
|
|
|
impl sealed::Group for peripherals::$type {}
|
|
|
|
impl Group for peripherals::$type {
|
|
|
|
fn number(&self) -> usize {
|
|
|
|
$number
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl_group!(PPI_GROUP0, 0);
|
|
|
|
impl_group!(PPI_GROUP1, 1);
|
|
|
|
impl_group!(PPI_GROUP2, 2);
|
|
|
|
impl_group!(PPI_GROUP3, 3);
|
|
|
|
#[cfg(not(feature = "nrf51"))]
|
|
|
|
impl_group!(PPI_GROUP4, 4);
|
|
|
|
#[cfg(not(feature = "nrf51"))]
|
|
|
|
impl_group!(PPI_GROUP5, 5);
|