1 /* ieee802154_kw41z.c - NXP KW41Z driver */
2 
3 /*
4  * Copyright (c) 2017 Linaro Limited
5  *
6  * SPDX-License-Identifier: Apache-2.0
7  */
8 
9 #define DT_DRV_COMPAT nxp_kw41z_ieee802154
10 
11 #define LOG_MODULE_NAME ieee802154_kw41z
12 #define LOG_LEVEL CONFIG_IEEE802154_DRIVER_LOG_LEVEL
13 
14 #include <zephyr/logging/log.h>
15 LOG_MODULE_REGISTER(LOG_MODULE_NAME);
16 
17 #include <zephyr/kernel.h>
18 #include <zephyr/device.h>
19 #include <zephyr/init.h>
20 #include <zephyr/irq.h>
21 #include <zephyr/net/ieee802154_radio.h>
22 #include <zephyr/net/net_if.h>
23 #include <zephyr/net/net_pkt.h>
24 #include <zephyr/sys/byteorder.h>
25 #include <zephyr/random/random.h>
26 
27 #include "fsl_xcvr.h"
28 
29 #if defined(CONFIG_NET_L2_OPENTHREAD)
30 #include <zephyr/net/openthread.h>
31 #endif
32 
33 
34 /*
35  * For non-invasive tracing of IRQ events. Sometimes the print logs
36  * will shift the timings around so this trace buffer can be used to
37  * post inspect conditions to see what sequence of events occurred.
38  */
39 
40 #define KW41_DBG_TRACE_WTRM	0
41 #define KW41_DBG_TRACE_RX	1
42 #define KW41_DBG_TRACE_TX	2
43 #define KW41_DBG_TRACE_CCA	3
44 #define KW41_DBG_TRACE_TMR3	0xFF
45 
46 #if defined(CONFIG_KW41_DBG_TRACE)
47 
48 #define KW41_DBG_TRACE_SIZE 30
49 
50 struct kw41_dbg_trace {
51 	uint8_t	type;
52 	uint32_t	time;
53 	uint32_t	irqsts;
54 	uint32_t	phy_ctrl;
55 	uint32_t	seq_state;
56 };
57 
58 struct kw41_dbg_trace kw41_dbg[KW41_DBG_TRACE_SIZE];
59 int kw41_dbg_idx;
60 
61 #define KW_DBG_TRACE(_type, _irqsts, _phy_ctrl, _seq_state) \
62 	do { \
63 		kw41_dbg[kw41_dbg_idx].type = (_type); \
64 		kw41_dbg[kw41_dbg_idx].time = \
65 			ZLL->EVENT_TMR >> ZLL_EVENT_TMR_EVENT_TMR_SHIFT; \
66 		kw41_dbg[kw41_dbg_idx].irqsts = (_irqsts); \
67 		kw41_dbg[kw41_dbg_idx].phy_ctrl = (_phy_ctrl); \
68 		kw41_dbg[kw41_dbg_idx].seq_state = (_seq_state); \
69 		if (++kw41_dbg_idx == KW41_DBG_TRACE_SIZE) { \
70 			kw41_dbg_idx = 0; \
71 		} \
72 	} while (false)
73 
74 #else
75 
76 #define KW_DBG_TRACE(_type, _irqsts, _phy_ctrl, _seq_state)
77 
78 #endif
79 
80 #define KW41Z_DEFAULT_CHANNEL		26
81 #define KW41Z_CCA_TIME			8
82 #define KW41Z_SHR_PHY_TIME		12
83 #define KW41Z_PER_BYTE_TIME		2
84 #define KW41Z_ACK_WAIT_TIME		54
85 #define KW41Z_PRE_RX_WAIT_TIME		1
86 #define KW40Z_POST_SEQ_WAIT_TIME	1
87 
88 #define RADIO_0_IRQ_PRIO		0x0
89 #define KW41Z_FCS_LENGTH		2
90 #define KW41Z_PSDU_LENGTH		125
91 #define KW41Z_OUTPUT_POWER_MAX		4
92 #define KW41Z_OUTPUT_POWER_MIN		(-31)
93 
94 #define IEEE802154_ACK_LENGTH		5
95 
96 #define BM_ZLL_IRQSTS_TMRxMSK (ZLL_IRQSTS_TMR1MSK_MASK | \
97 				ZLL_IRQSTS_TMR2MSK_MASK | \
98 				ZLL_IRQSTS_TMR3MSK_MASK | \
99 				ZLL_IRQSTS_TMR4MSK_MASK)
100 
101 /*
102  * Clear channel assessment types. Note that there is an extra one when
103  * bit 26 is included for "No CCA before transmit" if we are handling
104  * ACK frames but we will let the hardware handle that automatically.
105  */
106 enum {
107 	KW41Z_CCA_ED,       /* Energy detect */
108 	KW41Z_CCA_MODE1,    /* Energy above threshold   */
109 	KW41Z_CCA_MODE2,    /* Carrier sense only       */
110 	KW41Z_CCA_MODE3     /* Mode 1 + Mode 2          */
111 };
112 
113 /*
114  * KW41Z has a sequencer that can run in any of the following states.
115  */
116 enum {
117 	KW41Z_STATE_IDLE,
118 	KW41Z_STATE_RX,
119 	KW41Z_STATE_TX,
120 	KW41Z_STATE_CCA,
121 	KW41Z_STATE_TXRX,
122 	KW41Z_STATE_CCCA
123 };
124 
125 /* Lookup table for PA_PWR register */
126 static const uint8_t pa_pwr_lt[] = {
127 	1,                   /* -31.1 dBm: -31 */
128 	2, 2, 2, 2, 2, 2, 2, /* -25.0 dBm: -30, -29, -28, -27, -26, -25 */
129 	4, 4, 4, 4, 4,       /* -19.0 dBm: -24, -23, -22, -21, -20, -19 */
130 	6, 6, 6,             /* -15.6 dBm: -18, -17, -16 */
131 	8, 8,                /* -13.1 dBm: -15, -14 */
132 	10, 10,              /* -11.2 dBm: -13, -12 */
133 	12, 12,              /* - 9.6 dBm: -11, -10 */
134 	14,                  /* - 8.3 dBm: -9 */
135 	16,                  /* - 7.2 dBm: -8 */
136 	18,                  /* - 6.2 dBm: -7 */
137 	20,                  /* - 5.3 dBm: -6 */
138 	22,                  /* - 4.5 dBm: -5 */
139 	24,                  /* - 3.8 dBm: -4 */
140 	28,                  /* - 2.5 dBm: -3 */
141 	30,                  /* - 1.9 dBm: -2 */
142 	34,                  /* - 1.0 dBm: -1 */
143 	40,                  /* + 0.3 dBm:  0 */
144 	44,                  /* + 1.1 dBm: +1 */
145 	50,                  /* + 2.1 dBm: +2 */
146 	58,                  /* + 3.1 dBm: +3 */
147 	62                   /* + 3.5 dBm: +4 */
148 };
149 
150 struct kw41z_context {
151 	struct net_if *iface;
152 	uint8_t mac_addr[8];
153 
154 	struct k_sem seq_sync;
155 	atomic_t seq_retval;
156 
157 	uint32_t rx_warmup_time;
158 	uint32_t tx_warmup_time;
159 
160 	bool frame_pending; /* FP bit state from the most recent ACK frame. */
161 };
162 
163 static struct kw41z_context kw41z_context_data;
164 
kw41z_get_instant_state(void)165 static inline uint8_t kw41z_get_instant_state(void)
166 {
167 	return (ZLL->SEQ_STATE & ZLL_SEQ_STATE_SEQ_STATE_MASK) >>
168 	       ZLL_SEQ_STATE_SEQ_STATE_SHIFT;
169 }
170 
kw41z_get_seq_state(void)171 static inline uint8_t kw41z_get_seq_state(void)
172 {
173 	return (ZLL->PHY_CTRL & ZLL_PHY_CTRL_XCVSEQ_MASK) >>
174 	       ZLL_PHY_CTRL_XCVSEQ_SHIFT;
175 }
176 
kw41z_set_seq_state(uint8_t state)177 static inline void kw41z_set_seq_state(uint8_t state)
178 {
179 #if CONFIG_SOC_MKW40Z4
180 	/*
181 	 * KW40Z seems to require a small delay when switching to IDLE state
182 	 * after a programmed sequence is complete.
183 	 */
184 	if (state == KW41Z_STATE_IDLE) {
185 		k_busy_wait(KW40Z_POST_SEQ_WAIT_TIME);
186 	}
187 #endif
188 
189 	ZLL->PHY_CTRL = (ZLL->PHY_CTRL & ~ZLL_PHY_CTRL_XCVSEQ_MASK) |
190 			ZLL_PHY_CTRL_XCVSEQ(state);
191 }
192 
kw41z_wait_for_idle(void)193 static inline void kw41z_wait_for_idle(void)
194 {
195 	uint8_t state = kw41z_get_instant_state();
196 
197 	while (state != KW41Z_STATE_IDLE) {
198 		state = kw41z_get_instant_state();
199 	}
200 
201 	if (state != KW41Z_STATE_IDLE) {
202 		LOG_ERR("Error waiting for idle state");
203 	}
204 }
205 
kw41z_phy_abort(void)206 static void kw41z_phy_abort(void)
207 {
208 	unsigned int key;
209 
210 	key = irq_lock();
211 
212 	/* Mask SEQ interrupt */
213 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_SEQMSK_MASK;
214 	/* Disable timer trigger (for scheduled XCVSEQ) */
215 	if (ZLL->PHY_CTRL & ZLL_PHY_CTRL_TMRTRIGEN_MASK) {
216 		ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_TMRTRIGEN_MASK;
217 		/* give the FSM enough time to start if it was triggered */
218 		while ((XCVR_MISC->XCVR_CTRL &
219 			XCVR_CTRL_XCVR_STATUS_TSM_COUNT_MASK) == 0) {
220 		}
221 	}
222 
223 	/* If XCVR is not idle, abort current SEQ */
224 	if (ZLL->PHY_CTRL & ZLL_PHY_CTRL_XCVSEQ_MASK) {
225 		ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_XCVSEQ_MASK;
226 		/* wait for Sequence Idle (if not already) */
227 
228 		while (ZLL->SEQ_STATE & ZLL_SEQ_STATE_SEQ_STATE_MASK) {
229 		}
230 	}
231 
232 	/* Stop timers */
233 	ZLL->PHY_CTRL &= ~(ZLL_PHY_CTRL_TMR1CMP_EN_MASK |
234 			ZLL_PHY_CTRL_TMR2CMP_EN_MASK |
235 			ZLL_PHY_CTRL_TMR3CMP_EN_MASK |
236 			ZLL_PHY_CTRL_TC3TMOUT_MASK);
237 
238 	/*
239 	 * Clear all IRQ bits to avoid unexpected interrupts.
240 	 *
241 	 * For Coverity, this is a pointer to a register bank and the IRQSTS
242 	 * register bits get cleared when a 1 is written to them so doing a
243 	 * reg=reg may generate a warning but it is needed to clear the bits.
244 	 */
245 	ZLL->IRQSTS = ZLL->IRQSTS;
246 
247 	irq_unlock(key);
248 }
249 
kw41z_isr_timeout_cleanup(void)250 static void kw41z_isr_timeout_cleanup(void)
251 {
252 	uint32_t irqsts;
253 
254 	/*
255 	 * Set the PHY sequencer back to IDLE and disable TMR3 comparator
256 	 * and timeout
257 	 */
258 	ZLL->PHY_CTRL &= ~(ZLL_PHY_CTRL_TMR3CMP_EN_MASK |
259 			ZLL_PHY_CTRL_TC3TMOUT_MASK   |
260 			ZLL_PHY_CTRL_XCVSEQ_MASK);
261 
262 	/* Mask SEQ, RX, TX and CCA interrupts */
263 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_CCAMSK_MASK |
264 			ZLL_PHY_CTRL_RXMSK_MASK  |
265 			ZLL_PHY_CTRL_TXMSK_MASK  |
266 			ZLL_PHY_CTRL_SEQMSK_MASK;
267 
268 	while (ZLL->SEQ_STATE & ZLL_SEQ_STATE_SEQ_STATE_MASK) {
269 	}
270 
271 	irqsts = ZLL->IRQSTS;
272 	/* Mask TMR3 interrupt */
273 	irqsts |= ZLL_IRQSTS_TMR3MSK_MASK;
274 
275 	ZLL->IRQSTS = irqsts;
276 }
277 
kw41z_isr_seq_cleanup(void)278 static void kw41z_isr_seq_cleanup(void)
279 {
280 	uint32_t irqsts;
281 
282 	/* Set the PHY sequencer back to IDLE */
283 	ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_XCVSEQ_MASK;
284 	/* Mask SEQ, RX, TX and CCA interrupts */
285 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_CCAMSK_MASK |
286 			ZLL_PHY_CTRL_RXMSK_MASK  |
287 			ZLL_PHY_CTRL_TXMSK_MASK  |
288 			ZLL_PHY_CTRL_SEQMSK_MASK;
289 
290 	while (ZLL->SEQ_STATE & ZLL_SEQ_STATE_SEQ_STATE_MASK) {
291 	}
292 
293 	irqsts = ZLL->IRQSTS;
294 	/* Mask TMR3 interrupt */
295 	irqsts |= ZLL_IRQSTS_TMR3MSK_MASK;
296 
297 	/* Clear transceiver interrupts except TMRxIRQ */
298 	irqsts &= ~(ZLL_IRQSTS_TMR1IRQ_MASK |
299 		ZLL_IRQSTS_TMR2IRQ_MASK |
300 		ZLL_IRQSTS_TMR3IRQ_MASK |
301 		ZLL_IRQSTS_TMR4IRQ_MASK);
302 	ZLL->IRQSTS = irqsts;
303 }
304 
kw41z_enable_seq_irq(void)305 static inline void kw41z_enable_seq_irq(void)
306 {
307 	ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_SEQMSK_MASK;
308 }
309 
kw41z_disable_seq_irq(void)310 static inline void kw41z_disable_seq_irq(void)
311 {
312 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_SEQMSK_MASK;
313 }
314 
315 /*
316  * Set the T3CMP timer comparator. The 'timeout' value is an offset from
317  * now.
318  */
kw41z_tmr3_set_timeout(uint32_t timeout)319 static void kw41z_tmr3_set_timeout(uint32_t timeout)
320 {
321 	uint32_t irqsts;
322 
323 	/* Add in the current time so that we can get the comparator to
324 	 * match appropriately to our offset time.
325 	 */
326 	timeout += ZLL->EVENT_TMR >> ZLL_EVENT_TMR_EVENT_TMR_SHIFT;
327 
328 	/* disable TMR3 compare */
329 	ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_TMR3CMP_EN_MASK;
330 	ZLL->T3CMP = timeout & ZLL_T3CMP_T3CMP_MASK;
331 
332 	/* acknowledge TMR3 IRQ */
333 	irqsts  = ZLL->IRQSTS & BM_ZLL_IRQSTS_TMRxMSK;
334 	irqsts |= ZLL_IRQSTS_TMR3IRQ_MASK;
335 	ZLL->IRQSTS = irqsts;
336 	/* enable TMR3 compare and autosequence stop by TC3 match */
337 	ZLL->PHY_CTRL |=
338 		(ZLL_PHY_CTRL_TMR3CMP_EN_MASK | ZLL_PHY_CTRL_TC3TMOUT_MASK);
339 }
340 
kw41z_tmr3_disable(void)341 static void kw41z_tmr3_disable(void)
342 {
343 	uint32_t irqsts;
344 
345 	/*
346 	 * disable TMR3 compare and disable autosequence stop by TC3
347 	 * match
348 	 */
349 	ZLL->PHY_CTRL &= ~(ZLL_PHY_CTRL_TMR3CMP_EN_MASK |
350 			ZLL_PHY_CTRL_TC3TMOUT_MASK);
351 	/* mask TMR3 interrupt (do not change other IRQ status) */
352 	irqsts  = ZLL->IRQSTS & BM_ZLL_IRQSTS_TMRxMSK;
353 	irqsts |= ZLL_IRQSTS_TMR3MSK_MASK;
354 	/* acknowledge TMR3 IRQ */
355 	irqsts |= ZLL_IRQSTS_TMR3IRQ_MASK;
356 
357 	ZLL->IRQSTS = irqsts;
358 }
359 
kw41z_get_capabilities(const struct device * dev)360 static enum ieee802154_hw_caps kw41z_get_capabilities(const struct device *dev)
361 {
362 	return IEEE802154_HW_FCS | IEEE802154_HW_FILTER |
363 	       IEEE802154_HW_TX_RX_ACK | IEEE802154_HW_RX_TX_ACK;
364 }
365 
kw41z_cca(const struct device * dev)366 static int kw41z_cca(const struct device *dev)
367 {
368 	struct kw41z_context *kw41z = dev->data;
369 
370 	kw41z_phy_abort();
371 
372 	k_sem_init(&kw41z->seq_sync, 0, 1);
373 
374 	kw41z_enable_seq_irq();
375 	ZLL->PHY_CTRL = (ZLL->PHY_CTRL & ~ZLL_PHY_CTRL_CCATYPE_MASK) |
376 			ZLL_PHY_CTRL_CCATYPE(KW41Z_CCA_MODE1);
377 
378 	kw41z_set_seq_state(KW41Z_STATE_CCA);
379 
380 	k_sem_take(&kw41z->seq_sync, K_FOREVER);
381 
382 	return kw41z->seq_retval;
383 }
384 
kw41z_set_channel(const struct device * dev,uint16_t channel)385 static int kw41z_set_channel(const struct device *dev, uint16_t channel)
386 {
387 	if (channel < 11 || channel > 26) {
388 		return channel < 11 ? -ENOTSUP : -EINVAL;
389 	}
390 
391 	ZLL->CHANNEL_NUM0 = channel;
392 	return 0;
393 }
394 
kw41z_set_pan_id(const struct device * dev,uint16_t pan_id)395 static int kw41z_set_pan_id(const struct device *dev, uint16_t pan_id)
396 {
397 	ZLL->MACSHORTADDRS0 = (ZLL->MACSHORTADDRS0 &
398 			       ~ZLL_MACSHORTADDRS0_MACPANID0_MASK) |
399 			      ZLL_MACSHORTADDRS0_MACPANID0(pan_id);
400 	return 0;
401 }
402 
kw41z_set_short_addr(const struct device * dev,uint16_t short_addr)403 static int kw41z_set_short_addr(const struct device *dev, uint16_t short_addr)
404 {
405 	ZLL->MACSHORTADDRS0 = (ZLL->MACSHORTADDRS0 &
406 			       ~ZLL_MACSHORTADDRS0_MACSHORTADDRS0_MASK) |
407 			      ZLL_MACSHORTADDRS0_MACSHORTADDRS0(short_addr);
408 	return 0;
409 }
410 
kw41z_set_ieee_addr(const struct device * dev,const uint8_t * ieee_addr)411 static int kw41z_set_ieee_addr(const struct device *dev,
412 			       const uint8_t *ieee_addr)
413 {
414 	uint32_t val;
415 
416 	memcpy(&val, ieee_addr, sizeof(val));
417 	ZLL->MACLONGADDRS0_LSB = val;
418 
419 	memcpy(&val, ieee_addr + sizeof(val), sizeof(val));
420 	ZLL->MACLONGADDRS0_MSB = val;
421 
422 	return 0;
423 }
424 
kw41z_filter(const struct device * dev,bool set,enum ieee802154_filter_type type,const struct ieee802154_filter * filter)425 static int kw41z_filter(const struct device *dev,
426 			bool set,
427 			enum ieee802154_filter_type type,
428 			const struct ieee802154_filter *filter)
429 {
430 	LOG_DBG("Applying filter %u", type);
431 
432 	if (!set) {
433 		return -ENOTSUP;
434 	}
435 
436 	if (type == IEEE802154_FILTER_TYPE_IEEE_ADDR) {
437 		return kw41z_set_ieee_addr(dev, filter->ieee_addr);
438 	} else if (type == IEEE802154_FILTER_TYPE_SHORT_ADDR) {
439 		return kw41z_set_short_addr(dev, filter->short_addr);
440 	} else if (type == IEEE802154_FILTER_TYPE_PAN_ID) {
441 		return kw41z_set_pan_id(dev, filter->pan_id);
442 	}
443 
444 	return -ENOTSUP;
445 }
446 
kw41z_set_txpower(const struct device * dev,int16_t dbm)447 static int kw41z_set_txpower(const struct device *dev, int16_t dbm)
448 {
449 	if (dbm < KW41Z_OUTPUT_POWER_MIN) {
450 		LOG_INF("TX-power %d dBm below min of %d dBm, using %d dBm",
451 			    dbm,
452 			    KW41Z_OUTPUT_POWER_MIN,
453 			    KW41Z_OUTPUT_POWER_MIN);
454 		dbm = KW41Z_OUTPUT_POWER_MIN;
455 	} else if (dbm > KW41Z_OUTPUT_POWER_MAX) {
456 		LOG_INF("TX-power %d dBm above max of %d dBm, using %d dBm",
457 			    dbm,
458 			    KW41Z_OUTPUT_POWER_MAX,
459 			    KW41Z_OUTPUT_POWER_MAX);
460 		dbm = KW41Z_OUTPUT_POWER_MAX;
461 	}
462 
463 	ZLL->PA_PWR = pa_pwr_lt[dbm - KW41Z_OUTPUT_POWER_MIN];
464 
465 	return 0;
466 }
467 
kw41z_start(const struct device * dev)468 static int kw41z_start(const struct device *dev)
469 {
470 	irq_enable(Radio_1_IRQn);
471 
472 	kw41z_set_seq_state(KW41Z_STATE_RX);
473 	kw41z_enable_seq_irq();
474 
475 	return 0;
476 }
477 
kw41z_stop(const struct device * dev)478 static int kw41z_stop(const struct device *dev)
479 {
480 	irq_disable(Radio_1_IRQn);
481 
482 	kw41z_disable_seq_irq();
483 	kw41z_set_seq_state(KW41Z_STATE_IDLE);
484 
485 	return 0;
486 }
487 
kw41z_convert_lqi(uint8_t hw_lqi)488 static uint8_t kw41z_convert_lqi(uint8_t hw_lqi)
489 {
490 	if (hw_lqi >= 220U) {
491 		return 255;
492 	} else {
493 		return (hw_lqi * 51U) / 44;
494 	}
495 }
496 
kw41z_rx(struct kw41z_context * kw41z,uint8_t len)497 static inline void kw41z_rx(struct kw41z_context *kw41z, uint8_t len)
498 {
499 	struct net_pkt *pkt = NULL;
500 	struct net_buf *buf = NULL;
501 	uint8_t pkt_len, hw_lqi;
502 	int rslt;
503 
504 	LOG_DBG("ENTRY: len: %d", len);
505 
506 #if defined(CONFIG_NET_L2_OPENTHREAD)
507 	/*
508 	 * OpenThread stack expects a receive frame to include the FCS
509 	 */
510 	pkt_len = len;
511 #else
512 	pkt_len = len - KW41Z_FCS_LENGTH;
513 #endif
514 
515 	pkt = net_pkt_rx_alloc_with_buffer(kw41z->iface, pkt_len,
516 					   AF_UNSPEC, 0, K_NO_WAIT);
517 	if (!pkt) {
518 		LOG_ERR("No buf available");
519 		goto out;
520 	}
521 
522 	buf = pkt->buffer;
523 
524 #if CONFIG_SOC_MKW41Z4
525 	/* PKT_BUFFER_RX needs to be accessed aligned to 16 bits */
526 	for (uint16_t reg_val = 0, i = 0; i < pkt_len; i++) {
527 		if (i % 2 == 0U) {
528 			reg_val = ZLL->PKT_BUFFER_RX[i/2U];
529 			buf->data[i] = reg_val & 0xFF;
530 		} else {
531 			buf->data[i] = reg_val >> 8;
532 		}
533 	}
534 #else /* CONFIG_SOC_MKW40Z4 */
535 	/* PKT_BUFFER needs to be accessed aligned to 32 bits */
536 	for (uint32_t reg_val = 0, i = 0; i < pkt_len; i++) {
537 		switch (i % 4) {
538 		case 0:
539 			reg_val = ZLL->PKT_BUFFER[i/4U];
540 			buf->data[i] = reg_val & 0xFF;
541 			break;
542 		case 1:
543 			buf->data[i] = (reg_val >> 8) & 0xFF;
544 			break;
545 		case 2:
546 			buf->data[i] = (reg_val >> 16) & 0xFF;
547 			break;
548 		default:
549 			buf->data[i] = reg_val >> 24;
550 		}
551 	}
552 #endif
553 
554 	net_buf_add(buf, pkt_len);
555 
556 	hw_lqi = (ZLL->LQI_AND_RSSI & ZLL_LQI_AND_RSSI_LQI_VALUE_MASK) >>
557 		 ZLL_LQI_AND_RSSI_LQI_VALUE_SHIFT;
558 	net_pkt_set_ieee802154_lqi(pkt, kw41z_convert_lqi(hw_lqi));
559 	/* ToDo: get the rssi as well and use net_pkt_set_ieee802154_rssi() */
560 
561 	rslt = net_recv_data(kw41z->iface, pkt);
562 	if (rslt < 0) {
563 		LOG_ERR("RCV Packet dropped by NET stack: %d", rslt);
564 		goto out;
565 	}
566 
567 	return;
568 out:
569 	if (pkt) {
570 		net_pkt_unref(pkt);
571 	}
572 }
573 
574 #define ACK_FRAME_LEN 3
575 #define ACK_FRAME_TYPE (2 << 0)
576 #define ACK_FRAME_PENDING_BIT (1 << 4)
577 
handle_ack(struct kw41z_context * kw41z,uint8_t seq_number)578 static void handle_ack(struct kw41z_context *kw41z, uint8_t seq_number)
579 {
580 	struct net_pkt *ack_pkt;
581 	uint8_t ack_psdu[ACK_FRAME_LEN];
582 
583 	ack_pkt = net_pkt_rx_alloc_with_buffer(kw41z->iface, ACK_FRAME_LEN,
584 					       AF_UNSPEC, 0, K_NO_WAIT);
585 	if (!ack_pkt) {
586 		LOG_ERR("No free packet available.");
587 		return;
588 	}
589 
590 	/* Re-create ACK frame. */
591 	ack_psdu[0] = kw41z_context_data.frame_pending ?
592 		      ACK_FRAME_TYPE | ACK_FRAME_PENDING_BIT : ACK_FRAME_TYPE;
593 	ack_psdu[1] = 0;
594 	ack_psdu[2] = seq_number;
595 
596 	if (net_pkt_write(ack_pkt, ack_psdu, sizeof(ack_psdu)) < 0) {
597 		LOG_ERR("Failed to write to a packet.");
598 		goto out;
599 	}
600 
601 	/* Use some fake values for LQI and RSSI. */
602 	(void)net_pkt_set_ieee802154_lqi(ack_pkt, 80);
603 	(void)net_pkt_set_ieee802154_rssi_dbm(ack_pkt, -40);
604 
605 	net_pkt_cursor_init(ack_pkt);
606 
607 	if (ieee802154_handle_ack(kw41z->iface, ack_pkt) != NET_OK) {
608 		LOG_INF("ACK packet not handled - releasing.");
609 	}
610 
611 out:
612 	net_pkt_unref(ack_pkt);
613 }
614 
kw41z_tx(const struct device * dev,enum ieee802154_tx_mode mode,struct net_pkt * pkt,struct net_buf * frag)615 static int kw41z_tx(const struct device *dev, enum ieee802154_tx_mode mode,
616 		    struct net_pkt *pkt, struct net_buf *frag)
617 {
618 	struct kw41z_context *kw41z = dev->data;
619 	uint8_t payload_len = frag->len;
620 	uint32_t tx_timeout;
621 	uint8_t xcvseq;
622 	unsigned int key;
623 
624 	if (mode != IEEE802154_TX_MODE_DIRECT) {
625 		NET_ERR("TX mode %d not supported", mode);
626 		return -ENOTSUP;
627 	}
628 
629 	/*
630 	 * The transmit requests are preceded by the CCA request. On
631 	 * completion of the CCA the sequencer should be in the IDLE
632 	 * state.
633 	 */
634 	if (kw41z_get_seq_state() != KW41Z_STATE_IDLE) {
635 		LOG_WRN("Can't initiate new SEQ state");
636 		return -EBUSY;
637 	}
638 
639 	if (payload_len > KW41Z_PSDU_LENGTH) {
640 		LOG_ERR("Payload too long");
641 		return 0;
642 	}
643 
644 	key = irq_lock();
645 
646 	/* Disable the 802.15.4 radio IRQ */
647 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_TRCV_MSK_MASK;
648 	kw41z_disable_seq_irq();
649 
650 #if CONFIG_SOC_MKW41Z4
651 	((uint8_t *)ZLL->PKT_BUFFER_TX)[0] = payload_len + KW41Z_FCS_LENGTH;
652 	memcpy(((uint8_t *)ZLL->PKT_BUFFER_TX) + 1,
653 		(void *)frag->data, payload_len);
654 #else /* CONFIG_SOC_MKW40Z4 */
655 	((uint8_t *)ZLL->PKT_BUFFER)[0] = payload_len + KW41Z_FCS_LENGTH;
656 	memcpy(((uint8_t *)ZLL->PKT_BUFFER) + 1,
657 		(void *)frag->data, payload_len);
658 #endif
659 
660 	/* Set CCA mode */
661 	ZLL->PHY_CTRL = (ZLL->PHY_CTRL & ~ZLL_PHY_CTRL_CCATYPE_MASK) |
662 			ZLL_PHY_CTRL_CCATYPE(KW41Z_CCA_MODE1);
663 
664 	/* Clear all IRQ flags */
665 	ZLL->IRQSTS = ZLL->IRQSTS;
666 
667 	/* Perform automatic reception of ACK frame, if required */
668 	if (ieee802154_is_ar_flag_set(frag)) {
669 		tx_timeout = kw41z->tx_warmup_time + KW41Z_SHR_PHY_TIME +
670 				 payload_len * KW41Z_PER_BYTE_TIME + 10 +
671 				 KW41Z_ACK_WAIT_TIME;
672 
673 		LOG_DBG("AUTOACK ENABLED: len: %d, timeout: %d, seq: %d",
674 			payload_len, tx_timeout, frag->data[2]);
675 
676 		kw41z_tmr3_set_timeout(tx_timeout);
677 		ZLL->PHY_CTRL |= ZLL_PHY_CTRL_RXACKRQD_MASK;
678 		xcvseq = KW41Z_STATE_TXRX;
679 	} else {
680 		LOG_DBG("AUTOACK DISABLED: len: %d, seq: %d",
681 			payload_len, frag->data[2]);
682 
683 		ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_RXACKRQD_MASK;
684 		xcvseq = KW41Z_STATE_TX;
685 	}
686 
687 	kw41z_enable_seq_irq();
688 	/*
689 	 * PHY_CTRL is sensitive to multiple writes that can kick off
690 	 * the sequencer engine causing TX with AR request to send the
691 	 * TX frame multiple times.
692 	 *
693 	 * To minimize, ensure there is only one write to PHY_CTRL with
694 	 * TXRX sequence enable and the 802.15.4 radio IRQ.
695 	 */
696 	ZLL->PHY_CTRL = (ZLL->PHY_CTRL & ~ZLL_PHY_CTRL_TRCV_MSK_MASK) | xcvseq;
697 	irq_unlock(key);
698 	k_sem_take(&kw41z->seq_sync, K_FOREVER);
699 
700 	if ((kw41z->seq_retval == 0) && ieee802154_is_ar_flag_set(frag)) {
701 		handle_ack(kw41z, frag->data[2]);
702 	}
703 
704 	LOG_DBG("seq_retval: %ld", kw41z->seq_retval);
705 	return kw41z->seq_retval;
706 }
707 
kw41z_isr(int unused)708 static void kw41z_isr(int unused)
709 {
710 	uint32_t irqsts = ZLL->IRQSTS;
711 	uint8_t state = kw41z_get_seq_state();
712 	uint8_t restart_rx = 1U;
713 	uint32_t rx_len;
714 
715 	/*
716 	 * Variable is used in debug output to capture the state of the
717 	 * sequencer at interrupt.
718 	 */
719 	uint32_t seq_state = ZLL->SEQ_STATE;
720 
721 	LOG_DBG("ENTRY: irqsts: 0x%08X, PHY_CTRL: 0x%08X, "
722 		"SEQ_STATE: 0x%08X, SEQ_CTRL: 0x%08X, TMR: %d, state: %d",
723 		irqsts, (unsigned int)ZLL->PHY_CTRL,
724 		(unsigned int)seq_state,
725 		(unsigned int)ZLL->SEQ_CTRL_STS,
726 		(unsigned int)(ZLL->EVENT_TMR >> ZLL_EVENT_TMR_EVENT_TMR_SHIFT),
727 		state);
728 
729 	/* Clear interrupts */
730 	ZLL->IRQSTS = irqsts;
731 
732 	if (irqsts & ZLL_IRQSTS_FILTERFAIL_IRQ_MASK) {
733 		LOG_DBG("Incoming RX failed packet filtering rules: "
734 			"CODE: 0x%08X, irqsts: 0x%08X, PHY_CTRL: 0x%08X, "
735 			"SEQ_STATE: 0x%08X, state: %d",
736 			(unsigned int)ZLL->FILTERFAIL_CODE,
737 			irqsts,
738 			(unsigned int)ZLL->PHY_CTRL,
739 			(unsigned int)seq_state, state);
740 
741 		restart_rx = 0U;
742 
743 	} else if ((!(ZLL->PHY_CTRL & ZLL_PHY_CTRL_RX_WMRK_MSK_MASK)) &&
744 	    (irqsts & ZLL_IRQSTS_RXWTRMRKIRQ_MASK)) {
745 		/*
746 		 * There is a bug in the KW41Z where in noisy environments
747 		 * the RX sequence can get lost. The watermark mask IRQ can
748 		 * start TMR3 to complete the rest of the read or to assert
749 		 * IRQ if the sequencer gets lost so we can reset things.
750 		 * Note that a TX from the upper layers will also reset
751 		 * things so the problem is contained a bit in normal
752 		 * operation.
753 		 */
754 		rx_len = (irqsts & ZLL_IRQSTS_RX_FRAME_LENGTH_MASK)
755 			>> ZLL_IRQSTS_RX_FRAME_LENGTH_SHIFT;
756 
757 		KW_DBG_TRACE(KW41_DBG_TRACE_WTRM, irqsts,
758 			(unsigned int)ZLL->PHY_CTRL, seq_state);
759 
760 		if (rx_len > IEEE802154_ACK_LENGTH) {
761 
762 			LOG_DBG("WMRK irq: seq_state: 0x%08x, rx_len: %d",
763 				seq_state, rx_len);
764 			/*
765 			 * Assume the RX includes an auto-ACK so set the
766 			 * timer to include the RX frame size, crc, IFS,
767 			 * and ACK length and convert to symbols.
768 			 *
769 			 * IFS is 12 symbols
770 			 *
771 			 * ACK frame is 11 bytes: 4 preamble, 1 start of
772 			 * frame, 1 frame length, 2 frame control,
773 			 * 1 sequence, 2 FCS. Times two to convert to symbols.
774 			 */
775 			rx_len = rx_len * 2U + 12 + 22 + 2;
776 			kw41z_tmr3_set_timeout(rx_len);
777 		}
778 		restart_rx = 0U;
779 	}
780 
781 	/* Sequence done IRQ */
782 	if ((state != KW41Z_STATE_IDLE) && (irqsts & ZLL_IRQSTS_SEQIRQ_MASK)) {
783 		/*
784 		 * PLL unlock, the autosequence has been aborted due to
785 		 * PLL unlock
786 		 */
787 		if (irqsts & ZLL_IRQSTS_PLL_UNLOCK_IRQ_MASK) {
788 			LOG_ERR("PLL unlock error");
789 			kw41z_isr_seq_cleanup();
790 			restart_rx = 1U;
791 		}
792 		/*
793 		 * TMR3 timeout, the autosequence has been aborted due to
794 		 * TMR3 timeout
795 		 */
796 		else if ((irqsts & ZLL_IRQSTS_TMR3IRQ_MASK) &&
797 			(!(irqsts & ZLL_IRQSTS_RXIRQ_MASK)) &&
798 			(state != KW41Z_STATE_TX)) {
799 
800 			LOG_DBG("a) TMR3 timeout: irqsts: 0x%08X, "
801 				"seq_state: 0x%08X, PHY_CTRL: 0x%08X, "
802 				"state: %d",
803 				irqsts, seq_state,
804 				(unsigned int)ZLL->PHY_CTRL, state);
805 
806 			KW_DBG_TRACE(KW41_DBG_TRACE_TMR3, irqsts,
807 				(unsigned int)ZLL->PHY_CTRL, seq_state);
808 
809 			kw41z_isr_timeout_cleanup();
810 			restart_rx = 1U;
811 
812 			if (state == KW41Z_STATE_TXRX) {
813 				/* TODO: What is the right error for no ACK? */
814 				atomic_set(&kw41z_context_data.seq_retval,
815 					   -EBUSY);
816 				k_sem_give(&kw41z_context_data.seq_sync);
817 			}
818 		} else {
819 			kw41z_isr_seq_cleanup();
820 
821 			switch (state) {
822 			case KW41Z_STATE_RX:
823 				LOG_DBG("RX seq done: SEQ_STATE: 0x%08X",
824 					(unsigned int)seq_state);
825 
826 				KW_DBG_TRACE(KW41_DBG_TRACE_RX, irqsts,
827 					(unsigned int)ZLL->PHY_CTRL, seq_state);
828 
829 				kw41z_tmr3_disable();
830 
831 				rx_len = (ZLL->IRQSTS &
832 					  ZLL_IRQSTS_RX_FRAME_LENGTH_MASK) >>
833 					  ZLL_IRQSTS_RX_FRAME_LENGTH_SHIFT;
834 
835 				if (irqsts & ZLL_IRQSTS_RXIRQ_MASK) {
836 					if (rx_len != 0U) {
837 						kw41z_rx(&kw41z_context_data,
838 							rx_len);
839 					}
840 				}
841 				restart_rx = 1U;
842 				break;
843 			case KW41Z_STATE_TXRX:
844 				LOG_DBG("TXRX seq done");
845 				kw41z_tmr3_disable();
846 				/* Store the frame pending bit status. */
847 				kw41z_context_data.frame_pending =
848 					irqsts & ZLL_IRQSTS_RX_FRM_PEND_MASK;
849 			case KW41Z_STATE_TX:
850 				LOG_DBG("TX seq done");
851 				KW_DBG_TRACE(KW41_DBG_TRACE_TX, irqsts,
852 					(unsigned int)ZLL->PHY_CTRL, seq_state);
853 				if (irqsts & ZLL_IRQSTS_CCA_MASK) {
854 					atomic_set(
855 						&kw41z_context_data.seq_retval,
856 						-EBUSY);
857 				} else {
858 					atomic_set(
859 						&kw41z_context_data.seq_retval,
860 						0);
861 				}
862 
863 				k_sem_give(&kw41z_context_data.seq_sync);
864 				restart_rx = 1U;
865 
866 				break;
867 			case KW41Z_STATE_CCA:
868 				LOG_DBG("CCA seq done");
869 				KW_DBG_TRACE(KW41_DBG_TRACE_CCA, irqsts,
870 					(unsigned int)ZLL->PHY_CTRL, seq_state);
871 				if (irqsts & ZLL_IRQSTS_CCA_MASK) {
872 					atomic_set(
873 						&kw41z_context_data.seq_retval,
874 						-EBUSY);
875 					restart_rx = 1U;
876 				} else {
877 					atomic_set(
878 						&kw41z_context_data.seq_retval,
879 						0);
880 					restart_rx = 0U;
881 				}
882 
883 				k_sem_give(&kw41z_context_data.seq_sync);
884 				break;
885 			default:
886 				LOG_DBG("Unhandled state: %d", state);
887 				restart_rx = 1U;
888 				break;
889 			}
890 		}
891 	} else {
892 		/* Timer 3 Compare Match */
893 		if ((irqsts & ZLL_IRQSTS_TMR3IRQ_MASK) &&
894 			(!(irqsts & ZLL_IRQSTS_TMR3MSK_MASK))) {
895 
896 			LOG_DBG("b) TMR3 timeout: irqsts: 0x%08X, "
897 				"seq_state: 0x%08X, state: %d",
898 				irqsts, seq_state, state);
899 
900 			kw41z_tmr3_disable();
901 			restart_rx = 0U;
902 			if (state != KW41Z_STATE_IDLE) {
903 				kw41z_isr_timeout_cleanup();
904 				restart_rx = 1U;
905 				/* If we are not running an automated
906 				 * sequence then handle event. TMR3 can expire
907 				 * during Recv/Ack sequence where the transmit
908 				 * of the ACK is not being interrupted.
909 				 */
910 			}
911 		}
912 	}
913 
914 	/* Restart RX */
915 	if (restart_rx) {
916 		LOG_DBG("RESET RX");
917 		kw41z_phy_abort();
918 		kw41z_set_seq_state(KW41Z_STATE_RX);
919 		kw41z_enable_seq_irq();
920 	}
921 }
922 
get_mac(const struct device * dev)923 static inline uint8_t *get_mac(const struct device *dev)
924 {
925 	struct kw41z_context *kw41z = dev->data;
926 
927 	/*
928 	 * The KW40Z has two 32-bit registers for the MAC address where
929 	 * 40 bits of the registers are factory programmed to be unique
930 	 * and the rest are to be assigned as the "company-specific" value.
931 	 * 802.15.4 defines a EUI-64 64-bit address with company specific
932 	 * being 24 or 36 bits with the unique value being 24 or 40 bits.
933 	 *
934 	 * TODO: Grab from RSIM->MAC_LSB/MAC_MSB for the unique 40 bits
935 	 *       and how to allow for a OUI portion?
936 	 */
937 
938 	sys_rand_get(kw41z->mac_addr, sizeof(kw41z->mac_addr));
939 
940 	/*
941 	 * Clear bit 0 to ensure it isn't a multicast address and set
942 	 * bit 1 to indicate address is locally administered and may
943 	 * not be globally unique.
944 	 */
945 	kw41z->mac_addr[0] = (kw41z->mac_addr[0] & ~0x01) | 0x02;
946 
947 	return kw41z->mac_addr;
948 }
949 
kw41z_init(const struct device * dev)950 static int kw41z_init(const struct device *dev)
951 {
952 	struct kw41z_context *kw41z = dev->data;
953 	xcvrStatus_t xcvrStatus;
954 
955 	xcvrStatus = XCVR_Init(ZIGBEE_MODE, DR_500KBPS);
956 	if (xcvrStatus != gXcvrSuccess_c) {
957 		return -EIO;
958 	}
959 
960 	/* Disable all timers, enable AUTOACK, mask all interrupts */
961 	ZLL->PHY_CTRL = ZLL_PHY_CTRL_CCATYPE(KW41Z_CCA_MODE1)	|
962 			ZLL_PHY_CTRL_CRC_MSK_MASK		|
963 			ZLL_PHY_CTRL_PLL_UNLOCK_MSK_MASK	|
964 			/*ZLL_PHY_CTRL_FILTERFAIL_MSK_MASK	|*/
965 			ZLL_PHY_CTRL_RX_WMRK_MSK_MASK	|
966 			ZLL_PHY_CTRL_CCAMSK_MASK		|
967 			ZLL_PHY_CTRL_RXMSK_MASK			|
968 			ZLL_PHY_CTRL_TXMSK_MASK			|
969 			ZLL_PHY_CTRL_CCABFRTX_MASK		|
970 			ZLL_PHY_CTRL_SEQMSK_MASK;
971 
972 #if CONFIG_SOC_MKW41Z4
973 	ZLL->PHY_CTRL |= ZLL_IRQSTS_WAKE_IRQ_MASK;
974 #endif
975 
976 	ZLL->PHY_CTRL |= ZLL_PHY_CTRL_AUTOACK_MASK;
977 
978 	/*
979 	 * Clear all PP IRQ bits to avoid unexpected interrupts immediately
980 	 * after init disable all timer interrupts
981 	 */
982 	ZLL->IRQSTS = ZLL->IRQSTS;
983 
984 	/* Clear HW indirect queue */
985 	ZLL->SAM_TABLE |= ZLL_SAM_TABLE_INVALIDATE_ALL_MASK;
986 
987 	/* Accept FrameVersion 0 and 1 packets, reject all others */
988 	ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_PROMISCUOUS_MASK;
989 	ZLL->RX_FRAME_FILTER &= ~ZLL_RX_FRAME_FILTER_FRM_VER_FILTER_MASK;
990 	ZLL->RX_FRAME_FILTER = ZLL_RX_FRAME_FILTER_FRM_VER_FILTER(3)	|
991 			       ZLL_RX_FRAME_FILTER_CMD_FT_MASK		|
992 			       ZLL_RX_FRAME_FILTER_DATA_FT_MASK		|
993 			       ZLL_RX_FRAME_FILTER_ACK_FT_MASK		|
994 			       ZLL_RX_FRAME_FILTER_BEACON_FT_MASK;
995 
996 	/* Set prescaler to obtain 1 symbol (16us) timebase */
997 	ZLL->TMR_PRESCALE = 0x05;
998 
999 	kw41z_tmr3_disable();
1000 
1001 	/* Compute warmup times (scaled to 16us) */
1002 	kw41z->rx_warmup_time = (XCVR_TSM->END_OF_SEQ &
1003 				 XCVR_TSM_END_OF_SEQ_END_OF_RX_WU_MASK) >>
1004 				XCVR_TSM_END_OF_SEQ_END_OF_RX_WU_SHIFT;
1005 	kw41z->tx_warmup_time = (XCVR_TSM->END_OF_SEQ &
1006 				 XCVR_TSM_END_OF_SEQ_END_OF_TX_WU_MASK) >>
1007 				XCVR_TSM_END_OF_SEQ_END_OF_TX_WU_SHIFT;
1008 
1009 	if (kw41z->rx_warmup_time & 0x0F) {
1010 		kw41z->rx_warmup_time = 1 + (kw41z->rx_warmup_time >> 4);
1011 	} else {
1012 		kw41z->rx_warmup_time = kw41z->rx_warmup_time >> 4;
1013 	}
1014 
1015 	if (kw41z->tx_warmup_time & 0x0F) {
1016 		kw41z->tx_warmup_time = 1 + (kw41z->tx_warmup_time >> 4);
1017 	} else {
1018 		kw41z->tx_warmup_time = kw41z->tx_warmup_time >> 4;
1019 	}
1020 
1021 	/* Set CCA threshold to -75 dBm */
1022 	ZLL->CCA_LQI_CTRL &= ~ZLL_CCA_LQI_CTRL_CCA1_THRESH_MASK;
1023 	ZLL->CCA_LQI_CTRL |= ZLL_CCA_LQI_CTRL_CCA1_THRESH(0xB5);
1024 
1025 	/* Set the default power level */
1026 	kw41z_set_txpower(dev, 0);
1027 
1028 	/* Adjust ACK delay to fulfill the 802.15.4 turnaround requirements */
1029 	ZLL->ACKDELAY &= ~ZLL_ACKDELAY_ACKDELAY_MASK;
1030 	ZLL->ACKDELAY |= ZLL_ACKDELAY_ACKDELAY(-8);
1031 
1032 	/* Adjust LQI compensation */
1033 	ZLL->CCA_LQI_CTRL &= ~ZLL_CCA_LQI_CTRL_LQI_OFFSET_COMP_MASK;
1034 	ZLL->CCA_LQI_CTRL |= ZLL_CCA_LQI_CTRL_LQI_OFFSET_COMP(96);
1035 
1036 	/* Enable the RxWatermark IRQ  */
1037 	ZLL->PHY_CTRL &= ~(ZLL_PHY_CTRL_RX_WMRK_MSK_MASK);
1038 	/* Set Rx watermark level */
1039 	ZLL->RX_WTR_MARK = 0;
1040 
1041 
1042 	/* Set default channel to 2405 MHZ */
1043 	kw41z_set_channel(dev, KW41Z_DEFAULT_CHANNEL);
1044 
1045 	/* Unmask Transceiver Global Interrupts */
1046 	ZLL->PHY_CTRL &= ~ZLL_PHY_CTRL_TRCV_MSK_MASK;
1047 
1048 	/* Configure Radio IRQ */
1049 	NVIC_ClearPendingIRQ(Radio_1_IRQn);
1050 	IRQ_CONNECT(Radio_1_IRQn, RADIO_0_IRQ_PRIO, kw41z_isr, 0, 0);
1051 
1052 	return 0;
1053 }
1054 
kw41z_iface_init(struct net_if * iface)1055 static void kw41z_iface_init(struct net_if *iface)
1056 {
1057 	const struct device *dev = net_if_get_device(iface);
1058 	struct kw41z_context *kw41z = dev->data;
1059 	uint8_t *mac = get_mac(dev);
1060 
1061 #if defined(CONFIG_KW41_DBG_TRACE)
1062 	kw41_dbg_idx = 0;
1063 #endif
1064 
1065 	net_if_set_link_addr(iface, mac, 8, NET_LINK_IEEE802154);
1066 	kw41z->iface = iface;
1067 	ieee802154_init(iface);
1068 }
1069 
kw41z_configure(const struct device * dev,enum ieee802154_config_type type,const struct ieee802154_config * config)1070 static int kw41z_configure(const struct device *dev,
1071 			   enum ieee802154_config_type type,
1072 			   const struct ieee802154_config *config)
1073 {
1074 	return 0;
1075 }
1076 
1077 /* driver-allocated attribute memory - constant across all driver instances */
1078 IEEE802154_DEFINE_PHY_SUPPORTED_CHANNELS(drv_attr, 11, 26);
1079 
kw41z_attr_get(const struct device * dev,enum ieee802154_attr attr,struct ieee802154_attr_value * value)1080 static int kw41z_attr_get(const struct device *dev, enum ieee802154_attr attr,
1081 			  struct ieee802154_attr_value *value)
1082 {
1083 	ARG_UNUSED(dev);
1084 
1085 	return ieee802154_attr_get_channel_page_and_range(
1086 		attr, IEEE802154_ATTR_PHY_CHANNEL_PAGE_ZERO_OQPSK_2450_BPSK_868_915,
1087 		&drv_attr.phy_supported_channels, value);
1088 }
1089 
1090 static const struct ieee802154_radio_api kw41z_radio_api = {
1091 	.iface_api.init	= kw41z_iface_init,
1092 
1093 	.get_capabilities	= kw41z_get_capabilities,
1094 	.cca			= kw41z_cca,
1095 	.set_channel		= kw41z_set_channel,
1096 	.filter			= kw41z_filter,
1097 	.set_txpower		= kw41z_set_txpower,
1098 	.start			= kw41z_start,
1099 	.stop			= kw41z_stop,
1100 	.tx			= kw41z_tx,
1101 	.configure		= kw41z_configure,
1102 	.attr_get		= kw41z_attr_get,
1103 };
1104 
1105 #if defined(CONFIG_NET_L2_IEEE802154)
1106 
1107 #define L2 IEEE802154_L2
1108 #define L2_CTX_TYPE NET_L2_GET_CTX_TYPE(IEEE802154_L2)
1109 #define MTU KW41Z_PSDU_LENGTH
1110 
1111 #elif defined(CONFIG_NET_L2_OPENTHREAD)
1112 
1113 #define L2 OPENTHREAD_L2
1114 #define L2_CTX_TYPE NET_L2_GET_CTX_TYPE(OPENTHREAD_L2)
1115 #define MTU 1280
1116 
1117 #endif
1118 
1119 NET_DEVICE_DT_INST_DEFINE(
1120 	0,
1121 	kw41z_init,                         /* Initialization Function */
1122 	NULL,              /* No PM API support */
1123 	&kw41z_context_data,                /* Context data */
1124 	NULL,                               /* Configuration info */
1125 	CONFIG_IEEE802154_KW41Z_INIT_PRIO,  /* Initial priority */
1126 	&kw41z_radio_api,                   /* API interface functions */
1127 	L2,                                 /* L2 */
1128 	L2_CTX_TYPE,                        /* L2 context type */
1129 	MTU);                               /* MTU size */
1130