1 /*
2 * Copyright (c) 2024 Nordic Semiconductor ASA
3 * SPDX-License-Identifier: Apache-2.0
4 */
5
6 #define DT_DRV_COMPAT nordic_nrf_bellboard_rx
7
8 #include <zephyr/devicetree.h>
9 #include <zephyr/drivers/mbox.h>
10 #include <zephyr/sys/util.h>
11 #include <zephyr/sys/__assert.h>
12
13 #include <hal/nrf_bellboard.h>
14
15 #define BELLBOARD_NUM_IRQS 4U
16
17 BUILD_ASSERT(DT_NUM_IRQS(DT_DRV_INST(0)) <= BELLBOARD_NUM_IRQS, "# interrupt exceeds maximum");
18
19 BUILD_ASSERT((DT_INST_PROP_LEN(0, nordic_interrupt_mapping) % 2) == 0,
20 "# interrupt mappings not specified in pairs");
21
22 /* BELLBOARD event mappings */
23 #define EVT_MAPPING_ITEM(idx) DT_INST_PROP_BY_IDX(0, nordic_interrupt_mapping, idx)
24 #define BELLBOARD_GET_EVT_MAPPING(idx, _) \
25 COND_CODE_1( \
26 DT_INST_PROP_HAS_IDX(0, nordic_interrupt_mapping, UTIL_INC(UTIL_X2(idx))), \
27 ([EVT_MAPPING_ITEM(UTIL_INC(UTIL_X2(idx)))] = EVT_MAPPING_ITEM(UTIL_X2(idx)),), \
28 ())
29
30 static const uint32_t evt_mappings[BELLBOARD_NUM_IRQS] = {
31 LISTIFY(DT_NUM_IRQS(DT_DRV_INST(0)), BELLBOARD_GET_EVT_MAPPING, ())};
32
33 /* BELLBOARD instance */
34 static NRF_BELLBOARD_Type *bellboard = (NRF_BELLBOARD_Type *)DT_INST_REG_ADDR(0);
35
36 /* BELLBOARD runtime resources */
37 static mbox_callback_t cbs[NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT];
38 static void *cbs_ctx[NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT];
39 static uint32_t evt_enabled_masks[BELLBOARD_NUM_IRQS];
40
bellboard_rx_isr(const void * parameter)41 static void bellboard_rx_isr(const void *parameter)
42 {
43 uint8_t irq_idx = (uint8_t)(uintptr_t)parameter;
44 uint32_t int_pend;
45
46 int_pend = nrf_bellboard_int_pending_get(bellboard, irq_idx);
47
48 for (uint8_t i = 0U; i < NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT; i++) {
49 nrf_bellboard_event_t event = nrf_bellboard_triggered_event_get(i);
50
51 if ((int_pend & BIT(i)) != 0U) {
52 /* Only clear those events that have their corresponding bit set
53 * in INTPEND at the time we read it. Otherwise, if two (or more)
54 * events are generated in quick succession, INTPEND may be set for
55 * only one of events, but we clear the EVENTS_TRIGGERED bit for
56 * all of them, thus losing them.
57 *
58 * Assume nrf_bellboard_event_check() is true for the event
59 * that raised this interrupt.
60 */
61 __ASSERT_NO_MSG(nrf_bellboard_event_check(bellboard, event));
62
63 nrf_bellboard_event_clear(bellboard, event);
64
65 if (cbs[i] != NULL) {
66 cbs[i](DEVICE_DT_INST_GET(0), i, cbs_ctx[i], NULL);
67 }
68 }
69 }
70 }
71
bellboard_rx_max_channels_get(const struct device * dev)72 static uint32_t bellboard_rx_max_channels_get(const struct device *dev)
73 {
74 ARG_UNUSED(dev);
75
76 return NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT;
77 }
78
bellboard_rx_register_callback(const struct device * dev,uint32_t id,mbox_callback_t cb,void * user_data)79 static int bellboard_rx_register_callback(const struct device *dev, uint32_t id, mbox_callback_t cb,
80 void *user_data)
81 {
82 ARG_UNUSED(dev);
83
84 if (id >= NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT) {
85 return -EINVAL;
86 }
87
88 cbs[id] = cb;
89 cbs_ctx[id] = user_data;
90
91 return 0;
92 }
93
bellboard_rx_set_enabled(const struct device * dev,uint32_t id,bool enable)94 static int bellboard_rx_set_enabled(const struct device *dev, uint32_t id, bool enable)
95 {
96 bool valid_found = false;
97
98 ARG_UNUSED(dev);
99
100 if (id >= NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT) {
101 return -EINVAL;
102 }
103
104 for (uint8_t i = 0U; i < BELLBOARD_NUM_IRQS; i++) {
105 uint32_t *evt_enabled_mask;
106
107 if ((evt_mappings[i] == 0U) || ((evt_mappings[i] & BIT(id)) == 0U)) {
108 continue;
109 }
110
111 valid_found = true;
112 evt_enabled_mask = &evt_enabled_masks[i];
113
114 if (enable) {
115 if ((*evt_enabled_mask & BIT(id)) != 0U) {
116 return -EALREADY;
117 }
118
119 *evt_enabled_mask |= BIT(id);
120 nrf_bellboard_int_enable(bellboard, i, BIT(id));
121 } else {
122 if ((*evt_enabled_mask & BIT(id)) == 0U) {
123 return -EALREADY;
124 }
125
126 *evt_enabled_mask &= ~BIT(id);
127 nrf_bellboard_int_disable(bellboard, i, BIT(id));
128 }
129 }
130
131 if (!valid_found) {
132 return -EINVAL;
133 }
134
135 return 0;
136 }
137
138 static DEVICE_API(mbox, bellboard_rx_driver_api) = {
139 .max_channels_get = bellboard_rx_max_channels_get,
140 .register_callback = bellboard_rx_register_callback,
141 .set_enabled = bellboard_rx_set_enabled,
142 };
143
144 #define BELLBOARD_IRQ_CONFIGURE(name, idx) \
145 COND_CODE_1(DT_INST_IRQ_HAS_NAME(0, name), \
146 (IRQ_CONNECT(DT_INST_IRQ_BY_NAME(0, name, irq), \
147 DT_INST_IRQ_BY_NAME(0, name, priority), bellboard_rx_isr, \
148 (const void *)idx, 0); \
149 irq_enable(DT_INST_IRQ_BY_NAME(0, name, irq));), \
150 ())
151
bellboard_rx_init(const struct device * dev)152 static int bellboard_rx_init(const struct device *dev)
153 {
154 uint32_t evt_all_mappings =
155 evt_mappings[0] | evt_mappings[1] | evt_mappings[2] | evt_mappings[3];
156
157 ARG_UNUSED(dev);
158
159 nrf_bellboard_int_disable(bellboard, 0, evt_mappings[0]);
160 nrf_bellboard_int_disable(bellboard, 1, evt_mappings[1]);
161 nrf_bellboard_int_disable(bellboard, 2, evt_mappings[2]);
162 nrf_bellboard_int_disable(bellboard, 3, evt_mappings[3]);
163
164 for (uint8_t i = 0U; i < NRF_BELLBOARD_EVENTS_TRIGGERED_COUNT; i++) {
165 if ((evt_all_mappings & BIT(i)) != 0U) {
166 nrf_bellboard_event_clear(bellboard, nrf_bellboard_triggered_event_get(i));
167 }
168 }
169
170 BELLBOARD_IRQ_CONFIGURE(irq0, 0);
171 BELLBOARD_IRQ_CONFIGURE(irq1, 1);
172 BELLBOARD_IRQ_CONFIGURE(irq2, 2);
173 BELLBOARD_IRQ_CONFIGURE(irq3, 3);
174
175 return 0;
176 }
177
178 DEVICE_DT_INST_DEFINE(0, bellboard_rx_init, NULL, NULL, NULL, POST_KERNEL,
179 CONFIG_MBOX_INIT_PRIORITY, &bellboard_rx_driver_api);
180