blob: 54dd21b9624ff4c1dbd13ca213df73895b65e5b4 [file] [log] [blame]
/*
* Copyright (c) 2024 Nordic Semiconductor ASA
* SPDX-License-Identifier: Apache-2.0
*/
#define DT_DRV_COMPAT nordic_nrf_bellboard_rx
#include <zephyr/devicetree.h>
#include <zephyr/drivers/mbox.h>
#include <zephyr/sys/util.h>
#include <zephyr/sys/__assert.h>
#include <hal/nrf_bellboard.h>
#define BELLBOARD_NUM_IRQS 4U
BUILD_ASSERT(DT_NUM_IRQS(DT_DRV_INST(0)) <= BELLBOARD_NUM_IRQS, "# interrupt exceeds maximum");
BUILD_ASSERT((DT_INST_PROP_LEN(0, nordic_interrupt_mapping) % 2) == 0,
"# interrupt mappings not specified in pairs");
/* BELLBOARD event mappings */
#define EVT_MAPPING_ITEM(idx) DT_INST_PROP_BY_IDX(0, nordic_interrupt_mapping, idx)
#define BELLBOARD_GET_EVT_MAPPING(idx, _) \
COND_CODE_1( \
DT_INST_PROP_HAS_IDX(0, nordic_interrupt_mapping, UTIL_INC(UTIL_X2(idx))), \
([EVT_MAPPING_ITEM(UTIL_INC(UTIL_X2(idx)))] = EVT_MAPPING_ITEM(UTIL_X2(idx)),), \
())
static const uint32_t evt_mappings[BELLBOARD_NUM_IRQS] = {
LISTIFY(DT_NUM_IRQS(DT_DRV_INST(0)), BELLBOARD_GET_EVT_MAPPING, ())};
/* BELLBOARD instance */
static NRF_BELLBOARD_Type *bellboard = (NRF_BELLBOARD_Type *)DT_INST_REG_ADDR(0);
/* BELLBOARD runtime resources */
static mbox_callback_t cbs[NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT];
static void *cbs_ctx[NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT];
static uint32_t evt_enabled_masks[BELLBOARD_NUM_IRQS];
static void bellboard_rx_isr(const void *parameter)
{
uint8_t irq_idx = (uint8_t)(uintptr_t)parameter;
uint32_t int_pend;
int_pend = nrf_bellboard_int_pending_get(bellboard, irq_idx);
for (uint8_t i = 0U; i < NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT; i++) {
nrf_bellboard_event_t event = nrf_bellboard_triggered_event_get(i);
if ((int_pend & BIT(i)) != 0U) {
/* Only clear those events that have their corresponding bit set
* in INTPEND at the time we read it. Otherwise, if two (or more)
* events are generated in quick succession, INTPEND may be set for
* only one of events, but we clear the EVENTS_TRIGGERED bit for
* all of them, thus losing them.
*
* Assume nrf_bellboard_event_check() is true for the event
* that raised this interrupt.
*/
__ASSERT_NO_MSG(nrf_bellboard_event_check(bellboard, event));
nrf_bellboard_event_clear(bellboard, event);
if (cbs[i] != NULL) {
cbs[i](DEVICE_DT_INST_GET(0), i, cbs_ctx[i], NULL);
}
}
}
}
static uint32_t bellboard_rx_max_channels_get(const struct device *dev)
{
ARG_UNUSED(dev);
return NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT;
}
static int bellboard_rx_register_callback(const struct device *dev, uint32_t id, mbox_callback_t cb,
void *user_data)
{
ARG_UNUSED(dev);
if (id >= NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT) {
return -EINVAL;
}
cbs[id] = cb;
cbs_ctx[id] = user_data;
return 0;
}
static int bellboard_rx_set_enabled(const struct device *dev, uint32_t id, bool enable)
{
bool valid_found = false;
ARG_UNUSED(dev);
if (id >= NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT) {
return -EINVAL;
}
for (uint8_t i = 0U; i < BELLBOARD_NUM_IRQS; i++) {
uint32_t *evt_enabled_mask;
if ((evt_mappings[i] == 0U) || ((evt_mappings[i] & BIT(id)) == 0U)) {
continue;
}
valid_found = true;
evt_enabled_mask = &evt_enabled_masks[i];
if (enable) {
if ((*evt_enabled_mask & BIT(id)) != 0U) {
return -EALREADY;
}
*evt_enabled_mask |= BIT(id);
nrf_bellboard_int_enable(bellboard, i, BIT(id));
} else {
if ((*evt_enabled_mask & BIT(id)) == 0U) {
return -EALREADY;
}
*evt_enabled_mask &= ~BIT(id);
nrf_bellboard_int_disable(bellboard, i, BIT(id));
}
}
if (!valid_found) {
return -EINVAL;
}
return 0;
}
static const struct mbox_driver_api bellboard_rx_driver_api = {
.max_channels_get = bellboard_rx_max_channels_get,
.register_callback = bellboard_rx_register_callback,
.set_enabled = bellboard_rx_set_enabled,
};
#define BELLBOARD_IRQ_CONFIGURE(name, idx) \
COND_CODE_1(DT_INST_IRQ_HAS_NAME(0, name), \
(IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, name, irq), \
DT_INST_IRQ_BY_NAME(0, name, priority), bellboard_rx_isr, \
(const void *)idx, 0); \
irq_enable(DT_INST_IRQ_BY_NAME(0, name, irq));), \
())
static int bellboard_rx_init(const struct device *dev)
{
uint32_t evt_all_mappings =
evt_mappings[0] | evt_mappings[1] | evt_mappings[2] | evt_mappings[3];
ARG_UNUSED(dev);
nrf_bellboard_int_disable(bellboard, 0, evt_mappings[0]);
nrf_bellboard_int_disable(bellboard, 1, evt_mappings[1]);
nrf_bellboard_int_disable(bellboard, 2, evt_mappings[2]);
nrf_bellboard_int_disable(bellboard, 3, evt_mappings[3]);
for (uint8_t i = 0U; i < NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT; i++) {
if ((evt_all_mappings & BIT(i)) != 0U) {
nrf_bellboard_event_clear(bellboard, nrf_bellboard_triggered_event_get(i));
}
}
BELLBOARD_IRQ_CONFIGURE(irq0, 0);
BELLBOARD_IRQ_CONFIGURE(irq1, 1);
BELLBOARD_IRQ_CONFIGURE(irq2, 2);
BELLBOARD_IRQ_CONFIGURE(irq3, 3);
return 0;
}
DEVICE_DT_INST_DEFINE(0, bellboard_rx_init, NULL, NULL, NULL, POST_KERNEL,
CONFIG_MBOX_INIT_PRIORITY, &bellboard_rx_driver_api);