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