-
Notifications
You must be signed in to change notification settings - Fork 204
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Clocking API V2 (for thumbv7em) (#450)
Add a new API for the `clock` module on `thumbv7em` targets Add a new `clock` module API that enforces correctness by construction. It is impossible to create an invalid clock tree without using `unsafe`. Specifically, use typed tokens and clocks to guarantee memory safety and act as proof that a clock is correctly configured, and use type-level numbers to count consumer clocks at compile-time and restrict a given clock's API while it is in use. This commit comes after two years of work, starting with #272, then with #429 and culminating with PR #450. Co-authored-by: Bradley Harden <bradleyharden@gmail.com> Co-authored-by: Gabriel Górski <gabriel.gorski@volvocars.com> Co-authored-by: Henrik Tjäder <henrik.tjader@volvocars.com>
- Loading branch information
1 parent
e8e2203
commit d2ba3da
Showing
20 changed files
with
10,465 additions
and
547 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,171 @@ | ||
#![no_main] | ||
#![no_std] | ||
|
||
use panic_halt as _; | ||
|
||
use core::fmt::Write as _; | ||
|
||
use atsamd_hal::{ | ||
clock::v2::{ | ||
self as clock, | ||
dpll::Dpll, | ||
gclk::{Gclk, GclkDiv16, GclkDiv8}, | ||
osculp32k::OscUlp32k, | ||
pclk::Pclk, | ||
rtcosc::RtcOsc, | ||
xosc32k::{ControlGainMode, Xosc1k, Xosc32k, Xosc32kBase}, | ||
}, | ||
ehal::serial::Read as _, | ||
ehal::serial::Write, | ||
gpio::{Pins, PA04, PA05}, | ||
rtc::{ClockMode, Rtc}, | ||
sercom::{ | ||
uart::{self, BaudMode, Flags, Oversampling}, | ||
IoSet3, Sercom0, | ||
}, | ||
time::U32Ext, | ||
}; | ||
|
||
use rtic::app; | ||
|
||
type Pads = uart::PadsFromIds<Sercom0, IoSet3, PA05, PA04>; | ||
type Uart = uart::Uart<uart::Config<Pads>, uart::Duplex>; | ||
|
||
#[app(device = atsamd_hal::pac, peripherals = true)] | ||
mod app { | ||
use super::*; | ||
|
||
#[shared] | ||
struct SharedResources { | ||
uart: Uart, | ||
rtc: Rtc<ClockMode>, | ||
} | ||
|
||
#[local] | ||
struct LocalResources {} | ||
|
||
#[init] | ||
fn init(cx: init::Context) -> (SharedResources, LocalResources, init::Monotonics()) { | ||
let mut device = cx.device; | ||
|
||
// Get the clocks & tokens | ||
let (_buses, clocks, tokens) = clock::clock_system_at_reset( | ||
device.OSCCTRL, | ||
device.OSC32KCTRL, | ||
device.GCLK, | ||
device.MCLK, | ||
&mut device.NVMCTRL, | ||
); | ||
|
||
// This is required because the `sercom` and `rtc` modules have not yet | ||
// been update to use `clock::v2` | ||
let (_, _, _, mut mclk) = unsafe { clocks.pac.steal() }; | ||
|
||
// Get the pins | ||
let pins = Pins::new(device.PORT); | ||
|
||
// Take `Dfll` 48 MHz, divide down to `2 MHz` through `Gclk1` | ||
let (gclk1, dfll) = Gclk::from_source(tokens.gclks.gclk1, clocks.dfll); | ||
let gclk1 = gclk1.div(GclkDiv16::Div(24)).enable(); | ||
|
||
// Output `Gclk1` on PB15 pin | ||
let (gclk1, _gclk1_out) = gclk1.enable_gclk_out(pins.pb15); | ||
|
||
// Setup a peripheral channel to power up `Dpll0` from `Gclk1` | ||
let (pclk_dpll0, gclk1) = Pclk::enable(tokens.pclks.dpll0, gclk1); | ||
|
||
// Configure `Dpll0` with `2 * 60 + 0/32 = 120 MHz` frequency | ||
let dpll0 = Dpll::from_pclk(tokens.dpll0, pclk_dpll0) | ||
.loop_div(60, 0) | ||
.enable(); | ||
|
||
// Swap source of `Gclk0` from Dfll to Dpll0, `48 Mhz -> 120 MHz` | ||
let (gclk0, _dfll, _dpll0) = clocks.gclk0.swap_sources(dfll, dpll0); | ||
|
||
// Output `Gclk0` on pin PB14 | ||
let (gclk0, _gclk0_out) = gclk0.enable_gclk_out(pins.pb14); | ||
|
||
// Setup a peripheral channel to power up `Dpll1` from `Gclk1` | ||
let (pclk_dpll1, _gclk1) = Pclk::enable(tokens.pclks.dpll1, gclk1); | ||
|
||
// Configure `Dpll1` with `2 * 50 + 0/32 = 100 MHz` frequency | ||
let dpll1 = Dpll::from_pclk(tokens.dpll1, pclk_dpll1) | ||
.loop_div(50, 0) | ||
.enable(); | ||
|
||
// Output `Dpll1` on PB20 pin via `Gclk6`, divided by 200 resulting in 0.5 MHz | ||
// output frequency | ||
let (gclk6, _dpll1) = Gclk::from_source(tokens.gclks.gclk6, dpll1); | ||
let gclk6 = gclk6.div(GclkDiv8::Div(200)).enable(); | ||
let (_gclk6, _gclk6_out) = gclk6.enable_gclk_out(pins.pb12); | ||
|
||
// Configure `Xosc32k` with both outputs (1kHz, 32kHz) activated | ||
let xosc32k_base = Xosc32kBase::from_crystal(tokens.xosc32k.base, pins.pa00, pins.pa01) | ||
.control_gain_mode(ControlGainMode::HighSpeed) | ||
.on_demand(false) | ||
.run_standby(true) | ||
.enable(); | ||
let (xosc1k, xosc32k_base) = Xosc1k::enable(tokens.xosc32k.xosc1k, xosc32k_base); | ||
let (xosc32k, _xosc32k_base) = Xosc32k::enable(tokens.xosc32k.xosc32k, xosc32k_base); | ||
|
||
// Output `Xosc32k` on PB16 pin via `Gclk2`, divided by 2 resulting in 16 kHz | ||
// output frequency | ||
let (gclk2, _xosc32k) = Gclk::from_source(tokens.gclks.gclk2, xosc32k); | ||
let gclk2 = gclk2.div(GclkDiv8::Div(2)).enable(); | ||
let (_gclk2, _gclk2_out) = gclk2.enable_gclk_out(pins.pb16); | ||
|
||
// Output `OscUlp32k` on PB11 pin via `Gclk5`, without any division resulting in | ||
// 32 kHz output frequency | ||
let (osculp32k, _osculp_base) = | ||
OscUlp32k::enable(tokens.osculp32k.osculp32k, clocks.osculp32k_base); | ||
let (gclk5, _osculp32k) = Gclk::from_source(tokens.gclks.gclk5, osculp32k); | ||
let gclk5 = gclk5.enable(); | ||
let (_gclk5, _gclk5_out) = gclk5.enable_gclk_out(pins.pb11); | ||
|
||
// Setup a peripheral channel to power up `Uart` from `Gclk0` | ||
let (pclk_sercom0, _gclk0) = Pclk::enable(tokens.pclks.sercom0, gclk0); | ||
|
||
use atsamd_hal::sercom::uart; | ||
|
||
let pads = uart::Pads::default().rx(pins.pa05).tx(pins.pa04); | ||
// In the future, the `Uart` will take ownership of the `Pclk` and will | ||
// take an `ApbClk` instead of `&MCLK` | ||
let mut uart = uart::Config::new(&mclk, device.SERCOM0, pads, pclk_sercom0.freq()) | ||
.baud(115_200.hz(), BaudMode::Arithmetic(Oversampling::Bits16)) | ||
.enable(); | ||
uart.enable_interrupts(Flags::RXC); | ||
|
||
// Initialize the RTC oscillator from the 1 kHz output of XOSC32K | ||
let (rtc_osc, _xosc1k) = RtcOsc::enable(tokens.rtcosc, xosc1k); | ||
|
||
// Setup an `Rtc` in `ClockMode` | ||
// In the future, the `Rtc` will take ownership of the `RtcOsc` | ||
let rtc = Rtc::clock_mode(device.RTC, rtc_osc.freq(), &mut mclk); | ||
|
||
writeln!(&mut uart as &mut dyn Write<_, Error = _>, "RTIC booted!").unwrap(); | ||
|
||
( | ||
SharedResources { uart, rtc }, | ||
LocalResources {}, | ||
init::Monotonics(), | ||
) | ||
} | ||
|
||
#[task(binds = SERCOM0_2, shared = [uart, rtc])] | ||
fn uart(cx: uart::Context) { | ||
let mut uart = cx.shared.uart; | ||
let mut rtc = cx.shared.rtc; | ||
// Read from `Uart` to clean interrupt flag | ||
let _ = uart.lock(|u| u.read().unwrap()); | ||
|
||
// Print out `DateTime` coming from `Rtc` | ||
uart.lock(|u| { | ||
writeln!( | ||
u as &mut dyn Write<_, Error = _>, | ||
"{:#?}", | ||
rtc.lock(|r| r.current_time()) | ||
) | ||
.unwrap() | ||
}); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.