|
|
@@ -0,0 +1,478 @@
|
|
|
+
|
|
|
+/*
|
|
|
+ * Licensed to the Apache Software Foundation (ASF) under one
|
|
|
+ * or more contributor license agreements. See the NOTICE file
|
|
|
+ * distributed with this work for additional information
|
|
|
+ * regarding copyright ownership. The ASF licenses this file
|
|
|
+ * to you under the Apache License, Version 2.0 (the
|
|
|
+ * "License"); you may not use this file except in compliance
|
|
|
+ * with the License. You may obtain a copy of the License at
|
|
|
+ *
|
|
|
+ * http://www.apache.org/licenses/LICENSE-2.0
|
|
|
+ *
|
|
|
+ * Unless required by applicable law or agreed to in writing,
|
|
|
+ * software distributed under the License is distributed on an
|
|
|
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
|
+ * KIND, either express or implied. See the License for the
|
|
|
+ * specific language governing permissions and limitations
|
|
|
+ * under the License.
|
|
|
+ */
|
|
|
+
|
|
|
+#include <assert.h>
|
|
|
+#include <stddef.h>
|
|
|
+#include "syscfg/syscfg.h"
|
|
|
+#include "mcu/da1469x_hal.h"
|
|
|
+#include <mcu/mcu.h>
|
|
|
+#include "hal/hal_gpio.h"
|
|
|
+
|
|
|
+/* GPIO interrupts */
|
|
|
+#define HAL_GPIO_MAX_IRQ MYNEWT_VAL(MCU_GPIO_MAX_IRQ)
|
|
|
+
|
|
|
+#define GPIO_REG(name) ((__IO uint32_t *)(GPIO_BASE + offsetof(GPIO_Type, name)))
|
|
|
+#define WAKEUP_REG(name) ((__IO uint32_t *)(WAKEUP_BASE + offsetof(WAKEUP_Type, name)))
|
|
|
+#define CRG_TOP_REG(name) ((__IO uint32_t *)(CRG_TOP_BASE + offsetof(CRG_TOP_Type, name)))
|
|
|
+
|
|
|
+#ifndef MCU_GPIO_PORT0_PIN_COUNT
|
|
|
+#define MCU_GPIO_PORT0_PIN_COUNT 32
|
|
|
+#endif
|
|
|
+
|
|
|
+#if (MCU_GPIO_PORT0_PIN_COUNT) == 32
|
|
|
+#define GPIO_PORT(pin) (((unsigned)(pin)) >> 5U)
|
|
|
+#define GPIO_PORT_PIN(pin) (((unsigned)(pin)) & 31U)
|
|
|
+#else
|
|
|
+#define GPIO_PORT(pin) (((unsigned)(pin)) < MCU_GPIO_PORT0_PIN_COUNT ? 0 : 1)
|
|
|
+#define GPIO_PORT_PIN(pin) ((unsigned)(pin) < MCU_GPIO_PORT0_PIN_COUNT ? \
|
|
|
+ (pin) : (pin) - MCU_GPIO_PORT0_PIN_COUNT)
|
|
|
+#endif
|
|
|
+
|
|
|
+#define GPIO_PIN_BIT(pin) (1 << GPIO_PORT_PIN(pin))
|
|
|
+
|
|
|
+#define GPIO_PIN_DATA_REG_ADDR(pin) (GPIO_REG(P0_DATA_REG) + GPIO_PORT(pin))
|
|
|
+#define GPIO_PIN_DATA_REG(pin) *GPIO_PIN_DATA_REG_ADDR(pin)
|
|
|
+#define GPIO_PIN_SET_DATA_REG_ADDR(pin) (GPIO_REG(P0_SET_DATA_REG) + GPIO_PORT(pin))
|
|
|
+#define GPIO_PIN_SET_DATA_REG(pin) *GPIO_PIN_SET_DATA_REG_ADDR(pin)
|
|
|
+#define GPIO_PIN_RESET_DATA_REG_ADDR(pin) (GPIO_REG(P0_RESET_DATA_REG) + GPIO_PORT(pin))
|
|
|
+#define GPIO_PIN_RESET_DATA_REG(pin) *GPIO_PIN_RESET_DATA_REG_ADDR(pin)
|
|
|
+#define GPIO_PIN_MODE_REG_ADDR(pin) (GPIO_REG(P0_00_MODE_REG) + (pin))
|
|
|
+#define GPIO_PIN_MODE_REG(pin) *GPIO_PIN_MODE_REG_ADDR(pin)
|
|
|
+#define GPIO_PIN_PADPWR_CTRL_REG_ADDR(pin) (GPIO_REG(P0_PADPWR_CTRL_REG) + GPIO_PORT(pin))
|
|
|
+#define GPIO_PIN_PADPWR_CTRL_REG(pin) *GPIO_PIN_PADPWR_CTRL_REG_ADDR(pin)
|
|
|
+#define GPIO_PIN_UNLATCH_ADDR(pin) (CRG_TOP_REG(P0_SET_PAD_LATCH_REG) + GPIO_PORT(pin) * 3)
|
|
|
+#define GPIO_PIN_LATCH_ADDR(pin) (CRG_TOP_REG(P0_RESET_PAD_LATCH_REG) + GPIO_PORT(pin) * 3)
|
|
|
+
|
|
|
+#define WKUP_CTRL_REG_ADDR (WAKEUP_REG(WKUP_CTRL_REG))
|
|
|
+#define WKUP_RESET_IRQ_REG_ADDR (WAKEUP_REG(WKUP_RESET_IRQ_REG))
|
|
|
+#define WKUP_SELECT_PX_REG_ADDR(pin) (WAKEUP_REG(WKUP_SELECT_P0_REG) + GPIO_PORT(pin))
|
|
|
+#define WKUP_SELECT_PX_REG(pin) *(WKUP_SELECT_PX_REG_ADDR(pin))
|
|
|
+#define WKUP_POL_PX_REG_ADDR(pin) (WAKEUP_REG(WKUP_POL_P0_REG) + GPIO_PORT(pin))
|
|
|
+#define WKUP_POL_PX_SET_FALLING(pin) do { *(WKUP_POL_PX_REG_ADDR(pin)) |= GPIO_PIN_BIT(pin); } while (0)
|
|
|
+#define WKUP_POL_PX_SET_RISING(pin) do { *(WKUP_POL_PX_REG_ADDR(pin)) &= ~GPIO_PIN_BIT(pin); } while (0)
|
|
|
+#define WKUP_STAT_PX_REG_ADDR(pin) (WAKEUP_REG(WKUP_STATUS_P0_REG) + GPIO_PORT(pin))
|
|
|
+#define WKUP_STAT(pin) ((*(WKUP_STAT_PX_REG_ADDR(pin)) >> GPIO_PORT_PIN(pin)) & 1)
|
|
|
+#define WKUP_CLEAR_PX_REG_ADDR(pin) (WAKEUP_REG(WKUP_CLEAR_P0_REG) + GPIO_PORT(pin))
|
|
|
+#define WKUP_CLEAR_PX(pin) do { (*(WKUP_CLEAR_PX_REG_ADDR(pin)) = GPIO_PIN_BIT(pin)); } while (0)
|
|
|
+#define WKUP_SEL_GPIO_PX_REG_ADDR(pin) (WAKEUP_REG(WKUP_SEL_GPIO_P0_REG) + GPIO_PORT(pin))
|
|
|
+#define WKUP_SEL_GPIO_PX_REG(pin) *(WKUP_SEL_GPIO_PX_REG_ADDR(pin))
|
|
|
+
|
|
|
+/* Storage for GPIO callbacks. */
|
|
|
+struct hal_gpio_irq {
|
|
|
+ int pin;
|
|
|
+ hal_gpio_irq_handler_t func;
|
|
|
+ void *arg;
|
|
|
+};
|
|
|
+
|
|
|
+static struct hal_gpio_irq hal_gpio_irqs[HAL_GPIO_MAX_IRQ];
|
|
|
+
|
|
|
+#if MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM) >= 0
|
|
|
+static uint32_t g_mcu_gpio_latch_state[2];
|
|
|
+static uint8_t g_mcu_gpio_retained_num;
|
|
|
+static struct da1469x_retreg g_mcu_gpio_retained[MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM)];
|
|
|
+#endif
|
|
|
+
|
|
|
+/*
|
|
|
+ * We assume that any latched pin has default configuration, i.e. was either
|
|
|
+ * not configured or was deinited. Any unlatched pin is considered to be used
|
|
|
+ * by someone.
|
|
|
+ *
|
|
|
+ * By default, all pins are assumed to have default configuration and are
|
|
|
+ * latched. This allows PD_COM to be disabled (if no other peripheral needs
|
|
|
+ * it) since we do not need GPIO mux to be active.
|
|
|
+ *
|
|
|
+ * Configuration of any pin shall be done as follows, with interrupts disabled:
|
|
|
+ * 1. call mcu_gpio_unlatch_prepare() to enable PD_COM if needed
|
|
|
+ * 2. configure pin
|
|
|
+ * 3. call mcu_gpio_unlatch() to actually unlatch pin
|
|
|
+ *
|
|
|
+ * Once pin is restored to default configuration it shall be latched again by
|
|
|
+ * calling mcu_gpio_latch().
|
|
|
+ */
|
|
|
+
|
|
|
+#if MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM) >= 0
|
|
|
+static void
|
|
|
+mcu_gpio_retained_add_port(uint32_t latch_val, volatile uint32_t *base_reg)
|
|
|
+{
|
|
|
+ struct da1469x_retreg *retreg;
|
|
|
+ int pin;
|
|
|
+
|
|
|
+ retreg = &g_mcu_gpio_retained[g_mcu_gpio_retained_num];
|
|
|
+
|
|
|
+ while (latch_val) {
|
|
|
+ assert(g_mcu_gpio_retained_num < MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM));
|
|
|
+
|
|
|
+ pin = __builtin_ctz(latch_val);
|
|
|
+ latch_val &= ~(1 << pin);
|
|
|
+
|
|
|
+ da1469x_retreg_assign(retreg, &base_reg[pin]);
|
|
|
+
|
|
|
+ g_mcu_gpio_retained_num++;
|
|
|
+ retreg++;
|
|
|
+ }
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+static void
|
|
|
+mcu_gpio_retained_refresh(void)
|
|
|
+{
|
|
|
+#if MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM) >= 0
|
|
|
+ g_mcu_gpio_retained_num = 0;
|
|
|
+
|
|
|
+ mcu_gpio_retained_add_port(CRG_TOP->P0_PAD_LATCH_REG, &GPIO->P0_00_MODE_REG);
|
|
|
+ mcu_gpio_retained_add_port(CRG_TOP->P1_PAD_LATCH_REG, &GPIO->P1_00_MODE_REG);
|
|
|
+#endif
|
|
|
+}
|
|
|
+
|
|
|
+static inline void
|
|
|
+mcu_gpio_unlatch_prepare(int pin)
|
|
|
+{
|
|
|
+ __HAL_ASSERT_CRITICAL();
|
|
|
+ (void)pin;
|
|
|
+
|
|
|
+ /* Acquire PD_COM if first pin will be unlatched */
|
|
|
+// if ((CRG_TOP->P0_PAD_LATCH_REG | CRG_TOP->P1_PAD_LATCH_REG) == 0) {
|
|
|
+// da1469x_pd_acquire(MCU_PD_DOMAIN_COM);
|
|
|
+// }
|
|
|
+}
|
|
|
+
|
|
|
+static inline void
|
|
|
+mcu_gpio_unlatch(int pin)
|
|
|
+{
|
|
|
+ __HAL_ASSERT_CRITICAL();
|
|
|
+
|
|
|
+ *GPIO_PIN_UNLATCH_ADDR(pin) = GPIO_PIN_BIT(pin);
|
|
|
+ mcu_gpio_retained_refresh();
|
|
|
+}
|
|
|
+
|
|
|
+static inline void
|
|
|
+mcu_gpio_latch(int pin)
|
|
|
+{
|
|
|
+ (void)pin;
|
|
|
+// uint32_t primask;
|
|
|
+// uint32_t latch_pre;
|
|
|
+// uint32_t latch_post;
|
|
|
+//
|
|
|
+// __HAL_DISABLE_INTERRUPTS(primask);
|
|
|
+//
|
|
|
+// latch_pre = CRG_TOP->P0_PAD_LATCH_REG | CRG_TOP->P1_PAD_LATCH_REG;
|
|
|
+//
|
|
|
+// *GPIO_PIN_LATCH_ADDR(pin) = GPIO_PIN_BIT(pin);
|
|
|
+// mcu_gpio_retained_refresh();
|
|
|
+//
|
|
|
+// latch_post = CRG_TOP->P0_PAD_LATCH_REG | CRG_TOP->P1_PAD_LATCH_REG;
|
|
|
+//
|
|
|
+// /* Release PD_COM if last pin was latched */
|
|
|
+// if (latch_pre && !latch_post) {
|
|
|
+// da1469x_pd_release(MCU_PD_DOMAIN_COM);
|
|
|
+// }
|
|
|
+//
|
|
|
+// __HAL_ENABLE_INTERRUPTS(primask);
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_init_in(int pin, hal_gpio_pull_t pull)
|
|
|
+{
|
|
|
+ volatile uint32_t *px_xx_mod_reg = GPIO_PIN_MODE_REG_ADDR(pin);
|
|
|
+ uint32_t regval;
|
|
|
+ uint32_t primask;
|
|
|
+
|
|
|
+ switch (pull) {
|
|
|
+ case HAL_GPIO_PULL_UP:
|
|
|
+ regval = MCU_GPIO_FUNC_GPIO | MCU_GPIO_MODE_INPUT_PULLUP;
|
|
|
+ break;
|
|
|
+ case HAL_GPIO_PULL_DOWN:
|
|
|
+ regval = MCU_GPIO_FUNC_GPIO | MCU_GPIO_MODE_INPUT_PULLDOWN;
|
|
|
+ break;
|
|
|
+ case HAL_GPIO_PULL_NONE:
|
|
|
+ regval = MCU_GPIO_FUNC_GPIO | MCU_GPIO_MODE_INPUT;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+
|
|
|
+ __HAL_DISABLE_INTERRUPTS(primask);
|
|
|
+
|
|
|
+ mcu_gpio_unlatch_prepare(pin);
|
|
|
+
|
|
|
+ *px_xx_mod_reg = regval;
|
|
|
+
|
|
|
+ mcu_gpio_unlatch(pin);
|
|
|
+
|
|
|
+ __HAL_ENABLE_INTERRUPTS(primask);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_init_out(int pin, int val)
|
|
|
+{
|
|
|
+ uint32_t primask;
|
|
|
+
|
|
|
+ __HAL_DISABLE_INTERRUPTS(primask);
|
|
|
+
|
|
|
+ mcu_gpio_unlatch_prepare(pin);
|
|
|
+
|
|
|
+ GPIO_PIN_MODE_REG(pin) = MCU_GPIO_MODE_OUTPUT;
|
|
|
+
|
|
|
+ if (val) {
|
|
|
+ GPIO_PIN_SET_DATA_REG(pin) = GPIO_PIN_BIT(pin);
|
|
|
+ } else {
|
|
|
+ GPIO_PIN_RESET_DATA_REG(pin) = GPIO_PIN_BIT(pin);
|
|
|
+ }
|
|
|
+
|
|
|
+ mcu_gpio_unlatch(pin);
|
|
|
+
|
|
|
+ __HAL_ENABLE_INTERRUPTS(primask);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_deinit(int pin)
|
|
|
+{
|
|
|
+ /* Reset mode to default value and latch pin */
|
|
|
+ GPIO_PIN_MODE_REG(pin) = 0x200;
|
|
|
+ GPIO_PIN_RESET_DATA_REG(pin) = GPIO_PIN_BIT(pin);
|
|
|
+
|
|
|
+ mcu_gpio_latch(pin);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+hal_gpio_write(int pin, int val)
|
|
|
+{
|
|
|
+ if (val) {
|
|
|
+ GPIO_PIN_SET_DATA_REG(pin) = GPIO_PIN_BIT(pin);
|
|
|
+ } else {
|
|
|
+ GPIO_PIN_RESET_DATA_REG(pin) = GPIO_PIN_BIT(pin);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_read(int pin)
|
|
|
+{
|
|
|
+ return (GPIO_PIN_DATA_REG(pin) >> GPIO_PORT_PIN(pin)) & 1;
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_toggle(int pin)
|
|
|
+{
|
|
|
+ int new_value = hal_gpio_read(pin) == 0;
|
|
|
+
|
|
|
+ hal_gpio_write(pin, new_value);
|
|
|
+
|
|
|
+ return new_value;
|
|
|
+}
|
|
|
+
|
|
|
+static void
|
|
|
+hal_gpio_irq_handler(void)
|
|
|
+{
|
|
|
+ struct hal_gpio_irq *irq;
|
|
|
+ uint32_t stat;
|
|
|
+ int i;
|
|
|
+
|
|
|
+ *WKUP_RESET_IRQ_REG_ADDR = 1;
|
|
|
+ NVIC_ClearPendingIRQ(KEY_WKUP_GPIO_IRQn);
|
|
|
+
|
|
|
+ for (i = 0; i < HAL_GPIO_MAX_IRQ; i++) {
|
|
|
+ irq = &hal_gpio_irqs[i];
|
|
|
+
|
|
|
+ /* Read latched status value from relevant GPIO port */
|
|
|
+ stat = WKUP_STAT(irq->pin);
|
|
|
+
|
|
|
+ if (irq->func && stat) {
|
|
|
+ irq->func(irq->arg);
|
|
|
+ }
|
|
|
+
|
|
|
+ WKUP_CLEAR_PX(irq->pin);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void
|
|
|
+hal_gpio_irq_setup(void)
|
|
|
+{
|
|
|
+ static uint8_t irq_setup;
|
|
|
+ int sr;
|
|
|
+
|
|
|
+ if (!irq_setup) {
|
|
|
+ __HAL_DISABLE_INTERRUPTS(sr);
|
|
|
+
|
|
|
+ irq_setup = 1;
|
|
|
+
|
|
|
+ NVIC_ClearPendingIRQ(GPIO_P0_IRQn);
|
|
|
+ NVIC_ClearPendingIRQ(GPIO_P1_IRQn);
|
|
|
+ NVIC_SetVector(GPIO_P0_IRQn, (uint32_t)hal_gpio_irq_handler);
|
|
|
+ NVIC_SetVector(GPIO_P1_IRQn, (uint32_t)hal_gpio_irq_handler);
|
|
|
+ WAKEUP->WKUP_CTRL_REG = 0;
|
|
|
+ WAKEUP->WKUP_CLEAR_P0_REG = 0xFFFFFFFF;
|
|
|
+ WAKEUP->WKUP_CLEAR_P1_REG = 0x007FFFFF;
|
|
|
+ WAKEUP->WKUP_SELECT_P0_REG = 0;
|
|
|
+ WAKEUP->WKUP_SELECT_P1_REG = 0;
|
|
|
+ WAKEUP->WKUP_SEL_GPIO_P0_REG = 0;
|
|
|
+ WAKEUP->WKUP_SEL_GPIO_P1_REG = 0;
|
|
|
+ WAKEUP->WKUP_RESET_IRQ_REG = 0;
|
|
|
+
|
|
|
+ CRG_TOP->CLK_TMR_REG |= CRG_TOP_CLK_TMR_REG_WAKEUPCT_ENABLE_Msk;
|
|
|
+
|
|
|
+ __HAL_ENABLE_INTERRUPTS(sr);
|
|
|
+ NVIC_EnableIRQ(GPIO_P0_IRQn);
|
|
|
+ NVIC_EnableIRQ(GPIO_P1_IRQn);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static int
|
|
|
+hal_gpio_find_empty_slot(void)
|
|
|
+{
|
|
|
+ int i;
|
|
|
+
|
|
|
+ for (i = 0; i < HAL_GPIO_MAX_IRQ; i++) {
|
|
|
+ if (hal_gpio_irqs[i].func == NULL) {
|
|
|
+ return i;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return -1;
|
|
|
+}
|
|
|
+
|
|
|
+int
|
|
|
+hal_gpio_irq_init(int pin, hal_gpio_irq_handler_t handler, void *arg,
|
|
|
+ hal_gpio_irq_trig_t trig, hal_gpio_pull_t pull)
|
|
|
+{
|
|
|
+ int i;
|
|
|
+
|
|
|
+ hal_gpio_irq_setup();
|
|
|
+
|
|
|
+ i = hal_gpio_find_empty_slot();
|
|
|
+ /* If assert failed increase syscfg value MCU_GPIO_MAX_IRQ */
|
|
|
+ assert(i >= 0);
|
|
|
+ if (i < 0) {
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+
|
|
|
+ hal_gpio_init_in(pin, pull);
|
|
|
+
|
|
|
+ switch (trig) {
|
|
|
+ case HAL_GPIO_TRIG_RISING:
|
|
|
+ WKUP_POL_PX_SET_RISING(pin);
|
|
|
+ break;
|
|
|
+ case HAL_GPIO_TRIG_FALLING:
|
|
|
+ WKUP_POL_PX_SET_FALLING(pin);
|
|
|
+ break;
|
|
|
+ case HAL_GPIO_TRIG_BOTH:
|
|
|
+ /* Not supported */
|
|
|
+ default:
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+
|
|
|
+ hal_gpio_irqs[i].pin = pin;
|
|
|
+ hal_gpio_irqs[i].func = handler;
|
|
|
+ hal_gpio_irqs[i].arg = arg;
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+hal_gpio_irq_release(int pin)
|
|
|
+{
|
|
|
+ int i;
|
|
|
+
|
|
|
+ hal_gpio_irq_disable(pin);
|
|
|
+
|
|
|
+ for (i = 0; i < HAL_GPIO_MAX_IRQ; i++) {
|
|
|
+ if (hal_gpio_irqs[i].pin == pin && hal_gpio_irqs[i].func) {
|
|
|
+ hal_gpio_irqs[i].pin = -1;
|
|
|
+ hal_gpio_irqs[i].arg = NULL;
|
|
|
+ hal_gpio_irqs[i].func = NULL;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+hal_gpio_irq_enable(int pin)
|
|
|
+{
|
|
|
+ WKUP_SEL_GPIO_PX_REG(pin) |= GPIO_PIN_BIT(pin);
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+hal_gpio_irq_disable(int pin)
|
|
|
+{
|
|
|
+ WKUP_SEL_GPIO_PX_REG(pin) &= ~GPIO_PIN_BIT(pin);
|
|
|
+ WKUP_CLEAR_PX(pin);
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+mcu_gpio_set_pin_function(int pin, int mode, mcu_gpio_func func)
|
|
|
+{
|
|
|
+ uint32_t primask;
|
|
|
+
|
|
|
+ __HAL_DISABLE_INTERRUPTS(primask);
|
|
|
+
|
|
|
+ mcu_gpio_unlatch_prepare(pin);
|
|
|
+
|
|
|
+ GPIO_PIN_MODE_REG(pin) = (func & GPIO_P0_00_MODE_REG_PID_Msk) |
|
|
|
+ (mode & (GPIO_P0_00_MODE_REG_PUPD_Msk | GPIO_P0_00_MODE_REG_PPOD_Msk));
|
|
|
+
|
|
|
+ mcu_gpio_unlatch(pin);
|
|
|
+
|
|
|
+ __HAL_ENABLE_INTERRUPTS(primask);
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+mcu_gpio_enter_sleep(void)
|
|
|
+{
|
|
|
+#if MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM) >= 0
|
|
|
+ if (g_mcu_gpio_retained_num == 0) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ g_mcu_gpio_latch_state[0] = CRG_TOP->P0_PAD_LATCH_REG;
|
|
|
+ g_mcu_gpio_latch_state[1] = CRG_TOP->P1_PAD_LATCH_REG;
|
|
|
+
|
|
|
+ da1469x_retreg_update(g_mcu_gpio_retained, g_mcu_gpio_retained_num);
|
|
|
+
|
|
|
+ CRG_TOP->P0_RESET_PAD_LATCH_REG = CRG_TOP_P0_PAD_LATCH_REG_P0_LATCH_EN_Msk;
|
|
|
+ CRG_TOP->P1_RESET_PAD_LATCH_REG = CRG_TOP_P1_PAD_LATCH_REG_P1_LATCH_EN_Msk;
|
|
|
+
|
|
|
+ da1469x_pd_release(MCU_PD_DOMAIN_COM);
|
|
|
+#endif
|
|
|
+}
|
|
|
+
|
|
|
+void
|
|
|
+mcu_gpio_exit_sleep(void)
|
|
|
+{
|
|
|
+#if MYNEWT_VAL(MCU_GPIO_RETAINABLE_NUM) >= 0
|
|
|
+ if (g_mcu_gpio_retained_num == 0) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ da1469x_pd_acquire(MCU_PD_DOMAIN_COM);
|
|
|
+
|
|
|
+ da1469x_retreg_restore(g_mcu_gpio_retained, g_mcu_gpio_retained_num);
|
|
|
+
|
|
|
+ /* Set pins states to their latched values */
|
|
|
+ GPIO->P0_DATA_REG = GPIO->P0_DATA_REG;
|
|
|
+ GPIO->P1_DATA_REG = GPIO->P1_DATA_REG;
|
|
|
+
|
|
|
+ CRG_TOP->P0_PAD_LATCH_REG = g_mcu_gpio_latch_state[0];
|
|
|
+ CRG_TOP->P1_PAD_LATCH_REG = g_mcu_gpio_latch_state[1];
|
|
|
+#endif
|
|
|
+}
|