1 /*
2 * Copyright (c) 2018 Foundries.io
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT wnc_m14a2a
8
9 #define LOG_DOMAIN modem_wncm14a2a
10 #define LOG_LEVEL CONFIG_MODEM_LOG_LEVEL
11 #include <zephyr/logging/log.h>
12 LOG_MODULE_REGISTER(LOG_DOMAIN);
13
14 #include <zephyr/types.h>
15 #include <stddef.h>
16 #include <stdlib.h>
17 #include <ctype.h>
18 #include <errno.h>
19 #include <zephyr/kernel.h>
20 #include <zephyr/drivers/gpio.h>
21 #include <zephyr/device.h>
22 #include <zephyr/init.h>
23 #include <zephyr/random/random.h>
24
25 #include <zephyr/net/net_context.h>
26 #include <zephyr/net/net_if.h>
27 #include <zephyr/net/net_offload.h>
28 #include <zephyr/net/offloaded_netdev.h>
29 #include <zephyr/net/net_pkt.h>
30 #if defined(CONFIG_NET_IPV6)
31 #include "ipv6.h"
32 #endif
33 #if defined(CONFIG_NET_IPV4)
34 #include "ipv4.h"
35 #endif
36 #if defined(CONFIG_NET_UDP)
37 #include "udp_internal.h"
38 #endif
39
40 #include "modem_receiver.h"
41
42 /* Uncomment the #define below to enable a hexdump of all incoming
43 * data from the modem receiver
44 */
45 /* #define ENABLE_VERBOSE_MODEM_RECV_HEXDUMP 1 */
46
47 /* pin settings */
48 enum mdm_control_pins {
49 MDM_BOOT_MODE_SEL = 0,
50 MDM_POWER,
51 MDM_KEEP_AWAKE,
52 MDM_RESET,
53 SHLD_3V3_1V8_SIG_TRANS_ENA,
54 #if DT_INST_NODE_HAS_PROP(0, mdm_send_ok_gpios)
55 MDM_SEND_OK,
56 #endif
57 MAX_MDM_CONTROL_PINS,
58 };
59
60 #define MDM_UART_DEV DEVICE_DT_GET(DT_INST_BUS(0))
61
62 #define MDM_BOOT_MODE_SPECIAL 0
63 #define MDM_BOOT_MODE_NORMAL 1
64
65 #define MDM_CMD_TIMEOUT (5 * MSEC_PER_SEC)
66 #define MDM_CMD_SEND_TIMEOUT (10 * MSEC_PER_SEC)
67 #define MDM_CMD_CONN_TIMEOUT (31 * MSEC_PER_SEC)
68
69 #define MDM_MAX_DATA_LENGTH 1500
70
71 #define MDM_RECV_MAX_BUF 30
72 #define MDM_RECV_BUF_SIZE 128
73
74 #define MDM_MAX_SOCKETS 6
75
76 #define BUF_ALLOC_TIMEOUT K_SECONDS(1)
77
78 #define CMD_HANDLER(cmd_, cb_) { \
79 .cmd = cmd_, \
80 .cmd_len = (uint16_t)sizeof(cmd_)-1, \
81 .func = on_cmd_ ## cb_ \
82 }
83
84 #define MDM_MANUFACTURER_LENGTH 10
85 #define MDM_MODEL_LENGTH 16
86 #define MDM_REVISION_LENGTH 64
87 #define MDM_IMEI_LENGTH 16
88
89 #define RSSI_TIMEOUT_SECS 30
90
91 NET_BUF_POOL_DEFINE(mdm_recv_pool, MDM_RECV_MAX_BUF, MDM_RECV_BUF_SIZE,
92 0, NULL);
93
94 static uint8_t mdm_recv_buf[MDM_MAX_DATA_LENGTH];
95
96 /* RX thread structures */
97 K_KERNEL_STACK_DEFINE(wncm14a2a_rx_stack,
98 CONFIG_MODEM_WNCM14A2A_RX_STACK_SIZE);
99 struct k_thread wncm14a2a_rx_thread;
100
101 /* RX thread work queue */
102 K_KERNEL_STACK_DEFINE(wncm14a2a_workq_stack,
103 CONFIG_MODEM_WNCM14A2A_RX_WORKQ_STACK_SIZE);
104 static struct k_work_q wncm14a2a_workq;
105
106 struct wncm14a2a_socket {
107 struct net_context *context;
108 sa_family_t family;
109 enum net_sock_type type;
110 enum net_ip_protocol ip_proto;
111 struct sockaddr src;
112 struct sockaddr dst;
113
114 int socket_id;
115
116 /** semaphore */
117 struct k_sem sock_send_sem;
118
119 /** socket callbacks */
120 struct k_work recv_cb_work;
121 net_context_recv_cb_t recv_cb;
122 struct net_pkt *recv_pkt;
123 void *recv_user_data;
124 };
125
126 struct wncm14a2a_config {
127 struct gpio_dt_spec gpio[MAX_MDM_CONTROL_PINS];
128 };
129
130 struct wncm14a2a_iface_ctx {
131 struct net_if *iface;
132 uint8_t mac_addr[6];
133
134 /* RX specific attributes */
135 struct mdm_receiver_context mdm_ctx;
136
137 /* socket data */
138 struct wncm14a2a_socket sockets[MDM_MAX_SOCKETS];
139 int last_socket_id;
140 int last_error;
141
142 /* semaphores */
143 struct k_sem response_sem;
144
145 /* RSSI work */
146 struct k_work_delayable rssi_query_work;
147
148 /* modem data */
149 char mdm_manufacturer[MDM_MANUFACTURER_LENGTH];
150 char mdm_model[MDM_MODEL_LENGTH];
151 char mdm_revision[MDM_REVISION_LENGTH];
152 char mdm_imei[MDM_IMEI_LENGTH];
153 int mdm_rssi;
154
155 /* modem state */
156 int ev_csps;
157 int ev_rrcstate;
158 };
159
160 struct cmd_handler {
161 const char *cmd;
162 uint16_t cmd_len;
163 void (*func)(struct net_buf **buf, uint16_t len);
164 };
165
166 const static struct wncm14a2a_config wncm14a2a_cfg = {
167 .gpio = {
168 GPIO_DT_SPEC_INST_GET(0, mdm_boot_mode_sel_gpios),
169 GPIO_DT_SPEC_INST_GET(0, mdm_power_gpios),
170 GPIO_DT_SPEC_INST_GET(0, mdm_keep_awake_gpios),
171 GPIO_DT_SPEC_INST_GET(0, mdm_reset_gpios),
172 GPIO_DT_SPEC_INST_GET(0, mdm_shld_trans_ena_gpios),
173 #if DT_INST_NODE_HAS_PROP(0, mdm_send_ok_gpios)
174 GPIO_DT_SPEC_INST_GET(0, mdm_send_ok_gpios),
175 #endif
176 },
177 };
178
179 static struct wncm14a2a_iface_ctx ictx;
180
181 static void wncm14a2a_read_rx(struct net_buf **buf);
182
183 /*** Verbose Debugging Functions ***/
184 #if defined(ENABLE_VERBOSE_MODEM_RECV_HEXDUMP)
hexdump(const uint8_t * packet,size_t length)185 static inline void hexdump(const uint8_t *packet, size_t length)
186 {
187 char output[sizeof("xxxxyyyy xxxxyyyy")];
188 int n = 0, k = 0;
189 uint8_t byte;
190
191 while (length--) {
192 if (n % 16 == 0) {
193 printk(" %08X ", n);
194 }
195
196 byte = *packet++;
197
198 printk("%02X ", byte);
199
200 if (byte < 0x20 || byte > 0x7f) {
201 output[k++] = '.';
202 } else {
203 output[k++] = byte;
204 }
205
206 n++;
207 if (n % 8 == 0) {
208 if (n % 16 == 0) {
209 output[k] = '\0';
210 printk(" [%s]\n", output);
211 k = 0;
212 } else {
213 printk(" ");
214 }
215 }
216 }
217
218 if (n % 16) {
219 int i;
220
221 output[k] = '\0';
222
223 for (i = 0; i < (16 - (n % 16)); i++) {
224 printk(" ");
225 }
226
227 if ((n % 16) < 8) {
228 printk(" "); /* one extra delimiter after 8 chars */
229 }
230
231 printk(" [%s]\n", output);
232 }
233 }
234 #else
235 #define hexdump(...)
236 #endif
237
socket_get(void)238 static struct wncm14a2a_socket *socket_get(void)
239 {
240 int i;
241 struct wncm14a2a_socket *sock = NULL;
242
243 for (i = 0; i < MDM_MAX_SOCKETS; i++) {
244 if (!ictx.sockets[i].context) {
245 sock = &ictx.sockets[i];
246 break;
247 }
248 }
249
250 return sock;
251 }
252
socket_from_id(int socket_id)253 static struct wncm14a2a_socket *socket_from_id(int socket_id)
254 {
255 int i;
256 struct wncm14a2a_socket *sock = NULL;
257
258 if (socket_id < 1) {
259 return NULL;
260 }
261
262 for (i = 0; i < MDM_MAX_SOCKETS; i++) {
263 if (ictx.sockets[i].socket_id == socket_id) {
264 sock = &ictx.sockets[i];
265 break;
266 }
267 }
268
269 return sock;
270 }
271
socket_put(struct wncm14a2a_socket * sock)272 static void socket_put(struct wncm14a2a_socket *sock)
273 {
274 if (!sock) {
275 return;
276 }
277
278 sock->context = NULL;
279 sock->socket_id = 0;
280 (void)memset(&sock->src, 0, sizeof(struct sockaddr));
281 (void)memset(&sock->dst, 0, sizeof(struct sockaddr));
282 }
283
wncm14a2a_sprint_ip_addr(const struct sockaddr * addr)284 char *wncm14a2a_sprint_ip_addr(const struct sockaddr *addr)
285 {
286 static char buf[NET_IPV6_ADDR_LEN];
287
288 #if defined(CONFIG_NET_IPV6)
289 if (addr->sa_family == AF_INET6) {
290 return net_addr_ntop(AF_INET6, &net_sin6(addr)->sin6_addr,
291 buf, sizeof(buf));
292 } else
293 #endif
294 #if defined(CONFIG_NET_IPV4)
295 if (addr->sa_family == AF_INET) {
296 return net_addr_ntop(AF_INET, &net_sin(addr)->sin_addr,
297 buf, sizeof(buf));
298 } else
299 #endif
300 {
301 LOG_ERR("Unknown IP address family:%d", addr->sa_family);
302 return NULL;
303 }
304 }
305
306 /* Send an AT command with a series of response handlers */
send_at_cmd(struct wncm14a2a_socket * sock,const uint8_t * data,int timeout)307 static int send_at_cmd(struct wncm14a2a_socket *sock,
308 const uint8_t *data, int timeout)
309 {
310 int ret;
311
312 ictx.last_error = 0;
313
314 LOG_DBG("OUT: [%s]", data);
315 mdm_receiver_send(&ictx.mdm_ctx, data, strlen(data));
316 mdm_receiver_send(&ictx.mdm_ctx, "\r\n", 2);
317
318 if (timeout == 0) {
319 return 0;
320 }
321
322 if (!sock) {
323 k_sem_reset(&ictx.response_sem);
324 ret = k_sem_take(&ictx.response_sem, K_MSEC(timeout));
325 } else {
326 k_sem_reset(&sock->sock_send_sem);
327 ret = k_sem_take(&sock->sock_send_sem, K_MSEC(timeout));
328 }
329
330 if (ret == 0) {
331 ret = ictx.last_error;
332 } else if (ret == -EAGAIN) {
333 ret = -ETIMEDOUT;
334 }
335
336 return ret;
337 }
338
send_data(struct wncm14a2a_socket * sock,struct net_pkt * pkt)339 static int send_data(struct wncm14a2a_socket *sock, struct net_pkt *pkt)
340 {
341 int ret;
342 struct net_buf *frag;
343 char buf[sizeof("AT@SOCKWRITE=#,####,1\r")];
344
345 if (!sock) {
346 return -EINVAL;
347 }
348
349 ictx.last_error = 0;
350
351 frag = pkt->frags;
352 /* use SOCKWRITE with binary mode formatting */
353 snprintk(buf, sizeof(buf), "AT@SOCKWRITE=%d,%zu,1\r",
354 sock->socket_id, net_buf_frags_len(frag));
355 mdm_receiver_send(&ictx.mdm_ctx, buf, strlen(buf));
356
357 /* Loop through packet data and send */
358 while (frag) {
359 mdm_receiver_send(&ictx.mdm_ctx,
360 frag->data, frag->len);
361 frag = frag->frags;
362 }
363
364 mdm_receiver_send(&ictx.mdm_ctx, "\r\n", 2);
365 k_sem_reset(&sock->sock_send_sem);
366 ret = k_sem_take(&sock->sock_send_sem, K_MSEC(MDM_CMD_SEND_TIMEOUT));
367 if (ret == 0) {
368 ret = ictx.last_error;
369 } else if (ret == -EAGAIN) {
370 ret = -ETIMEDOUT;
371 }
372
373 return ret;
374 }
375
376 /*** NET_BUF HELPERS ***/
377
is_crlf(uint8_t c)378 static bool is_crlf(uint8_t c)
379 {
380 if (c == '\n' || c == '\r') {
381 return true;
382 } else {
383 return false;
384 }
385 }
386
net_buf_skipcrlf(struct net_buf ** buf)387 static void net_buf_skipcrlf(struct net_buf **buf)
388 {
389 /* chop off any /n or /r */
390 while (*buf && is_crlf(*(*buf)->data)) {
391 net_buf_pull_u8(*buf);
392 if (!(*buf)->len) {
393 *buf = net_buf_frag_del(NULL, *buf);
394 }
395 }
396 }
397
net_buf_findcrlf(struct net_buf * buf,struct net_buf ** frag,uint16_t * offset)398 static uint16_t net_buf_findcrlf(struct net_buf *buf, struct net_buf **frag,
399 uint16_t *offset)
400 {
401 uint16_t len = 0U, pos = 0U;
402
403 while (buf && !is_crlf(*(buf->data + pos))) {
404 if (pos + 1 >= buf->len) {
405 len += buf->len;
406 buf = buf->frags;
407 pos = 0U;
408 } else {
409 pos++;
410 }
411 }
412
413 if (buf && is_crlf(*(buf->data + pos))) {
414 len += pos;
415 *offset = pos;
416 *frag = buf;
417 return len;
418 }
419
420 return 0;
421 }
422
423 /*** UDP / TCP Helper Function ***/
424
425 /* Setup IP header data to be used by some network applications.
426 * While much is dummy data, some fields such as dst, port and family are
427 * important.
428 * Return the IP + protocol header length.
429 */
pkt_setup_ip_data(struct net_pkt * pkt,struct wncm14a2a_socket * sock)430 static int pkt_setup_ip_data(struct net_pkt *pkt,
431 struct wncm14a2a_socket *sock)
432 {
433 int hdr_len = 0;
434 uint16_t src_port = 0U, dst_port = 0U;
435
436 #if defined(CONFIG_NET_IPV6)
437 if (net_pkt_family(pkt) == AF_INET6) {
438 if (net_ipv6_create(
439 pkt,
440 &((struct sockaddr_in6 *)&sock->dst)->sin6_addr,
441 &((struct sockaddr_in6 *)&sock->src)->sin6_addr)) {
442 return -1;
443 }
444 src_port = ntohs(net_sin6(&sock->src)->sin6_port);
445 dst_port = ntohs(net_sin6(&sock->dst)->sin6_port);
446
447 hdr_len = sizeof(struct net_ipv6_hdr);
448 } else
449 #endif
450 #if defined(CONFIG_NET_IPV4)
451 if (net_pkt_family(pkt) == AF_INET) {
452 if (net_ipv4_create(
453 pkt,
454 &((struct sockaddr_in *)&sock->dst)->sin_addr,
455 &((struct sockaddr_in *)&sock->src)->sin_addr)) {
456 return -1;
457 }
458 src_port = ntohs(net_sin(&sock->src)->sin_port);
459 dst_port = ntohs(net_sin(&sock->dst)->sin_port);
460
461 hdr_len = sizeof(struct net_ipv4_hdr);
462 } else
463 #endif
464 {
465 /* no error here as hdr_len is checked later for 0 value */
466 }
467
468 #if defined(CONFIG_NET_UDP)
469 if (sock->ip_proto == IPPROTO_UDP) {
470 if (net_udp_create(pkt, dst_port, src_port)) {
471 return -1;
472 }
473
474 hdr_len += NET_UDPH_LEN;
475 } else
476 #endif
477 #if defined(CONFIG_NET_TCP)
478 if (sock->ip_proto == IPPROTO_TCP) {
479 NET_PKT_DATA_ACCESS_DEFINE(tcp_access, struct net_tcp_hdr);
480 struct net_tcp_hdr *tcp;
481
482 tcp = (struct net_tcp_hdr *)net_pkt_get_data(pkt, &tcp_access);
483 if (!tcp) {
484 return -1;
485 }
486
487 (void)memset(tcp, 0, NET_TCPH_LEN);
488
489 /* Setup TCP header */
490 tcp->src_port = dst_port;
491 tcp->dst_port = src_port;
492
493 if (net_pkt_set_data(pkt, &tcp_access)) {
494 return -1;
495 }
496
497 hdr_len += NET_TCPH_LEN;
498 } else
499 #endif /* CONFIG_NET_TCP */
500 {
501 /* no error here as hdr_len is checked later for 0 value */
502 }
503
504 return hdr_len;
505 }
506
507 /*** MODEM RESPONSE HANDLERS ***/
508
509 /* Last Socket ID Handler */
on_cmd_atcmdecho(struct net_buf ** buf,uint16_t len)510 static void on_cmd_atcmdecho(struct net_buf **buf, uint16_t len)
511 {
512 char value[2];
513 /* make sure only a single digit is picked up for socket_id */
514 value[0] = net_buf_pull_u8(*buf);
515 ictx.last_socket_id = atoi(value);
516 }
517
518 /* Echo Handler for commands without related sockets */
on_cmd_atcmdecho_nosock(struct net_buf ** buf,uint16_t len)519 static void on_cmd_atcmdecho_nosock(struct net_buf **buf, uint16_t len)
520 {
521 /* clear last_socket_id */
522 ictx.last_socket_id = 0;
523 }
524
on_cmd_atcmdinfo_manufacturer(struct net_buf ** buf,uint16_t len)525 static void on_cmd_atcmdinfo_manufacturer(struct net_buf **buf, uint16_t len)
526 {
527 size_t out_len;
528
529 out_len = net_buf_linearize(ictx.mdm_manufacturer,
530 sizeof(ictx.mdm_manufacturer) - 1,
531 *buf, 0, len);
532 ictx.mdm_manufacturer[out_len] = 0;
533 LOG_INF("Manufacturer: %s", ictx.mdm_manufacturer);
534 }
535
on_cmd_atcmdinfo_model(struct net_buf ** buf,uint16_t len)536 static void on_cmd_atcmdinfo_model(struct net_buf **buf, uint16_t len)
537 {
538 size_t out_len;
539
540 out_len = net_buf_linearize(ictx.mdm_model,
541 sizeof(ictx.mdm_model) - 1,
542 *buf, 0, len);
543 ictx.mdm_model[out_len] = 0;
544 LOG_INF("Model: %s", ictx.mdm_model);
545 }
546
on_cmd_atcmdinfo_revision(struct net_buf ** buf,uint16_t len)547 static void on_cmd_atcmdinfo_revision(struct net_buf **buf, uint16_t len)
548 {
549 size_t out_len;
550
551 out_len = net_buf_linearize(ictx.mdm_revision,
552 sizeof(ictx.mdm_revision) - 1,
553 *buf, 0, len);
554 ictx.mdm_revision[out_len] = 0;
555 LOG_INF("Revision: %s", ictx.mdm_revision);
556 }
557
on_cmd_atcmdecho_nosock_imei(struct net_buf ** buf,uint16_t len)558 static void on_cmd_atcmdecho_nosock_imei(struct net_buf **buf, uint16_t len)
559 {
560 struct net_buf *frag = NULL;
561 uint16_t offset;
562 size_t out_len;
563
564 /* make sure IMEI data is received */
565 if (len < MDM_IMEI_LENGTH) {
566 LOG_DBG("Waiting for data");
567 /* wait for more data */
568 k_sleep(K_MSEC(500));
569 wncm14a2a_read_rx(buf);
570 }
571
572 net_buf_skipcrlf(buf);
573 if (!*buf) {
574 LOG_DBG("Unable to find IMEI (net_buf_skipcrlf)");
575 return;
576 }
577
578 frag = NULL;
579 len = net_buf_findcrlf(*buf, &frag, &offset);
580 if (!frag) {
581 LOG_DBG("Unable to find IMEI (net_buf_findcrlf)");
582 return;
583 }
584
585 out_len = net_buf_linearize(ictx.mdm_imei, sizeof(ictx.mdm_imei) - 1,
586 *buf, 0, len);
587 ictx.mdm_imei[out_len] = 0;
588
589 LOG_INF("IMEI: %s", ictx.mdm_imei);
590 }
591
592 /* Handler: %MEAS: RSSI:Reported= -68, Ant0= -63, Ant1= -251 */
on_cmd_atcmdinfo_rssi(struct net_buf ** buf,uint16_t len)593 static void on_cmd_atcmdinfo_rssi(struct net_buf **buf, uint16_t len)
594 {
595 int start = 0, i = 0;
596 size_t value_size;
597 char value[64];
598
599 value_size = sizeof(value);
600 (void)memset(value, 0, value_size);
601 while (*buf && len > 0 && i < value_size) {
602 value[i] = net_buf_pull_u8(*buf);
603 if (!(*buf)->len) {
604 *buf = net_buf_frag_del(NULL, *buf);
605 }
606
607 /* 2nd "=" marks the beginning of the RSSI value */
608 if (start < 2) {
609 if (value[i] == '=') {
610 start++;
611 }
612
613 continue;
614 }
615
616 /* "," marks the end of the RSSI value */
617 if (value[i] == ',') {
618 value[i] = '\0';
619 break;
620 }
621
622 i++;
623 }
624
625 if (i > 0) {
626 ictx.mdm_rssi = atoi(value);
627 LOG_INF("RSSI: %d", ictx.mdm_rssi);
628 } else {
629 LOG_WRN("Bad format found for RSSI");
630 }
631 }
632
633 /* Handler: OK */
on_cmd_sockok(struct net_buf ** buf,uint16_t len)634 static void on_cmd_sockok(struct net_buf **buf, uint16_t len)
635 {
636 struct wncm14a2a_socket *sock = NULL;
637
638 ictx.last_error = 0;
639 sock = socket_from_id(ictx.last_socket_id);
640 if (!sock) {
641 k_sem_give(&ictx.response_sem);
642 } else {
643 k_sem_give(&sock->sock_send_sem);
644 }
645 }
646
647 /* Handler: ERROR */
on_cmd_sockerror(struct net_buf ** buf,uint16_t len)648 static void on_cmd_sockerror(struct net_buf **buf, uint16_t len)
649 {
650 struct wncm14a2a_socket *sock = NULL;
651
652 ictx.last_error = -EIO;
653 sock = socket_from_id(ictx.last_socket_id);
654 if (!sock) {
655 k_sem_give(&ictx.response_sem);
656 } else {
657 k_sem_give(&sock->sock_send_sem);
658 }
659 }
660
661 /* Handler: @EXTERR:<exterror_id> */
on_cmd_sockexterror(struct net_buf ** buf,uint16_t len)662 static void on_cmd_sockexterror(struct net_buf **buf, uint16_t len)
663 {
664 char value[8];
665 size_t out_len;
666
667 struct wncm14a2a_socket *sock = NULL;
668
669 out_len = net_buf_linearize(value, sizeof(value) - 1, *buf, 0, len);
670 value[out_len] = 0;
671 ictx.last_error = -atoi(value);
672 LOG_ERR("@EXTERR:%d", ictx.last_error);
673 sock = socket_from_id(ictx.last_socket_id);
674 if (!sock) {
675 k_sem_give(&ictx.response_sem);
676 } else {
677 k_sem_give(&sock->sock_send_sem);
678 }
679 }
680
681 /* Handler: @SOCKDIAL:<status> */
on_cmd_sockdial(struct net_buf ** buf,uint16_t len)682 static void on_cmd_sockdial(struct net_buf **buf, uint16_t len)
683 {
684 char value[8];
685 size_t out_len;
686
687 out_len = net_buf_linearize(value, sizeof(value) - 1, *buf, 0, len);
688 value[out_len] = 0;
689 ictx.last_error = atoi(value);
690 k_sem_give(&ictx.response_sem);
691 }
692
693 /* Handler: @SOCKCREAT:<socket_id> */
on_cmd_sockcreat(struct net_buf ** buf,uint16_t len)694 static void on_cmd_sockcreat(struct net_buf **buf, uint16_t len)
695 {
696 char value[2];
697 struct wncm14a2a_socket *sock = NULL;
698
699 /* look up new socket by special id */
700 sock = socket_from_id(MDM_MAX_SOCKETS + 1);
701 if (sock) {
702 /* make sure only a single digit is picked up for socket_id */
703 value[0] = net_buf_pull_u8(*buf);
704 sock->socket_id = atoi(value);
705 }
706 /* don't give back semaphore -- OK to follow */
707 }
708
709 /* Handler: @SOCKWRITE:<actual_length> */
on_cmd_sockwrite(struct net_buf ** buf,uint16_t len)710 static void on_cmd_sockwrite(struct net_buf **buf, uint16_t len)
711 {
712 char value[8];
713 size_t out_len;
714 int write_len;
715 struct wncm14a2a_socket *sock = NULL;
716
717 /* TODO: check against what we wanted to send */
718 out_len = net_buf_linearize(value, sizeof(value) - 1, *buf, 0, len);
719 value[out_len] = 0;
720 write_len = atoi(value);
721 if (write_len <= 0) {
722 return;
723 }
724
725 sock = socket_from_id(ictx.last_socket_id);
726 if (sock) {
727 k_sem_give(&sock->sock_send_sem);
728 }
729 }
730
sockreadrecv_cb_work(struct k_work * work)731 static void sockreadrecv_cb_work(struct k_work *work)
732 {
733 struct wncm14a2a_socket *sock = NULL;
734 struct net_pkt *pkt;
735
736 sock = CONTAINER_OF(work, struct wncm14a2a_socket, recv_cb_work);
737
738 /* return data */
739 pkt = sock->recv_pkt;
740 sock->recv_pkt = NULL;
741 if (sock->recv_cb) {
742 sock->recv_cb(sock->context, pkt, NULL, NULL,
743 0, sock->recv_user_data);
744 } else {
745 net_pkt_unref(pkt);
746 }
747 }
748
749 /* Handler: @SOCKREAD:<actual_length>,"<hex encoded binary>" */
on_cmd_sockread(struct net_buf ** buf,uint16_t len)750 static void on_cmd_sockread(struct net_buf **buf, uint16_t len)
751 {
752 struct wncm14a2a_socket *sock = NULL;
753 uint8_t c = 0U;
754 int i, actual_length, hdr_len = 0;
755 size_t value_size;
756 char value[10];
757
758 /* first comma marks the end of actual_length */
759 i = 0;
760 value_size = sizeof(value);
761 (void)memset(value, 0, value_size);
762 while (*buf && i < value_size - 1) {
763 value[i++] = net_buf_pull_u8(*buf);
764 len--;
765 if (!(*buf)->len) {
766 *buf = net_buf_frag_del(NULL, *buf);
767 }
768
769 if (value[i-1] == ',') {
770 i--;
771 break;
772 }
773 }
774
775 /* make sure we still have buf data, the last pulled character was
776 * a comma and that the next char in the buffer is a quote.
777 */
778 if (!*buf || value[i] != ',' || *(*buf)->data != '\"') {
779 LOG_ERR("Incorrect format! Ignoring data!");
780 return;
781 }
782
783 /* clear the comma */
784 value[i] = '\0';
785 actual_length = atoi(value);
786
787 /* skip quote */
788 len--;
789 net_buf_pull_u8(*buf);
790 if (!(*buf)->len) {
791 *buf = net_buf_frag_del(NULL, *buf);
792 }
793
794 /* check that we have enough data */
795 if (!*buf || len > (actual_length * 2) + 1) {
796 LOG_ERR("Incorrect format! Ignoring data!");
797 return;
798 }
799
800 sock = socket_from_id(ictx.last_socket_id);
801 if (!sock) {
802 LOG_ERR("Socket not found! (%d)", ictx.last_socket_id);
803 return;
804 }
805
806 /* allocate an RX pkt */
807 sock->recv_pkt = net_pkt_rx_alloc_with_buffer(
808 net_context_get_iface(sock->context),
809 actual_length, sock->family, sock->ip_proto,
810 BUF_ALLOC_TIMEOUT);
811 if (!sock->recv_pkt) {
812 LOG_ERR("Failed net_pkt_get_reserve_rx!");
813 return;
814 }
815
816 /* set pkt data */
817 net_pkt_set_context(sock->recv_pkt, sock->context);
818
819 /* add IP / protocol headers */
820 hdr_len = pkt_setup_ip_data(sock->recv_pkt, sock);
821
822 /* move hex encoded data from the buffer to the recv_pkt */
823 for (i = 0; i < actual_length * 2; i++) {
824 char c2 = *(*buf)->data;
825
826 if (isdigit((int)c2) != 0) {
827 c += c2 - '0';
828 } else if (isalpha((int)c2) != 0) {
829 c += c2 - (isupper((int)c2) != 0 ? 'A' - 10 : 'a' - 10);
830 } else {
831 /* TODO: unexpected input! skip? */
832 }
833
834 if (i % 2) {
835 if (net_pkt_write_u8(sock->recv_pkt, c)) {
836 LOG_ERR("Unable to add data! Aborting!");
837 net_pkt_unref(sock->recv_pkt);
838 sock->recv_pkt = NULL;
839 return;
840 }
841
842 c = 0U;
843 } else {
844 c = c << 4;
845 }
846
847 /* pull data from buf and advance to the next frag if needed */
848 net_buf_pull_u8(*buf);
849 if (!(*buf)->len) {
850 *buf = net_buf_frag_del(NULL, *buf);
851 }
852 }
853
854 net_pkt_cursor_init(sock->recv_pkt);
855 net_pkt_set_overwrite(sock->recv_pkt, true);
856
857 if (hdr_len > 0) {
858 net_pkt_skip(sock->recv_pkt, hdr_len);
859 }
860
861 /* Let's do the callback processing in a different work queue in
862 * case the app takes a long time.
863 */
864 k_work_submit_to_queue(&wncm14a2a_workq, &sock->recv_cb_work);
865 }
866
867 /* Handler: @SOCKDATAIND: <socket_id>,<session_status>,<left_bytes> */
on_cmd_sockdataind(struct net_buf ** buf,uint16_t len)868 static void on_cmd_sockdataind(struct net_buf **buf, uint16_t len)
869 {
870 int socket_id, left_bytes;
871 size_t out_len;
872 char *delim1, *delim2;
873 char value[sizeof("#,#,#####\r")];
874 char sendbuf[sizeof("AT@SOCKREAD=-#####,-#####\r")];
875 struct wncm14a2a_socket *sock = NULL;
876
877 out_len = net_buf_linearize(value, sizeof(value) - 1, *buf, 0, len);
878 value[out_len] = 0;
879
880 /* First comma separator marks the end of socket_id */
881 delim1 = strchr(value, ',');
882 if (!delim1) {
883 LOG_ERR("Missing 1st comma");
884 return;
885 }
886
887 *delim1++ = '\0';
888 socket_id = atoi(value);
889
890 /* Second comma separator marks the end of session_status */
891 /* TODO: ignore for now, but maybe this is useful? */
892 delim2 = strchr(delim1, ',');
893 if (!delim2) {
894 LOG_ERR("Missing 2nd comma");
895 return;
896 }
897
898 *delim2++ = '\0';
899
900 /* Third param is for left_bytes */
901 /* TODO: ignore for now because we ask for max data len
902 * but maybe this is useful in the future?
903 */
904 left_bytes = atoi(delim2);
905
906 sock = socket_from_id(socket_id);
907 if (!sock) {
908 LOG_ERR("Unable to find socket_id:%d", socket_id);
909 return;
910 }
911
912 if (left_bytes > 0) {
913 LOG_DBG("socket_id:%d left_bytes:%d", socket_id, left_bytes);
914 snprintk(sendbuf, sizeof(sendbuf), "AT@SOCKREAD=%d,%d",
915 sock->socket_id, left_bytes);
916
917 /* We entered this trigger due to an unsolicited modem response.
918 * When we send the AT@SOCKREAD command it won't generate an
919 * "OK" response directly. The modem will respond with
920 * "@SOCKREAD ..." and the data requested and then "OK" or
921 * "ERROR". Let's not wait here by passing in a timeout to
922 * send_at_cmd(). Instead, when the resulting response is
923 * received, we trigger on_cmd_sockread() to handle it.
924 */
925 send_at_cmd(sock, sendbuf, 0);
926 }
927 }
928
on_cmd_socknotifyev(struct net_buf ** buf,uint16_t len)929 static void on_cmd_socknotifyev(struct net_buf **buf, uint16_t len)
930 {
931 char value[40];
932 size_t out_len;
933 int p1 = 0, p2 = 0;
934
935 out_len = net_buf_linearize(value, sizeof(value) - 1, *buf, 0, len);
936 value[out_len] = 0;
937
938 /* walk value till 1st quote */
939 while (p1 < len && value[p1] != '\"') {
940 p1++;
941 }
942
943 if (value[p1] != '\"') {
944 /* 1st quote not found */
945 return;
946 }
947
948 p1++;
949 p2 = p1;
950 while (p2 < len && value[p2] != '\"') {
951 p2++;
952 }
953
954 if (value[p2] != '\"') {
955 /* 2nd quote not found */
956 return;
957 }
958
959 /* clear quote */
960 value[p2] = '\0';
961 p2++;
962
963 /* skip comma if present */
964 if (value[p2] == ',') {
965 p2++;
966 }
967
968 /* CSPS: 0: Moved to PS mode, 1: Moved to CS/PS mode */
969 if (!strncmp(&value[p1], "CSPS", 4)) {
970 ictx.ev_csps = atoi(&value[p2]);
971 /* This also signifies that RRCSTATE = 1 */
972 ictx.ev_rrcstate = 1;
973 LOG_DBG("CSPS:%d", ictx.ev_csps);
974 /* RRCSTATE: 0: RRC Idle, 1: RRC Connected, 2: RRC Unknown */
975 } else if (!strncmp(&value[p1], "RRCSTATE", 8)) {
976 ictx.ev_rrcstate = atoi(&value[p2]);
977 LOG_DBG("RRCSTATE:%d", ictx.ev_rrcstate);
978 } else if (!strncmp(&value[p1], "LTIME", 5)) {
979 /* local time from network */
980 LOG_INF("LTIME:%s", &value[p2]);
981 } else if (!strncmp(&value[p1], "SIB1", 4)) {
982 /* do nothing? */
983 LOG_DBG("SIB1");
984 } else {
985 LOG_DBG("UNHANDLED: [%s:%s]", &value[p1], &value[p2]);
986 }
987 }
988
net_buf_ncmp(struct net_buf * buf,const uint8_t * s2,size_t n)989 static int net_buf_ncmp(struct net_buf *buf, const uint8_t *s2, size_t n)
990 {
991 struct net_buf *frag = buf;
992 uint16_t offset = 0U;
993
994 while ((n > 0) && (*(frag->data + offset) == *s2) && (*s2 != '\0')) {
995 if (offset == frag->len) {
996 if (!frag->frags) {
997 break;
998 }
999 frag = frag->frags;
1000 offset = 0U;
1001 } else {
1002 offset++;
1003 }
1004
1005 s2++;
1006 n--;
1007 }
1008
1009 return (n == 0) ? 0 : (*(frag->data + offset) - *s2);
1010 }
1011
read_rx_allocator(k_timeout_t timeout,void * user_data)1012 static inline struct net_buf *read_rx_allocator(k_timeout_t timeout,
1013 void *user_data)
1014 {
1015 return net_buf_alloc((struct net_buf_pool *)user_data, timeout);
1016 }
1017
wncm14a2a_read_rx(struct net_buf ** buf)1018 static void wncm14a2a_read_rx(struct net_buf **buf)
1019 {
1020 uint8_t uart_buffer[MDM_RECV_BUF_SIZE];
1021 size_t bytes_read = 0;
1022 int ret;
1023 uint16_t rx_len;
1024
1025 /* read all of the data from mdm_receiver */
1026 while (true) {
1027 ret = mdm_receiver_recv(&ictx.mdm_ctx,
1028 uart_buffer,
1029 sizeof(uart_buffer),
1030 &bytes_read);
1031 if (ret < 0 || bytes_read == 0) {
1032 /* mdm_receiver buffer is empty */
1033 break;
1034 }
1035
1036 hexdump(uart_buffer, bytes_read);
1037
1038 /* make sure we have storage */
1039 if (!*buf) {
1040 *buf = net_buf_alloc(&mdm_recv_pool, BUF_ALLOC_TIMEOUT);
1041 if (!*buf) {
1042 LOG_ERR("Can't allocate RX data! "
1043 "Skipping data!");
1044 break;
1045 }
1046 }
1047
1048 rx_len = net_buf_append_bytes(*buf, bytes_read, uart_buffer,
1049 BUF_ALLOC_TIMEOUT,
1050 read_rx_allocator,
1051 &mdm_recv_pool);
1052 if (rx_len < bytes_read) {
1053 LOG_ERR("Data was lost! read %u of %zu!",
1054 rx_len, bytes_read);
1055 }
1056 }
1057 }
1058
1059 /* RX thread */
wncm14a2a_rx(void)1060 static void wncm14a2a_rx(void)
1061 {
1062 struct net_buf *rx_buf = NULL;
1063 struct net_buf *frag = NULL;
1064 int i;
1065 uint16_t offset, len;
1066
1067 static const struct cmd_handler handlers[] = {
1068 /* NON-SOCKET COMMAND ECHOES to clear last_socket_id */
1069 CMD_HANDLER("ATE1", atcmdecho_nosock),
1070 CMD_HANDLER("AT%PDNSET=", atcmdecho_nosock),
1071 CMD_HANDLER("ATI", atcmdecho_nosock),
1072 CMD_HANDLER("AT+CGSN", atcmdecho_nosock_imei),
1073 CMD_HANDLER("AT%MEAS=", atcmdecho_nosock),
1074 CMD_HANDLER("AT@INTERNET=", atcmdecho_nosock),
1075 CMD_HANDLER("AT@SOCKDIAL=", atcmdecho_nosock),
1076 CMD_HANDLER("AT@SOCKCREAT=", atcmdecho_nosock),
1077
1078 /* SOCKET COMMAND ECHOES for last_socket_id processing */
1079 CMD_HANDLER("AT@SOCKCONN=", atcmdecho),
1080 CMD_HANDLER("AT@SOCKWRITE=", atcmdecho),
1081 CMD_HANDLER("AT@SOCKREAD=", atcmdecho),
1082 CMD_HANDLER("AT@SOCKCLOSE=", atcmdecho),
1083
1084 /* MODEM Information */
1085 CMD_HANDLER("Manufacturer: ", atcmdinfo_manufacturer),
1086 CMD_HANDLER("Model: ", atcmdinfo_model),
1087 CMD_HANDLER("Revision: ", atcmdinfo_revision),
1088 CMD_HANDLER("%MEAS: RSSI:", atcmdinfo_rssi),
1089
1090 /* SOLICITED SOCKET RESPONSES */
1091 CMD_HANDLER("OK", sockok),
1092 CMD_HANDLER("ERROR", sockerror),
1093 CMD_HANDLER("@EXTERR:", sockexterror),
1094 CMD_HANDLER("@SOCKDIAL:", sockdial),
1095 CMD_HANDLER("@SOCKCREAT:", sockcreat),
1096 CMD_HANDLER("@OCKCREAT:", sockcreat), /* seeing this a lot */
1097 CMD_HANDLER("@SOCKWRITE:", sockwrite),
1098 CMD_HANDLER("@SOCKREAD:", sockread),
1099
1100 /* UNSOLICITED SOCKET RESPONSES */
1101 CMD_HANDLER("@SOCKDATAIND:", sockdataind),
1102 CMD_HANDLER("%NOTIFYEV:", socknotifyev),
1103 };
1104
1105 while (true) {
1106 /* wait for incoming data */
1107 (void)k_sem_take(&ictx.mdm_ctx.rx_sem, K_FOREVER);
1108
1109 wncm14a2a_read_rx(&rx_buf);
1110
1111 while (rx_buf) {
1112 net_buf_skipcrlf(&rx_buf);
1113 if (!rx_buf) {
1114 break;
1115 }
1116
1117 frag = NULL;
1118 len = net_buf_findcrlf(rx_buf, &frag, &offset);
1119 if (!frag) {
1120 break;
1121 }
1122
1123 /* look for matching data handlers */
1124 i = -1;
1125 for (i = 0; i < ARRAY_SIZE(handlers); i++) {
1126 if (net_buf_ncmp(rx_buf, handlers[i].cmd,
1127 handlers[i].cmd_len) == 0) {
1128 /* found a matching handler */
1129 LOG_DBG("MATCH %s (len:%u)",
1130 handlers[i].cmd, len);
1131
1132 /* skip cmd_len */
1133 rx_buf = net_buf_skip(rx_buf,
1134 handlers[i].cmd_len);
1135
1136 /* locate next cr/lf */
1137 frag = NULL;
1138 len = net_buf_findcrlf(rx_buf,
1139 &frag, &offset);
1140 if (!frag) {
1141 break;
1142 }
1143
1144 /* call handler */
1145 if (handlers[i].func) {
1146 handlers[i].func(&rx_buf, len);
1147 }
1148
1149 frag = NULL;
1150 /* make sure buf still has data */
1151 if (!rx_buf) {
1152 break;
1153 }
1154
1155 /*
1156 * We've handled the current line
1157 * and need to exit the "search for
1158 * handler loop". Let's skip any
1159 * "extra" data and look for the next
1160 * CR/LF, leaving us ready for the
1161 * next handler search. Ignore the
1162 * length returned.
1163 */
1164 (void)net_buf_findcrlf(rx_buf,
1165 &frag, &offset);
1166 break;
1167 }
1168 }
1169
1170 if (frag && rx_buf) {
1171 /* clear out processed line (buffers) */
1172 while (frag && rx_buf != frag) {
1173 rx_buf = net_buf_frag_del(NULL, rx_buf);
1174 }
1175
1176 net_buf_pull(rx_buf, offset);
1177 }
1178 }
1179
1180 /* give up time if we have a solid stream of data */
1181 k_yield();
1182 }
1183 }
1184
modem_pin_init(void)1185 static int modem_pin_init(void)
1186 {
1187 LOG_INF("Setting Modem Pins");
1188
1189 /* Hard reset the modem (>5 seconds required)
1190 * (doesn't go through the signal level translator)
1191 */
1192 LOG_DBG("MDM_RESET_PIN -> ASSERTED");
1193 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_RESET], 1);
1194 k_sleep(K_SECONDS(7));
1195 LOG_DBG("MDM_RESET_PIN -> NOT_ASSERTED");
1196 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_RESET], 0);
1197
1198 /* disable signal level translator (necessary
1199 * for the modem to boot properly). All signals
1200 * except mdm_reset go through the level translator
1201 * and have internal pull-up/down in the module. While
1202 * the level translator is disabled, these pins will
1203 * be in the correct state.
1204 */
1205 LOG_DBG("SIG_TRANS_ENA_PIN -> DISABLED");
1206 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[SHLD_3V3_1V8_SIG_TRANS_ENA], 0);
1207
1208 /* While the level translator is disabled and output pins
1209 * are tristated, make sure the inputs are in the same state
1210 * as the WNC Module pins so that when the level translator is
1211 * enabled, there are no differences.
1212 */
1213 LOG_DBG("MDM_BOOT_MODE_SEL_PIN -> NORMAL");
1214 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_BOOT_MODE_SEL], MDM_BOOT_MODE_NORMAL);
1215 LOG_DBG("MDM_POWER_PIN -> ENABLE");
1216 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_POWER], 1);
1217 LOG_DBG("MDM_KEEP_AWAKE_PIN -> ENABLED");
1218 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_KEEP_AWAKE], 1);
1219 #if DT_INST_NODE_HAS_PROP(0, mdm_send_ok_gpios)
1220 LOG_DBG("MDM_SEND_OK_PIN -> ENABLED");
1221 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_SEND_OK], 1);
1222 #endif
1223
1224 /* wait for the WNC Module to perform its initial boot correctly */
1225 k_sleep(K_SECONDS(1));
1226
1227 /* Enable the level translator.
1228 * The input pins should now be the same as how the M14A module is
1229 * driving them with internal pull ups/downs.
1230 * When enabled, there will be no changes in the above 4 pins...
1231 */
1232 LOG_DBG("SIG_TRANS_ENA_PIN -> ENABLED");
1233 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[SHLD_3V3_1V8_SIG_TRANS_ENA], 1);
1234
1235 LOG_INF("... Done!");
1236
1237 return 0;
1238 }
1239
modem_wakeup_pin_fix(void)1240 static void modem_wakeup_pin_fix(void)
1241 {
1242 /* AT&T recommend toggling the KEEP_AWAKE signal to reduce missed
1243 * UART characters.
1244 */
1245 LOG_DBG("Toggling MDM_KEEP_AWAKE_PIN to avoid missed characters");
1246 k_sleep(K_MSEC(20));
1247 LOG_DBG("MDM_KEEP_AWAKE_PIN -> DISABLED");
1248 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_KEEP_AWAKE], 0);
1249 k_sleep(K_SECONDS(2));
1250 LOG_DBG("MDM_KEEP_AWAKE_PIN -> ENABLED");
1251 gpio_pin_set_dt(&wncm14a2a_cfg.gpio[MDM_KEEP_AWAKE], 1);
1252 k_sleep(K_MSEC(20));
1253 }
1254
wncm14a2a_rssi_query_work(struct k_work * work)1255 static void wncm14a2a_rssi_query_work(struct k_work *work)
1256 {
1257 int ret;
1258
1259 /* query modem RSSI */
1260 ret = send_at_cmd(NULL, "AT%MEAS=\"23\"", MDM_CMD_TIMEOUT);
1261 if (ret < 0) {
1262 LOG_ERR("AT%%MEAS ret:%d", ret);
1263 }
1264
1265 /* re-start RSSI query work */
1266 k_work_reschedule_for_queue(&wncm14a2a_workq, &ictx.rssi_query_work,
1267 K_SECONDS(RSSI_TIMEOUT_SECS));
1268 }
1269
wncm14a2a_modem_reset(void)1270 static void wncm14a2a_modem_reset(void)
1271 {
1272 int ret = 0, retry_count = 0, counter = 0;
1273
1274 /* bring down network interface */
1275 net_if_carrier_off(ictx.iface);
1276
1277 restart:
1278 /* stop RSSI delay work */
1279 k_work_cancel_delayable(&ictx.rssi_query_work);
1280
1281 modem_pin_init();
1282
1283 LOG_INF("Waiting for modem to respond");
1284
1285 /* Give the modem a while to start responding to simple 'AT' commands.
1286 * Also wait for CSPS=1 or RRCSTATE=1 notification
1287 */
1288 ret = -1;
1289 while (counter++ < 50 && ret < 0) {
1290 k_sleep(K_SECONDS(2));
1291 ret = send_at_cmd(NULL, "AT", MDM_CMD_TIMEOUT);
1292 if (ret < 0 && ret != -ETIMEDOUT) {
1293 break;
1294 }
1295 }
1296
1297 if (ret < 0) {
1298 LOG_ERR("MODEM WAIT LOOP ERROR: %d", ret);
1299 goto error;
1300 }
1301
1302 LOG_INF("Setting modem to always stay awake");
1303 modem_wakeup_pin_fix();
1304
1305 ret = send_at_cmd(NULL, "ATE1", MDM_CMD_TIMEOUT);
1306 if (ret < 0) {
1307 LOG_ERR("ATE1 ret:%d", ret);
1308 goto error;
1309 }
1310
1311 ret = send_at_cmd(NULL, "AT%PDNSET=1,\"" CONFIG_MODEM_WNCM14A2A_APN_NAME
1312 "\",\"IPV4V6\"", MDM_CMD_TIMEOUT);
1313 if (ret < 0) {
1314 LOG_ERR("AT%%PDNSET ret:%d", ret);
1315 goto error;
1316 }
1317
1318 /* query modem info */
1319 LOG_INF("Querying modem information");
1320 ret = send_at_cmd(NULL, "ATI", MDM_CMD_TIMEOUT);
1321 if (ret < 0) {
1322 LOG_ERR("ATI ret:%d", ret);
1323 goto error;
1324 }
1325
1326 /* query modem IMEI */
1327 ret = send_at_cmd(NULL, "AT+CGSN", MDM_CMD_TIMEOUT);
1328 if (ret < 0) {
1329 LOG_ERR("AT+CGSN ret:%d", ret);
1330 goto error;
1331 }
1332
1333 LOG_INF("Waiting for network");
1334
1335 /* query modem RSSI */
1336 wncm14a2a_rssi_query_work(NULL);
1337 k_sleep(K_SECONDS(2));
1338
1339 counter = 0;
1340 /* wait for RSSI > -1000 and != 0 */
1341 while (counter++ < 15 &&
1342 (ictx.mdm_rssi <= -1000 ||
1343 ictx.mdm_rssi == 0)) {
1344 /* stop RSSI delay work */
1345 k_work_cancel_delayable(&ictx.rssi_query_work);
1346 wncm14a2a_rssi_query_work(NULL);
1347 k_sleep(K_SECONDS(2));
1348 }
1349
1350 if (ictx.mdm_rssi <= -1000 || ictx.mdm_rssi == 0) {
1351 retry_count++;
1352 if (retry_count > 3) {
1353 LOG_ERR("Failed network init. Too many attempts!");
1354 goto error;
1355 }
1356
1357 LOG_ERR("Failed network init. Restarting process.");
1358 goto restart;
1359 }
1360
1361 LOG_INF("Network is ready.");
1362
1363 ret = send_at_cmd(NULL, "AT@INTERNET=1", MDM_CMD_TIMEOUT);
1364 if (ret < 0) {
1365 LOG_ERR("AT@INTERNET ret:%d", ret);
1366 goto error;
1367 }
1368
1369 ret = send_at_cmd(NULL, "AT@SOCKDIAL=1", MDM_CMD_TIMEOUT);
1370 if (ret < 0) {
1371 LOG_ERR("SOCKDIAL=1 CHECK ret:%d", ret);
1372 /* don't report this as an error, we retry later */
1373 }
1374
1375 /* Set iface up */
1376 net_if_carrier_on(ictx.iface);
1377
1378 error:
1379 return;
1380 }
1381
wncm14a2a_init(const struct device * dev)1382 static int wncm14a2a_init(const struct device *dev)
1383 {
1384 int i, ret = 0;
1385
1386 ARG_UNUSED(dev);
1387
1388 (void)memset(&ictx, 0, sizeof(ictx));
1389 for (i = 0; i < MDM_MAX_SOCKETS; i++) {
1390 k_work_init(&ictx.sockets[i].recv_cb_work,
1391 sockreadrecv_cb_work);
1392 k_sem_init(&ictx.sockets[i].sock_send_sem, 0, 1);
1393 }
1394 k_sem_init(&ictx.response_sem, 0, 1);
1395
1396 /* initialize the work queue */
1397 k_work_queue_start(&wncm14a2a_workq, wncm14a2a_workq_stack,
1398 K_KERNEL_STACK_SIZEOF(wncm14a2a_workq_stack),
1399 K_PRIO_COOP(7), NULL);
1400
1401 ictx.last_socket_id = 0;
1402
1403 /* setup port devices and pin directions */
1404 for (i = 0; i < MAX_MDM_CONTROL_PINS; i++) {
1405 if (!gpio_is_ready_dt(&wncm14a2a_cfg.gpio[i])) {
1406 LOG_ERR("gpio port (%s) not ready!",
1407 wncm14a2a_cfg.gpio[i].port->name);
1408 return -ENODEV;
1409 }
1410
1411 gpio_pin_configure_dt(&wncm14a2a_cfg.gpio[i], GPIO_OUTPUT);
1412 }
1413
1414 /* Set modem data storage */
1415 ictx.mdm_ctx.data_manufacturer = ictx.mdm_manufacturer;
1416 ictx.mdm_ctx.data_model = ictx.mdm_model;
1417 ictx.mdm_ctx.data_revision = ictx.mdm_revision;
1418 #ifdef CONFIG_MODEM_SIM_NUMBERS
1419 ictx.mdm_ctx.data_imei = ictx.mdm_imei;
1420 #endif
1421 ictx.mdm_ctx.data_rssi = &ictx.mdm_rssi;
1422
1423 ret = mdm_receiver_register(&ictx.mdm_ctx, MDM_UART_DEV,
1424 mdm_recv_buf, sizeof(mdm_recv_buf));
1425 if (ret < 0) {
1426 LOG_ERR("Error registering modem receiver (%d)!", ret);
1427 goto error;
1428 }
1429
1430 /* start RX thread */
1431 k_thread_create(&wncm14a2a_rx_thread, wncm14a2a_rx_stack,
1432 K_KERNEL_STACK_SIZEOF(wncm14a2a_rx_stack),
1433 (k_thread_entry_t) wncm14a2a_rx,
1434 NULL, NULL, NULL, K_PRIO_COOP(7), 0, K_NO_WAIT);
1435
1436 /* init RSSI query */
1437 k_work_init_delayable(&ictx.rssi_query_work, wncm14a2a_rssi_query_work);
1438
1439 wncm14a2a_modem_reset();
1440
1441 error:
1442 return ret;
1443 }
1444
1445 /*** OFFLOAD FUNCTIONS ***/
1446
offload_get(sa_family_t family,enum net_sock_type type,enum net_ip_protocol ip_proto,struct net_context ** context)1447 static int offload_get(sa_family_t family,
1448 enum net_sock_type type,
1449 enum net_ip_protocol ip_proto,
1450 struct net_context **context)
1451 {
1452 int ret;
1453 char buf[sizeof("AT@SOCKCREAT=###,#\r")];
1454 struct wncm14a2a_socket *sock = NULL;
1455
1456 /* new socket */
1457 sock = socket_get();
1458 if (!sock) {
1459 return -ENOMEM;
1460 }
1461
1462 (*context)->offload_context = sock;
1463 sock->family = family;
1464 sock->type = type;
1465 sock->ip_proto = ip_proto;
1466 sock->context = *context;
1467 sock->socket_id = MDM_MAX_SOCKETS + 1; /* socket # needs assigning */
1468
1469 snprintk(buf, sizeof(buf), "AT@SOCKCREAT=%d,%d", type,
1470 family == AF_INET ? 0 : 1);
1471 ret = send_at_cmd(NULL, buf, MDM_CMD_TIMEOUT);
1472 if (ret < 0) {
1473 LOG_ERR("AT@SOCKCREAT ret:%d", ret);
1474 socket_put(sock);
1475 }
1476
1477 return ret;
1478 }
1479
offload_bind(struct net_context * context,const struct sockaddr * addr,socklen_t addrlen)1480 static int offload_bind(struct net_context *context,
1481 const struct sockaddr *addr,
1482 socklen_t addrlen)
1483 {
1484 struct wncm14a2a_socket *sock = NULL;
1485
1486 if (!context) {
1487 return -EINVAL;
1488 }
1489
1490 sock = (struct wncm14a2a_socket *)context->offload_context;
1491 if (!sock) {
1492 LOG_ERR("Can't locate socket for net_ctx:%p!", context);
1493 return -EINVAL;
1494 }
1495
1496 /* save bind address information */
1497 sock->src.sa_family = addr->sa_family;
1498 #if defined(CONFIG_NET_IPV6)
1499 if (addr->sa_family == AF_INET6) {
1500 net_ipaddr_copy(&net_sin6(&sock->src)->sin6_addr,
1501 &net_sin6(addr)->sin6_addr);
1502 net_sin6(&sock->src)->sin6_port = net_sin6(addr)->sin6_port;
1503 } else
1504 #endif
1505 #if defined(CONFIG_NET_IPV4)
1506 if (addr->sa_family == AF_INET) {
1507 net_ipaddr_copy(&net_sin(&sock->src)->sin_addr,
1508 &net_sin(addr)->sin_addr);
1509 net_sin(&sock->src)->sin_port = net_sin(addr)->sin_port;
1510 } else
1511 #endif
1512 {
1513 return -EPFNOSUPPORT;
1514 }
1515
1516 return 0;
1517 }
1518
offload_listen(struct net_context * context,int backlog)1519 static int offload_listen(struct net_context *context, int backlog)
1520 {
1521 /* NOT IMPLEMENTED */
1522 return -ENOTSUP;
1523 }
1524
offload_connect(struct net_context * context,const struct sockaddr * addr,socklen_t addrlen,net_context_connect_cb_t cb,int32_t timeout,void * user_data)1525 static int offload_connect(struct net_context *context,
1526 const struct sockaddr *addr,
1527 socklen_t addrlen,
1528 net_context_connect_cb_t cb,
1529 int32_t timeout,
1530 void *user_data)
1531 {
1532 int ret, dst_port = -1;
1533 int32_t timeout_sec = -1; /* if not changed, this will be min timeout */
1534 char buf[sizeof("AT@SOCKCONN=#,###.###.###.###,#####,#####\r")];
1535 struct wncm14a2a_socket *sock;
1536
1537 if (timeout > 0) {
1538 timeout_sec = timeout / MSEC_PER_SEC;
1539 }
1540
1541 if (!context || !addr) {
1542 return -EINVAL;
1543 }
1544
1545 sock = (struct wncm14a2a_socket *)context->offload_context;
1546 if (!sock) {
1547 LOG_ERR("Can't locate socket for net_ctx:%p!", context);
1548 return -EINVAL;
1549 }
1550
1551 if (sock->socket_id < 1) {
1552 LOG_ERR("Invalid socket_id(%d) for net_ctx:%p!",
1553 sock->socket_id, context);
1554 return -EINVAL;
1555 }
1556
1557 sock->dst.sa_family = addr->sa_family;
1558
1559 #if defined(CONFIG_NET_IPV6)
1560 if (addr->sa_family == AF_INET6) {
1561 net_ipaddr_copy(&net_sin6(&sock->dst)->sin6_addr,
1562 &net_sin6(addr)->sin6_addr);
1563 dst_port = ntohs(net_sin6(addr)->sin6_port);
1564 net_sin6(&sock->dst)->sin6_port = dst_port;
1565 } else
1566 #endif
1567 #if defined(CONFIG_NET_IPV4)
1568 if (addr->sa_family == AF_INET) {
1569 net_ipaddr_copy(&net_sin(&sock->dst)->sin_addr,
1570 &net_sin(addr)->sin_addr);
1571 dst_port = ntohs(net_sin(addr)->sin_port);
1572 net_sin(&sock->dst)->sin_port = dst_port;
1573 } else
1574 #endif
1575 {
1576 return -EINVAL;
1577 }
1578
1579 if (dst_port < 0) {
1580 LOG_ERR("Invalid port: %d", dst_port);
1581 return -EINVAL;
1582 }
1583
1584 /*
1585 * AT@SOCKCONN timeout param has minimum value of 30 seconds and
1586 * maximum value of 360 seconds, otherwise an error is generated
1587 */
1588 timeout_sec = CLAMP(timeout_sec, 30, 360);
1589
1590 snprintk(buf, sizeof(buf), "AT@SOCKCONN=%d,\"%s\",%d,%d",
1591 sock->socket_id, wncm14a2a_sprint_ip_addr(addr),
1592 dst_port, timeout_sec);
1593 ret = send_at_cmd(sock, buf, MDM_CMD_CONN_TIMEOUT);
1594 if (!ret) {
1595 net_context_set_state(sock->context, NET_CONTEXT_CONNECTED);
1596 } else {
1597 LOG_ERR("AT@SOCKCONN ret:%d", ret);
1598 }
1599
1600 if (cb) {
1601 cb(context, ret, user_data);
1602 }
1603
1604 return ret;
1605 }
1606
offload_accept(struct net_context * context,net_tcp_accept_cb_t cb,int32_t timeout,void * user_data)1607 static int offload_accept(struct net_context *context,
1608 net_tcp_accept_cb_t cb,
1609 int32_t timeout,
1610 void *user_data)
1611 {
1612 /* NOT IMPLEMENTED */
1613 return -ENOTSUP;
1614 }
1615
offload_sendto(struct net_pkt * pkt,const struct sockaddr * dst_addr,socklen_t addrlen,net_context_send_cb_t cb,int32_t timeout,void * user_data)1616 static int offload_sendto(struct net_pkt *pkt,
1617 const struct sockaddr *dst_addr,
1618 socklen_t addrlen,
1619 net_context_send_cb_t cb,
1620 int32_t timeout,
1621 void *user_data)
1622 {
1623 struct net_context *context = net_pkt_context(pkt);
1624 struct wncm14a2a_socket *sock;
1625 int ret = 0;
1626
1627 if (!context) {
1628 return -EINVAL;
1629 }
1630
1631 sock = (struct wncm14a2a_socket *)context->offload_context;
1632 if (!sock) {
1633 LOG_ERR("Can't locate socket for net_ctx:%p!", context);
1634 return -EINVAL;
1635 }
1636
1637 ret = send_data(sock, pkt);
1638 if (ret < 0) {
1639 LOG_ERR("send_data error: %d", ret);
1640 } else {
1641 net_pkt_unref(pkt);
1642 }
1643
1644 if (cb) {
1645 cb(context, ret, user_data);
1646 }
1647
1648 return ret;
1649 }
1650
offload_send(struct net_pkt * pkt,net_context_send_cb_t cb,int32_t timeout,void * user_data)1651 static int offload_send(struct net_pkt *pkt,
1652 net_context_send_cb_t cb,
1653 int32_t timeout,
1654 void *user_data)
1655 {
1656 struct net_context *context = net_pkt_context(pkt);
1657 socklen_t addrlen;
1658
1659 #if defined(CONFIG_NET_IPV6)
1660 if (net_pkt_family(pkt) == AF_INET6) {
1661 addrlen = sizeof(struct sockaddr_in6);
1662 } else
1663 #endif /* CONFIG_NET_IPV6 */
1664 #if defined(CONFIG_NET_IPV4)
1665 if (net_pkt_family(pkt) == AF_INET) {
1666 addrlen = sizeof(struct sockaddr_in);
1667 } else
1668 #endif /* CONFIG_NET_IPV4 */
1669 {
1670 return -EPFNOSUPPORT;
1671 }
1672
1673 return offload_sendto(pkt, &context->remote, addrlen, cb,
1674 timeout, user_data);
1675 }
1676
offload_recv(struct net_context * context,net_context_recv_cb_t cb,int32_t timeout,void * user_data)1677 static int offload_recv(struct net_context *context,
1678 net_context_recv_cb_t cb,
1679 int32_t timeout,
1680 void *user_data)
1681 {
1682 struct wncm14a2a_socket *sock;
1683
1684 if (!context) {
1685 return -EINVAL;
1686 }
1687
1688 sock = (struct wncm14a2a_socket *)context->offload_context;
1689 if (!sock) {
1690 LOG_ERR("Can't locate socket for net_ctx:%p!", context);
1691 return -EINVAL;
1692 }
1693
1694 sock->recv_cb = cb;
1695 sock->recv_user_data = user_data;
1696
1697 return 0;
1698 }
1699
offload_put(struct net_context * context)1700 static int offload_put(struct net_context *context)
1701 {
1702 struct wncm14a2a_socket *sock;
1703 char buf[sizeof("AT@SOCKCLOSE=#\r")];
1704 int ret;
1705
1706 if (!context) {
1707 return -EINVAL;
1708 }
1709
1710 sock = (struct wncm14a2a_socket *)context->offload_context;
1711 if (!sock) {
1712 /* socket was already closed? Exit quietly here. */
1713 return 0;
1714 }
1715
1716 snprintk(buf, sizeof(buf), "AT@SOCKCLOSE=%d", sock->socket_id);
1717
1718 ret = send_at_cmd(sock, buf, MDM_CMD_TIMEOUT);
1719 if (ret < 0) {
1720 LOG_ERR("AT@SOCKCLOSE ret:%d", ret);
1721 }
1722
1723 /* clear last_socket_id */
1724 ictx.last_socket_id = 0;
1725
1726 socket_put(sock);
1727 net_context_unref(context);
1728 if (sock->type == SOCK_STREAM) {
1729 /* TCP contexts are referenced twice,
1730 * once for the app and once for the stack.
1731 * Since TCP stack is not used for offload,
1732 * unref a second time.
1733 */
1734 net_context_unref(context);
1735 }
1736
1737 return 0;
1738 }
1739
1740 static struct net_offload offload_funcs = {
1741 .get = offload_get,
1742 .bind = offload_bind,
1743 .listen = offload_listen, /* TODO */
1744 .connect = offload_connect,
1745 .accept = offload_accept, /* TODO */
1746 .send = offload_send,
1747 .sendto = offload_sendto,
1748 .recv = offload_recv,
1749 .put = offload_put,
1750 };
1751
wncm14a2a_get_mac(const struct device * dev)1752 static inline uint8_t *wncm14a2a_get_mac(const struct device *dev)
1753 {
1754 struct wncm14a2a_iface_ctx *ctx = dev->data;
1755
1756 ctx->mac_addr[0] = 0x00;
1757 ctx->mac_addr[1] = 0x10;
1758
1759 UNALIGNED_PUT(sys_cpu_to_be32(sys_rand32_get()),
1760 (uint32_t *)(ctx->mac_addr + 2));
1761
1762 return ctx->mac_addr;
1763 }
1764
offload_iface_init(struct net_if * iface)1765 static void offload_iface_init(struct net_if *iface)
1766 {
1767 const struct device *dev = net_if_get_device(iface);
1768 struct wncm14a2a_iface_ctx *ctx = dev->data;
1769
1770 iface->if_dev->offload = &offload_funcs;
1771 net_if_set_link_addr(iface, wncm14a2a_get_mac(dev),
1772 sizeof(ctx->mac_addr),
1773 NET_LINK_ETHERNET);
1774 ctx->iface = iface;
1775 }
1776
1777 static struct offloaded_if_api api_funcs = {
1778 .iface_api.init = offload_iface_init,
1779 };
1780
1781 NET_DEVICE_DT_INST_OFFLOAD_DEFINE(0, wncm14a2a_init, NULL,
1782 &ictx, &wncm14a2a_cfg,
1783 CONFIG_MODEM_WNCM14A2A_INIT_PRIORITY,
1784 &api_funcs,
1785 MDM_MAX_DATA_LENGTH);
1786