RFC: separate pad configurations from the HAL #73
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
RFC: separate pad configurations from the HAL
The PR introduces the
imxrt-iomuxc
crate family. The goal is to de-couple pad definitions from the HAL, supporting a split HAL (#56). Additional, the crates introduce a new pin muxing interface, reaching the ideal described in #26. Finally, the pad configurations and interfaces are re-usable without requiring any HAL, which supports out-of-tree driver development, and also encourages alternative HAL implementations, such as fully-asyc
HALs.I'm looking for feedback on the approach. If you'd like to try this out today, you can use the
iomuxc-integration
branch of theteensy4-rs
project. I've updated and tested all (non-RTIC) examples with the new interface.Proposal
iomuxc
module.imxrt-iomuxc
, which providesimxrt106x-iomuxc
. The crate provides the pad implementations previously available in the HAL'siomuxc
module.imxrt-iomuxc-build
, which is used by pin crate implementers. The crate lets us auto-generate Rust pad definitions at build time.imxrt-iomuxc
interfaces into the HAL.imxrt106x-iomuxc
from the current HAL.imxrt106x-iomuxc
.imxrt-iomuxc
crate, and pin implementation crates, for HAL and driver designers.Walkthrough
We introduce
imxrt-iomuxc
, the top-level crate for the IOMUXC crate family. In the spirit of the IOMUXC hardware peripheral, which provides pad configuration and multiplexing, we name these crates to includeiomuxc
.imxrt-iomuxc
defines pin traits. An examplePin
trait for a UART-compatible pin resembleswhere
ALT
is the alternate value.DAISY
is the daisy register.Direction
is a type tag for UART signals; currently eitherTX
orRX
.Module
is atypenum
constant.The alternate value is an associated constant. In today's
iomuxc
module, it's a type state. The type state leaks into user code, code that doesn't care about this implementation detail. The detail also leaks into our pad definition macros, which needs to know what alternate values a pad may take (see below). The associated constant lets us mark it once, then never worry about it again.We favor
typenum
type-level constants. These replace the hand-rolled type constants that the HAL was using and exposing.We have similar traits in today's
iomuxc
module. The only difference from today is in dependency management. We can use these traits independent of the HAL, which is not possible today.A driver implementer uses these traits in their interfaces:
Notice the calls to
prepare()
.imxrt-iomuxc
provides all functions for configuring pads. These functions are designed to a common trait, such that they're usable with strongly-typed pads or type-erased pads. Theimxrt-iomuxc
's interface is typicallyunsafe
, which puts the onus on driver and HAL designers to guarantee safety. Theunsafe
interface also signals that these functions might not be suitable for normal user code. See the crate documentation for justification on theunsafe
interfaces.Also note the
ErasedPad
types. The new crates provide type-erased pads, which users may prefer over strongly-typed pads (at the cost of memory overhead). The documentation provides guidance on how to design interfaces to strongly-typed and type-erased pads.We introduce the
imxrt106x-iomuxc
crate. The crateimxrt-iomuxc
pin tratsAn example of 1 is
Unlike today's
iomuxc
module, which heavily relies on macros, we can simply implement allPin
traits with associated constants and types from theimxrt-iomuxc
crate. The constants prefixed withDAISY_*
may be written by hand, or auto-generated from an i.MX RT SVD file. We introduce a script that generates these constants.Note the
AD_B1_03
pad type. This leads to 2: defining the processor pads. We generate pads at build-time using theimxrt-iomuxc-build
crate:We introduce the
imxrt-iomuxc-build
crate. Processor-specific crates likeimxrt106x-iomuxc
use build scripts to define the pads. The build scripts depend onimxrt-iomuxc-build
. The build scripts generatePin
traits, since all pads may be used as GPIOs.Pads
structs, which can be used to acquire pad objects.Auto-generating the pads lets us quickly support other i.MX RT processor variants. We prefer to add pad definitions and pin implementations in separate crates (like
imxrt101x-iomuxc
), but we could provide definitions in the processor-specific HALs.This PR includes all of these new crates, and it integrates the crates through the HAL. The crates have other details that aren't covered here, but are commented in source. Also see the documentation in the new crates.
This is a breaking change. The ways users reference and pass pad objects to HAL code is different. Additionally, there's a new GPIO driver which takes advantage of the new pad objects.
Discussion
The updated documentation, as well as the integration in the
teensy4-rs
project, shows that this achieves the goals described by #26. Do we think this is a sufficient approach for pin configuration?What are our thoughts on using this as the foundation for split HALs, as part of #56? Pad definitions are one of the most processor-specific features we need to support, and this approach should let us de-couple the pad definitions from more general peripheral drivers.
I favor separating the pad definitions from the HAL. Specifically, I favor having a
imxrt106x-iomuxc
crate that's used by aimxrt106x-hal
, rather than incorporating the contents directly into the HAL crate. These pin traits and pad implementations could have value for others, and keeping them separate from any HAL implementation could help others with prototyping. Do we think that's a worthy goal? Or, are there reasons that pad definitions should live in processor-specific HALs?I went back and forth on how much Rust code we should auto-generate. At one point, I was writing a custom SVD -> Rust translator that would auto-generate all of the
Pin
implementations based on the SVD definitions. I eventually settled on this middle-ground, where we auto-generate the pad definitions, then manually implementPin
traits. I thought a fully-automated approach would be overkill, particularly since the number of processors we need to support is bounded (I have six different reference manuals; there's probably six crates we'd need to implement). The proposed approach does require us to manually check against the reference manuals. Should we consider more, or less, automation?Any preference on maintaining these new crates in this repo? Or, should they live in a separate repo? This PR adds the new crates to this repo, but my first plan was to maintain them in a separate repo (the crates were all developed outside of the HAL).
TODOs
iomux::pin_config
module.enum
with no variants, instead ofstruct
, for symbols that are intended as type-level tags.