blob: 7d376d5b29cc75e3af354aa14233ca4153408271 [file] [log] [blame]
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +01001/*
2 * Copyright (c) 2019 Manivannan Sadhasivam
3 * Copyright (c) 2020 Andreas Sandberg
4 *
5 * SPDX-License-Identifier: Apache-2.0
6 */
7
8#include <drivers/gpio.h>
9#include <drivers/lora.h>
10#include <drivers/spi.h>
11#include <zephyr.h>
12
13#include <sx126x/sx126x.h>
14
15#include "sx12xx_common.h"
16
17#include <logging/log.h>
18LOG_MODULE_REGISTER(sx126x, CONFIG_LORA_LOG_LEVEL);
19
20#if DT_HAS_COMPAT_STATUS_OKAY(semtech_sx1261)
21#define DT_DRV_COMPAT semtech_sx1261
22#define SX126X_DEVICE_ID SX1261
23#elif DT_HAS_COMPAT_STATUS_OKAY(semtech_sx1262)
24#define DT_DRV_COMPAT semtech_sx1262
25#define SX126X_DEVICE_ID SX1262
26#else
27#error No SX126x instance in device tree.
28#endif
29
30BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(semtech_sx1261) +
31 DT_NUM_INST_STATUS_OKAY(semtech_sx1262) <= 1,
32 "Multiple SX12xx instances in DT");
33
34#define HAVE_GPIO_CS DT_INST_SPI_DEV_HAS_CS_GPIOS(0)
Andreas Sandbergb5bff2c2020-07-09 22:46:14 +010035#define HAVE_GPIO_ANTENNA_ENABLE \
36 DT_INST_NODE_HAS_PROP(0, antenna_enable_gpios)
Andreas Sandberg78cc7df2020-04-17 13:36:36 +010037#define HAVE_GPIO_TX_ENABLE DT_INST_NODE_HAS_PROP(0, tx_enable_gpios)
38#define HAVE_GPIO_RX_ENABLE DT_INST_NODE_HAS_PROP(0, rx_enable_gpios)
Andreas Sandbergb5bff2c2020-07-09 22:46:14 +010039
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010040#define GPIO_CS_LABEL DT_INST_SPI_DEV_CS_GPIOS_LABEL(0)
41#define GPIO_CS_PIN DT_INST_SPI_DEV_CS_GPIOS_PIN(0)
Andreas Sandberg33d473d2020-07-05 21:44:27 +010042#define GPIO_CS_FLAGS DT_INST_SPI_DEV_CS_GPIOS_FLAGS(0)
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010043
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010044#define GPIO_RESET_PIN DT_INST_GPIO_PIN(0, reset_gpios)
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010045#define GPIO_BUSY_PIN DT_INST_GPIO_PIN(0, busy_gpios)
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010046#define GPIO_DIO1_PIN DT_INST_GPIO_PIN(0, dio1_gpios)
Andreas Sandbergb5bff2c2020-07-09 22:46:14 +010047#define GPIO_ANTENNA_ENABLE_PIN DT_INST_GPIO_PIN(0, antenna_enable_gpios)
Andreas Sandberg78cc7df2020-04-17 13:36:36 +010048#define GPIO_TX_ENABLE_PIN DT_INST_GPIO_PIN(0, tx_enable_gpios)
49#define GPIO_RX_ENABLE_PIN DT_INST_GPIO_PIN(0, rx_enable_gpios)
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010050
51#define DIO2_TX_ENABLE DT_INST_PROP(0, dio2_tx_enable)
52
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010053#define HAVE_DIO3_TCXO DT_INST_NODE_HAS_PROP(0, dio3_tcxo_voltage)
54#if HAVE_DIO3_TCXO
55#define TCXO_DIO3_VOLTAGE DT_INST_PROP(0, dio3_tcxo_voltage)
56#endif
57
58#if DT_INST_NODE_HAS_PROP(0, tcxo_power_startup_delay_ms)
59#define TCXO_POWER_STARTUP_DELAY_MS \
60 DT_INST_PROP(0, tcxo_power_startup_delay_ms)
61#else
62#define TCXO_POWER_STARTUP_DELAY_MS 0
63#endif
64
65#define SX126X_CALIBRATION_ALL 0x7f
66
67struct sx126x_data {
Tomasz Bursztykae18fcbb2020-04-30 20:33:38 +020068 const struct device *reset;
69 const struct device *busy;
70 const struct device *dio1;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010071 struct gpio_callback dio1_irq_callback;
72 struct k_work dio1_irq_work;
73 DioIrqHandler *radio_dio_irq;
74#if HAVE_GPIO_ANTENNA_ENABLE
Tomasz Bursztykae18fcbb2020-04-30 20:33:38 +020075 const struct device *antenna_enable;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010076#endif
Andreas Sandberg78cc7df2020-04-17 13:36:36 +010077#if HAVE_GPIO_TX_ENABLE
78 const struct device *tx_enable;
79#endif
80#if HAVE_GPIO_RX_ENABLE
81 const struct device *rx_enable;
82#endif
Tomasz Bursztykae18fcbb2020-04-30 20:33:38 +020083 const struct device *spi;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010084 struct spi_config spi_cfg;
85#if HAVE_GPIO_CS
86 struct spi_cs_control spi_cs;
87#endif
Andreas Sandberg78cc7df2020-04-17 13:36:36 +010088 RadioOperatingModes_t mode;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +010089} dev_data;
90
91
92void SX126xWaitOnBusy(void);
93
Andreas Sandberg78cc7df2020-04-17 13:36:36 +010094#define MODE(m) [MODE_##m] = #m
95static const char *const mode_names[] = {
96 MODE(SLEEP),
97 MODE(STDBY_RC),
98 MODE(STDBY_XOSC),
99 MODE(FS),
100 MODE(TX),
101 MODE(RX),
102 MODE(RX_DC),
103 MODE(CAD),
104};
105#undef MODE
106
107static const char *sx126x_mode_name(RadioOperatingModes_t m)
108{
109 static const char *unknown_mode = "unknown";
110
111 if (m < ARRAY_SIZE(mode_names) && mode_names[m]) {
112 return mode_names[m];
113 } else {
114 return unknown_mode;
115 }
116}
117
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100118static int sx126x_spi_transceive(uint8_t *req_tx, uint8_t *req_rx,
119 size_t req_len, void *data_tx, void *data_rx,
120 size_t data_len)
121{
122 int ret;
123
124 const struct spi_buf tx_buf[] = {
125 {
126 .buf = req_tx,
127 .len = req_len,
128 },
129 {
130 .buf = data_tx,
131 .len = data_len
132 }
133 };
134
135 const struct spi_buf rx_buf[] = {
136 {
137 .buf = req_rx,
138 .len = req_len,
139 },
140 {
141 .buf = data_rx,
142 .len = data_len
143 }
144 };
145
146 const struct spi_buf_set tx = {
147 .buffers = tx_buf,
148 .count = ARRAY_SIZE(tx_buf),
149 };
150
151 const struct spi_buf_set rx = {
152 .buffers = rx_buf,
153 .count = ARRAY_SIZE(rx_buf)
154 };
155
156 /* Wake the device if necessary */
157 SX126xCheckDeviceReady();
158
159 if (!req_rx && !data_rx) {
160 ret = spi_write(dev_data.spi, &dev_data.spi_cfg, &tx);
161 } else {
162 ret = spi_transceive(dev_data.spi, &dev_data.spi_cfg,
163 &tx, &rx);
164 }
165
166 if (ret < 0) {
167 LOG_ERR("SPI transaction failed: %i", ret);
168 }
169
170 if (req_len >= 1 && req_tx[0] != RADIO_SET_SLEEP) {
171 SX126xWaitOnBusy();
172 }
173 return ret;
174}
175
176uint8_t SX126xReadRegister(uint16_t address)
177{
178 uint8_t data;
179
180 SX126xReadRegisters(address, &data, 1);
181
182 return data;
183}
184
185void SX126xReadRegisters(uint16_t address, uint8_t *buffer, uint16_t size)
186{
187 uint8_t req[] = {
188 RADIO_READ_REGISTER,
189 (address >> 8) & 0xff,
190 address & 0xff,
191 0,
192 };
193
194 LOG_DBG("Reading %" PRIu16 " registers @ 0x%" PRIx16, size, address);
195 sx126x_spi_transceive(req, NULL, sizeof(req), NULL, buffer, size);
196 LOG_HEXDUMP_DBG(buffer, size, "register_value");
197}
198
199void SX126xWriteRegister(uint16_t address, uint8_t value)
200{
201 SX126xWriteRegisters(address, &value, 1);
202}
203
204void SX126xWriteRegisters(uint16_t address, uint8_t *buffer, uint16_t size)
205{
206 uint8_t req[] = {
207 RADIO_WRITE_REGISTER,
208 (address >> 8) & 0xff,
209 address & 0xff,
210 };
211
212 LOG_DBG("Writing %" PRIu16 " registers @ 0x%" PRIx16
213 ": 0x%" PRIx8 " , ...",
214 size, address, buffer[0]);
215 sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
216}
217
218uint8_t SX126xReadCommand(RadioCommands_t opcode,
219 uint8_t *buffer, uint16_t size)
220{
221 uint8_t tx_req[] = {
222 opcode,
223 0x00,
224 };
225
226 uint8_t rx_req[sizeof(tx_req)];
227
228 LOG_DBG("Issuing opcode 0x%x (data size: %" PRIx16 ")",
229 opcode, size);
230 sx126x_spi_transceive(tx_req, rx_req, sizeof(rx_req),
231 NULL, buffer, size);
232 LOG_DBG("-> status: 0x%" PRIx8, rx_req[1]);
233 return rx_req[1];
234}
235
236void SX126xWriteCommand(RadioCommands_t opcode, uint8_t *buffer, uint16_t size)
237{
238 uint8_t req[] = {
239 opcode,
240 };
241
242 LOG_DBG("Issuing opcode 0x%x w. %" PRIu16 " bytes of data",
243 opcode, size);
244 sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
245}
246
247void SX126xReadBuffer(uint8_t offset, uint8_t *buffer, uint8_t size)
248{
249 uint8_t req[] = {
250 RADIO_READ_BUFFER,
251 offset,
252 0x00,
253 };
254
255 LOG_DBG("Reading buffers @ 0x%" PRIx8 " (%" PRIu8 " bytes)",
256 offset, size);
257 sx126x_spi_transceive(req, NULL, sizeof(req), NULL, buffer, size);
258}
259
260void SX126xWriteBuffer(uint8_t offset, uint8_t *buffer, uint8_t size)
261{
262 uint8_t req[] = {
263 RADIO_WRITE_BUFFER,
264 offset,
265 };
266
267 LOG_DBG("Writing buffers @ 0x%" PRIx8 " (%" PRIu8 " bytes)",
268 offset, size);
269 sx126x_spi_transceive(req, NULL, sizeof(req), buffer, NULL, size);
270}
271
272void SX126xAntSwOn(void)
273{
274#if HAVE_GPIO_ANTENNA_ENABLE
275 LOG_DBG("Enabling antenna switch");
276 gpio_pin_set(dev_data.antenna_enable, GPIO_ANTENNA_ENABLE_PIN, 1);
277#else
278 LOG_DBG("No antenna switch configured");
279#endif
280}
281
282void SX126xAntSwOff(void)
283{
284#if HAVE_GPIO_ANTENNA_ENABLE
285 LOG_DBG("Disabling antenna switch");
286 gpio_pin_set(dev_data.antenna_enable, GPIO_ANTENNA_ENABLE_PIN, 0);
287#else
288 LOG_DBG("No antenna switch configured");
289#endif
290}
291
Andreas Sandberg78cc7df2020-04-17 13:36:36 +0100292static void sx126x_set_tx_enable(int value)
293{
294#if HAVE_GPIO_TX_ENABLE
295 gpio_pin_set(dev_data.tx_enable, GPIO_TX_ENABLE_PIN, value);
296#endif
297}
298
299static void sx126x_set_rx_enable(int value)
300{
301#if HAVE_GPIO_RX_ENABLE
302 gpio_pin_set(dev_data.rx_enable, GPIO_RX_ENABLE_PIN, value);
303#endif
304}
305
306RadioOperatingModes_t SX126xGetOperatingMode(void)
307{
308 return dev_data.mode;
309}
310
311void SX126xSetOperatingMode(RadioOperatingModes_t mode)
312{
313 LOG_DBG("SetOperatingMode: %s (%i)", sx126x_mode_name(mode), mode);
314
315 dev_data.mode = mode;
316
317 /* To avoid inadvertently putting the RF switch in an
318 * undefined state, first disable the port we don't want to
319 * use and then enable the other one.
320 */
321 switch (mode) {
322 case MODE_TX:
323 sx126x_set_rx_enable(0);
324 sx126x_set_tx_enable(1);
325 break;
326
327 case MODE_RX:
328 case MODE_RX_DC:
329 case MODE_CAD:
330 sx126x_set_tx_enable(0);
331 sx126x_set_rx_enable(1);
332 break;
333
Jordan Yatese756ad62021-01-24 16:55:32 +1000334 case MODE_SLEEP:
335 /* Additionally disable the DIO1 interrupt to save power */
336 gpio_pin_interrupt_configure(dev_data.dio1, GPIO_DIO1_PIN,
337 GPIO_INT_DISABLE);
338 __fallthrough;
Andreas Sandberg78cc7df2020-04-17 13:36:36 +0100339 default:
340 sx126x_set_rx_enable(0);
341 sx126x_set_tx_enable(0);
342 break;
343 }
344}
345
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100346uint32_t SX126xGetBoardTcxoWakeupTime(void)
347{
348 return TCXO_POWER_STARTUP_DELAY_MS;
349}
350
351uint8_t SX126xGetDeviceId(void)
352{
353 return SX126X_DEVICE_ID;
354}
355
356void SX126xIoIrqInit(DioIrqHandler dioIrq)
357{
358 LOG_DBG("Configuring DIO IRQ callback");
359 dev_data.radio_dio_irq = dioIrq;
360}
361
362void SX126xIoTcxoInit(void)
363{
364#if HAVE_DIO3_TCXO
365 CalibrationParams_t cal = {
366 .Value = SX126X_CALIBRATION_ALL,
367 };
368
369 LOG_DBG("TCXO on DIO3");
370
371 /* Delay in units of 15.625 us (1/64 ms) */
372 SX126xSetDio3AsTcxoCtrl(TCXO_DIO3_VOLTAGE,
373 TCXO_POWER_STARTUP_DELAY_MS << 6);
374 SX126xCalibrate(cal);
375#else
376 LOG_DBG("No TCXO configured");
377#endif
378}
379
Andreas Sandberg78cc7df2020-04-17 13:36:36 +0100380void SX126xIoRfSwitchInit(void)
381{
382 LOG_DBG("Configuring DIO2");
383 SX126xSetDio2AsRfSwitchCtrl(DIO2_TX_ENABLE);
384}
385
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100386void SX126xReset(void)
387{
388 LOG_DBG("Resetting radio");
389 gpio_pin_set(dev_data.reset, GPIO_RESET_PIN, 1);
390 k_sleep(K_MSEC(20));
391 gpio_pin_set(dev_data.reset, GPIO_RESET_PIN, 0);
392 k_sleep(K_MSEC(10));
Jordan Yates67765352021-01-24 16:38:13 +1000393 /* Device transitions to standby on reset */
394 dev_data.mode = MODE_STDBY_RC;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100395}
396
397void SX126xSetRfTxPower(int8_t power)
398{
399 LOG_DBG("power: %" PRIi8, power);
400 SX126xSetTxParams(power, RADIO_RAMP_40_US);
401}
402
403void SX126xWaitOnBusy(void)
404{
405 while (gpio_pin_get(dev_data.busy, GPIO_BUSY_PIN)) {
406 k_sleep(K_MSEC(1));
407 }
408}
409
410void SX126xWakeup(void)
411{
412 int ret;
413
Jordan Yatese756ad62021-01-24 16:55:32 +1000414 /* Reenable DIO1 when waking up */
415 gpio_pin_interrupt_configure(dev_data.dio1, GPIO_DIO1_PIN,
416 GPIO_INT_EDGE_TO_ACTIVE);
417
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100418 uint8_t req[] = { RADIO_GET_STATUS, 0 };
419 const struct spi_buf tx_buf = {
420 .buf = req,
421 .len = sizeof(req),
422 };
423
424 const struct spi_buf_set tx = {
425 .buffers = &tx_buf,
426 .count = 1,
427 };
428
429 LOG_DBG("Sending GET_STATUS");
430 ret = spi_write(dev_data.spi, &dev_data.spi_cfg, &tx);
431 if (ret < 0) {
432 LOG_ERR("SPI transaction failed: %i", ret);
433 return;
434 }
435
436 LOG_DBG("Waiting for device...");
437 SX126xWaitOnBusy();
438 LOG_DBG("Device ready");
Jordan Yates67765352021-01-24 16:38:13 +1000439 /* This function is only called from sleep mode
440 * All edges on the SS SPI pin will transition the modem to
441 * standby mode (via startup)
442 */
443 dev_data.mode = MODE_STDBY_RC;
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100444}
445
446static void sx126x_dio1_irq_work_handler(struct k_work *work)
447{
448 LOG_DBG("Processing DIO1 interrupt");
449 if (!dev_data.radio_dio_irq) {
450 LOG_WRN("DIO1 interrupt without valid HAL IRQ callback.");
451 return;
452 }
453
454 dev_data.radio_dio_irq(NULL);
455 if (Radio.IrqProcess) {
456 Radio.IrqProcess();
457 }
458}
459
Tomasz Bursztykae18fcbb2020-04-30 20:33:38 +0200460static void sx126x_dio1_irq_callback(const struct device *dev,
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100461 struct gpio_callback *cb, uint32_t pins)
462{
463 if (pins & BIT(GPIO_DIO1_PIN)) {
464 k_work_submit(&dev_data.dio1_irq_work);
465 }
466}
467
Tomasz Bursztykae18fcbb2020-04-30 20:33:38 +0200468static int sx126x_lora_init(const struct device *dev)
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100469{
470 int ret;
471
472 LOG_DBG("Initializing %s", DT_INST_LABEL(0));
473
Andreas Sandbergb5bff2c2020-07-09 22:46:14 +0100474 if (sx12xx_configure_pin(reset, GPIO_OUTPUT_ACTIVE) ||
475 sx12xx_configure_pin(busy, GPIO_INPUT) ||
476 sx12xx_configure_pin(dio1, GPIO_INPUT | GPIO_INT_DEBOUNCE) ||
Andreas Sandberg78cc7df2020-04-17 13:36:36 +0100477 sx12xx_configure_pin(antenna_enable, GPIO_OUTPUT_INACTIVE) ||
478 sx12xx_configure_pin(rx_enable, GPIO_OUTPUT_INACTIVE) ||
479 sx12xx_configure_pin(tx_enable, GPIO_OUTPUT_INACTIVE)) {
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100480 return -EIO;
481 }
482
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100483 k_work_init(&dev_data.dio1_irq_work, sx126x_dio1_irq_work_handler);
484
485 gpio_init_callback(&dev_data.dio1_irq_callback,
486 sx126x_dio1_irq_callback, BIT(GPIO_DIO1_PIN));
487 if (gpio_add_callback(dev_data.dio1, &dev_data.dio1_irq_callback) < 0) {
488 LOG_ERR("Could not set GPIO callback for DIO1 interrupt.");
489 return -EIO;
490 }
491
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100492 dev_data.spi = device_get_binding(DT_INST_BUS_LABEL(0));
493 if (!dev_data.spi) {
494 LOG_ERR("Cannot get pointer to %s device",
495 DT_INST_BUS_LABEL(0));
496 return -EINVAL;
497 }
498
499#if HAVE_GPIO_CS
500 dev_data.spi_cs.gpio_dev = device_get_binding(GPIO_CS_LABEL);
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100501 if (!dev_data.spi_cs.gpio_dev) {
502 LOG_ERR("Cannot get pointer to %s device", GPIO_CS_LABEL);
503 return -EIO;
504 }
505
Andreas Sandberg33d473d2020-07-05 21:44:27 +0100506 dev_data.spi_cs.gpio_pin = GPIO_CS_PIN;
507 dev_data.spi_cs.gpio_dt_flags = GPIO_CS_FLAGS;
508 dev_data.spi_cs.delay = 0U;
509
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100510 dev_data.spi_cfg.cs = &dev_data.spi_cs;
511#endif
512 dev_data.spi_cfg.operation = SPI_WORD_SET(8) | SPI_TRANSFER_MSB;
513 dev_data.spi_cfg.frequency = DT_INST_PROP(0, spi_max_frequency);
514 dev_data.spi_cfg.slave = DT_INST_REG_ADDR(0);
515
Andreas Sandberg78cc7df2020-04-17 13:36:36 +0100516
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100517 ret = sx12xx_init(dev);
518 if (ret < 0) {
519 LOG_ERR("Failed to initialize SX12xx common");
520 return ret;
521 }
522
523 return 0;
524}
525
526static const struct lora_driver_api sx126x_lora_api = {
527 .config = sx12xx_lora_config,
528 .send = sx12xx_lora_send,
529 .recv = sx12xx_lora_recv,
530 .test_cw = sx12xx_lora_test_cw,
531};
532
Gerard Marull-Paretas496cdac2021-04-28 11:16:35 +0200533DEVICE_DT_INST_DEFINE(0, &sx126x_lora_init, NULL, NULL,
Andreas Sandberg6a4ebe52020-04-17 13:36:36 +0100534 NULL, POST_KERNEL, CONFIG_LORA_INIT_PRIORITY,
535 &sx126x_lora_api);