1 /*
2 * Copyright (c) 2019 Manivannan Sadhasivam
3 * Copyright (c) 2020 Grinn
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 "sx12xx_common.h"
14
15 #include <zephyr/logging/log.h>
16 LOG_MODULE_REGISTER(sx127x, CONFIG_LORA_LOG_LEVEL);
17
18 #if DT_HAS_COMPAT_STATUS_OKAY(semtech_sx1272)
19
20 #include <sx1272/sx1272.h>
21
22 #define DT_DRV_COMPAT semtech_sx1272
23
24 #define SX127xCheckRfFrequency SX1272CheckRfFrequency
25 #define SX127xGetBoardTcxoWakeupTime SX1272GetBoardTcxoWakeupTime
26 #define SX127xSetAntSwLowPower SX1272SetAntSwLowPower
27 #define SX127xSetBoardTcxo SX1272SetBoardTcxo
28 #define SX127xSetAntSw SX1272SetAntSw
29 #define SX127xReset SX1272Reset
30 #define SX127xIoIrqInit SX1272IoIrqInit
31 #define SX127xWriteBuffer SX1272WriteBuffer
32 #define SX127xReadBuffer SX1272ReadBuffer
33 #define SX127xSetRfTxPower SX1272SetRfTxPower
34 #define SX127xGetDio1PinState SX1272GetDio1PinState
35 #define SX127xInit SX1272Init
36 #define SX127xGetStatus SX1272GetStatus
37 #define SX127xSetModem SX1272SetModem
38 #define SX127xSetChannel SX1272SetChannel
39 #define SX127xIsChannelFree SX1272IsChannelFree
40 #define SX127xRandom SX1272Random
41 #define SX127xSetRxConfig SX1272SetRxConfig
42 #define SX127xSetTxConfig SX1272SetTxConfig
43 #define SX127xGetTimeOnAir SX1272GetTimeOnAir
44 #define SX127xSend SX1272Send
45 #define SX127xSetSleep SX1272SetSleep
46 #define SX127xSetStby SX1272SetStby
47 #define SX127xSetRx SX1272SetRx
48 #define SX127xWrite SX1272Write
49 #define SX127xRead SX1272Read
50 #define SX127xSetMaxPayloadLength SX1272SetMaxPayloadLength
51 #define SX127xSetPublicNetwork SX1272SetPublicNetwork
52 #define SX127xGetWakeupTime SX1272GetWakeupTime
53 #define SX127xSetTxContinuousWave SX1272SetTxContinuousWave
54
55 #elif DT_HAS_COMPAT_STATUS_OKAY(semtech_sx1276)
56
57 #include <sx1276/sx1276.h>
58
59 #define DT_DRV_COMPAT semtech_sx1276
60
61 #define SX127xCheckRfFrequency SX1276CheckRfFrequency
62 #define SX127xGetBoardTcxoWakeupTime SX1276GetBoardTcxoWakeupTime
63 #define SX127xSetAntSwLowPower SX1276SetAntSwLowPower
64 #define SX127xSetBoardTcxo SX1276SetBoardTcxo
65 #define SX127xSetAntSw SX1276SetAntSw
66 #define SX127xReset SX1276Reset
67 #define SX127xIoIrqInit SX1276IoIrqInit
68 #define SX127xWriteBuffer SX1276WriteBuffer
69 #define SX127xReadBuffer SX1276ReadBuffer
70 #define SX127xSetRfTxPower SX1276SetRfTxPower
71 #define SX127xGetDio1PinState SX1276GetDio1PinState
72 #define SX127xInit SX1276Init
73 #define SX127xGetStatus SX1276GetStatus
74 #define SX127xSetModem SX1276SetModem
75 #define SX127xSetChannel SX1276SetChannel
76 #define SX127xIsChannelFree SX1276IsChannelFree
77 #define SX127xRandom SX1276Random
78 #define SX127xSetRxConfig SX1276SetRxConfig
79 #define SX127xSetTxConfig SX1276SetTxConfig
80 #define SX127xGetTimeOnAir SX1276GetTimeOnAir
81 #define SX127xSend SX1276Send
82 #define SX127xSetSleep SX1276SetSleep
83 #define SX127xSetStby SX1276SetStby
84 #define SX127xSetRx SX1276SetRx
85 #define SX127xWrite SX1276Write
86 #define SX127xRead SX1276Read
87 #define SX127xSetMaxPayloadLength SX1276SetMaxPayloadLength
88 #define SX127xSetPublicNetwork SX1276SetPublicNetwork
89 #define SX127xGetWakeupTime SX1276GetWakeupTime
90 #define SX127xSetTxContinuousWave SX1276SetTxContinuousWave
91
92 #else
93 #error No SX127x instance in device tree.
94 #endif
95
96 BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(semtech_sx1272) +
97 DT_NUM_INST_STATUS_OKAY(semtech_sx1276) <= 1,
98 "Multiple SX127x instances in DT");
99
100 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_startup_delay_ms)
101 #define TCXO_POWER_STARTUP_DELAY_MS \
102 DT_INST_PROP(0, tcxo_power_startup_delay_ms)
103 #else
104 #define TCXO_POWER_STARTUP_DELAY_MS 0
105 #endif
106
107 /*
108 * Those macros must be in sync with 'power-amplifier-output' dts property.
109 */
110 #define SX127X_PA_RFO 0
111 #define SX127X_PA_BOOST 1
112
113 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios) && \
114 DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
115 #define SX127X_PA_OUTPUT(power) \
116 ((power) > 14 ? SX127X_PA_BOOST : SX127X_PA_RFO)
117 #elif DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios)
118 #define SX127X_PA_OUTPUT(power) SX127X_PA_RFO
119 #elif DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
120 #define SX127X_PA_OUTPUT(power) SX127X_PA_BOOST
121 #elif DT_INST_NODE_HAS_PROP(0, power_amplifier_output)
122 #define SX127X_PA_OUTPUT(power) \
123 DT_INST_ENUM_IDX(0, power_amplifier_output)
124 #else
125 BUILD_ASSERT(0, "None of rfo-enable-gpios, pa-boost-enable-gpios and "
126 "power-amplifier-output has been specified. "
127 "Look at semtech,sx127x-base.yaml to fix that.");
128 #endif
129
130 #define SX127X_PADAC_20DBM_ON (RF_PADAC_20DBM_ON)
131 #define SX127X_PADAC_20DBM_OFF (RF_PADAC_20DBM_OFF)
132 #define SX127X_PADAC_20DBM_MASK (~RF_PADAC_20DBM_MASK)
133
134 #define SX127X_PACONFIG_PASELECT_PABOOST (RF_PACONFIG_PASELECT_PABOOST)
135 #define SX127X_PACONFIG_OUTPUTPOWER_MASK (~RF_PACONFIG_OUTPUTPOWER_MASK)
136
137 #ifdef RF_PACONFIG_MAX_POWER_MASK
138 #define SX127X_PACONFIG_MAX_POWER_SHIFT 4
139 #endif
140
141 extern DioIrqHandler *DioIrq[];
142
143 /* Helper macro that UTIL_LISTIFY can use and produces an element with comma */
144 #define SX127X_DIO_GPIO_LEN(inst) \
145 DT_INST_PROP_LEN(inst, dio_gpios)
146
147 #define SX127X_DIO_GPIO_ELEM(idx, inst) \
148 GPIO_DT_SPEC_INST_GET_BY_IDX(inst, dio_gpios, idx)
149
150 #define SX127X_DIO_GPIO_INIT(n) \
151 LISTIFY(SX127X_DIO_GPIO_LEN(n), SX127X_DIO_GPIO_ELEM, (,), n)
152
153 static const struct gpio_dt_spec sx127x_dios[] = { SX127X_DIO_GPIO_INIT(0) };
154
155 #define SX127X_MAX_DIO ARRAY_SIZE(sx127x_dios)
156
157 struct sx127x_config {
158 struct spi_dt_spec bus;
159 struct gpio_dt_spec reset;
160 #if DT_INST_NODE_HAS_PROP(0, antenna_enable_gpios)
161 struct gpio_dt_spec antenna_enable;
162 #endif
163 #if DT_INST_NODE_HAS_PROP(0, rfi_enable_gpios)
164 struct gpio_dt_spec rfi_enable;
165 #endif
166 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios)
167 struct gpio_dt_spec rfo_enable;
168 #endif
169 #if DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
170 struct gpio_dt_spec pa_boost_enable;
171 #endif
172 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_gpios)
173 struct gpio_dt_spec tcxo_power;
174 #endif
175 };
176
177 static const struct sx127x_config dev_config = {
178 .bus = SPI_DT_SPEC_INST_GET(0, SPI_WORD_SET(8) | SPI_TRANSFER_MSB, 0),
179 .reset = GPIO_DT_SPEC_INST_GET(0, reset_gpios),
180 #if DT_INST_NODE_HAS_PROP(0, antenna_enable_gpios)
181 .antenna_enable = GPIO_DT_SPEC_INST_GET(0, antenna_enable_gpios),
182 #endif
183 #if DT_INST_NODE_HAS_PROP(0, rfi_enable_gpios)
184 .rfi_enable = GPIO_DT_SPEC_INST_GET(0, rfi_enable_gpios),
185 #endif
186 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios)
187 .rfo_enable = GPIO_DT_SPEC_INST_GET(0, rfo_enable_gpios),
188 #endif
189 #if DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
190 .pa_boost_enable = GPIO_DT_SPEC_INST_GET(0, pa_boost_enable_gpios),
191 #endif
192 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_gpios)
193 .tcxo_power = GPIO_DT_SPEC_INST_GET(0, tcxo_power_gpios),
194 #endif
195 };
196
197 static struct sx127x_data {
198 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios) && \
199 DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
200 uint8_t tx_power;
201 #endif
202 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_gpios)
203 bool tcxo_power_enabled;
204 #endif
205 struct k_work dio_work[SX127X_MAX_DIO];
206 } dev_data;
207
clamp_int8(int8_t x,int8_t min,int8_t max)208 static int8_t clamp_int8(int8_t x, int8_t min, int8_t max)
209 {
210 if (x < min) {
211 return min;
212 } else if (x > max) {
213 return max;
214 } else {
215 return x;
216 }
217 }
218
SX127xCheckRfFrequency(uint32_t frequency)219 bool SX127xCheckRfFrequency(uint32_t frequency)
220 {
221 /* TODO */
222 return true;
223 }
224
SX127xGetBoardTcxoWakeupTime(void)225 uint32_t SX127xGetBoardTcxoWakeupTime(void)
226 {
227 return TCXO_POWER_STARTUP_DELAY_MS;
228 }
229
sx127x_antenna_enable(int val)230 static inline void sx127x_antenna_enable(int val)
231 {
232 #if DT_INST_NODE_HAS_PROP(0, antenna_enable_gpios)
233 gpio_pin_set_dt(&dev_config.antenna_enable, val);
234 #endif
235 }
236
sx127x_rfi_enable(int val)237 static inline void sx127x_rfi_enable(int val)
238 {
239 #if DT_INST_NODE_HAS_PROP(0, rfi_enable_gpios)
240 gpio_pin_set_dt(&dev_config.rfi_enable, val);
241 #endif
242 }
243
sx127x_rfo_enable(int val)244 static inline void sx127x_rfo_enable(int val)
245 {
246 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios)
247 gpio_pin_set_dt(&dev_config.rfo_enable, val);
248 #endif
249 }
250
sx127x_pa_boost_enable(int val)251 static inline void sx127x_pa_boost_enable(int val)
252 {
253 #if DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
254 gpio_pin_set_dt(&dev_config.pa_boost_enable, val);
255 #endif
256 }
257
SX127xSetAntSwLowPower(bool low_power)258 void SX127xSetAntSwLowPower(bool low_power)
259 {
260 if (low_power) {
261 /* force inactive (low power) state of all antenna paths */
262 sx127x_rfi_enable(0);
263 sx127x_rfo_enable(0);
264 sx127x_pa_boost_enable(0);
265
266 sx127x_antenna_enable(0);
267 } else {
268 sx127x_antenna_enable(1);
269
270 /* rely on SX127xSetAntSw() to configure proper antenna path */
271 }
272 }
273
SX127xSetBoardTcxo(uint8_t state)274 void SX127xSetBoardTcxo(uint8_t state)
275 {
276 #if DT_INST_NODE_HAS_PROP(0, tcxo_power_gpios)
277 bool enable = state;
278
279 if (enable == dev_data.tcxo_power_enabled) {
280 return;
281 }
282
283 if (enable) {
284 gpio_pin_set_dt(&dev_config.tcxo_power, 1);
285
286 if (TCXO_POWER_STARTUP_DELAY_MS > 0) {
287 k_sleep(K_MSEC(TCXO_POWER_STARTUP_DELAY_MS));
288 }
289 } else {
290 gpio_pin_set_dt(&dev_config.tcxo_power, 0);
291 }
292
293 dev_data.tcxo_power_enabled = enable;
294 #endif
295 }
296
SX127xSetAntSw(uint8_t opMode)297 void SX127xSetAntSw(uint8_t opMode)
298 {
299 switch (opMode) {
300 case RFLR_OPMODE_TRANSMITTER:
301 sx127x_rfi_enable(0);
302
303 if (SX127X_PA_OUTPUT(dev_data.tx_power) == SX127X_PA_BOOST) {
304 sx127x_rfo_enable(0);
305 sx127x_pa_boost_enable(1);
306 } else {
307 sx127x_pa_boost_enable(0);
308 sx127x_rfo_enable(1);
309 }
310 break;
311 default:
312 sx127x_rfo_enable(0);
313 sx127x_pa_boost_enable(0);
314 sx127x_rfi_enable(1);
315 break;
316 }
317 }
318
SX127xReset(void)319 void SX127xReset(void)
320 {
321 SX127xSetBoardTcxo(true);
322
323 gpio_pin_set_dt(&dev_config.reset, 1);
324
325 k_sleep(K_MSEC(1));
326
327 gpio_pin_set_dt(&dev_config.reset, 0);
328
329 k_sleep(K_MSEC(6));
330 }
331
sx127x_dio_work_handle(struct k_work * work)332 static void sx127x_dio_work_handle(struct k_work *work)
333 {
334 int dio = work - dev_data.dio_work;
335
336 (*DioIrq[dio])(NULL);
337 }
338
sx127x_irq_callback(const struct device * dev,struct gpio_callback * cb,uint32_t pins)339 static void sx127x_irq_callback(const struct device *dev,
340 struct gpio_callback *cb, uint32_t pins)
341 {
342 unsigned int i, pin;
343
344 pin = find_lsb_set(pins) - 1;
345
346 for (i = 0; i < SX127X_MAX_DIO; i++) {
347 if (dev == sx127x_dios[i].port &&
348 pin == sx127x_dios[i].pin) {
349 k_work_submit(&dev_data.dio_work[i]);
350 }
351 }
352 }
353
SX127xIoIrqInit(DioIrqHandler ** irqHandlers)354 void SX127xIoIrqInit(DioIrqHandler **irqHandlers)
355 {
356 unsigned int i;
357 static struct gpio_callback callbacks[SX127X_MAX_DIO];
358
359 /* Setup DIO gpios */
360 for (i = 0; i < SX127X_MAX_DIO; i++) {
361 if (!irqHandlers[i]) {
362 continue;
363 }
364
365 if (!gpio_is_ready_dt(&sx127x_dios[i])) {
366 LOG_ERR("GPIO port %s not ready",
367 sx127x_dios[i].port->name);
368 return;
369 }
370
371 k_work_init(&dev_data.dio_work[i], sx127x_dio_work_handle);
372
373 gpio_pin_configure_dt(&sx127x_dios[i], GPIO_INPUT);
374
375 gpio_init_callback(&callbacks[i],
376 sx127x_irq_callback,
377 BIT(sx127x_dios[i].pin));
378
379 if (gpio_add_callback(sx127x_dios[i].port, &callbacks[i]) < 0) {
380 LOG_ERR("Could not set gpio callback.");
381 return;
382 }
383 gpio_pin_interrupt_configure_dt(&sx127x_dios[i],
384 GPIO_INT_EDGE_TO_ACTIVE);
385 }
386
387 }
388
sx127x_transceive(uint8_t reg,bool write,void * data,size_t length)389 static int sx127x_transceive(uint8_t reg, bool write, void *data, size_t length)
390 {
391 const struct spi_buf buf[2] = {
392 {
393 .buf = ®,
394 .len = sizeof(reg)
395 },
396 {
397 .buf = data,
398 .len = length
399 }
400 };
401
402 struct spi_buf_set tx = {
403 .buffers = buf,
404 .count = ARRAY_SIZE(buf),
405 };
406
407 if (!write) {
408 const struct spi_buf_set rx = {
409 .buffers = buf,
410 .count = ARRAY_SIZE(buf)
411 };
412
413 return spi_transceive_dt(&dev_config.bus, &tx, &rx);
414 }
415
416 return spi_write_dt(&dev_config.bus, &tx);
417 }
418
sx127x_read(uint8_t reg_addr,uint8_t * data,uint8_t len)419 int sx127x_read(uint8_t reg_addr, uint8_t *data, uint8_t len)
420 {
421 return sx127x_transceive(reg_addr, false, data, len);
422 }
423
sx127x_write(uint8_t reg_addr,uint8_t * data,uint8_t len)424 int sx127x_write(uint8_t reg_addr, uint8_t *data, uint8_t len)
425 {
426 return sx127x_transceive(reg_addr | BIT(7), true, data, len);
427 }
428
SX127xWriteBuffer(uint32_t addr,uint8_t * buffer,uint8_t size)429 void SX127xWriteBuffer(uint32_t addr, uint8_t *buffer, uint8_t size)
430 {
431 int ret;
432
433 ret = sx127x_write(addr, buffer, size);
434 if (ret < 0) {
435 LOG_ERR("Unable to write address: 0x%x", addr);
436 }
437 }
438
SX127xReadBuffer(uint32_t addr,uint8_t * buffer,uint8_t size)439 void SX127xReadBuffer(uint32_t addr, uint8_t *buffer, uint8_t size)
440 {
441 int ret;
442
443 ret = sx127x_read(addr, buffer, size);
444 if (ret < 0) {
445 LOG_ERR("Unable to read address: 0x%x", addr);
446 }
447 }
448
SX127xSetRfTxPower(int8_t power)449 void SX127xSetRfTxPower(int8_t power)
450 {
451 int ret;
452 uint8_t pa_config = 0;
453 uint8_t pa_dac = 0;
454
455 ret = sx127x_read(REG_PADAC, &pa_dac, 1);
456 if (ret < 0) {
457 LOG_ERR("Unable to read PA dac");
458 return;
459 }
460
461 pa_dac &= ~SX127X_PADAC_20DBM_MASK;
462
463 if (SX127X_PA_OUTPUT(power) == SX127X_PA_BOOST) {
464 power = clamp_int8(power, 2, 20);
465
466 pa_config |= SX127X_PACONFIG_PASELECT_PABOOST;
467 if (power > 17) {
468 pa_dac |= SX127X_PADAC_20DBM_ON;
469 pa_config |= (power - 5) & SX127X_PACONFIG_OUTPUTPOWER_MASK;
470 } else {
471 pa_dac |= SX127X_PADAC_20DBM_OFF;
472 pa_config |= (power - 2) & SX127X_PACONFIG_OUTPUTPOWER_MASK;
473 }
474 } else {
475 #ifdef SX127X_PACONFIG_MAX_POWER_SHIFT
476 power = clamp_int8(power, -4, 15);
477
478 pa_dac |= SX127X_PADAC_20DBM_OFF;
479 if (power > 0) {
480 /* Set the power range to 0 -- 10.8+0.6*7 dBm */
481 pa_config |= 7 << SX127X_PACONFIG_MAX_POWER_SHIFT;
482 pa_config |= power & SX127X_PACONFIG_OUTPUTPOWER_MASK;
483 } else {
484 /* Set the power range to -4.2 -- 10.8+0.6*0 dBm */
485 pa_config |= (power + 4) & SX127X_PACONFIG_OUTPUTPOWER_MASK;
486 }
487 #else
488 power = clamp_int8(power, -1, 14);
489
490 pa_dac |= SX127X_PADAC_20DBM_OFF;
491 pa_config |= (power + 1) & SX127X_PACONFIG_OUTPUTPOWER_MASK;
492 #endif
493 }
494
495 #if DT_INST_NODE_HAS_PROP(0, rfo_enable_gpios) && \
496 DT_INST_NODE_HAS_PROP(0, pa_boost_enable_gpios)
497 dev_data.tx_power = power;
498 #endif
499
500 ret = sx127x_write(REG_PACONFIG, &pa_config, 1);
501 if (ret < 0) {
502 LOG_ERR("Unable to write PA config");
503 return;
504 }
505
506 ret = sx127x_write(REG_PADAC, &pa_dac, 1);
507 if (ret < 0) {
508 LOG_ERR("Unable to write PA dac");
509 return;
510 }
511 }
512
SX127xGetDio1PinState(void)513 uint32_t SX127xGetDio1PinState(void)
514 {
515 #if SX127X_DIO_GPIO_LEN(0) >= 2
516 if (gpio_pin_get_dt(&sx127x_dios[1]) > 0) {
517 return 1U;
518 }
519 #endif
520
521 return 0U;
522 }
523
524 /* Initialize Radio driver callbacks */
525 const struct Radio_s Radio = {
526 .Init = SX127xInit,
527 .GetStatus = SX127xGetStatus,
528 .SetModem = SX127xSetModem,
529 .SetChannel = SX127xSetChannel,
530 .IsChannelFree = SX127xIsChannelFree,
531 .Random = SX127xRandom,
532 .SetRxConfig = SX127xSetRxConfig,
533 .SetTxConfig = SX127xSetTxConfig,
534 .CheckRfFrequency = SX127xCheckRfFrequency,
535 .TimeOnAir = SX127xGetTimeOnAir,
536 .Send = SX127xSend,
537 .Sleep = SX127xSetSleep,
538 .Standby = SX127xSetStby,
539 .Rx = SX127xSetRx,
540 .Write = SX127xWrite,
541 .Read = SX127xRead,
542 .WriteBuffer = SX127xWriteBuffer,
543 .ReadBuffer = SX127xReadBuffer,
544 .SetMaxPayloadLength = SX127xSetMaxPayloadLength,
545 .SetPublicNetwork = SX127xSetPublicNetwork,
546 .GetWakeupTime = SX127xGetWakeupTime,
547 .IrqProcess = NULL,
548 .RxBoosted = NULL,
549 .SetRxDutyCycle = NULL,
550 .SetTxContinuousWave = SX127xSetTxContinuousWave,
551 };
552
sx127x_antenna_configure(void)553 static int sx127x_antenna_configure(void)
554 {
555 int ret;
556
557 ret = sx12xx_configure_pin(antenna_enable, GPIO_OUTPUT_INACTIVE);
558 if (ret) {
559 return ret;
560 }
561
562 ret = sx12xx_configure_pin(rfi_enable, GPIO_OUTPUT_INACTIVE);
563 if (ret) {
564 return ret;
565 }
566
567 ret = sx12xx_configure_pin(rfo_enable, GPIO_OUTPUT_INACTIVE);
568 if (ret) {
569 return ret;
570 }
571
572 ret = sx12xx_configure_pin(pa_boost_enable, GPIO_OUTPUT_INACTIVE);
573 if (ret) {
574 return ret;
575 }
576
577 return 0;
578 }
579
sx127x_lora_init(const struct device * dev)580 static int sx127x_lora_init(const struct device *dev)
581 {
582 int ret;
583 uint8_t regval;
584
585 if (!spi_is_ready_dt(&dev_config.bus)) {
586 LOG_ERR("SPI device not ready");
587 return -ENODEV;
588 }
589
590 ret = sx12xx_configure_pin(tcxo_power, GPIO_OUTPUT_INACTIVE);
591 if (ret) {
592 return ret;
593 }
594
595 /* Setup Reset gpio and perform soft reset */
596 ret = sx12xx_configure_pin(reset, GPIO_OUTPUT_ACTIVE);
597 if (ret) {
598 return ret;
599 }
600
601 k_sleep(K_MSEC(100));
602 gpio_pin_set_dt(&dev_config.reset, 0);
603 k_sleep(K_MSEC(100));
604
605 ret = sx127x_read(REG_VERSION, ®val, 1);
606 if (ret < 0) {
607 LOG_ERR("Unable to read version info");
608 return -EIO;
609 }
610
611 LOG_INF("SX127x version 0x%02x found", regval);
612
613 ret = sx127x_antenna_configure();
614 if (ret < 0) {
615 LOG_ERR("Unable to configure antenna");
616 return -EIO;
617 }
618
619 ret = sx12xx_init(dev);
620 if (ret < 0) {
621 LOG_ERR("Failed to initialize SX12xx common");
622 return ret;
623 }
624
625 return 0;
626 }
627
628 static DEVICE_API(lora, sx127x_lora_api) = {
629 .config = sx12xx_lora_config,
630 .send = sx12xx_lora_send,
631 .send_async = sx12xx_lora_send_async,
632 .recv = sx12xx_lora_recv,
633 .recv_async = sx12xx_lora_recv_async,
634 .test_cw = sx12xx_lora_test_cw,
635 };
636
637 DEVICE_DT_INST_DEFINE(0, &sx127x_lora_init, NULL, NULL,
638 NULL, POST_KERNEL, CONFIG_LORA_INIT_PRIORITY,
639 &sx127x_lora_api);
640