Skip to content
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

Implement Observable POC #157

Draft
wants to merge 12 commits into
base: staged-pac
Choose a base branch
from
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ version = "0.0.2"
nb = "0.1.1"
#stm32g4 = { git = "https://github.com/stm32-rs/stm32-rs-nightlies" } #"0.15.1"
stm32g4 = { version = "0.19.0", package = "stm32g4-staging" }
proto-hal = { git = "https://github.com/AdinAck/proto-hal", rev = "9030895" }
paste = "1.0"
bitflags = "1.2"
vcell = "0.1"
Expand Down
8 changes: 4 additions & 4 deletions examples/comp.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ use rt::entry;

#[entry]
fn main() -> ! {
use hal::comparator::{ComparatorExt, ComparatorSplit, Config, Hysteresis, RefintInput};
use hal::comparator::{refint_input, ComparatorExt, ComparatorSplit, Config, Hysteresis};
use hal::gpio::GpioExt;
use hal::prelude::OutputPin;
use hal::rcc::RccExt;
Expand All @@ -31,16 +31,16 @@ fn main() -> ! {

let pa1 = gpioa.pa1.into_analog();
let pa0 = gpioa.pa0.into_analog();
let comp1 = comp1.comparator(&pa1, pa0, Config::default(), &rcc.clocks);
let comp1 = comp1.comparator(pa1, pa0, Config::default(), &rcc.clocks);
let comp1 = comp1.enable();

// led1 pa1 will be updated manually when to match comp1 value
let mut led1 = gpioa.pa5.into_push_pull_output();

let pa7 = gpioa.pa7.into_analog();
let comp2 = comp2.comparator(
&pa7,
RefintInput::VRefintM12,
pa7,
refint_input::VRefintM12,
Config::default()
.hysteresis(Hysteresis::None)
.output_inverted(),
Expand Down
8 changes: 5 additions & 3 deletions examples/comp_w_dac.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ mod utils;
extern crate cortex_m_rt as rt;

use rt::entry;
use stm32g4xx_hal::observable::Observable;

#[entry]
fn main() -> ! {
Expand All @@ -30,15 +31,16 @@ fn main() -> ! {
// Set up DAC to output to pa4 and to internal signal Dac1IntSig1
// which just so happens is compatible with comp1
let dac1ch1 = dp.DAC1.constrain((gpioa.pa4, Dac1IntSig1), &mut rcc);
let mut dac = dac1ch1.calibrate_buffer(&mut delay).enable();
let dac = dac1ch1.calibrate_buffer(&mut delay).enable();
let (mut dac, [dac_token]) = dac.observe();

let (comp1, _comp2, ..) = dp.COMP.split(&mut rcc);
let pa1 = gpioa.pa1.into_analog();

// Set up comparator with pa1 as positive, and the DAC as negative input
let comp = comp1.comparator(
&pa1,
&dac,
pa1,
dac_token,
comparator::Config::default().hysteresis(comparator::Hysteresis::None),
&rcc.clocks,
);
Expand Down
64 changes: 64 additions & 0 deletions examples/observe.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
//! ## Origin
//!
//! This code has been taken from the stm32g0xx-hal project and modified slightly to support
//! STM32G4xx MCUs.

//#![deny(warnings)]
#![deny(unsafe_code)]
#![no_main]
#![no_std]

mod utils;
use utils::logger::info;
extern crate cortex_m_rt as rt;

use fugit::ExtU32 as _;
use hal::{
adc::AdcClaim as _,
comparator::{ComparatorExt, ComparatorSplit, Config},
delay::SYSTDelayExt as _,
gpio::GpioExt,
rcc::RccExt,
stm32,
};
use proto_hal::stasis::Freeze;
use rt::entry;
use stm32g4xx_hal::{self as hal, adc::config::SampleTime, delay::DelayExt as _};

#[entry]
fn main() -> ! {
let cp = cortex_m::Peripherals::take().unwrap();
let dp = stm32::Peripherals::take().unwrap();
let mut rcc = dp.RCC.constrain();

let gpioa = dp.GPIOA.split(&mut rcc);

let (comp1, ..) = dp.COMP.split(&mut rcc);

let (pa1, [pa1_token]) = gpioa // <- The pin to keep track of
.pa1
.into_analog()
.freeze();
let pa0 = gpioa.pa0.into_analog(); // <- Reference voltage

// Only pa1_token and pa0 consumed here
let comp1 = comp1.comparator(pa1_token, pa0, Config::default(), &rcc.clocks);
let _comp1 = comp1.enable(); // <-- TODO: Do things with comparator

let mut delay = cp.SYST.delay(&rcc.clocks);
let mut adc = dp.ADC1.claim_and_configure(
stm32g4xx_hal::adc::ClockSource::SystemClock,
&rcc,
stm32g4xx_hal::adc::config::AdcConfig::default(),
&mut delay,
false,
);

// Can not reconfigure pa1 here
loop {
// Can still use pa1 here
let sample = adc.convert(&pa1, SampleTime::Cycles_640_5);
info!("Reading: {}", sample);
delay.delay(1000.millis());
}
}
28 changes: 14 additions & 14 deletions examples/opamp.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

use stm32g4xx_hal::adc::AdcClaim;
use stm32g4xx_hal::adc::ClockSource;
use stm32g4xx_hal::opamp::{Gain, InternalOutput};
use stm32g4xx_hal::opamp::Gain;
use stm32g4xx_hal::prelude::*;
use stm32g4xx_hal::pwr::PwrExt;

Expand Down Expand Up @@ -34,12 +34,20 @@ fn main() -> ! {
// setup opamps
let (opamp1, opamp2, opamp3, ..) = dp.OPAMP.split(&mut rcc);

let pa1 = gpioa.pa1.into_analog();
let pa2 = gpioa.pa2.into_analog();
let pa7 = gpioa.pa7.into_analog();

let pb0 = gpiob.pb0.into_analog();
let pb1 = gpiob.pb1.into_analog();
let pb2 = gpiob.pb2.into_analog();

// Set up opamp1 and opamp2 in follower mode
let opamp1 = opamp1.follower(gpioa.pa1, gpioa.pa2);
let opamp2 = opamp2.follower(gpioa.pa7, InternalOutput);
let opamp1 = opamp1.follower(pa1).enable_output(pa2);
let opamp2 = opamp2.follower(pa7);

// Set up opamp1 and opamp2 in open loop mode
let opamp3 = opamp3.open_loop(gpiob.pb0, gpiob.pb2, gpiob.pb1);
let opamp3 = opamp3.open_loop(pb0, pb2).enable_output(pb1);

// disable opamps
let (opamp1, pa1, pa2) = opamp1.disable();
Expand All @@ -48,18 +56,10 @@ fn main() -> ! {
let (_opamp3, _pb0, _pb2, _pb1) = opamp3.disable();

// Configure opamp1 with pa1 as non-inverting input and set gain to x2
let _opamp1 = opamp1.pga(
pa1,
pa2, // Route output to pin pa2
Gain::Gain2,
);
let _opamp1 = opamp1.pga(pa1, Gain::Gain2).enable_output(pa2); // Route output to pin pa2

// Configure op with pa7 as non-inverting input and set gain to x4
let opamp2 = opamp2.pga(
pa7,
InternalOutput, // Do not route output to any external pin, use internal AD instead
Gain::Gain4,
);
let opamp2 = opamp2.pga(pa7, Gain::Gain4); // Do not route output to any external pin, use internal AD instead

// Lock opamp2. After the opamp is locked its registers cannot be written
// until the device is reset (even if using unsafe register accesses).
Expand Down
21 changes: 16 additions & 5 deletions src/adc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ use embedded_hal::{
adc::{Channel, OneShot},
blocking::delay::DelayUs,
};
use proto_hal::stasis;

use self::config::ExternalTrigger12;

Expand Down Expand Up @@ -136,7 +137,9 @@ impl Temperature {
macro_rules! adc_pins {
($($pin:ty => ($adc:ident, $chan:expr)),+ $(,)*) => {
$(
impl Channel<stm32::$adc> for $pin {
impl<P> Channel<stm32::$adc> for P
where P: stasis::EntitlementLock<Resource = $pin>
{
Comment on lines -139 to +142
Copy link
Contributor Author

@usbalbin usbalbin Dec 28, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This does not work. Both Channel and EntitlementLock (or rather P i suppose) are from outside this crate. There are also errors about conflicting implementations with the other opamps below.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the simplest solution is to make a local trait like AdcChannel which requires embedded_hal::adc::Channel and then it would work.

I'm not very familiar with what you're doing here though. I'll think on it.

You could also do concrete implementations since this is macro generated anyway.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You could also do concrete implementations since this is macro generated anyway.

Yeah that would probably be the simplest solution. I'll give it a go

type ID = u8;
fn channel() -> u8 { $chan }
}
Expand All @@ -147,22 +150,30 @@ macro_rules! adc_pins {
macro_rules! adc_opamp {
($($opamp:ty => ($adc:ident, $chan:expr)),+ $(,)*) => {
$(
impl<A> Channel<stm32::$adc> for opamp::Follower<$opamp, A, InternalOutput> {
impl<A, P> Channel<stm32::$adc> for P
where P: stasis::EntitlementLock<Resource = opamp::Follower<$opamp, A, InternalOutput>>
{
type ID = u8;
fn channel() -> u8 { $chan }
}

impl<A, B> Channel<stm32::$adc> for opamp::OpenLoop<$opamp, A, B, InternalOutput> {
impl<A, B, P> Channel<stm32::$adc> for P
where P: stasis::EntitlementLock<Resource = opamp::OpenLoop<$opamp, A, B, InternalOutput>>
{
type ID = u8;
fn channel() -> u8 { $chan }
}

impl<A> Channel<stm32::$adc> for opamp::Pga<$opamp, A, InternalOutput> {
impl<A, P> Channel<stm32::$adc> for P
where P: stasis::EntitlementLock<Resource = opamp::Pga<$opamp, A, InternalOutput>>
{
type ID = u8;
fn channel() -> u8 { $chan }
}

impl Channel<stm32::$adc> for opamp::Locked<$opamp, InternalOutput> {
impl<P> Channel<stm32::$adc> for P
where P: stasis::EntitlementLock<Resource = opamp::Locked<$opamp, InternalOutput>>
{
type ID = u8;
fn channel() -> u8 { $chan }
}
Expand Down
Loading
Loading