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

Add PTC support for SAMD11 and SAMD21 #420

Open
wants to merge 24 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
60bf956
Add PTC support for samd21e.
AuroransSolis Mar 22, 2021
3df65a6
Mapped all pins and handled X line channels.
AuroransSolis Mar 22, 2021
126a563
Hide the whole module behind SAMD21.
AuroransSolis Mar 22, 2021
76cfa1f
Couldn't stop thinking about it. Added PTC support for G and J.
AuroransSolis Mar 22, 2021
af09284
The `rustfmt.sh` script likes to remove this tag for some reason.
AuroransSolis Mar 22, 2021
1441ebb
Merge branch 'master' into master
AuroransSolis Mar 25, 2021
aadb194
Realized that for the time being, it makes more sense to put PTC stuf…
AuroransSolis Mar 26, 2021
6937385
Undid manual SVD patching and added PTC patch to the SAMD21 XSL file.
AuroransSolis Mar 26, 2021
729a92d
Merge branch 'master' into master
AuroransSolis Mar 27, 2021
9848467
Added PTC peripheral to the SAMD11 SVD file and generalized the PTC p…
AuroransSolis Apr 5, 2021
d8c9cff
Bring branch up to date with upstream.
AuroransSolis Aug 3, 2021
a5828f9
Lotsa stuff going on here, but in short, SAMD{1,2}D should have worki…
AuroransSolis Aug 3, 2021
e3fc4a7
Finally back from vacation and able to make this simple fix.
AuroransSolis Aug 13, 2021
ea256d5
Add PTC support for samd21e.
AuroransSolis Mar 22, 2021
6e4bc13
Mapped all pins and handled X line channels.
AuroransSolis Mar 22, 2021
9d537a9
Hide the whole module behind SAMD21.
AuroransSolis Mar 22, 2021
22d65af
Couldn't stop thinking about it. Added PTC support for G and J.
AuroransSolis Mar 22, 2021
3ceac4b
The `rustfmt.sh` script likes to remove this tag for some reason.
AuroransSolis Mar 22, 2021
5b46304
Realized that for the time being, it makes more sense to put PTC stuf…
AuroransSolis Mar 26, 2021
fe4c8bf
Undid manual SVD patching and added PTC patch to the SAMD21 XSL file.
AuroransSolis Mar 26, 2021
f13ea26
Added PTC peripheral to the SAMD11 SVD file and generalized the PTC p…
AuroransSolis Apr 5, 2021
91f5a1e
Lotsa stuff going on here, but in short, SAMD{1,2}D should have worki…
AuroransSolis Aug 3, 2021
db5949e
Finally back from vacation and able to make this simple fix.
AuroransSolis Aug 13, 2021
7a5613c
Merge branch 'master' of github.com:AuroransSolis/atsamd
AuroransSolis Aug 14, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions hal/src/thumbv6m/clock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -415,6 +415,7 @@ clock_generator!(
(ac_ana, AcAnaClock, AC_ANA),
(ac_dig, AcDigClock, AC_DIG),
(dac, DacClock, DAC),
(ptc, PtcClock, PTC),
);
// samd21
#[cfg(feature = "samd21")]
Expand Down Expand Up @@ -449,6 +450,7 @@ clock_generator!(
(ac_ana, AcAnaClock, AC_ANA),
(ac_dig, AcDigClock, AC_DIG),
(dac, DacClock, DAC),
(ptc, PtcClock, PTC),
(i2s0, I2S0Clock, I2S_0),
(i2s1, I2S1Clock, I2S_1),
);
Expand Down
3 changes: 3 additions & 0 deletions hal/src/thumbv6m/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,4 +23,7 @@ pub mod watchdog;
#[cfg(feature = "samd21")]
pub mod usb;

#[cfg(feature = "unproven")]
pub mod ptc;

pub(crate) mod sercom;
290 changes: 290 additions & 0 deletions hal/src/thumbv6m/ptc.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,290 @@
//! Peripheral Touch Controller (PTC)
//! Please refer to the relevant documentation for your chip for information on
//! how to use this device peripheral.
//! - SAMD11: [Section 34 (page 855)](http://ww1.microchip.com/downloads/en/DeviceDoc/Atmel-42363-SAM-D11_Datasheet.pdf)
//! - SAMD21: [Section 36 (page 1040)](https://ww1.microchip.com/downloads/en/DeviceDoc/SAM_D21_DA1_Family_DataSheet_DS40001882F.pdf)
//!
//! Note that Y channels map to IDs `[0, 16)` and X channels to `[16, 32)`.

use crate::clock::GenericClockController;
use crate::gpio::v1;
use crate::gpio::v2::*;
use crate::hal::adc::{Channel, OneShot};
use crate::target_device::{
ptc::{
convctrl::ADCACCUM_A, freqctrl::SAMPLEDELAY_A, serres::RESISTOR_A, xselect::XMUX_A,
yselect::YMUX_A,
},
PM, PTC,
};

pub struct Ptc<PTC> {
ptc: PTC,
}

impl Ptc<PTC> {
pub fn ptc(ptc: PTC, pm: &mut PM, clocks: &mut GenericClockController) -> Self {
// Enable PTC in the APBC mask
pm.apbcmask.modify(|_, w| w.ptc_().set_bit());
let gclk1 = clocks.gclk1();
// Enable the PTC clock
clocks.ptc(&gclk1).expect("ptc clock setup failed");
while ptc.ctrlb.read().syncflag().bit_is_set() {}

// Reset the PTC module
ptc.ctrla.modify(|_, w| w.swrst().set_bit());
while ptc.ctrlb.read().syncflag().bit_is_set() {}

// Magic writes? Honestly dunno what these are for.
// f7 => 11110111
// fb => 11111011
// fc => 11111100
ptc.unk4c04.write(|w| unsafe { w.bits(0xf7) });
ptc.unk4c04.write(|w| unsafe { w.bits(0xfb) });
ptc.unk4c04.write(|w| unsafe { w.bits(0xfc) });
while ptc.ctrlb.read().syncflag().bit_is_set() {}

// Next in the init sequence in the FreeTouch repo, writes of the following two
// values are made to FREQCTRL:
// 9f => 10011111
// ---baaaa
// ef => 11101111
// ---baaaa
// The upper three bits are unused, so I'm unsure what the point of this is
// beyond setting all of the SAMPLEDELAY field to 1 and toggling
// FREQSPREADEN. Furthermore, the next thing done is setting SAMPLEDELAY
// to 0, thus ending up with (in theory): 11101111
// & 11110000
// ----------
// 11100000
// So honestly, I'm just going to set them to 0 in one step.
ptc.freqctrl.write(|w| {
w.freqspreaden().clear_bit();
w.sampledelay().variant(SAMPLEDELAY_A::FREQHOP1)
});
while ptc.ctrlb.read().syncflag().bit_is_set() {}

// Software init
ptc.ctrlc.write(|w| w.init().set_bit());
// Set to run in standby
ptc.ctrla.write(|w| w.runstdby().set_bit());
while ptc.ctrlb.read().syncflag().bit_is_set() {}

// Set interrupt enables
ptc.intenclr.write(|w| {
w.wco().set_bit();
w.eoc().set_bit()
});
while ptc.ctrlb.read().syncflag().bit_is_set() {}

Self { ptc }
}

pub fn compcap(&mut self, compcap: u16) {
self.ptc
.compcap
.write(|w| unsafe { w.value().bits(compcap) });
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

pub fn intcap(&mut self, intcap: u8) {
self.ptc.intcap.write(|w| unsafe { w.value().bits(intcap) });
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

pub fn oversample(&mut self, oversample: ADCACCUM_A) {
self.ptc
.convctrl
.write(|w| w.adcaccum().variant(oversample));
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

pub fn sample_delay(&mut self, sampledelay: SAMPLEDELAY_A) {
match sampledelay {
SAMPLEDELAY_A::FREQHOP1 => self.ptc.freqctrl.write(|w| w.freqspreaden().clear_bit()),
_ => self.ptc.freqctrl.write(|w| w.freqspreaden().set_bit()),
}
self.ptc
.freqctrl
.write(|w| w.sampledelay().variant(sampledelay));
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

pub fn series_resistance(&mut self, serres: RESISTOR_A) {
self.ptc.serres.write(|w| w.resistor().variant(serres));
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

pub fn xselect(&mut self, xselect: XMUX_A) {
self.ptc.xselect.write(|w| w.xmux().variant(xselect));
}

pub fn yselect(&mut self, yselect: YMUX_A) {
self.ptc.yselect.write(|w| w.ymux().variant(yselect));
}

fn power_up(&mut self) {
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
self.ptc.ctrla.modify(|_, w| w.enable().set_bit());
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

fn power_down(&mut self) {
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
self.ptc.ctrla.modify(|_, w| w.enable().clear_bit());
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}
}

fn convert(&mut self) -> u16 {
self.ptc
.burstmode
.write(|w| unsafe { w.burstmode().bits(0xa4) });
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}

self.ptc.convctrl.write(|w| w.convert().set_bit());
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}

self.ptc.result.read().result().bits()
}
}

impl<WORD, PIN> OneShot<PTC, WORD, PIN> for Ptc<PTC>
where
WORD: From<u16>,
PIN: Channel<PTC, ID = u8>,
{
type Error = ();

fn read(&mut self, _pin: &mut PIN) -> nb::Result<WORD, Self::Error> {
let channel = PIN::channel();
while self.ptc.ctrlb.read().syncflag().bit_is_set() {}

// Select and enable specified channel
if channel > 15 {
let channel = channel - 16;
self.ptc
.xselect
.write(|w| unsafe { w.xmux().bits(1 << channel) });
self.ptc
.xselecten
.modify(|r, w| unsafe { w.bits(r.bits() | (1 << channel)) });
} else {
self.ptc
.yselect
.write(|w| unsafe { w.ymux().bits(1 << channel) });
self.ptc
.yselecten
.modify(|r, w| unsafe { w.bits(r.bits() | (1 << channel)) });
};
self.power_up();
let result = self.convert();
self.power_down();

Ok(result.into())
}
}

pub struct XAndY {
pub x: u8,
pub y: u8,
}

macro_rules! ptc_pins {
($($PinId:ident: $Chan:literal$( / $Chan2:literal)?),+) => {$(
ptc_pins!{@$PinId: $Chan$( / $Chan2)?}
)+};
(@$PinId:ident: $Chan:literal) => {
impl Channel<PTC> for Pin<$PinId, AlternateB> {
type ID = u8;
fn channel() -> u8 { $Chan }
}
};
(@$PinId:ident: $XChan:literal / $YChan:literal) => {
impl Channel<PTC> for Pin<$PinId, AlternateB> {
type ID = XAndY;
fn channel() -> XAndY { XAndY { x: $XChan, y: $YChan } }
}
};
}

/// Implement ['Channel`] for [`v1::Pin`]s based on the implementations for
/// `v2` [`Pin`]s
impl<I> Channel<PTC> for v1::Pin<I, v1::PfB>
where
I: PinId,
Pin<I, AlternateB>: Channel<PTC, ID = u8>,
{
type ID = u8;
fn channel() -> u8 {
Pin::<I, AlternateB>::channel()
}
}

#[cfg(feature = "samd11")]
ptc_pins! {
PA02: 0,
PA04: 2,
PA05: 3,
PA14: 16 / 6,
PA15: 17 / 7,
PA24: 24 / 14,
PA25: 25 / 15
}

#[cfg(feature = "samd11d")]
ptc_pins! {
PA03: 1,
PA06: 4,
PA06: 5,
PA10: 18 / 8,
PA11: 19 / 9,
PA16: 20 / 10,
PA17: 21 / 11,
PA22: 22 / 12,
PA23: 23 / 13,
PA27: 26
}

#[cfg(feature = "samd21")]
ptc_pins! {
PA02: 0,
PA03: 1,
PA04: 2,
PA05: 3,
PA06: 4,
PA07: 5,
PA08: 16,
PA09: 17,
PA10: 18,
PA11: 19,
PA16: 20,
PA17: 21,
PA18: 22,
PA19: 23,
PA22: 26,
PA23: 27
}

#[cfg(any(feature = "samd21g", feature = "samd21j"))]
ptc_pins! {
PB02: 8,
PB03: 9,
PB08: 14,
PB09: 15,
PA20: 24,
PA21: 25
}

#[cfg(feature = "samd21j")]
ptc_pins! {
PB00: 6,
PB01: 7,
PB04: 10,
PB05: 11,
PB06: 12,
PB07: 13,
PB12: 28,
PB13: 29,
PB14: 30,
PB15: 31
}
1 change: 1 addition & 0 deletions pac/atsamd11c/device.x
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,5 @@ PROVIDE(TC2 = DefaultHandler);
PROVIDE(ADC = DefaultHandler);
PROVIDE(AC = DefaultHandler);
PROVIDE(DAC = DefaultHandler);
PROVIDE(PTC = DefaultHandler);

13 changes: 13 additions & 0 deletions pac/atsamd11c/src/gclk/clkctrl.rs
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,8 @@ pub enum ID_A {
AC_ANA = 21,
#[doc = "22: DAC"]
DAC = 22,
#[doc = "23: PTC"]
PTC = 23,
}
impl From<ID_A> for u8 {
#[inline(always)]
Expand Down Expand Up @@ -98,6 +100,7 @@ impl ID_R {
20 => Val(ID_A::AC_DIG),
21 => Val(ID_A::AC_ANA),
22 => Val(ID_A::DAC),
23 => Val(ID_A::PTC),
i => Res(i),
}
}
Expand Down Expand Up @@ -216,6 +219,11 @@ impl ID_R {
pub fn is_dac(&self) -> bool {
*self == ID_A::DAC
}
#[doc = "Checks if the value of the field is `PTC`"]
#[inline(always)]
pub fn is_ptc(&self) -> bool {
*self == ID_A::PTC
}
}
#[doc = "Write proxy for field `ID`"]
pub struct ID_W<'a> {
Expand Down Expand Up @@ -342,6 +350,11 @@ impl<'a> ID_W<'a> {
pub fn dac(self) -> &'a mut W {
self.variant(ID_A::DAC)
}
#[doc = "PTC"]
#[inline(always)]
pub fn ptc(self) -> &'a mut W {
self.variant(ID_A::PTC)
}
#[doc = r"Writes raw bits to the field"]
#[inline(always)]
pub unsafe fn bits(self, value: u8) -> &'a mut W {
Expand Down
Loading