1 /*
2  * Copyright (c) 2019 Manivannan Sadhasivam
3  * Copyright (c) 2020 Andreas Sandberg
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #include <zephyr/drivers/gpio.h>
9 #include <zephyr/drivers/lora.h>
10 #include <zephyr/drivers/spi.h>
11 #include <zephyr/kernel.h>
12 
13 #include <sx126x/sx126x.h>
14 
15 #include "sx12xx_common.h"
16 #include "sx126x_common.h"
17 
18 #include <zephyr/logging/log.h>
19 LOG_MODULE_REGISTER(sx126x, CONFIG_LORA_LOG_LEVEL);
20 
21 BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(semtech_sx1261) +
22 	     DT_NUM_INST_STATUS_OKAY(semtech_sx1262) +
23 	     DT_NUM_INST_STATUS_OKAY(st_stm32wl_subghz_radio) <= 1,
24 	     "Multiple SX126x instances in DT");
25 
26 #define DIO2_TX_ENABLE DT_INST_PROP(0, dio2_tx_enable)
27 
28 #define HAVE_DIO3_TCXO		DT_INST_NODE_HAS_PROP(0, dio3_tcxo_voltage)
29 #if HAVE_DIO3_TCXO
30 #define TCXO_DIO3_VOLTAGE	DT_INST_PROP(0, dio3_tcxo_voltage)
31 #endif
32 
33 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_startup_delay_ms)
34 #define TCXO_POWER_STARTUP_DELAY_MS			\
35 	DT_INST_PROP(0, tcxo_power_startup_delay_ms)
36 #else
37 #define TCXO_POWER_STARTUP_DELAY_MS	0
38 #endif
39 
40 #define SX126X_CALIBRATION_ALL 0x7f
41 
42 static const struct sx126x_config dev_config = {
43 	.bus = SPI_DT_SPEC_INST_GET(0, SPI_WORD_SET(8) | SPI_TRANSFER_MSB, 0),
44 #if HAVE_GPIO_ANTENNA_ENABLE
45 	.antenna_enable = GPIO_DT_SPEC_INST_GET(0, antenna_enable_gpios),
46 #endif
47 #if HAVE_GPIO_TX_ENABLE
48 	.tx_enable = GPIO_DT_SPEC_INST_GET(0, tx_enable_gpios),
49 #endif
50 #if HAVE_GPIO_RX_ENABLE
51 	.rx_enable = GPIO_DT_SPEC_INST_GET(0, rx_enable_gpios),
52 #endif
53 };
54 
55 static struct sx126x_data dev_data;
56 
57 void SX126xWaitOnBusy(void);
58 
59 #define MODE(m) [MODE_##m] = #m
60 static const char *const mode_names[] = {
61 	MODE(SLEEP),
62 	MODE(STDBY_RC),
63 	MODE(STDBY_XOSC),
64 	MODE(FS),
65 	MODE(TX),
66 	MODE(RX),
67 	MODE(RX_DC),
68 	MODE(CAD),
69 };
70 #undef MODE
71 
sx126x_mode_name(RadioOperatingModes_t m)72 static const char *sx126x_mode_name(RadioOperatingModes_t m)
73 {
74 	static const char *unknown_mode = "unknown";
75 
76 	if (m < ARRAY_SIZE(mode_names) && mode_names[m]) {
77 		return mode_names[m];
78 	} else {
79 		return unknown_mode;
80 	}
81 }
82 
sx126x_spi_transceive(uint8_t * req_tx,uint8_t * req_rx,size_t req_len,void * data_tx,void * data_rx,size_t data_len)83 static int sx126x_spi_transceive(uint8_t *req_tx, uint8_t *req_rx,
84 				 size_t req_len, void *data_tx, void *data_rx,
85 				 size_t data_len)
86 {
87 	int ret;
88 
89 	const struct spi_buf tx_buf[] = {
90 		{
91 			.buf = req_tx,
92 			.len = req_len,
93 		},
94 		{
95 			.buf = data_tx,
96 			.len = data_len
97 		}
98 	};
99 
100 	const struct spi_buf rx_buf[] = {
101 		{
102 			.buf = req_rx,
103 			.len = req_len,
104 		},
105 		{
106 			.buf = data_rx,
107 			.len = data_len
108 		}
109 	};
110 
111 	const struct spi_buf_set tx = {
112 		.buffers = tx_buf,
113 		.count = ARRAY_SIZE(tx_buf),
114 	};
115 
116 	const struct spi_buf_set rx = {
117 		.buffers = rx_buf,
118 		.count = ARRAY_SIZE(rx_buf)
119 	};
120 
121 	/* Wake the device if necessary */
122 	SX126xCheckDeviceReady();
123 
124 	if (!req_rx && !data_rx) {
125 		ret = spi_write_dt(&dev_config.bus, &tx);
126 	} else {
127 		ret = spi_transceive_dt(&dev_config.bus, &tx, &rx);
128 	}
129 
130 	if (ret < 0) {
131 		LOG_ERR("SPI transaction failed: %i", ret);
132 	}
133 
134 	if (req_len >= 1 && req_tx[0] != RADIO_SET_SLEEP) {
135 		SX126xWaitOnBusy();
136 	}
137 	return ret;
138 }
139 
SX126xReadRegister(uint16_t address)140 uint8_t SX126xReadRegister(uint16_t address)
141 {
142 	uint8_t data;
143 
144 	SX126xReadRegisters(address, &data, 1);
145 
146 	return data;
147 }
148 
SX126xReadRegisters(uint16_t address,uint8_t * buffer,uint16_t size)149 void SX126xReadRegisters(uint16_t address, uint8_t *buffer, uint16_t size)
150 {
151 	uint8_t req[] = {
152 		RADIO_READ_REGISTER,
153 		(address >> 8) & 0xff,
154 		address & 0xff,
155 		0,
156 	};
157 
158 	LOG_DBG("Reading %" PRIu16 " registers @ 0x%" PRIx16, size, address);
159 	sx126x_spi_transceive(req, NULL, sizeof(req), NULL, buffer, size);
160 	LOG_HEXDUMP_DBG(buffer, size, "register_value");
161 }
162 
SX126xWriteRegister(uint16_t address,uint8_t value)163 void SX126xWriteRegister(uint16_t address, uint8_t value)
164 {
165 	SX126xWriteRegisters(address, &value, 1);
166 }
167 
SX126xWriteRegisters(uint16_t address,uint8_t * buffer,uint16_t size)168 void SX126xWriteRegisters(uint16_t address, uint8_t *buffer, uint16_t size)
169 {
170 	uint8_t req[] = {
171 		RADIO_WRITE_REGISTER,
172 		(address >> 8) & 0xff,
173 		address & 0xff,
174 	};
175 
176 	LOG_DBG("Writing %" PRIu16 " registers @ 0x%" PRIx16
177 		": 0x%" PRIx8 " , ...",
178 		size, address, buffer[0]);
179 	sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
180 }
181 
SX126xReadCommand(RadioCommands_t opcode,uint8_t * buffer,uint16_t size)182 uint8_t SX126xReadCommand(RadioCommands_t opcode,
183 			  uint8_t *buffer, uint16_t size)
184 {
185 	uint8_t tx_req[] = {
186 		opcode,
187 		0x00,
188 	};
189 
190 	uint8_t rx_req[sizeof(tx_req)];
191 
192 	LOG_DBG("Issuing opcode 0x%x (data size: %" PRIx16 ")",
193 		opcode, size);
194 	sx126x_spi_transceive(tx_req, rx_req, sizeof(rx_req),
195 			      NULL, buffer, size);
196 	LOG_DBG("-> status: 0x%" PRIx8, rx_req[1]);
197 	return rx_req[1];
198 }
199 
SX126xWriteCommand(RadioCommands_t opcode,uint8_t * buffer,uint16_t size)200 void SX126xWriteCommand(RadioCommands_t opcode, uint8_t *buffer, uint16_t size)
201 {
202 	uint8_t req[] = {
203 		opcode,
204 	};
205 
206 	LOG_DBG("Issuing opcode 0x%x w. %" PRIu16 " bytes of data",
207 		opcode, size);
208 	sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
209 }
210 
SX126xReadBuffer(uint8_t offset,uint8_t * buffer,uint8_t size)211 void SX126xReadBuffer(uint8_t offset, uint8_t *buffer, uint8_t size)
212 {
213 	uint8_t req[] = {
214 		RADIO_READ_BUFFER,
215 		offset,
216 		0x00,
217 	};
218 
219 	LOG_DBG("Reading buffers @ 0x%" PRIx8 " (%" PRIu8 " bytes)",
220 		offset, size);
221 	sx126x_spi_transceive(req, NULL, sizeof(req), NULL, buffer, size);
222 }
223 
SX126xWriteBuffer(uint8_t offset,uint8_t * buffer,uint8_t size)224 void SX126xWriteBuffer(uint8_t offset, uint8_t *buffer, uint8_t size)
225 {
226 	uint8_t req[] = {
227 		RADIO_WRITE_BUFFER,
228 		offset,
229 	};
230 
231 	LOG_DBG("Writing buffers @ 0x%" PRIx8 " (%" PRIu8 " bytes)",
232 		offset, size);
233 	sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
234 }
235 
SX126xAntSwOn(void)236 void SX126xAntSwOn(void)
237 {
238 #if HAVE_GPIO_ANTENNA_ENABLE
239 	LOG_DBG("Enabling antenna switch");
240 	gpio_pin_set_dt(&dev_config.antenna_enable, 1);
241 #else
242 	LOG_DBG("No antenna switch configured");
243 #endif
244 }
245 
SX126xAntSwOff(void)246 void SX126xAntSwOff(void)
247 {
248 #if HAVE_GPIO_ANTENNA_ENABLE
249 	LOG_DBG("Disabling antenna switch");
250 	gpio_pin_set_dt(&dev_config.antenna_enable, 0);
251 #else
252 	LOG_DBG("No antenna switch configured");
253 #endif
254 }
255 
sx126x_set_tx_enable(int value)256 static void sx126x_set_tx_enable(int value)
257 {
258 #if HAVE_GPIO_TX_ENABLE
259 	gpio_pin_set_dt(&dev_config.tx_enable, value);
260 #endif
261 }
262 
sx126x_set_rx_enable(int value)263 static void sx126x_set_rx_enable(int value)
264 {
265 #if HAVE_GPIO_RX_ENABLE
266 	gpio_pin_set_dt(&dev_config.rx_enable, value);
267 #endif
268 }
269 
SX126xGetOperatingMode(void)270 RadioOperatingModes_t SX126xGetOperatingMode(void)
271 {
272 	return dev_data.mode;
273 }
274 
SX126xSetOperatingMode(RadioOperatingModes_t mode)275 void SX126xSetOperatingMode(RadioOperatingModes_t mode)
276 {
277 	LOG_DBG("SetOperatingMode: %s (%i)", sx126x_mode_name(mode), mode);
278 
279 	dev_data.mode = mode;
280 
281 	/* To avoid inadvertently putting the RF switch in an
282 	 * undefined state, first disable the port we don't want to
283 	 * use and then enable the other one.
284 	 */
285 	switch (mode) {
286 	case MODE_TX:
287 		sx126x_set_rx_enable(0);
288 		sx126x_set_tx_enable(1);
289 		break;
290 
291 	case MODE_RX:
292 	case MODE_RX_DC:
293 	case MODE_CAD:
294 		sx126x_set_tx_enable(0);
295 		sx126x_set_rx_enable(1);
296 		break;
297 
298 	case MODE_SLEEP:
299 		/* Additionally disable the DIO1 interrupt to save power */
300 		sx126x_dio1_irq_disable(&dev_data);
301 		__fallthrough;
302 	default:
303 		sx126x_set_rx_enable(0);
304 		sx126x_set_tx_enable(0);
305 		break;
306 	}
307 }
308 
SX126xGetBoardTcxoWakeupTime(void)309 uint32_t SX126xGetBoardTcxoWakeupTime(void)
310 {
311 	return TCXO_POWER_STARTUP_DELAY_MS;
312 }
313 
SX126xGetDeviceId(void)314 uint8_t SX126xGetDeviceId(void)
315 {
316 	return SX126X_DEVICE_ID;
317 }
318 
SX126xIoIrqInit(DioIrqHandler dioIrq)319 void SX126xIoIrqInit(DioIrqHandler dioIrq)
320 {
321 	LOG_DBG("Configuring DIO IRQ callback");
322 	dev_data.radio_dio_irq = dioIrq;
323 }
324 
SX126xIoTcxoInit(void)325 void SX126xIoTcxoInit(void)
326 {
327 #if HAVE_DIO3_TCXO
328 	CalibrationParams_t cal = {
329 		.Value = SX126X_CALIBRATION_ALL,
330 	};
331 
332 	LOG_DBG("TCXO on DIO3");
333 
334 	/* Delay in units of 15.625 us (1/64 ms) */
335 	SX126xSetDio3AsTcxoCtrl(TCXO_DIO3_VOLTAGE,
336 				TCXO_POWER_STARTUP_DELAY_MS << 6);
337 	SX126xCalibrate(cal);
338 #else
339 	LOG_DBG("No TCXO configured");
340 #endif
341 }
342 
SX126xIoRfSwitchInit(void)343 void SX126xIoRfSwitchInit(void)
344 {
345 	LOG_DBG("Configuring DIO2");
346 	SX126xSetDio2AsRfSwitchCtrl(DIO2_TX_ENABLE);
347 }
348 
SX126xReset(void)349 void SX126xReset(void)
350 {
351 	LOG_DBG("Resetting radio");
352 
353 	sx126x_reset(&dev_data);
354 
355 	/* Device transitions to standby on reset */
356 	dev_data.mode = MODE_STDBY_RC;
357 }
358 
SX126xSetRfTxPower(int8_t power)359 void SX126xSetRfTxPower(int8_t power)
360 {
361 	LOG_DBG("power: %" PRIi8, power);
362 	sx126x_set_tx_params(power, RADIO_RAMP_40_US);
363 }
364 
SX126xWaitOnBusy(void)365 void SX126xWaitOnBusy(void)
366 {
367 	while (sx126x_is_busy(&dev_data)) {
368 		k_sleep(K_MSEC(1));
369 	}
370 }
371 
SX126xWakeup(void)372 void SX126xWakeup(void)
373 {
374 	int ret;
375 
376 	/* Reenable DIO1 when waking up */
377 	sx126x_dio1_irq_enable(&dev_data);
378 
379 	uint8_t req[] = { RADIO_GET_STATUS, 0 };
380 	const struct spi_buf tx_buf = {
381 		.buf = req,
382 		.len = sizeof(req),
383 	};
384 
385 	const struct spi_buf_set tx = {
386 		.buffers = &tx_buf,
387 		.count = 1,
388 	};
389 
390 	LOG_DBG("Sending GET_STATUS");
391 	ret = spi_write_dt(&dev_config.bus, &tx);
392 	if (ret < 0) {
393 		LOG_ERR("SPI transaction failed: %i", ret);
394 		return;
395 	}
396 
397 	LOG_DBG("Waiting for device...");
398 	SX126xWaitOnBusy();
399 	LOG_DBG("Device ready");
400 	/* This function is only called from sleep mode
401 	 * All edges on the SS SPI pin will transition the modem to
402 	 * standby mode (via startup)
403 	 */
404 	dev_data.mode = MODE_STDBY_RC;
405 }
406 
SX126xGetDio1PinState(void)407 uint32_t SX126xGetDio1PinState(void)
408 {
409 	return sx126x_get_dio1_pin_state(&dev_data);
410 }
411 
sx126x_dio1_irq_work_handler(struct k_work * work)412 static void sx126x_dio1_irq_work_handler(struct k_work *work)
413 {
414 	LOG_DBG("Processing DIO1 interrupt");
415 	if (!dev_data.radio_dio_irq) {
416 		LOG_WRN("DIO1 interrupt without valid HAL IRQ callback.");
417 		return;
418 	}
419 
420 	dev_data.radio_dio_irq(NULL);
421 	if (Radio.IrqProcess) {
422 		Radio.IrqProcess();
423 	}
424 
425 	/* Re-enable the interrupt if we are not in sleep mode */
426 	if (dev_data.mode != MODE_SLEEP) {
427 		sx126x_dio1_irq_enable(&dev_data);
428 	}
429 }
430 
sx126x_lora_init(const struct device * dev)431 static int sx126x_lora_init(const struct device *dev)
432 {
433 	const struct sx126x_config *config = dev->config;
434 	int ret;
435 
436 	LOG_DBG("Initializing sx126x");
437 
438 	if (sx12xx_configure_pin(antenna_enable, GPIO_OUTPUT_INACTIVE) ||
439 	    sx12xx_configure_pin(rx_enable, GPIO_OUTPUT_INACTIVE) ||
440 	    sx12xx_configure_pin(tx_enable, GPIO_OUTPUT_INACTIVE)) {
441 		return -EIO;
442 	}
443 
444 	k_work_init(&dev_data.dio1_irq_work, sx126x_dio1_irq_work_handler);
445 
446 	ret = sx126x_variant_init(dev);
447 	if (ret) {
448 		LOG_ERR("Variant initialization failed");
449 		return ret;
450 	}
451 
452 	if (!spi_is_ready_dt(&config->bus)) {
453 		LOG_ERR("SPI device not ready");
454 		return -ENODEV;
455 	}
456 
457 	ret = sx12xx_init(dev);
458 	if (ret < 0) {
459 		LOG_ERR("Failed to initialize SX12xx common");
460 		return ret;
461 	}
462 
463 	return 0;
464 }
465 
466 static DEVICE_API(lora, sx126x_lora_api) = {
467 	.config = sx12xx_lora_config,
468 	.send = sx12xx_lora_send,
469 	.send_async = sx12xx_lora_send_async,
470 	.recv = sx12xx_lora_recv,
471 	.recv_async = sx12xx_lora_recv_async,
472 	.test_cw = sx12xx_lora_test_cw,
473 };
474 
475 DEVICE_DT_INST_DEFINE(0, &sx126x_lora_init, NULL, &dev_data,
476 		      &dev_config, POST_KERNEL, CONFIG_LORA_INIT_PRIORITY,
477 		      &sx126x_lora_api);
478