-
Notifications
You must be signed in to change notification settings - Fork 48
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow to "freeze" gpio type state #138
base: main
Are you sure you want to change the base?
Conversation
8b96486
to
1b80d08
Compare
@AdinAck do you have any input on this? |
I have also been thinking about this! It's really quite unfortunate that references to ZSTs are not ZSTs themselves, otherwise I believe holding references would have been the correct solution, the lifetime troubles could certainly be worked out. However, since we cannot do that without introducing significant overhead, this is a clever solution. I can't tell if it's complete though... I believe that the notion that a peripheral shall be owned by one thing and one thing only is an invariance that should not be broken. Seeing as the goal of the HAL is to maintain a certain parity with the hardware, what if instead we change the HAL pin structs a bit to reflect the hardware? Clearly comparators have an additional special access to select pins. So the pins could have channels: struct PXi<Mode> {
ch1: PXi1<Mode>,
ch2: PXi2<Mode>, That way, it can be used like so: let pin = gpiox.pi.into_input_pull_up().split(); // `split()` will only be implemented for inputs
// i don't know either of these peripheral interfaces
let adc = p.ADC.constrain(pin.ch1);
let comp = p.COMP.constrain(pin.ch2); Conversion methods ( let pin = pin.common.recombine((adc.reset_and_release().1, comp.reset_and_release().1)).into_push_pull_output();
I'm still thinking about all this, so it's all in flux. I think this better adheres to the design patterns already present in the HAL, but maybe it's too complicated, let me know what you think. I am not against your proposal, it's certainly a step in the right direction nonetheless. Edit: Thinking about it more, it may not be immediately clear how many "channels" each pin should have, we could just arbitrarily give all pins more than enough channels (say 8), but that may be a bit weird. Also, unlike the Edit 2: I have done a lot more thinking and I think this is not the right idea. Pins are not the only peripherals that may be "read" by multiple other peripherals, and there is no fixed maximum number of "observers". While it is unfortunate that the |
Example of how let gpiox = p.GPIOX.split();
let px1 = gpiox.px1.into_analog().freeze();
let px2 = gpiox.px2.into_floating_input().freeze();
let (dacx_ch1, ..) = p.DACX.constrain(); // iirc dac channels are intrinsically frozen
let exti = p.EXTI.split();
let (comp1, ..) = p.COMP.split();
let comp1 = comp1.constrain(&dacx_ch1, &px1); // requires both sources are frozen
let comp1 = comp1.freeze(); // now even the comp itself is frozen for use with EXTI
exti.ch2.constrain(&px2).enable(); // requires `px2` is frozen
exti.ch23.constrain(&comp1).enable(); // requires `comp1` is frozen This example also demonstrates an improved and more consistent EXTI interface. |
There should also be a Like for comparators: pub trait PositiveInput<C>: Frozen {
fn setup(&self, comp: &C);
} The unsafe trait Frozen {} There could also be a trait Freeze {
type Frozen: Frozen;
fn freeze(self) -> Self::Frozen;
} Edit: Oh whoops you already had a lot of this 😅 |
What about something like: fn split<const N: usize>(self) -> [Channel<N>; N] {
todo!()
}
fn recombine(..., channels: [Channel<N>; N]) -> ... {
todo!()
} The user is free to split into however many channels they want. The channels then remember how many siblings they are so that we can require them all when recombining |
A fantastic idea! I will need time to think about this :) |
Ok here's what I've come up with: What we're dealing with here is a new way that peripherals interact with each other. In this case, EXTI, COMP, and ADC are observing GPIO. To facilitate this, consider: struct Observed<P, const OBSERVERS: usize> {
p: P,
}
struct ObservationToken<P> {
_p: PhantomData<P>,
}
trait Observable {
fn observe<const N: usize>(self) -> (Observed<Self, N>, [ObservationToken<Self>; N]) {
(
Observed { p: self },
core::array::from_fn(|| ObservationToken { _p: PhantomData }), // this may be introduce overhead, will have to investigate
)
}
}
Usage: let (pin, [ob_exti, ob_comp, ob_adc]) = gpiox.pxi.into_analog().observe();
// pin type is `Observed<PXi<Analog>, 3>`
// ob_* type is `ObservationToken<PXi<Analog>>`
exti.source(ob_exti).enable();
comp.source(dac_ch1, ob_comp);
adc.constrain(Sequence { .., third: ob_adc }); // very pseudo
let (_, ob_exti) = exti.reset_and_release();
let (_, ob_comp) = comp.reset_and_release();
let (_, ob_adc) = adc.reset_and_release();
let pin = pin.release([ob_exti, ob_comp, ob_adc]);
pin.get_level(); // we can use the pin in software again The key to my reasoning for this design is these peripheral interfaces only need to "lock" the observed peripheral configuration, and do not invoke any of its methods because the interaction is purely via hardware. Only software interfaces should borrow peripherals by reference. So HAL interfaces will look like: fn constrain<INP, INM>(rb: COMP1, inp: ObservationToken<INP>, inm: ObservationToken<INM>)
where
INP: PositiveInput,
INM: NegativeInput,
{ .. } Since the HAL never uses the software methods (really it shouldn't anyways) the usage of This whole design philosophy relies on the invariant fact that up to one instance of any peripheral type exists. Still more thinking to be done, some holes to patch, please let me know your thoughts. |
I made a quick test in godbolt and rust playground and I think the usage of |
Just trying to wrap my head around this :) Is Or is it more locked down than that? If so, then what use do we even have for pin besides keeping track of the number of observers? Is this better than the observers themselves keeping track of how many siblings they are? Thus removing the need for the
What does "use the pin in software" mean? Was Lots of questions... Just trying to wrap my head around what consequences something like this would result in. :) |
Good point! It should be accessible. Since it's of type impl<P, const N: usize> AsRef<P> for Observed<P, N> {
fn as_ref(&self) -> &P {
self.p
}
} This is great because methods like
A software trigger of the ADC would not be the same as calling In conclusion, if I understand your questions correctly:
It could be argued that one should be able to mutate a peripheral while it is observed, this would be useful for testing, not production. I feel that additional interface is misleading and extraneous, but I could understand the desire for its presence. Also, I share your concern for introducing this interface, it's a rather sizable change, I agree it is important to make sure this idea is absolutely the correct decision, so I appreciate the questions! :) |
You know, adding an impl for |
Thanks a lot for your input on this :)
Correct me if I am wrong, but a lot (perhaps not all but still, a lot) of this should be possible to do in a backwards compatible way by still allowing the owned(not splitted) peripheral/pin in places where a Observer or Observed are expected, right? |
Hm, I'm not sure! Only one way to find out, I'll test it out and see what happens :) |
Ah, yes. We can implement |
See #157 for an attempt at implementing that idea |
Some random thoughts: Side note, it would be quite interesting to see what could be done with a system like this for the rcc and its consumers. At the very least i suppose that would allow a safe way to change clock frequency once tokens from all consumer are collected without risk of forgetting any peripheral that might not like the clock settings to change. However doing something like that well is likely not trivial, perhaps requiring something more fine grained? Would if the RTC is clocked by the LSE then changing other parts of the clock tree should not affect the RTC etc. Side note to the Side note. Thera are probably a lot of interesting things that can be done to the rcc system more generally. Lots of weird hardware limitations with the different steps in the pll system having different max frequencies etc. Unless am completely misremembering I believe the max frequency for the ADC is 60MHz but only if you just use one of them, if you use more then I think just north of 50. The hrtim has a minimum input frequency of 100MHz. There are probably lots of more weird things that would be interesting to see if it would be possible to check for at compile time. Then there are different limitations depending on input voltage etc. What if the user could use a const fn to generate a clock config that was checked at compile time, that config was then applied at runtime. Perhaps one could even have multiple configs in different constants and it would be completely safe to dynamicly change between them(assuming it would be possible to solve both things above). |
I have been thinking about this for many months and believe that it is the perfect opportunity to flex why Rust is so good for embedded.
I am working on a system that procedurally generates type-enforced peripheral access interfaces and have some examples where this works and stops the developer from forgetting to enable peripheral clocks before usage. gpio example. I am very excited to begin reasoning about clock speeds in the type system.
This is something I have done in the past in const PLL_CFG: rcc::PllConfig = rcc::RawPllConfig {
mux: rcc::PLLSrc::HSI, // 16MHz
m: 1, // /1
n: 16, // x16
r: 4, // /4 = 64MHz
q: Some(2), // /2 = 128MHz
p: None,
}
.validate();
// in RTIC::init...
let mut rcc = ctx.device.RCC.freeze(rcc::Config::pll().pll_cfg(PLL_CFG));
|
Nice! I will take a look at that :)
Very nice! Great to have that level of protection all the way down How/does this interact with svd-files or does this require every peripheral to be manually specified? |
No SVD, they are always riddled with errors from the vendor, which is why we patch them so much. They are hard to read, don't encode any hardware relationships at all, just suck to use. It is so much easier to just scroll through a datasheet and type in the register/field offsets. Some brief examples: #[block(
base_addr = 0x4800_0000,
entitlements = [super::super::rcc::ahb2enr::gpioaen::Enabled],
auto_increment,
)]
pub mod gpioa {
#[register(auto_increment)]
mod moder {
#[schema(width = 2, auto_increment)]
mod mode {
#[state]
struct Input;
#[state]
struct Output;
#[state]
struct Alternate;
#[state]
struct Analog;
}
#[field_array(range = ..13, schema = mode, read, write, reset = Analog)]
mod modeX {}
#[field(schema = mode, read, write, reset = Alternate)]
mod mode13 {}
#[field(schema = mode, read, write, reset = Alternate)]
mod mode14 {}
#[field(schema = mode, read, write, reset = Alternate)]
mod mode15 {}
}
#[block(
base_addr = 0x4002_0c00,
entitlements = [super::rcc::ahb1enr::cordicen::Enabled],
auto_increment,
erase_mod,
)]
mod cordic {
#[register(auto_increment)]
mod csr {
#[field(width = 4, read, write, reset = Cos, auto_increment)]
mod func {
#[state(entitlements = [scale::N0])]
struct Cos;
#[state(entitlements = [scale::N0])]
struct Sin;
#[state(entitlements = [scale::N0])]
struct ATan2;
#[state(entitlements = [scale::N0])]
struct Magnitude;
#[state]
struct ATan;
#[state(entitlements = [scale::N1])]
struct CosH;
#[state(entitlements = [scale::N1])]
struct SinH;
#[state(entitlements = [scale::N1])]
struct ATanH;
#[state(entitlements = [scale::N1, scale::N2, scale::N3, scale::N4])]
struct Ln;
#[state(entitlements = [scale::N0, scale::N1, scale::N2])]
struct Sqrt;
}
Remember all those crazy types and traits and structures that occupied 1200 lines of code to enforce the CORDIC's hardware invariances? Now they are described in 30 lines of code. Reading the datasheet is trivial, and this macro statically validates a lot in case mistakes are made. I'm worried my design of this system doesn't scale but I just keep implementing peripherals and adding things as needed. Hopefully eventually it will encapsulate everything. |
Yeah, cant say I disagree
Yes :) Wow!
Very nice
I think this might be related to stm32-rs/meta#4 . |
Very interesting! I'll read this and see if there's anything I can contribute to the conversation, thanks! |
Thinking about scaling as in getting something like this to work on many families in the future. As is discussed in the linked issue, there are a lot of devices with a lot of peripherals... Do you think it would be possible and make sense to somehow get this to work with the pacs of today(lots of it which after all the patches of stm32-rs works quite good, yet without all the nice magic type safety things)? Would it even be possible to add the entitlement info and whatnot (sorry have not quite wrapped my head around it) to svd2rust's yaml format so this could be added in peripheral by peripheral while still making use of all the things we already have? |
This was my first approach before bailing and making My motivation for creating So far implementing peripherals with the macro has been very easy and quick, so I'm not really worried about redoing work already done in other HALs/PACs, but maybe I'll run into issues soon. |
One perhaps not great idea but at least low effort way would perhaps be if your pac types implemented |
Hm, yeah. Some adaptor between the two would be nice. I'll think on that, thanks! |
This observation-thing feels like it could be useful for other hals and perhaps not only parts from stm. Would it make sense to put src/observation.rs into its own crate to reduce duplication and make it easier to make crates that work with different hals? |
Yep! I already have it in It's in the stasis file. I moved on from the "observation" naming scheme because since the beginning we've known that's not the best name. For example if we consider the output level of a pin to not contribute to the state of that pin, then an Peripherals are entitled to states. Obviously I am using this for far more complicated peripheral interface generation stuff in I would be open to moving these structures to their own crate all by themselves if you really want. But I feel like it makes sense to keep them in |
The reason I created |
I just migrated over most of the important design features to stasis. |
Oh my bad, sorry I did not get that Entitlements was the same thing as ObservationToken :). Makes sense
Already seems to be quite broken out, no need to break it out further The rough gist from what I have seen so far(which is not a lot) is:
Is that somewhat accurate? Very interesting stuff :) |
100% accurate!
I agree! Thanks for all your help! |
1. Changing pin state
Today a gpio pin can be turned into another mode using for example
let pin = pin.into_push_pull_output();
orlet pin = pin.into_alternate()
. This can also be done multiple times for example:This is of course very convenient.
2. Same pin, Multiple peripherals
Some analog peripherals like for example the comparators need their input pins to be set in analog mode. Since the pin is set to analog mode it should also be possible to perform AD-readings on it at the same time as the comparator is connected.
This may for example be useful for having near instant over current protection while still allowing regular AD-readings all on the same pin.
3. Peripherals
When setting up a peripheral such as the comparator, the input pins are typically provided by value, thus consuming the pin. This to ensure that no one else will change the mode of the pin.
The problem
Currently
1
and3
works, but not2
. Changing peripherals to take a pin by reference and store the reference would prove quite cumbersome since the pin would have to live as long as the peripheral and not be moved etc.Only having having the init function take a reference and then not storing it would not prevent the user from then changing pin mode after peripheral init.
Proposed solution
Add type parameter
F
to the pin types which may be one of the following types:The type parameter default to
IsNotFrozen
so as to keep backwards compatibility and1
from above works as before. However once the user knows they will want the pin to stay in a particular state for the rest of the program,let pin = pin.freeze();
may be called. Which returns the same type but withIsFrozen
this disabling theinto_alternate/analog/{x}_input/output
methods.This then allows peripherals init to take the pin by reference (assuming
F=IsFrozen
) and not having to hold on to it. (only done for some of the peripherals so far)