diff --git a/src/gpio.rs b/src/gpio.rs index 7bb05cef..9c3df234 100644 --- a/src/gpio.rs +++ b/src/gpio.rs @@ -277,13 +277,13 @@ macro_rules! gpio { fn set_high(&mut self) -> Result<(), ()> { // NOTE(unsafe) atomic write to a stateless register - unsafe { (*$GPIOX::ptr()).bsrr().write(|w| w.bits(1 << self.i)) }; + unsafe { (*$GPIOX::ptr()).bsrr().write(|w| w.bs(self.i).set_bit()) }; Ok(()) } fn set_low(&mut self) -> Result<(), ()> { // NOTE(unsafe) atomic write to a stateless register - unsafe { (*$GPIOX::ptr()).bsrr().write(|w| w.bits(1 << (self.i + 16))) }; + unsafe { (*$GPIOX::ptr()).bsrr().write(|w| w.br(self.i).set_bit()) }; Ok(()) } } @@ -296,7 +296,7 @@ macro_rules! gpio { fn is_set_low(&self) -> Result { // NOTE(unsafe) atomic read with no side effects - let is_set_low = unsafe { (*$GPIOX::ptr()).odr().read().bits() & (1 << self.i) == 0 }; + let is_set_low = unsafe { (*$GPIOX::ptr()).odr().read().odr(self.i).is_low() }; Ok(is_set_low) } } @@ -314,7 +314,7 @@ macro_rules! gpio { fn is_low(&self) -> Result { // NOTE(unsafe) atomic read with no side effects - let is_low = unsafe { (*$GPIOX::ptr()).idr().read().bits() & (1 << self.i) == 0 }; + let is_low = unsafe { (*$GPIOX::ptr()).idr().read().idr(self.i).is_low() }; Ok(is_low) } } @@ -329,7 +329,7 @@ macro_rules! gpio { fn is_low(&self) -> Result { // NOTE(unsafe) atomic read with no side effects - let is_low = unsafe { (*$GPIOX::ptr()).idr().read().bits() & (1 << self.i) == 0 }; + let is_low = unsafe { (*$GPIOX::ptr()).idr().read().idr(self.i).is_low() }; Ok(is_low) } } @@ -382,97 +382,89 @@ macro_rules! gpio { impl $PXi { /// Configures the pin to operate as a floating input pin pub fn into_floating_input(self) -> $PXi> { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).floating() }); - gpio.moder().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.moder().modify(|_, w| { + w.moder($i).input() }) - }; + }); $PXi { _mode: PhantomData } } /// Configures the pin to operate as a pulled down input pin pub fn into_pull_down_input(self) -> $PXi> { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b10 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).pull_down() }); - gpio.moder().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) - }) - }; + + }); $PXi { _mode: PhantomData } } /// Configures the pin to operate as a pulled up input pin pub fn into_pull_up_input(self) -> $PXi> { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b01 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).pull_up() }); - gpio.moder().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.moder().modify(|_, w| { + w.moder($i).input() }) - }; + }); $PXi { _mode: PhantomData } } /// Configures the pin to operate as an analog pin pub fn into_analog(self) -> $PXi { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).floating() }); - gpio.moder().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b11 << offset)) - }); - } + gpio.moder().modify(|_, w| { + w.moder($i).analog() + }) + }); $PXi { _mode: PhantomData } } /// Configures the pin to operate as an open drain output pin pub fn into_open_drain_output(self) -> $PXi> { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).floating() }); - gpio.otyper().modify(|r, w| { - w.bits(r.bits() | (0b1 << $i)) + gpio.otyper().modify(|_, w| { + w.ot($i).open_drain() }); - gpio.moder().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b01 << offset)) - }) - }; + gpio.moder().modify(|_, w| { + w.moder($i).output() + }); + }); $PXi { _mode: PhantomData } } /// Configures the pin to operate as an push pull output pin pub fn into_push_pull_output(self) -> $PXi> { - let offset = 2 * $i; - unsafe { + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - gpio.pupdr().modify(|r, w| { - w.bits(r.bits() & !(0b11 << offset)) + gpio.pupdr().modify(|_, w| { + w.pupdr($i).floating() }); - gpio.otyper().modify(|r, w| { - w.bits(r.bits() & !(0b1 << $i)) + gpio.otyper().modify(|_, w| { + w.ot($i).push_pull() }); - gpio.moder().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b01 << offset)) - }) - }; + gpio.moder().modify(|_, w| { + w.moder($i).output() + }); + }); $PXi { _mode: PhantomData } } @@ -484,64 +476,59 @@ macro_rules! gpio { /// Set pin speed pub fn set_speed(self, speed: Speed) -> Self { - let offset = 2 * $i; - unsafe { - (*$GPIOX::ptr()).ospeedr().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | ((speed as u32) << offset)) + cortex_m::interrupt::free(|_| unsafe { + (*$GPIOX::ptr()).ospeedr().modify(|_, w| { + w.ospeedr($i).set(speed as u8) }); - } + }); self } pub fn into_alternate(self) -> $PXi> { - let mode = A as u32; - let offset = 2 * $i; - let offset2 = 4 * $i; - unsafe { + let mode = A as u8; + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - if offset2 < 32 { - gpio.afrl().modify(|r, w| { - w.bits((r.bits() & !(0b1111 << offset2)) | (mode << offset2)) + if $i < 8 { + gpio.afrl().modify(|_, w| { + w.afr($i).set(mode) }); } else { - let offset2 = offset2 - 32; - gpio.afrh().modify(|r, w| { - w.bits((r.bits() & !(0b1111 << offset2)) | (mode << offset2)) + let i = $i - 8; + gpio.afrh().modify(|_, w| { + w.afr(i).set(mode) }); } - gpio.moder().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b10 << offset)) + gpio.moder().modify(|_, w| { + w.moder($i).alternate() }); - gpio.otyper().modify(|r, w| { - w.bits(r.bits() & !(0b1 << $i)) + gpio.otyper().modify(|_, w| { + w.ot($i).push_pull() }); - } + }); $PXi { _mode: PhantomData } } pub fn into_alternate_open_drain(self) -> $PXi> { - let mode = A as u32; - let offset = 2 * $i; - let offset2 = 4 * $i; - unsafe { + let mode = A as u8; + cortex_m::interrupt::free(|_| unsafe { let gpio = &(*$GPIOX::ptr()); - if offset2 < 32 { - gpio.afrl().modify(|r, w| { - w.bits((r.bits() & !(0b1111 << offset2)) | (mode << offset2)) + if $i < 8 { + gpio.afrl().modify(|_, w| { + w.afr($i).set(mode) }); } else { - let offset2 = offset2 - 32; - gpio.afrh().modify(|r, w| { - w.bits((r.bits() & !(0b1111 << offset2)) | (mode << offset2)) + let i = $i - 8; + gpio.afrh().modify(|_, w| { + w.afr(i).set(mode) }); } - gpio.otyper().modify(|r, w| { - w.bits(r.bits() | (0b1 << $i)) + gpio.moder().modify(|_, w| { + w.moder($i).alternate() }); - gpio.moder().modify(|r, w| { - w.bits((r.bits() & !(0b11 << offset)) | (0b10 << offset)) + gpio.otyper().modify(|_, w| { + w.ot($i).open_drain() }); - } + }); $PXi { _mode: PhantomData } } }