summaryrefslogtreecommitdiffstats
path: root/watch-library/hardware/watch/watch_deepsleep.c
diff options
context:
space:
mode:
authorAlexsander Akers <me@a2.io>2022-01-25 15:03:22 -0500
committerGitHub <noreply@github.com>2022-01-25 15:03:22 -0500
commitb8de35658ffd78ad8b22f91ccbbd3d63663afda9 (patch)
tree1f265ddfcc8e5abf0316b81b15f80bf5c70fa7b7 /watch-library/hardware/watch/watch_deepsleep.c
parent9e24f6c336773c7404139ab4db0eaab2f99504e2 (diff)
downloadSensor-Watch-b8de35658ffd78ad8b22f91ccbbd3d63663afda9.tar.gz
Sensor-Watch-b8de35658ffd78ad8b22f91ccbbd3d63663afda9.tar.bz2
Sensor-Watch-b8de35658ffd78ad8b22f91ccbbd3d63663afda9.zip
Sensor Watch Simulator (#35)
* Put something on screen * Use the 32bit watch_date_time repr to pass from JS * Implement periodic callbacks * Clear display on enabling * Hook up watch_set_led_color() to SVG (green-only) * Make debug output full-width * Remove default Emscripten canvas * Implement sleep and button clicks * Fix time zone conversion bug in beats-time app * Clean up warnings * Fix pin levels * Set time zone to browser value (if available) * Add basic backup data saving * Silence format specifier warnings in both targets * Remove unnecessary, copied files * Use RTC pointer to clear callbacks (if available) * Use preprocessor define to avoid hardcoding MOVEMENT_NUM_FACES * Change each face to const preprocessor definition * Remove Intl.DateTimeFormat usage * Update shell.html title, header * Add touch start/end event handlers on SVG buttons * Update shell.html * Update folder structure (shared, simulator, hardware under watch-library) * Tease out shared components from watch_slcd * Clean up simulator watch_slcd.c inline JS calls * Fix missing newlines at end of file * Add simulator warnings (except format, unused-paremter) * Implement remaining watch_rtc functions * Fix button bug on mouse down then drag out * Implement remaining watch_slcd functions * Link keyboard events to buttons (for keys A, L, M) * Rewrite event handling (mouse, touch, keyboard) in C * Set explicit text UTF-8 charset in shell.html * Address PR comments * Remove unused directories from include paths
Diffstat (limited to 'watch-library/hardware/watch/watch_deepsleep.c')
-rw-r--r--watch-library/hardware/watch/watch_deepsleep.c209
1 files changed, 209 insertions, 0 deletions
diff --git a/watch-library/hardware/watch/watch_deepsleep.c b/watch-library/hardware/watch/watch_deepsleep.c
new file mode 100644
index 00000000..1813ff24
--- /dev/null
+++ b/watch-library/hardware/watch/watch_deepsleep.c
@@ -0,0 +1,209 @@
+/*
+ * MIT License
+ *
+ * Copyright (c) 2020 Joey Castillo
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in all
+ * copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+ * SOFTWARE.
+ */
+
+#include "watch_extint.h"
+
+// this warning only appears when you `make BOARD=OSO-SWAT-A1-02`. it's annoying,
+// but i'd rather have it warn us at build-time than fail silently at run-time.
+// besides, no one but me really has any of these boards anyway.
+#if BTN_ALARM != GPIO(GPIO_PORTA, 2)
+#warning This board revision does not support external wake on BTN_ALARM, so watch_register_extwake_callback will not work with it. Use watch_register_interrupt_callback instead.
+#endif
+
+void watch_register_extwake_callback(uint8_t pin, ext_irq_cb_t callback, bool level) {
+ uint32_t pinmux;
+ hri_rtc_tampctrl_reg_t config = RTC->MODE2.TAMPCTRL.reg;
+
+ switch (pin) {
+ case A4:
+ a4_callback = callback;
+ pinmux = PINMUX_PB00G_RTC_IN0;
+ config &= ~(3 << RTC_TAMPCTRL_IN0ACT_Pos);
+ config &= ~(1 << RTC_TAMPCTRL_TAMLVL0_Pos);
+ config |= 1 << RTC_TAMPCTRL_IN0ACT_Pos;
+ config |= 1 << RTC_TAMPCTRL_DEBNC0_Pos;
+ if (level) config |= 1 << RTC_TAMPCTRL_TAMLVL0_Pos;
+ break;
+ case A2:
+ a2_callback = callback;
+ pinmux = PINMUX_PB02G_RTC_IN1;
+ config &= ~(3 << RTC_TAMPCTRL_IN1ACT_Pos);
+ config &= ~(1 << RTC_TAMPCTRL_TAMLVL1_Pos);
+ config |= 1 << RTC_TAMPCTRL_IN1ACT_Pos;
+ config |= 1 << RTC_TAMPCTRL_DEBNC1_Pos;
+ if (level) config |= 1 << RTC_TAMPCTRL_TAMLVL1_Pos;
+ break;
+ case BTN_ALARM:
+ gpio_set_pin_pull_mode(pin, GPIO_PULL_DOWN);
+ btn_alarm_callback = callback;
+ pinmux = PINMUX_PA02G_RTC_IN2;
+ config &= ~(3 << RTC_TAMPCTRL_IN2ACT_Pos);
+ config &= ~(1 << RTC_TAMPCTRL_TAMLVL2_Pos);
+ config |= 1 << RTC_TAMPCTRL_IN2ACT_Pos;
+ config |= 1 << RTC_TAMPCTRL_DEBNC2_Pos;
+ if (level) config |= 1 << RTC_TAMPCTRL_TAMLVL2_Pos;
+ break;
+ default:
+ return;
+ }
+ gpio_set_pin_direction(pin, GPIO_DIRECTION_IN);
+ gpio_set_pin_function(pin, pinmux);
+
+ // disable the RTC
+ RTC->MODE2.CTRLA.bit.ENABLE = 0;
+ while (RTC->MODE2.SYNCBUSY.bit.ENABLE);
+
+ // update the configuration
+ RTC->MODE2.TAMPCTRL.reg = config;
+ // re-enable the RTC
+ RTC->MODE2.CTRLA.bit.ENABLE = 1;
+
+ NVIC_ClearPendingIRQ(RTC_IRQn);
+ NVIC_EnableIRQ(RTC_IRQn);
+ RTC->MODE2.INTENSET.reg = RTC_MODE2_INTENSET_TAMPER;
+}
+
+void watch_disable_extwake_interrupt(uint8_t pin) {
+ hri_rtc_tampctrl_reg_t config = hri_rtc_get_TAMPCTRL_reg(RTC, 0xFFFFFFFF);
+
+ switch (pin) {
+ case A4:
+ a4_callback = NULL;
+ config &= ~(3 << RTC_TAMPCTRL_IN0ACT_Pos);
+ break;
+ case A2:
+ a2_callback = NULL;
+ config &= ~(3 << RTC_TAMPCTRL_IN1ACT_Pos);
+ break;
+ case BTN_ALARM:
+ btn_alarm_callback = NULL;
+ config &= ~(3 << RTC_TAMPCTRL_IN2ACT_Pos);
+ break;
+ default:
+ return;
+ }
+
+ if (hri_rtcmode0_get_CTRLA_ENABLE_bit(RTC)) {
+ hri_rtcmode0_clear_CTRLA_ENABLE_bit(RTC);
+ hri_rtcmode0_wait_for_sync(RTC, RTC_MODE0_SYNCBUSY_ENABLE);
+ }
+ hri_rtc_write_TAMPCTRL_reg(RTC, config);
+ hri_rtcmode0_set_CTRLA_ENABLE_bit(RTC);
+}
+
+void watch_store_backup_data(uint32_t data, uint8_t reg) {
+ if (reg < 8) {
+ RTC->MODE0.BKUP[reg].reg = data;
+ }
+}
+
+uint32_t watch_get_backup_data(uint8_t reg) {
+ if (reg < 8) {
+ return RTC->MODE0.BKUP[reg].reg;
+ }
+
+ return 0;
+}
+
+static void _watch_disable_all_pins_except_rtc(void) {
+ uint32_t config = RTC->MODE0.TAMPCTRL.reg;
+ uint32_t portb_pins_to_disable = 0xFFFFFFFF;
+
+ // if there's an action set on RTC/IN[0], leave PB00 configured
+ if (config & RTC_TAMPCTRL_IN0ACT_Msk) portb_pins_to_disable &= 0xFFFFFFFE;
+ // same with RTC/IN[1] and PB02
+ if (config & RTC_TAMPCTRL_IN1ACT_Msk) portb_pins_to_disable &= 0xFFFFFFFB;
+
+ // port A: always keep PA02 configured as-is; that's our ALARM button.
+ gpio_set_port_direction(0, 0xFFFFFFFB, GPIO_DIRECTION_OFF);
+ // port B: disable all pins we didn't save above.
+ gpio_set_port_direction(1, portb_pins_to_disable, GPIO_DIRECTION_OFF);
+}
+
+static void _watch_disable_all_peripherals_except_slcd(void) {
+ _watch_disable_tcc();
+ watch_disable_adc();
+ watch_disable_external_interrupts();
+ watch_disable_i2c();
+ // TODO: replace this with a proper function when we remove the debug UART
+ SERCOM3->USART.CTRLA.reg &= ~SERCOM_USART_CTRLA_ENABLE;
+ MCLK->APBCMASK.reg &= ~MCLK_APBCMASK_SERCOM3;
+}
+
+void watch_enter_sleep_mode(void) {
+ // disable all other peripherals
+ _watch_disable_all_peripherals_except_slcd();
+
+ // disable tick interrupt
+ watch_rtc_disable_all_periodic_callbacks();
+
+ // disable brownout detector interrupt, which could inadvertently wake us up.
+ SUPC->INTENCLR.bit.BOD33DET = 1;
+
+ // disable all pins
+ _watch_disable_all_pins_except_rtc();
+
+ // enter standby (4); we basically hang out here until an interrupt wakes us.
+ sleep(4);
+
+ // and we awake! re-enable the brownout detector
+ SUPC->INTENSET.bit.BOD33DET = 1;
+
+ // call app_setup so the app can re-enable everything we disabled.
+ app_setup();
+
+ // and call app_wake_from_standby (since main won't have a chance to do it)
+ app_wake_from_standby();
+}
+
+void watch_enter_deep_sleep_mode(void) {
+ // identical to sleep mode except we disable the LCD first.
+ slcd_sync_deinit(&SEGMENT_LCD_0);
+ hri_mclk_clear_APBCMASK_SLCD_bit(SLCD);
+
+ watch_enter_sleep_mode();
+}
+
+void watch_enter_backup_mode(void) {
+ watch_rtc_disable_all_periodic_callbacks();
+ _watch_disable_all_peripherals_except_slcd();
+ slcd_sync_deinit(&SEGMENT_LCD_0);
+ hri_mclk_clear_APBCMASK_SLCD_bit(SLCD);
+ _watch_disable_all_pins_except_rtc();
+
+ // go into backup sleep mode (5). when we exit, the reset controller will take over.
+ sleep(5);
+}
+
+// deprecated
+void watch_enter_shallow_sleep(bool display_on) {
+ if (display_on) watch_enter_sleep_mode();
+ else watch_enter_deep_sleep_mode();
+}
+
+// deprecated
+void watch_enter_deep_sleep(void) {
+ watch_register_extwake_callback(BTN_ALARM, NULL, true);
+ watch_enter_backup_mode();
+}