2438 lines
56 KiB
C
2438 lines
56 KiB
C
/*
|
|
* Copyright (C) 2021 metraTec GmbH
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#define DT_DRV_COMPAT simcom_sim7080
|
|
|
|
#include <zephyr/logging/log.h>
|
|
#include <zephyr/net/offloaded_netdev.h>
|
|
LOG_MODULE_REGISTER(modem_simcom_sim7080, CONFIG_MODEM_LOG_LEVEL);
|
|
|
|
#include <zephyr/drivers/modem/simcom-sim7080.h>
|
|
#include "simcom-sim7080.h"
|
|
|
|
#define SMS_TP_UDHI_HEADER 0x40
|
|
|
|
static struct k_thread modem_rx_thread;
|
|
static struct k_work_q modem_workq;
|
|
static struct sim7080_data mdata;
|
|
static struct modem_context mctx;
|
|
static const struct socket_op_vtable offload_socket_fd_op_vtable;
|
|
|
|
static struct zsock_addrinfo dns_result;
|
|
static struct sockaddr dns_result_addr;
|
|
static char dns_result_canonname[DNS_MAX_NAME_SIZE + 1];
|
|
|
|
static struct sim7080_gnss_data gnss_data;
|
|
|
|
static K_KERNEL_STACK_DEFINE(modem_rx_stack, CONFIG_MODEM_SIMCOM_SIM7080_RX_STACK_SIZE);
|
|
static K_KERNEL_STACK_DEFINE(modem_workq_stack, CONFIG_MODEM_SIMCOM_SIM7080_RX_WORKQ_STACK_SIZE);
|
|
NET_BUF_POOL_DEFINE(mdm_recv_pool, MDM_RECV_MAX_BUF, MDM_RECV_BUF_SIZE, 0, NULL);
|
|
|
|
/* pin settings */
|
|
static const struct gpio_dt_spec power_gpio = GPIO_DT_SPEC_INST_GET(0, mdm_power_gpios);
|
|
|
|
static void socket_close(struct modem_socket *sock);
|
|
static const struct socket_dns_offload offload_dns_ops;
|
|
|
|
static inline uint32_t hash32(char *str, int len)
|
|
{
|
|
#define HASH_MULTIPLIER 37
|
|
uint32_t h = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
h = (h * HASH_MULTIPLIER) + str[i];
|
|
}
|
|
|
|
return h;
|
|
}
|
|
|
|
static inline uint8_t *modem_get_mac(const struct device *dev)
|
|
{
|
|
struct sim7080_data *data = dev->data;
|
|
uint32_t hash_value;
|
|
|
|
data->mac_addr[0] = 0x00;
|
|
data->mac_addr[1] = 0x10;
|
|
|
|
/* use IMEI for mac_addr */
|
|
hash_value = hash32(mdata.mdm_imei, strlen(mdata.mdm_imei));
|
|
|
|
UNALIGNED_PUT(hash_value, (uint32_t *)(data->mac_addr + 2));
|
|
|
|
return data->mac_addr;
|
|
}
|
|
|
|
static int offload_socket(int family, int type, int proto);
|
|
|
|
/* Setup the Modem NET Interface. */
|
|
static void modem_net_iface_init(struct net_if *iface)
|
|
{
|
|
const struct device *dev = net_if_get_device(iface);
|
|
struct sim7080_data *data = dev->data;
|
|
|
|
net_if_set_link_addr(iface, modem_get_mac(dev), sizeof(data->mac_addr), NET_LINK_ETHERNET);
|
|
|
|
data->netif = iface;
|
|
|
|
socket_offload_dns_register(&offload_dns_ops);
|
|
|
|
net_if_socket_offload_set(iface, offload_socket);
|
|
}
|
|
|
|
/**
|
|
* Changes the operating state of the sim7080.
|
|
*
|
|
* @param state The new state.
|
|
*/
|
|
static void change_state(enum sim7080_state state)
|
|
{
|
|
LOG_DBG("Changing state to (%d)", state);
|
|
mdata.state = state;
|
|
}
|
|
|
|
/**
|
|
* Get the current operating state of the sim7080.
|
|
*
|
|
* @return The current state.
|
|
*/
|
|
static enum sim7080_state get_state(void)
|
|
{
|
|
return mdata.state;
|
|
}
|
|
|
|
/*
|
|
* Parses the +CAOPEN command and gives back the
|
|
* connect semaphore.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_caopen)
|
|
{
|
|
int result = atoi(argv[1]);
|
|
|
|
LOG_INF("+CAOPEN: %d", result);
|
|
modem_cmd_handler_set_error(data, result);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Unlock the tx ready semaphore if '> ' is received.
|
|
*/
|
|
MODEM_CMD_DIRECT_DEFINE(on_cmd_tx_ready)
|
|
{
|
|
k_sem_give(&mdata.sem_tx_ready);
|
|
return len;
|
|
}
|
|
|
|
/*
|
|
* Connects an modem socket. Protocol can either be TCP or UDP.
|
|
*/
|
|
static int offload_connect(void *obj, const struct sockaddr *addr, socklen_t addrlen)
|
|
{
|
|
struct modem_socket *sock = (struct modem_socket *)obj;
|
|
uint16_t dst_port = 0;
|
|
char *protocol;
|
|
struct modem_cmd cmd[] = { MODEM_CMD("+CAOPEN: ", on_cmd_caopen, 2U, ",") };
|
|
char buf[sizeof("AT+CAOPEN: #,#,#####,#xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx#,####")];
|
|
char ip_str[NET_IPV6_ADDR_LEN];
|
|
int ret;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (modem_socket_is_allocated(&mdata.socket_config, sock) == false) {
|
|
LOG_ERR("Invalid socket id %d from fd %d", sock->id, sock->sock_fd);
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
|
|
if (sock->is_connected == true) {
|
|
LOG_ERR("Socket is already connected! id: %d, fd: %d", sock->id, sock->sock_fd);
|
|
errno = EISCONN;
|
|
return -1;
|
|
}
|
|
|
|
/* get the destination port */
|
|
if (addr->sa_family == AF_INET6) {
|
|
dst_port = ntohs(net_sin6(addr)->sin6_port);
|
|
} else if (addr->sa_family == AF_INET) {
|
|
dst_port = ntohs(net_sin(addr)->sin_port);
|
|
}
|
|
|
|
/* Get protocol */
|
|
protocol = (sock->type == SOCK_STREAM) ? "TCP" : "UDP";
|
|
|
|
ret = modem_context_sprint_ip_addr(addr, ip_str, sizeof(ip_str));
|
|
if (ret != 0) {
|
|
LOG_ERR("Failed to format IP!");
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
ret = snprintk(buf, sizeof(buf), "AT+CAOPEN=%d,%d,\"%s\",\"%s\",%d", 0, sock->id,
|
|
protocol, ip_str, dst_port);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to build connect command. ID: %d, FD: %d", sock->id, sock->sock_fd);
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmd, ARRAY_SIZE(cmd), buf,
|
|
&mdata.sem_response, MDM_CONNECT_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("%s ret: %d", buf, ret);
|
|
socket_close(sock);
|
|
goto error;
|
|
}
|
|
|
|
ret = modem_cmd_handler_get_error(&mdata.cmd_handler_data);
|
|
if (ret != 0) {
|
|
LOG_ERR("Closing the socket!");
|
|
socket_close(sock);
|
|
goto error;
|
|
}
|
|
|
|
sock->is_connected = true;
|
|
errno = 0;
|
|
return 0;
|
|
error:
|
|
errno = -ret;
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
* Send data over a given socket.
|
|
*
|
|
* First we signal the module that we want to send data over a socket.
|
|
* This is done by sending AT+CASEND=<sockfd>,<nbytes>\r\n.
|
|
* If The module is ready to send data it will send back
|
|
* an UNTERMINATED prompt '> '. After that data can be sent to the modem.
|
|
* As terminating byte a STRG+Z (0x1A) is sent. The module will
|
|
* then send a OK or ERROR.
|
|
*/
|
|
static ssize_t offload_sendto(void *obj, const void *buf, size_t len, int flags,
|
|
const struct sockaddr *dest_addr, socklen_t addrlen)
|
|
{
|
|
int ret;
|
|
struct modem_socket *sock = (struct modem_socket *)obj;
|
|
char send_buf[sizeof("AT+CASEND=#,####")] = { 0 };
|
|
char ctrlz = 0x1A;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Do some sanity checks. */
|
|
if (!buf || len == 0) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
|
|
/* Socket has to be connected. */
|
|
if (!sock->is_connected) {
|
|
errno = ENOTCONN;
|
|
return -1;
|
|
}
|
|
|
|
/* Only send up to MTU bytes. */
|
|
if (len > MDM_MAX_DATA_LENGTH) {
|
|
len = MDM_MAX_DATA_LENGTH;
|
|
}
|
|
|
|
ret = snprintk(send_buf, sizeof(send_buf), "AT+CASEND=%d,%ld", sock->id, (long)len);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to build send command!!");
|
|
errno = ENOMEM;
|
|
return -1;
|
|
}
|
|
|
|
/* Make sure only one send can be done at a time. */
|
|
k_sem_take(&mdata.cmd_handler_data.sem_tx_lock, K_FOREVER);
|
|
k_sem_reset(&mdata.sem_tx_ready);
|
|
|
|
/* Send CASEND */
|
|
mdata.current_sock_written = len;
|
|
ret = modem_cmd_send_nolock(&mctx.iface, &mctx.cmd_handler, NULL, 0U, send_buf, NULL,
|
|
K_NO_WAIT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to send CASEND!!");
|
|
goto exit;
|
|
}
|
|
|
|
/* Wait for '> ' */
|
|
ret = k_sem_take(&mdata.sem_tx_ready, K_SECONDS(2));
|
|
if (ret < 0) {
|
|
LOG_ERR("Timeout while waiting for tx");
|
|
goto exit;
|
|
}
|
|
|
|
/* Send data */
|
|
mctx.iface.write(&mctx.iface, buf, len);
|
|
mctx.iface.write(&mctx.iface, &ctrlz, 1);
|
|
|
|
/* Wait for the OK */
|
|
k_sem_reset(&mdata.sem_response);
|
|
ret = k_sem_take(&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Timeout waiting for OK");
|
|
}
|
|
|
|
exit:
|
|
k_sem_give(&mdata.cmd_handler_data.sem_tx_lock);
|
|
/* Data was successfully sent */
|
|
|
|
if (ret < 0) {
|
|
errno = -ret;
|
|
return -1;
|
|
}
|
|
|
|
errno = 0;
|
|
return mdata.current_sock_written;
|
|
}
|
|
|
|
/*
|
|
* Read data from a given socket.
|
|
*
|
|
* The response has the form +CARECV: <length>,data\r\nOK\r\n
|
|
*/
|
|
static int sockread_common(int sockfd, struct modem_cmd_handler_data *data, int socket_data_length,
|
|
uint16_t len)
|
|
{
|
|
struct modem_socket *sock;
|
|
struct socket_read_data *sock_data;
|
|
int ret, packet_size;
|
|
|
|
if (!len) {
|
|
LOG_ERR("Invalid length, aborting");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (!data->rx_buf) {
|
|
LOG_ERR("Incorrect format! Ignoring data!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (socket_data_length <= 0) {
|
|
LOG_ERR("Length error (%d)", socket_data_length);
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (net_buf_frags_len(data->rx_buf) < socket_data_length) {
|
|
LOG_DBG("Not enough data -- wait!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
sock = modem_socket_from_fd(&mdata.socket_config, sockfd);
|
|
if (!sock) {
|
|
LOG_ERR("Socket not found! (%d)", sockfd);
|
|
ret = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
sock_data = (struct socket_read_data *)sock->data;
|
|
if (!sock_data) {
|
|
LOG_ERR("Socket data not found! (%d)", sockfd);
|
|
ret = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
ret = net_buf_linearize(sock_data->recv_buf, sock_data->recv_buf_len, data->rx_buf, 0,
|
|
(uint16_t)socket_data_length);
|
|
data->rx_buf = net_buf_skip(data->rx_buf, ret);
|
|
sock_data->recv_read_len = ret;
|
|
if (ret != socket_data_length) {
|
|
LOG_ERR("Total copied data is different then received data!"
|
|
" copied:%d vs. received:%d",
|
|
ret, socket_data_length);
|
|
ret = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
exit:
|
|
/* Indication only sets length to a dummy value. */
|
|
packet_size = modem_socket_next_packet_size(&mdata.socket_config, sock);
|
|
modem_socket_packet_size_update(&mdata.socket_config, sock, -packet_size);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Handler for carecv response.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_carecv)
|
|
{
|
|
return sockread_common(mdata.current_sock_fd, data, atoi(argv[0]), len);
|
|
}
|
|
|
|
/*
|
|
* Read data from a given socket.
|
|
*/
|
|
static ssize_t offload_recvfrom(void *obj, void *buf, size_t max_len, int flags,
|
|
struct sockaddr *src_addr, socklen_t *addrlen)
|
|
{
|
|
struct modem_socket *sock = (struct modem_socket *)obj;
|
|
char sendbuf[sizeof("AT+CARECV=##,####")];
|
|
int ret, packet_size;
|
|
struct socket_read_data sock_data;
|
|
|
|
struct modem_cmd data_cmd[] = { MODEM_CMD("+CARECV: ", on_cmd_carecv, 1U, ",") };
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (!buf || max_len == 0) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
|
|
if (flags & ZSOCK_MSG_PEEK) {
|
|
errno = ENOTSUP;
|
|
return -1;
|
|
}
|
|
|
|
packet_size = modem_socket_next_packet_size(&mdata.socket_config, sock);
|
|
if (!packet_size) {
|
|
if (flags & ZSOCK_MSG_DONTWAIT) {
|
|
errno = EAGAIN;
|
|
return -1;
|
|
}
|
|
|
|
modem_socket_wait_data(&mdata.socket_config, sock);
|
|
packet_size = modem_socket_next_packet_size(&mdata.socket_config, sock);
|
|
}
|
|
|
|
max_len = (max_len > MDM_MAX_DATA_LENGTH) ? MDM_MAX_DATA_LENGTH : max_len;
|
|
snprintk(sendbuf, sizeof(sendbuf), "AT+CARECV=%d,%zd", sock->id, max_len);
|
|
|
|
memset(&sock_data, 0, sizeof(sock_data));
|
|
sock_data.recv_buf = buf;
|
|
sock_data.recv_buf_len = max_len;
|
|
sock_data.recv_addr = src_addr;
|
|
sock->data = &sock_data;
|
|
mdata.current_sock_fd = sock->sock_fd;
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, data_cmd, ARRAY_SIZE(data_cmd),
|
|
sendbuf, &mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
errno = -ret;
|
|
ret = -1;
|
|
goto exit;
|
|
}
|
|
|
|
/* HACK: use dst address as src */
|
|
if (src_addr && addrlen) {
|
|
*addrlen = sizeof(sock->dst);
|
|
memcpy(src_addr, &sock->dst, *addrlen);
|
|
}
|
|
|
|
errno = 0;
|
|
ret = sock_data.recv_read_len;
|
|
|
|
exit:
|
|
/* clear socket data */
|
|
mdata.current_sock_fd = -1;
|
|
sock->data = NULL;
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Sends messages to the modem.
|
|
*/
|
|
static ssize_t offload_sendmsg(void *obj, const struct msghdr *msg, int flags)
|
|
{
|
|
struct modem_socket *sock = obj;
|
|
ssize_t sent = 0;
|
|
const char *buf;
|
|
size_t len;
|
|
int ret;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (sock->type == SOCK_DGRAM) {
|
|
/*
|
|
* Current implementation only handles single contiguous fragment at a time, so
|
|
* prevent sending multiple datagrams.
|
|
*/
|
|
if (msghdr_non_empty_iov_count(msg) > 1) {
|
|
errno = EMSGSIZE;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < msg->msg_iovlen; i++) {
|
|
buf = msg->msg_iov[i].iov_base;
|
|
len = msg->msg_iov[i].iov_len;
|
|
|
|
while (len > 0) {
|
|
ret = offload_sendto(obj, buf, len, flags, msg->msg_name, msg->msg_namelen);
|
|
if (ret < 0) {
|
|
if (ret == -EAGAIN) {
|
|
k_sleep(K_SECONDS(1));
|
|
} else {
|
|
return ret;
|
|
}
|
|
} else {
|
|
sent += ret;
|
|
buf += ret;
|
|
len -= ret;
|
|
}
|
|
}
|
|
}
|
|
|
|
return sent;
|
|
}
|
|
|
|
/*
|
|
* Closes a given socket.
|
|
*/
|
|
static void socket_close(struct modem_socket *sock)
|
|
{
|
|
char buf[sizeof("AT+CACLOSE=##")];
|
|
int ret;
|
|
|
|
snprintk(buf, sizeof(buf), "AT+CACLOSE=%d", sock->id);
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buf, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("%s ret: %d", buf, ret);
|
|
}
|
|
|
|
modem_socket_put(&mdata.socket_config, sock->sock_fd);
|
|
}
|
|
|
|
/*
|
|
* Offloads read by reading from a given socket.
|
|
*/
|
|
static ssize_t offload_read(void *obj, void *buffer, size_t count)
|
|
{
|
|
return offload_recvfrom(obj, buffer, count, 0, NULL, 0);
|
|
}
|
|
|
|
/*
|
|
* Offloads write by writing to a given socket.
|
|
*/
|
|
static ssize_t offload_write(void *obj, const void *buffer, size_t count)
|
|
{
|
|
return offload_sendto(obj, buffer, count, 0, NULL, 0);
|
|
}
|
|
|
|
/*
|
|
* Offloads close by terminating the connection and freeing the socket.
|
|
*/
|
|
static int offload_close(void *obj)
|
|
{
|
|
struct modem_socket *sock = (struct modem_socket *)obj;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Make sure socket is allocated */
|
|
if (modem_socket_is_allocated(&mdata.socket_config, sock) == false) {
|
|
return 0;
|
|
}
|
|
|
|
/* Close the socket only if it is connected. */
|
|
if (sock->is_connected) {
|
|
socket_close(sock);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Polls a given socket.
|
|
*/
|
|
static int offload_poll(struct zsock_pollfd *fds, int nfds, int msecs)
|
|
{
|
|
int i;
|
|
void *obj;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Only accept modem sockets. */
|
|
for (i = 0; i < nfds; i++) {
|
|
if (fds[i].fd < 0) {
|
|
continue;
|
|
}
|
|
|
|
/* If vtable matches, then it's modem socket. */
|
|
obj = zvfs_get_fd_obj(fds[i].fd,
|
|
(const struct fd_op_vtable *)&offload_socket_fd_op_vtable,
|
|
EINVAL);
|
|
if (obj == NULL) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return modem_socket_poll(&mdata.socket_config, fds, nfds, msecs);
|
|
}
|
|
|
|
/*
|
|
* Offloads ioctl. Only supported ioctl is poll_offload.
|
|
*/
|
|
static int offload_ioctl(void *obj, unsigned int request, va_list args)
|
|
{
|
|
switch (request) {
|
|
case ZFD_IOCTL_POLL_PREPARE:
|
|
return -EXDEV;
|
|
|
|
case ZFD_IOCTL_POLL_UPDATE:
|
|
return -EOPNOTSUPP;
|
|
|
|
case ZFD_IOCTL_POLL_OFFLOAD: {
|
|
/* Poll on the given socket. */
|
|
struct zsock_pollfd *fds;
|
|
int nfds, timeout;
|
|
|
|
fds = va_arg(args, struct zsock_pollfd *);
|
|
nfds = va_arg(args, int);
|
|
timeout = va_arg(args, int);
|
|
|
|
return offload_poll(fds, nfds, timeout);
|
|
}
|
|
|
|
default:
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static const struct socket_op_vtable offload_socket_fd_op_vtable = {
|
|
.fd_vtable = {
|
|
.read = offload_read,
|
|
.write = offload_write,
|
|
.close = offload_close,
|
|
.ioctl = offload_ioctl,
|
|
},
|
|
.bind = NULL,
|
|
.connect = offload_connect,
|
|
.sendto = offload_sendto,
|
|
.recvfrom = offload_recvfrom,
|
|
.listen = NULL,
|
|
.accept = NULL,
|
|
.sendmsg = offload_sendmsg,
|
|
.getsockopt = NULL,
|
|
.setsockopt = NULL,
|
|
};
|
|
|
|
/*
|
|
* Parses the dns response from the modem.
|
|
*
|
|
* Response on success:
|
|
* +CDNSGIP: 1,<domain name>,<IPv4>[,<IPv6>]
|
|
*
|
|
* Response on failure:
|
|
* +CDNSGIP: 0,<err>
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cdnsgip)
|
|
{
|
|
int state;
|
|
char ips[256];
|
|
size_t out_len;
|
|
int ret = -1;
|
|
|
|
state = atoi(argv[0]);
|
|
if (state == 0) {
|
|
LOG_ERR("DNS lookup failed with error %s", argv[1]);
|
|
goto exit;
|
|
}
|
|
|
|
/* Offset to skip the leading " */
|
|
out_len = net_buf_linearize(ips, sizeof(ips) - 1, data->rx_buf, 1, len);
|
|
ips[out_len] = '\0';
|
|
|
|
/* find trailing " */
|
|
char *ipv4 = strstr(ips, "\"");
|
|
|
|
if (!ipv4) {
|
|
LOG_ERR("Malformed DNS response!!");
|
|
goto exit;
|
|
}
|
|
|
|
*ipv4 = '\0';
|
|
net_addr_pton(dns_result.ai_family, ips,
|
|
&((struct sockaddr_in *)&dns_result_addr)->sin_addr);
|
|
ret = 0;
|
|
|
|
exit:
|
|
k_sem_give(&mdata.sem_dns);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Perform a dns lookup.
|
|
*/
|
|
static int offload_getaddrinfo(const char *node, const char *service,
|
|
const struct zsock_addrinfo *hints, struct zsock_addrinfo **res)
|
|
{
|
|
struct modem_cmd cmd[] = { MODEM_CMD("+CDNSGIP: ", on_cmd_cdnsgip, 2U, ",") };
|
|
char sendbuf[sizeof("AT+CDNSGIP=\"\",##,#####") + 128];
|
|
uint32_t port = 0;
|
|
int ret;
|
|
|
|
/* Modem is not attached to the network. */
|
|
if (get_state() != SIM7080_STATE_NETWORKING) {
|
|
LOG_ERR("Modem currently not attached to the network!");
|
|
return DNS_EAI_AGAIN;
|
|
}
|
|
|
|
/* init result */
|
|
(void)memset(&dns_result, 0, sizeof(dns_result));
|
|
(void)memset(&dns_result_addr, 0, sizeof(dns_result_addr));
|
|
|
|
/* Currently only support IPv4. */
|
|
dns_result.ai_family = AF_INET;
|
|
dns_result_addr.sa_family = AF_INET;
|
|
dns_result.ai_addr = &dns_result_addr;
|
|
dns_result.ai_addrlen = sizeof(dns_result_addr);
|
|
dns_result.ai_canonname = dns_result_canonname;
|
|
dns_result_canonname[0] = '\0';
|
|
|
|
if (service) {
|
|
port = atoi(service);
|
|
if (port < 1 || port > USHRT_MAX) {
|
|
return DNS_EAI_SERVICE;
|
|
}
|
|
}
|
|
|
|
if (port > 0U) {
|
|
if (dns_result.ai_family == AF_INET) {
|
|
net_sin(&dns_result_addr)->sin_port = htons(port);
|
|
}
|
|
}
|
|
|
|
/* Check if node is an IP address */
|
|
if (net_addr_pton(dns_result.ai_family, node,
|
|
&((struct sockaddr_in *)&dns_result_addr)->sin_addr) == 0) {
|
|
*res = &dns_result;
|
|
return 0;
|
|
}
|
|
|
|
/* user flagged node as numeric host, but we failed net_addr_pton */
|
|
if (hints && hints->ai_flags & AI_NUMERICHOST) {
|
|
return DNS_EAI_NONAME;
|
|
}
|
|
|
|
snprintk(sendbuf, sizeof(sendbuf), "AT+CDNSGIP=\"%s\",10,20000", node);
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmd, ARRAY_SIZE(cmd), sendbuf,
|
|
&mdata.sem_dns, MDM_DNS_TIMEOUT);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
*res = (struct zsock_addrinfo *)&dns_result;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Free addrinfo structure.
|
|
*/
|
|
static void offload_freeaddrinfo(struct zsock_addrinfo *res)
|
|
{
|
|
/* No need to free static memory. */
|
|
ARG_UNUSED(res);
|
|
}
|
|
|
|
/*
|
|
* DNS vtable.
|
|
*/
|
|
static const struct socket_dns_offload offload_dns_ops = {
|
|
.getaddrinfo = offload_getaddrinfo,
|
|
.freeaddrinfo = offload_freeaddrinfo,
|
|
};
|
|
|
|
static struct offloaded_if_api api_funcs = {
|
|
.iface_api.init = modem_net_iface_init,
|
|
};
|
|
|
|
static bool offload_is_supported(int family, int type, int proto)
|
|
{
|
|
if (family != AF_INET &&
|
|
family != AF_INET6) {
|
|
return false;
|
|
}
|
|
|
|
if (type != SOCK_DGRAM &&
|
|
type != SOCK_STREAM) {
|
|
return false;
|
|
}
|
|
|
|
if (proto != IPPROTO_TCP &&
|
|
proto != IPPROTO_UDP) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int offload_socket(int family, int type, int proto)
|
|
{
|
|
int ret;
|
|
|
|
ret = modem_socket_get(&mdata.socket_config, family, type, proto);
|
|
if (ret < 0) {
|
|
errno = -ret;
|
|
return -1;
|
|
}
|
|
|
|
errno = 0;
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Process all messages received from the modem.
|
|
*/
|
|
static void modem_rx(void *p1, void *p2, void *p3)
|
|
{
|
|
ARG_UNUSED(p1);
|
|
ARG_UNUSED(p2);
|
|
ARG_UNUSED(p3);
|
|
|
|
while (true) {
|
|
/* Wait for incoming data */
|
|
modem_iface_uart_rx_wait(&mctx.iface, K_FOREVER);
|
|
|
|
modem_cmd_handler_process(&mctx.cmd_handler, &mctx.iface);
|
|
}
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_cmd_ok)
|
|
{
|
|
modem_cmd_handler_set_error(data, 0);
|
|
k_sem_give(&mdata.sem_response);
|
|
return 0;
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_cmd_error)
|
|
{
|
|
modem_cmd_handler_set_error(data, -EIO);
|
|
k_sem_give(&mdata.sem_response);
|
|
return 0;
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_cmd_exterror)
|
|
{
|
|
modem_cmd_handler_set_error(data, -EIO);
|
|
k_sem_give(&mdata.sem_response);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handles pdp context urc.
|
|
*
|
|
* The urc has the form +APP PDP: <index>,<state>.
|
|
* State can either be ACTIVE for activation or
|
|
* DEACTIVE if disabled.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_urc_app_pdp)
|
|
{
|
|
mdata.pdp_active = strcmp(argv[1], "ACTIVE") == 0;
|
|
LOG_INF("PDP context: %u", mdata.pdp_active);
|
|
k_sem_give(&mdata.sem_response);
|
|
return 0;
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_urc_sms)
|
|
{
|
|
LOG_INF("SMS: %s", argv[0]);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handles socket data notification.
|
|
*
|
|
* The sim modem sends and unsolicited +CADATAIND: <cid>
|
|
* if data can be read from a socket.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_urc_cadataind)
|
|
{
|
|
struct modem_socket *sock;
|
|
int sock_fd;
|
|
|
|
sock_fd = atoi(argv[0]);
|
|
|
|
sock = modem_socket_from_fd(&mdata.socket_config, sock_fd);
|
|
if (!sock) {
|
|
return 0;
|
|
}
|
|
|
|
/* Modem does not tell packet size. Set dummy for receive. */
|
|
modem_socket_packet_size_update(&mdata.socket_config, sock, 1);
|
|
|
|
LOG_INF("Data available on socket: %d", sock_fd);
|
|
modem_socket_data_ready(&mdata.socket_config, sock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handles the castate response.
|
|
*
|
|
* +CASTATE: <cid>,<state>
|
|
*
|
|
* Cid is the connection id (socket fd) and
|
|
* state can be:
|
|
* 0 - Closed by remote server or error
|
|
* 1 - Connected to remote server
|
|
* 2 - Listening
|
|
*/
|
|
MODEM_CMD_DEFINE(on_urc_castate)
|
|
{
|
|
struct modem_socket *sock;
|
|
int sockfd, state;
|
|
|
|
sockfd = atoi(argv[0]);
|
|
state = atoi(argv[1]);
|
|
|
|
sock = modem_socket_from_fd(&mdata.socket_config, sockfd);
|
|
if (!sock) {
|
|
return 0;
|
|
}
|
|
|
|
/* Only continue if socket was closed. */
|
|
if (state != 0) {
|
|
return 0;
|
|
}
|
|
|
|
LOG_INF("Socket close indication for socket: %d", sockfd);
|
|
|
|
sock->is_connected = false;
|
|
LOG_INF("Socket closed: %d", sockfd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Handles the ftpget urc.
|
|
*
|
|
* +FTPGET: <mode>,<error>
|
|
*
|
|
* Mode can be 1 for opening a session and
|
|
* reporting that data is available or 2 for
|
|
* reading data. This urc handler will only handle
|
|
* mode 1 because 2 will not occur as urc.
|
|
*
|
|
* Error can be either:
|
|
* - 1 for data available/opened session.
|
|
* - 0 If transfer is finished.
|
|
* - >0 for some error.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_urc_ftpget)
|
|
{
|
|
int error = atoi(argv[0]);
|
|
|
|
LOG_INF("+FTPGET: 1,%d", error);
|
|
|
|
/* Transfer finished. */
|
|
if (error == 0) {
|
|
mdata.ftp.state = SIM7080_FTP_CONNECTION_STATE_FINISHED;
|
|
} else if (error == 1) {
|
|
mdata.ftp.state = SIM7080_FTP_CONNECTION_STATE_CONNECTED;
|
|
} else {
|
|
mdata.ftp.state = SIM7080_FTP_CONNECTION_STATE_ERROR;
|
|
}
|
|
|
|
k_sem_give(&mdata.sem_ftp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read manufacturer identification.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cgmi)
|
|
{
|
|
size_t out_len = net_buf_linearize(
|
|
mdata.mdm_manufacturer, sizeof(mdata.mdm_manufacturer) - 1, data->rx_buf, 0, len);
|
|
mdata.mdm_manufacturer[out_len] = '\0';
|
|
LOG_INF("Manufacturer: %s", mdata.mdm_manufacturer);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read model identification.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cgmm)
|
|
{
|
|
size_t out_len = net_buf_linearize(mdata.mdm_model, sizeof(mdata.mdm_model) - 1,
|
|
data->rx_buf, 0, len);
|
|
mdata.mdm_model[out_len] = '\0';
|
|
LOG_INF("Model: %s", mdata.mdm_model);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read software release.
|
|
*
|
|
* Response will be in format RESPONSE: <revision>.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cgmr)
|
|
{
|
|
size_t out_len;
|
|
char *p;
|
|
|
|
out_len = net_buf_linearize(mdata.mdm_revision, sizeof(mdata.mdm_revision) - 1,
|
|
data->rx_buf, 0, len);
|
|
mdata.mdm_revision[out_len] = '\0';
|
|
|
|
/* The module prepends a Revision: */
|
|
p = strchr(mdata.mdm_revision, ':');
|
|
if (p) {
|
|
out_len = strlen(p + 1);
|
|
memmove(mdata.mdm_revision, p + 1, out_len + 1);
|
|
}
|
|
|
|
LOG_INF("Revision: %s", mdata.mdm_revision);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read serial number identification.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cgsn)
|
|
{
|
|
size_t out_len =
|
|
net_buf_linearize(mdata.mdm_imei, sizeof(mdata.mdm_imei) - 1, data->rx_buf, 0, len);
|
|
mdata.mdm_imei[out_len] = '\0';
|
|
LOG_INF("IMEI: %s", mdata.mdm_imei);
|
|
return 0;
|
|
}
|
|
|
|
#if defined(CONFIG_MODEM_SIM_NUMBERS)
|
|
/*
|
|
* Read international mobile subscriber identity.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cimi)
|
|
{
|
|
size_t out_len =
|
|
net_buf_linearize(mdata.mdm_imsi, sizeof(mdata.mdm_imsi) - 1, data->rx_buf, 0, len);
|
|
mdata.mdm_imsi[out_len] = '\0';
|
|
|
|
/* Log the received information. */
|
|
LOG_INF("IMSI: %s", mdata.mdm_imsi);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Read iccid.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_ccid)
|
|
{
|
|
size_t out_len = net_buf_linearize(mdata.mdm_iccid, sizeof(mdata.mdm_iccid) - 1,
|
|
data->rx_buf, 0, len);
|
|
mdata.mdm_iccid[out_len] = '\0';
|
|
|
|
/* Log the received information. */
|
|
LOG_INF("ICCID: %s", mdata.mdm_iccid);
|
|
return 0;
|
|
}
|
|
#endif /* defined(CONFIG_MODEM_SIM_NUMBERS) */
|
|
|
|
/*
|
|
* Parses the non urc C(E)REG and updates registration status.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cereg)
|
|
{
|
|
mdata.mdm_registration = atoi(argv[1]);
|
|
LOG_INF("CREG: %u", mdata.mdm_registration);
|
|
return 0;
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_cmd_cpin)
|
|
{
|
|
mdata.cpin_ready = strcmp(argv[0], "READY") == 0;
|
|
LOG_INF("CPIN: %d", mdata.cpin_ready);
|
|
return 0;
|
|
}
|
|
|
|
MODEM_CMD_DEFINE(on_cmd_cgatt)
|
|
{
|
|
mdata.mdm_cgatt = atoi(argv[0]);
|
|
LOG_INF("CGATT: %d", mdata.mdm_cgatt);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Handler for RSSI query.
|
|
*
|
|
* +CSQ: <rssi>,<ber>
|
|
* rssi: 0,-115dBm; 1,-111dBm; 2...30,-110...-54dBm; 31,-52dBm or greater.
|
|
* 99, ukn
|
|
* ber: Not used.
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_csq)
|
|
{
|
|
int rssi = atoi(argv[0]);
|
|
|
|
if (rssi == 0) {
|
|
mdata.mdm_rssi = -115;
|
|
} else if (rssi == 1) {
|
|
mdata.mdm_rssi = -111;
|
|
} else if (rssi > 1 && rssi < 31) {
|
|
mdata.mdm_rssi = -114 + 2 * rssi;
|
|
} else if (rssi == 31) {
|
|
mdata.mdm_rssi = -52;
|
|
} else {
|
|
mdata.mdm_rssi = -1000;
|
|
}
|
|
|
|
LOG_INF("RSSI: %d", mdata.mdm_rssi);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Queries modem RSSI.
|
|
*
|
|
* If a work queue parameter is provided query work will
|
|
* be scheduled. Otherwise rssi is queried once.
|
|
*/
|
|
static void modem_rssi_query_work(struct k_work *work)
|
|
{
|
|
struct modem_cmd cmd[] = { MODEM_CMD("+CSQ: ", on_cmd_csq, 2U, ",") };
|
|
static char *send_cmd = "AT+CSQ";
|
|
int ret;
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmd, ARRAY_SIZE(cmd), send_cmd,
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("AT+CSQ ret:%d", ret);
|
|
}
|
|
|
|
if (work) {
|
|
k_work_reschedule_for_queue(&modem_workq, &mdata.rssi_query_work,
|
|
K_SECONDS(RSSI_TIMEOUT_SECS));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Possible responses by the sim7080.
|
|
*/
|
|
static const struct modem_cmd response_cmds[] = {
|
|
MODEM_CMD("OK", on_cmd_ok, 0U, ""),
|
|
MODEM_CMD("ERROR", on_cmd_error, 0U, ""),
|
|
MODEM_CMD("+CME ERROR: ", on_cmd_exterror, 1U, ""),
|
|
MODEM_CMD_DIRECT(">", on_cmd_tx_ready),
|
|
};
|
|
|
|
/*
|
|
* Possible unsolicited commands.
|
|
*/
|
|
static const struct modem_cmd unsolicited_cmds[] = {
|
|
MODEM_CMD("+APP PDP: ", on_urc_app_pdp, 2U, ","),
|
|
MODEM_CMD("SMS ", on_urc_sms, 1U, ""),
|
|
MODEM_CMD("+CADATAIND: ", on_urc_cadataind, 1U, ""),
|
|
MODEM_CMD("+CASTATE: ", on_urc_castate, 2U, ","),
|
|
MODEM_CMD("+FTPGET: 1,", on_urc_ftpget, 1U, ""),
|
|
};
|
|
|
|
/*
|
|
* Activates the pdp context
|
|
*/
|
|
static int modem_pdp_activate(void)
|
|
{
|
|
int counter;
|
|
int ret = 0;
|
|
#if defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_GSM)
|
|
const char *buf = "AT+CREG?";
|
|
struct modem_cmd cmds[] = { MODEM_CMD("+CREG: ", on_cmd_cereg, 2U, ",") };
|
|
#else
|
|
const char *buf = "AT+CEREG?";
|
|
struct modem_cmd cmds[] = { MODEM_CMD("+CEREG: ", on_cmd_cereg, 2U, ",") };
|
|
#endif /* defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_GSM) */
|
|
|
|
struct modem_cmd cgatt_cmd[] = { MODEM_CMD("+CGATT: ", on_cmd_cgatt, 1U, "") };
|
|
|
|
counter = 0;
|
|
while (counter++ < MDM_MAX_CGATT_WAITS && mdata.mdm_cgatt != 1) {
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cgatt_cmd,
|
|
ARRAY_SIZE(cgatt_cmd), "AT+CGATT?", &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to query cgatt!!");
|
|
return -1;
|
|
}
|
|
|
|
k_sleep(K_SECONDS(1));
|
|
}
|
|
|
|
if (counter >= MDM_MAX_CGATT_WAITS) {
|
|
LOG_WRN("Network attach failed!!");
|
|
return -1;
|
|
}
|
|
|
|
if (!mdata.cpin_ready || mdata.mdm_cgatt != 1) {
|
|
LOG_ERR("Fatal: Modem is not attached to GPRS network!!");
|
|
return -1;
|
|
}
|
|
|
|
LOG_INF("Waiting for network");
|
|
|
|
/* Wait until the module is registered to the network.
|
|
* Registration will be set by urc.
|
|
*/
|
|
counter = 0;
|
|
while (counter++ < MDM_MAX_CEREG_WAITS && mdata.mdm_registration != 1 &&
|
|
mdata.mdm_registration != 5) {
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmds, ARRAY_SIZE(cmds), buf,
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to query registration!!");
|
|
return -1;
|
|
}
|
|
|
|
k_sleep(K_SECONDS(1));
|
|
}
|
|
|
|
if (counter >= MDM_MAX_CEREG_WAITS) {
|
|
LOG_WRN("Network registration failed!");
|
|
ret = -1;
|
|
goto error;
|
|
}
|
|
|
|
/* Set dual stack mode (IPv4/IPv6) */
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0, "AT+CNCFG=0,0",
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Could not configure pdp context!");
|
|
goto error;
|
|
}
|
|
|
|
/*
|
|
* Now activate the pdp context and wait for confirmation.
|
|
*/
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0, "AT+CNACT=0,1",
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Could not activate PDP context.");
|
|
goto error;
|
|
}
|
|
|
|
ret = k_sem_take(&mdata.sem_response, MDM_PDP_TIMEOUT);
|
|
if (ret < 0 || mdata.pdp_active == false) {
|
|
LOG_ERR("Failed to activate PDP context.");
|
|
ret = -1;
|
|
goto error;
|
|
}
|
|
|
|
LOG_INF("Network active.");
|
|
|
|
error:
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Toggles the modems power pin.
|
|
*/
|
|
static void modem_pwrkey(void)
|
|
{
|
|
/* Power pin should be high for 1.5 seconds. */
|
|
gpio_pin_set_dt(&power_gpio, 1);
|
|
k_sleep(K_MSEC(1500));
|
|
gpio_pin_set_dt(&power_gpio, 0);
|
|
k_sleep(K_SECONDS(5));
|
|
}
|
|
|
|
/*
|
|
* Commands to be sent at setup.
|
|
*/
|
|
static const struct setup_cmd setup_cmds[] = {
|
|
SETUP_CMD_NOHANDLE("ATH"),
|
|
SETUP_CMD("AT+CGMI", "", on_cmd_cgmi, 0U, ""),
|
|
SETUP_CMD("AT+CGMM", "", on_cmd_cgmm, 0U, ""),
|
|
SETUP_CMD("AT+CGMR", "", on_cmd_cgmr, 0U, ""),
|
|
SETUP_CMD("AT+CGSN", "", on_cmd_cgsn, 0U, ""),
|
|
#if defined(CONFIG_MODEM_SIM_NUMBERS)
|
|
SETUP_CMD("AT+CIMI", "", on_cmd_cimi, 0U, ""),
|
|
SETUP_CMD("AT+CCID", "", on_cmd_ccid, 0U, ""),
|
|
#endif /* defined(CONFIG_MODEM_SIM_NUMBERS) */
|
|
#if defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_NB1)
|
|
SETUP_CMD_NOHANDLE("AT+CNMP=38"),
|
|
SETUP_CMD_NOHANDLE("AT+CMNB=2"),
|
|
SETUP_CMD_NOHANDLE("AT+CBANDCFG=\"NB-IOT\"," MDM_LTE_BANDS),
|
|
#endif /* defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_NB1) */
|
|
#if defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_M1)
|
|
SETUP_CMD_NOHANDLE("AT+CNMP=38"),
|
|
SETUP_CMD_NOHANDLE("AT+CMNB=1"),
|
|
SETUP_CMD_NOHANDLE("AT+CBANDCFG=\"CAT-M\"," MDM_LTE_BANDS),
|
|
#endif /* defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_M1) */
|
|
#if defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_GSM)
|
|
SETUP_CMD_NOHANDLE("AT+CNMP=13"),
|
|
#endif /* defined(CONFIG_MODEM_SIMCOM_SIM7080_RAT_GSM) */
|
|
SETUP_CMD("AT+CPIN?", "+CPIN: ", on_cmd_cpin, 1U, ""),
|
|
};
|
|
|
|
/**
|
|
* Performs the autobaud sequence until modem answers or limit is reached.
|
|
*
|
|
* @return On successful boot 0 is returned. Otherwise <0 is returned.
|
|
*/
|
|
static int modem_autobaud(void)
|
|
{
|
|
int boot_tries = 0;
|
|
int counter = 0;
|
|
int ret;
|
|
|
|
while (boot_tries++ <= MDM_BOOT_TRIES) {
|
|
modem_pwrkey();
|
|
|
|
/*
|
|
* The sim7080 has a autobaud function.
|
|
* On startup multiple AT's are sent until
|
|
* a OK is received.
|
|
*/
|
|
counter = 0;
|
|
while (counter < MDM_MAX_AUTOBAUD) {
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT",
|
|
&mdata.sem_response, K_MSEC(500));
|
|
|
|
/* OK was received. */
|
|
if (ret == 0) {
|
|
/* Disable echo */
|
|
return modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U,
|
|
"ATE0", &mdata.sem_response, K_SECONDS(2));
|
|
}
|
|
|
|
counter++;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Get the next parameter from the gnss phrase.
|
|
*
|
|
* @param src The source string supported on first call.
|
|
* @param delim The delimiter of the parameter list.
|
|
* @param saveptr Pointer for subsequent parses.
|
|
* @return On success a pointer to the parameter. On failure
|
|
* or end of string NULL is returned.
|
|
*
|
|
* This function is used instead of strtok because strtok would
|
|
* skip empty parameters, which is not desired. The modem may
|
|
* omit parameters which could lead to a incorrect parse.
|
|
*/
|
|
static char *gnss_get_next_param(char *src, const char *delim, char **saveptr)
|
|
{
|
|
char *start, *del;
|
|
|
|
if (src) {
|
|
start = src;
|
|
} else {
|
|
start = *saveptr;
|
|
}
|
|
|
|
/* Illegal start string. */
|
|
if (!start) {
|
|
return NULL;
|
|
}
|
|
|
|
/* End of string reached. */
|
|
if (*start == '\0' || *start == '\r') {
|
|
return NULL;
|
|
}
|
|
|
|
del = strstr(start, delim);
|
|
if (!del) {
|
|
return NULL;
|
|
}
|
|
|
|
*del = '\0';
|
|
*saveptr = del + 1;
|
|
|
|
if (del == start) {
|
|
return NULL;
|
|
}
|
|
|
|
return start;
|
|
}
|
|
|
|
static void gnss_skip_param(char **saveptr)
|
|
{
|
|
gnss_get_next_param(NULL, ",", saveptr);
|
|
}
|
|
|
|
/**
|
|
* Splits float parameters of the CGNSINF response on '.'
|
|
*
|
|
* @param src Null terminated string containing the float.
|
|
* @param f1 Resulting number part of the float.
|
|
* @param f2 Resulting fraction part of the float.
|
|
* @return 0 if parsing was successful. Otherwise <0 is returned.
|
|
*
|
|
* If the number part of the float is negative f1 and f2 will be
|
|
* negative too.
|
|
*/
|
|
static int gnss_split_on_dot(const char *src, int32_t *f1, int32_t *f2)
|
|
{
|
|
char *dot = strchr(src, '.');
|
|
|
|
if (!dot) {
|
|
return -1;
|
|
}
|
|
|
|
*dot = '\0';
|
|
|
|
*f1 = (int32_t)strtol(src, NULL, 10);
|
|
*f2 = (int32_t)strtol(dot + 1, NULL, 10);
|
|
|
|
if (*f1 < 0) {
|
|
*f2 = -*f2;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Parses cgnsinf response into the gnss_data structure.
|
|
*
|
|
* @param gps_buf Null terminated buffer containing the response.
|
|
* @return 0 on successful parse. Otherwise <0 is returned.
|
|
*/
|
|
static int parse_cgnsinf(char *gps_buf)
|
|
{
|
|
char *saveptr;
|
|
int ret;
|
|
int32_t number, fraction;
|
|
|
|
char *run_status = gnss_get_next_param(gps_buf, ",", &saveptr);
|
|
|
|
if (run_status == NULL) {
|
|
goto error;
|
|
} else if (*run_status != '1') {
|
|
goto error;
|
|
}
|
|
|
|
char *fix_status = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
if (fix_status == NULL) {
|
|
goto error;
|
|
} else if (*fix_status != '1') {
|
|
goto error;
|
|
}
|
|
|
|
char *utc = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
if (utc == NULL) {
|
|
goto error;
|
|
}
|
|
|
|
char *lat = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
if (lat == NULL) {
|
|
goto error;
|
|
}
|
|
|
|
char *lon = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
if (lon == NULL) {
|
|
goto error;
|
|
}
|
|
|
|
char *alt = gnss_get_next_param(NULL, ",", &saveptr);
|
|
char *speed = gnss_get_next_param(NULL, ",", &saveptr);
|
|
char *course = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
/* discard fix mode and reserved*/
|
|
gnss_skip_param(&saveptr);
|
|
gnss_skip_param(&saveptr);
|
|
|
|
char *hdop = gnss_get_next_param(NULL, ",", &saveptr);
|
|
|
|
if (hdop == NULL) {
|
|
goto error;
|
|
}
|
|
|
|
gnss_data.run_status = 1;
|
|
gnss_data.fix_status = 1;
|
|
|
|
strncpy(gnss_data.utc, utc, sizeof(gnss_data.utc) - 1);
|
|
|
|
ret = gnss_split_on_dot(lat, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.lat = number * 10000000 + fraction * 10;
|
|
|
|
ret = gnss_split_on_dot(lon, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.lon = number * 10000000 + fraction * 10;
|
|
|
|
if (alt) {
|
|
ret = gnss_split_on_dot(alt, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.alt = number * 1000 + fraction;
|
|
} else {
|
|
gnss_data.alt = 0;
|
|
}
|
|
|
|
ret = gnss_split_on_dot(hdop, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.hdop = number * 100 + fraction * 10;
|
|
|
|
if (course) {
|
|
ret = gnss_split_on_dot(course, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.cog = number * 100 + fraction * 10;
|
|
} else {
|
|
gnss_data.cog = 0;
|
|
}
|
|
|
|
if (speed) {
|
|
ret = gnss_split_on_dot(speed, &number, &fraction);
|
|
if (ret != 0) {
|
|
goto error;
|
|
}
|
|
gnss_data.kmh = number * 10 + fraction / 10;
|
|
} else {
|
|
gnss_data.kmh = 0;
|
|
}
|
|
|
|
return 0;
|
|
error:
|
|
memset(&gnss_data, 0, sizeof(gnss_data));
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
* Parses the +CGNSINF Gnss response.
|
|
*
|
|
* The CGNSINF command has the following parameters but
|
|
* not all parameters are set by the module:
|
|
*
|
|
* +CGNSINF: <GNSS run status>,<Fix status>,<UTC date & Time>,
|
|
* <Latitude>,<Longitude>,<MSL Altitude>,<Speed Over Ground>,
|
|
* <Course Over Ground>,<Fix Mode>,<Reserved1>,<HDOP>,<PDOP>,
|
|
* <VDOP>,<Reserved2>,<GNSS Satellites in View>,<Reserved3>,
|
|
* <HPA>,<VPA>
|
|
*
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cgnsinf)
|
|
{
|
|
char gps_buf[MDM_GNSS_PARSER_MAX_LEN];
|
|
size_t out_len = net_buf_linearize(gps_buf, sizeof(gps_buf) - 1, data->rx_buf, 0, len);
|
|
|
|
gps_buf[out_len] = '\0';
|
|
return parse_cgnsinf(gps_buf);
|
|
}
|
|
|
|
int mdm_sim7080_query_gnss(struct sim7080_gnss_data *data)
|
|
{
|
|
int ret;
|
|
struct modem_cmd cmds[] = { MODEM_CMD("+CGNSINF: ", on_cmd_cgnsinf, 0U, NULL) };
|
|
|
|
if (get_state() != SIM7080_STATE_GNSS) {
|
|
LOG_ERR("GNSS functionality is not enabled!!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmds, ARRAY_SIZE(cmds), "AT+CGNSINF",
|
|
&mdata.sem_response, K_SECONDS(2));
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (!gnss_data.run_status || !gnss_data.fix_status) {
|
|
return -EAGAIN;
|
|
}
|
|
|
|
if (data) {
|
|
memcpy(data, &gnss_data, sizeof(gnss_data));
|
|
}
|
|
|
|
memset(&gnss_data, 0, sizeof(gnss_data));
|
|
|
|
return ret;
|
|
}
|
|
|
|
int mdm_sim7080_start_gnss(void)
|
|
{
|
|
int ret;
|
|
|
|
change_state(SIM7080_STATE_INIT);
|
|
k_work_cancel_delayable(&mdata.rssi_query_work);
|
|
|
|
ret = modem_autobaud();
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to start modem!!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT+CGNSCOLD",
|
|
&mdata.sem_response, K_SECONDS(2));
|
|
if (ret < 0) {
|
|
return -1;
|
|
}
|
|
|
|
change_state(SIM7080_STATE_GNSS);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Parse the +FTPGET response.
|
|
*
|
|
* +FTPGET: <mode>,<len>
|
|
*
|
|
* Mode is hard set to 2.
|
|
*
|
|
* Length is the number of bytes following (the ftp data).
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_ftpget)
|
|
{
|
|
int nbytes = atoi(argv[0]);
|
|
int bytes_to_skip;
|
|
size_t out_len;
|
|
|
|
if (nbytes == 0) {
|
|
mdata.ftp.nread = 0;
|
|
return 0;
|
|
}
|
|
|
|
/* Skip length parameter and trailing \r\n */
|
|
bytes_to_skip = strlen(argv[0]) + 2;
|
|
|
|
/* Wait until data is ready.
|
|
* >= to ensure buffer is not empty after skip.
|
|
*/
|
|
if (net_buf_frags_len(data->rx_buf) <= nbytes + bytes_to_skip) {
|
|
return -EAGAIN;
|
|
}
|
|
|
|
out_len = net_buf_linearize(mdata.ftp.read_buffer, mdata.ftp.nread, data->rx_buf,
|
|
bytes_to_skip, nbytes);
|
|
if (out_len != nbytes) {
|
|
LOG_WRN("FTP read size differs!");
|
|
}
|
|
data->rx_buf = net_buf_skip(data->rx_buf, nbytes + bytes_to_skip);
|
|
|
|
mdata.ftp.nread = nbytes;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mdm_sim7080_ftp_get_read(char *dst, size_t *size)
|
|
{
|
|
int ret;
|
|
char buffer[sizeof("AT+FTPGET=#,######")];
|
|
struct modem_cmd cmds[] = { MODEM_CMD("+FTPGET: 2,", on_cmd_ftpget, 1U, "") };
|
|
|
|
/* Some error occurred. */
|
|
if (mdata.ftp.state == SIM7080_FTP_CONNECTION_STATE_ERROR ||
|
|
mdata.ftp.state == SIM7080_FTP_CONNECTION_STATE_INITIAL) {
|
|
return SIM7080_FTP_RC_ERROR;
|
|
}
|
|
|
|
/* Setup buffer. */
|
|
mdata.ftp.read_buffer = dst;
|
|
mdata.ftp.nread = *size;
|
|
|
|
/* Read ftp data. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPGET=2,%zu", *size);
|
|
if (ret < 0) {
|
|
*size = 0;
|
|
return SIM7080_FTP_RC_ERROR;
|
|
}
|
|
|
|
/* Wait for data from the server. */
|
|
k_sem_take(&mdata.sem_ftp, K_MSEC(200));
|
|
|
|
if (mdata.ftp.state == SIM7080_FTP_CONNECTION_STATE_FINISHED) {
|
|
*size = 0;
|
|
return SIM7080_FTP_RC_FINISHED;
|
|
} else if (mdata.ftp.state == SIM7080_FTP_CONNECTION_STATE_ERROR) {
|
|
*size = 0;
|
|
return SIM7080_FTP_RC_ERROR;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmds, ARRAY_SIZE(cmds), buffer,
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
*size = 0;
|
|
return SIM7080_FTP_RC_ERROR;
|
|
}
|
|
|
|
/* Set read size. */
|
|
*size = mdata.ftp.nread;
|
|
|
|
return SIM7080_FTP_RC_OK;
|
|
}
|
|
|
|
int mdm_sim7080_ftp_get_start(const char *server, const char *user, const char *passwd,
|
|
const char *file, const char *path)
|
|
{
|
|
int ret;
|
|
char buffer[256];
|
|
|
|
/* Start network. */
|
|
ret = mdm_sim7080_start_network();
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to start network for FTP!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set connection id for ftp. */
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT+FTPCID=0",
|
|
&mdata.sem_response, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set FTP Cid!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp server. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPSERV=\"%s\"", server);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set FTP Cid!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp user. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPUN=\"%s\"", user);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set ftp user!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp password. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPPW=\"%s\"", passwd);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set ftp password!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp filename. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPGETNAME=\"%s\"", file);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set ftp filename!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp filename. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPGETNAME=\"%s\"", file);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set ftp filename!");
|
|
return -1;
|
|
}
|
|
|
|
/* Set ftp path. */
|
|
ret = snprintk(buffer, sizeof(buffer), "AT+FTPGETPATH=\"%s\"", path);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to build command!");
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, buffer, &mdata.sem_response,
|
|
MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to set ftp path!");
|
|
return -1;
|
|
}
|
|
|
|
/* Initialize ftp variables. */
|
|
mdata.ftp.read_buffer = NULL;
|
|
mdata.ftp.nread = 0;
|
|
mdata.ftp.state = SIM7080_FTP_CONNECTION_STATE_INITIAL;
|
|
|
|
/* Start the ftp session. */
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT+FTPGET=1",
|
|
&mdata.sem_ftp, MDM_CMD_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_WRN("Failed to start session!");
|
|
return -1;
|
|
}
|
|
|
|
if (mdata.ftp.state != SIM7080_FTP_CONNECTION_STATE_CONNECTED) {
|
|
LOG_WRN("Session state is not connected!");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Decode readable hex to "real" hex.
|
|
*/
|
|
static uint8_t mdm_pdu_decode_ascii(char byte)
|
|
{
|
|
if ((byte >= '0') && (byte <= '9')) {
|
|
return byte - '0';
|
|
} else if ((byte >= 'A') && (byte <= 'F')) {
|
|
return byte - 'A' + 10;
|
|
} else if ((byte >= 'a') && (byte <= 'f')) {
|
|
return byte - 'a' + 10;
|
|
} else {
|
|
return 255;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reads "byte" from pdu.
|
|
*
|
|
* @param pdu pdu to read from.
|
|
* @param index index of "byte".
|
|
*
|
|
* Sim module "encodes" one pdu byte as two human readable bytes
|
|
* this functions squashes these two bytes into one.
|
|
*/
|
|
static uint8_t mdm_pdu_read_byte(const char *pdu, size_t index)
|
|
{
|
|
return (mdm_pdu_decode_ascii(pdu[index * 2]) << 4 |
|
|
mdm_pdu_decode_ascii(pdu[index * 2 + 1]));
|
|
}
|
|
|
|
/**
|
|
* Decodes time from pdu.
|
|
*
|
|
* @param pdu pdu to read from.
|
|
* @param index index of "byte".
|
|
*/
|
|
static uint8_t mdm_pdu_read_time(const char *pdu, size_t index)
|
|
{
|
|
return (mdm_pdu_decode_ascii(pdu[index * 2]) +
|
|
mdm_pdu_decode_ascii(pdu[index * 2 + 1]) * 10);
|
|
}
|
|
|
|
/**
|
|
* Decode a sms from pdu mode.
|
|
*/
|
|
static int mdm_decode_pdu(const char *pdu, size_t pdu_len, struct sim7080_sms *target_buf)
|
|
{
|
|
size_t index;
|
|
|
|
/*
|
|
* GSM_03.38 to Unicode conversion table
|
|
*/
|
|
const short enc7_basic[128] = {
|
|
'@', 0xA3, '$', 0xA5, 0xE8, 0xE9, 0xF9, 0xEC, 0xF2, 0xE7,
|
|
'\n', 0xD8, 0xF8, '\r', 0xC5, 0xF8, 0x0394, '_', 0x03A6, 0x0393,
|
|
0x039B, 0x03A9, 0x03A0, 0x03A8, 0x03A3, 0x0398, 0x039E, '\x1b', 0xC6, 0xE6,
|
|
0xDF, 0xC9, ' ', '!', '\"', '#', 0xA4, '%', '&', '\'',
|
|
'(', ')', '*', '+', ',', '-', '.', '/', '0', '1',
|
|
'2', '3', '4', '5', '6', '7', '8', '9', ':', ';',
|
|
'<', '=', '>', '?', 0xA1, 'A', 'B', 'C', 'D', 'E',
|
|
'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O',
|
|
'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y',
|
|
'Z', 0xC4, 0xD6, 0xD1, 0xDC, 0xA7, 0xBF, 'a', 'b', 'c',
|
|
'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
|
|
'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w',
|
|
'x', 'y', 'z', 0xE4, 0xF6, 0xF1, 0xFC, 0xE0
|
|
};
|
|
|
|
/* two bytes in pdu are on real byte */
|
|
pdu_len = (pdu_len / 2);
|
|
|
|
/* first byte of pdu is length of trailing SMSC information
|
|
* skip it by setting index to SMSC length + 1.
|
|
*/
|
|
index = mdm_pdu_read_byte(pdu, 0) + 1;
|
|
|
|
if (index >= pdu_len) {
|
|
return -1;
|
|
}
|
|
|
|
/* read first octet */
|
|
target_buf->first_octet = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
if (index >= pdu_len) {
|
|
return -1;
|
|
}
|
|
|
|
/* pdu_index now points to the address field.
|
|
* first byte of addr field is the addr length -> skip it.
|
|
* address type is not included in addr len -> add +1.
|
|
* address is coded in semi octets
|
|
* + addr_len/2 if even
|
|
* + addr_len/2 + 1 if odd
|
|
*/
|
|
uint8_t addr_len = mdm_pdu_read_byte(pdu, index);
|
|
|
|
index += ((addr_len % 2) == 0) ? (addr_len / 2) + 2 : (addr_len / 2) + 3;
|
|
|
|
if (index >= pdu_len) {
|
|
return -1;
|
|
}
|
|
|
|
/* read protocol identifier */
|
|
target_buf->tp_pid = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
if (index >= pdu_len) {
|
|
return -1;
|
|
}
|
|
|
|
/* read coding scheme */
|
|
uint8_t tp_dcs = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
/* parse date and time */
|
|
if ((index + 7) >= pdu_len) {
|
|
return -1;
|
|
}
|
|
|
|
target_buf->time.year = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.month = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.day = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.hour = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.minute = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.second = mdm_pdu_read_time(pdu, index++);
|
|
target_buf->time.timezone = mdm_pdu_read_time(pdu, index++);
|
|
|
|
/* Read user data length */
|
|
uint8_t tp_udl = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
/* Discard header */
|
|
uint8_t header_skip = 0;
|
|
|
|
if (target_buf->first_octet & SMS_TP_UDHI_HEADER) {
|
|
uint8_t tp_udhl = mdm_pdu_read_byte(pdu, index);
|
|
|
|
index += tp_udhl + 1;
|
|
header_skip = tp_udhl + 1;
|
|
|
|
if (index >= pdu_len) {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/* Read data according to type set in TP-DCS */
|
|
if (tp_dcs == 0x00) {
|
|
/* 7 bit GSM coding */
|
|
uint8_t fill_level = 0;
|
|
uint16_t buf = 0;
|
|
|
|
if (target_buf->first_octet & SMS_TP_UDHI_HEADER) {
|
|
/* Initial fill because septets are aligned to
|
|
* septet boundary after header
|
|
*/
|
|
uint8_t fill_bits = 7 - ((header_skip * 8) % 7);
|
|
|
|
if (fill_bits == 7) {
|
|
fill_bits = 0;
|
|
}
|
|
|
|
buf = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
fill_level = 8 - fill_bits;
|
|
}
|
|
|
|
uint16_t data_index = 0;
|
|
|
|
for (unsigned int idx = 0; idx < tp_udl; idx++) {
|
|
if (fill_level < 7) {
|
|
uint8_t octet = mdm_pdu_read_byte(pdu, index++);
|
|
|
|
buf &= ((1 << fill_level) - 1);
|
|
buf |= (octet << fill_level);
|
|
fill_level += 8;
|
|
}
|
|
|
|
/*
|
|
* Convert 7-bit encoded data to Unicode and
|
|
* then to UTF-8
|
|
*/
|
|
short letter = enc7_basic[buf & 0x007f];
|
|
|
|
if (letter < 0x0080) {
|
|
target_buf->data[data_index++] = letter & 0x007f;
|
|
} else if (letter < 0x0800) {
|
|
target_buf->data[data_index++] = 0xc0 | ((letter & 0x07c0) >> 6);
|
|
target_buf->data[data_index++] = 0x80 | ((letter & 0x003f) >> 0);
|
|
}
|
|
buf >>= 7;
|
|
fill_level -= 7;
|
|
}
|
|
target_buf->data_len = data_index;
|
|
} else if (tp_dcs == 0x04) {
|
|
/* 8 bit binary coding */
|
|
for (int idx = 0; idx < tp_udl - header_skip; idx++) {
|
|
target_buf->data[idx] = mdm_pdu_read_byte(pdu, index++);
|
|
}
|
|
target_buf->data_len = tp_udl;
|
|
} else if (tp_dcs == 0x08) {
|
|
/* Unicode (16 bit per character) */
|
|
for (int idx = 0; idx < tp_udl - header_skip; idx++) {
|
|
target_buf->data[idx] = mdm_pdu_read_byte(pdu, index++);
|
|
}
|
|
target_buf->data_len = tp_udl;
|
|
} else {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Check if given char sequence is crlf.
|
|
*
|
|
* @param c The char sequence.
|
|
* @param len Total length of the fragment.
|
|
* @return @c true if char sequence is crlf.
|
|
* Otherwise @c false is returned.
|
|
*/
|
|
static bool is_crlf(uint8_t *c, uint8_t len)
|
|
{
|
|
/* crlf does not fit. */
|
|
if (len < 2) {
|
|
return false;
|
|
}
|
|
|
|
return c[0] == '\r' && c[1] == '\n';
|
|
}
|
|
|
|
/**
|
|
* Find terminating crlf in a netbuffer.
|
|
*
|
|
* @param buf The netbuffer.
|
|
* @param skip Bytes to skip before search.
|
|
* @return Length of the returned fragment or 0 if not found.
|
|
*/
|
|
static size_t net_buf_find_crlf(struct net_buf *buf, size_t skip)
|
|
{
|
|
size_t len = 0, pos = 0;
|
|
struct net_buf *frag = buf;
|
|
|
|
/* Skip to the start. */
|
|
while (frag && skip >= frag->len) {
|
|
skip -= frag->len;
|
|
frag = frag->frags;
|
|
}
|
|
|
|
/* Need to wait for more data. */
|
|
if (!frag) {
|
|
return 0;
|
|
}
|
|
|
|
pos = skip;
|
|
|
|
while (frag && !is_crlf(frag->data + pos, frag->len - pos)) {
|
|
if (pos + 1 >= frag->len) {
|
|
len += frag->len;
|
|
frag = frag->frags;
|
|
pos = 0U;
|
|
} else {
|
|
pos++;
|
|
}
|
|
}
|
|
|
|
if (frag && is_crlf(frag->data + pos, frag->len - pos)) {
|
|
len += pos;
|
|
return len - skip;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Parses list sms and add them to buffer.
|
|
* Format is:
|
|
*
|
|
* +CMGL: <index>,<stat>,,<length><CR><LF><pdu><CR><LF>
|
|
* +CMGL: <index>,<stat>,,<length><CR><LF><pdu><CR><LF>
|
|
* ...
|
|
* OK
|
|
*/
|
|
MODEM_CMD_DEFINE(on_cmd_cmgl)
|
|
{
|
|
int sms_index, sms_stat, ret;
|
|
char pdu_buffer[256];
|
|
size_t out_len, sms_len, param_len;
|
|
struct sim7080_sms *sms;
|
|
|
|
sms_index = atoi(argv[0]);
|
|
sms_stat = atoi(argv[1]);
|
|
|
|
/* Get the length of the "length" parameter.
|
|
* The last parameter will be stuck in the netbuffer.
|
|
* It is not the actual length of the trailing pdu so
|
|
* we have to search the next crlf.
|
|
*/
|
|
param_len = net_buf_find_crlf(data->rx_buf, 0);
|
|
if (param_len == 0) {
|
|
LOG_INF("No <CR><LF>");
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Get actual trailing pdu len. +2 to skip crlf. */
|
|
sms_len = net_buf_find_crlf(data->rx_buf, param_len + 2);
|
|
if (sms_len == 0) {
|
|
return -EAGAIN;
|
|
}
|
|
|
|
/* Skip to start of pdu. */
|
|
data->rx_buf = net_buf_skip(data->rx_buf, param_len + 2);
|
|
|
|
out_len = net_buf_linearize(pdu_buffer, sizeof(pdu_buffer) - 1, data->rx_buf, 0, sms_len);
|
|
pdu_buffer[out_len] = '\0';
|
|
|
|
data->rx_buf = net_buf_skip(data->rx_buf, sms_len);
|
|
|
|
/* No buffer specified. */
|
|
if (!mdata.sms_buffer) {
|
|
return 0;
|
|
}
|
|
|
|
/* No space left in buffer. */
|
|
if (mdata.sms_buffer_pos >= mdata.sms_buffer->nsms) {
|
|
return 0;
|
|
}
|
|
|
|
sms = &mdata.sms_buffer->sms[mdata.sms_buffer_pos];
|
|
|
|
ret = mdm_decode_pdu(pdu_buffer, out_len, sms);
|
|
if (ret < 0) {
|
|
return 0;
|
|
}
|
|
|
|
sms->stat = sms_stat;
|
|
sms->index = sms_index;
|
|
sms->data[sms->data_len] = '\0';
|
|
|
|
mdata.sms_buffer_pos++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mdm_sim7080_read_sms(struct sim7080_sms_buffer *buffer)
|
|
{
|
|
int ret;
|
|
struct modem_cmd cmds[] = { MODEM_CMD("+CMGL: ", on_cmd_cmgl, 4U, ",\r") };
|
|
|
|
mdata.sms_buffer = buffer;
|
|
mdata.sms_buffer_pos = 0;
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, cmds, ARRAY_SIZE(cmds), "AT+CMGL=4",
|
|
&mdata.sem_response, K_SECONDS(20));
|
|
if (ret < 0) {
|
|
return -1;
|
|
}
|
|
|
|
return mdata.sms_buffer_pos;
|
|
}
|
|
|
|
int mdm_sim7080_delete_sms(uint16_t index)
|
|
{
|
|
int ret;
|
|
char buf[sizeof("AT+CMGD=#####")] = { 0 };
|
|
|
|
ret = snprintk(buf, sizeof(buf), "AT+CMGD=%u", index);
|
|
if (ret < 0) {
|
|
return -1;
|
|
}
|
|
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0, buf, &mdata.sem_response,
|
|
K_SECONDS(5));
|
|
if (ret < 0) {
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Does the modem setup by starting it and
|
|
* bringing the modem to a PDP active state.
|
|
*/
|
|
static int modem_setup(void)
|
|
{
|
|
int ret = 0;
|
|
int counter = 0;
|
|
|
|
k_work_cancel_delayable(&mdata.rssi_query_work);
|
|
|
|
ret = modem_autobaud();
|
|
if (ret < 0) {
|
|
LOG_ERR("Booting modem failed!!");
|
|
goto error;
|
|
}
|
|
|
|
ret = modem_cmd_handler_setup_cmds(&mctx.iface, &mctx.cmd_handler, setup_cmds,
|
|
ARRAY_SIZE(setup_cmds), &mdata.sem_response,
|
|
MDM_REGISTRATION_TIMEOUT);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to send init commands!");
|
|
goto error;
|
|
}
|
|
|
|
k_sleep(K_SECONDS(3));
|
|
|
|
/* Wait for acceptable rssi values. */
|
|
modem_rssi_query_work(NULL);
|
|
k_sleep(MDM_WAIT_FOR_RSSI_DELAY);
|
|
|
|
counter = 0;
|
|
while (counter++ < MDM_WAIT_FOR_RSSI_COUNT &&
|
|
(mdata.mdm_rssi >= 0 || mdata.mdm_rssi <= -1000)) {
|
|
modem_rssi_query_work(NULL);
|
|
k_sleep(MDM_WAIT_FOR_RSSI_DELAY);
|
|
}
|
|
|
|
if (mdata.mdm_rssi >= 0 || mdata.mdm_rssi <= -1000) {
|
|
LOG_ERR("Network not reachable!!");
|
|
ret = -ENETUNREACH;
|
|
goto error;
|
|
}
|
|
|
|
ret = modem_pdp_activate();
|
|
if (ret < 0) {
|
|
goto error;
|
|
}
|
|
|
|
k_work_reschedule_for_queue(&modem_workq, &mdata.rssi_query_work,
|
|
K_SECONDS(RSSI_TIMEOUT_SECS));
|
|
|
|
change_state(SIM7080_STATE_NETWORKING);
|
|
|
|
error:
|
|
return ret;
|
|
}
|
|
|
|
int mdm_sim7080_start_network(void)
|
|
{
|
|
change_state(SIM7080_STATE_INIT);
|
|
return modem_setup();
|
|
}
|
|
|
|
int mdm_sim7080_power_on(void)
|
|
{
|
|
return modem_autobaud();
|
|
}
|
|
|
|
int mdm_sim7080_power_off(void)
|
|
{
|
|
int tries = 5;
|
|
int autobaud_tries;
|
|
int ret = 0;
|
|
|
|
k_work_cancel_delayable(&mdata.rssi_query_work);
|
|
|
|
/* Check if module is already off. */
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT", &mdata.sem_response,
|
|
K_MSEC(1000));
|
|
if (ret < 0) {
|
|
change_state(SIM7080_STATE_OFF);
|
|
return 0;
|
|
}
|
|
|
|
while (tries--) {
|
|
modem_pwrkey();
|
|
|
|
autobaud_tries = 5;
|
|
|
|
while (autobaud_tries--) {
|
|
ret = modem_cmd_send(&mctx.iface, &mctx.cmd_handler, NULL, 0U, "AT",
|
|
&mdata.sem_response, K_MSEC(500));
|
|
if (ret == 0) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ret < 0) {
|
|
change_state(SIM7080_STATE_OFF);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
const char *mdm_sim7080_get_manufacturer(void)
|
|
{
|
|
return mdata.mdm_manufacturer;
|
|
}
|
|
|
|
const char *mdm_sim7080_get_model(void)
|
|
{
|
|
return mdata.mdm_model;
|
|
}
|
|
|
|
const char *mdm_sim7080_get_revision(void)
|
|
{
|
|
return mdata.mdm_revision;
|
|
}
|
|
|
|
const char *mdm_sim7080_get_imei(void)
|
|
{
|
|
return mdata.mdm_imei;
|
|
}
|
|
|
|
/*
|
|
* Initializes modem handlers and context.
|
|
* After successful init this function calls
|
|
* modem_setup.
|
|
*/
|
|
static int modem_init(const struct device *dev)
|
|
{
|
|
int ret;
|
|
|
|
ARG_UNUSED(dev);
|
|
|
|
k_sem_init(&mdata.sem_response, 0, 1);
|
|
k_sem_init(&mdata.sem_tx_ready, 0, 1);
|
|
k_sem_init(&mdata.sem_dns, 0, 1);
|
|
k_sem_init(&mdata.sem_ftp, 0, 1);
|
|
k_work_queue_start(&modem_workq, modem_workq_stack,
|
|
K_KERNEL_STACK_SIZEOF(modem_workq_stack), K_PRIO_COOP(7), NULL);
|
|
|
|
/* Assume the modem is not registered to the network. */
|
|
mdata.mdm_registration = 0;
|
|
mdata.cpin_ready = false;
|
|
mdata.pdp_active = false;
|
|
|
|
mdata.sms_buffer = NULL;
|
|
mdata.sms_buffer_pos = 0;
|
|
|
|
/* Socket config. */
|
|
ret = modem_socket_init(&mdata.socket_config, &mdata.sockets[0], ARRAY_SIZE(mdata.sockets),
|
|
MDM_BASE_SOCKET_NUM, true, &offload_socket_fd_op_vtable);
|
|
if (ret < 0) {
|
|
goto error;
|
|
}
|
|
|
|
change_state(SIM7080_STATE_INIT);
|
|
|
|
/* Command handler. */
|
|
const struct modem_cmd_handler_config cmd_handler_config = {
|
|
.match_buf = &mdata.cmd_match_buf[0],
|
|
.match_buf_len = sizeof(mdata.cmd_match_buf),
|
|
.buf_pool = &mdm_recv_pool,
|
|
.alloc_timeout = BUF_ALLOC_TIMEOUT,
|
|
.eol = "\r\n",
|
|
.user_data = NULL,
|
|
.response_cmds = response_cmds,
|
|
.response_cmds_len = ARRAY_SIZE(response_cmds),
|
|
.unsol_cmds = unsolicited_cmds,
|
|
.unsol_cmds_len = ARRAY_SIZE(unsolicited_cmds),
|
|
};
|
|
|
|
ret = modem_cmd_handler_init(&mctx.cmd_handler, &mdata.cmd_handler_data,
|
|
&cmd_handler_config);
|
|
if (ret < 0) {
|
|
goto error;
|
|
}
|
|
|
|
/* Uart handler. */
|
|
const struct modem_iface_uart_config uart_config = {
|
|
.rx_rb_buf = &mdata.iface_rb_buf[0],
|
|
.rx_rb_buf_len = sizeof(mdata.iface_rb_buf),
|
|
.dev = MDM_UART_DEV,
|
|
.hw_flow_control = DT_PROP(MDM_UART_NODE, hw_flow_control),
|
|
};
|
|
|
|
ret = modem_iface_uart_init(&mctx.iface, &mdata.iface_data, &uart_config);
|
|
if (ret < 0) {
|
|
goto error;
|
|
}
|
|
|
|
mdata.current_sock_fd = -1;
|
|
mdata.current_sock_written = 0;
|
|
|
|
mdata.ftp.read_buffer = NULL;
|
|
mdata.ftp.nread = 0;
|
|
mdata.ftp.state = SIM7080_FTP_CONNECTION_STATE_INITIAL;
|
|
|
|
/* Modem data storage. */
|
|
mctx.data_manufacturer = mdata.mdm_manufacturer;
|
|
mctx.data_model = mdata.mdm_model;
|
|
mctx.data_revision = mdata.mdm_revision;
|
|
mctx.data_imei = mdata.mdm_imei;
|
|
#if defined(CONFIG_MODEM_SIM_NUMBERS)
|
|
mctx.data_imsi = mdata.mdm_imsi;
|
|
mctx.data_iccid = mdata.mdm_iccid;
|
|
#endif /* #if defined(CONFIG_MODEM_SIM_NUMBERS) */
|
|
mctx.data_rssi = &mdata.mdm_rssi;
|
|
|
|
ret = gpio_pin_configure_dt(&power_gpio, GPIO_OUTPUT_LOW);
|
|
if (ret < 0) {
|
|
LOG_ERR("Failed to configure %s pin", "power");
|
|
goto error;
|
|
}
|
|
|
|
mctx.driver_data = &mdata;
|
|
|
|
memset(&gnss_data, 0, sizeof(gnss_data));
|
|
|
|
ret = modem_context_register(&mctx);
|
|
if (ret < 0) {
|
|
LOG_ERR("Error registering modem context: %d", ret);
|
|
goto error;
|
|
}
|
|
|
|
k_thread_create(&modem_rx_thread, modem_rx_stack, K_KERNEL_STACK_SIZEOF(modem_rx_stack),
|
|
modem_rx, NULL, NULL, NULL, K_PRIO_COOP(7), 0, K_NO_WAIT);
|
|
|
|
/* Init RSSI query */
|
|
k_work_init_delayable(&mdata.rssi_query_work, modem_rssi_query_work);
|
|
|
|
return modem_setup();
|
|
error:
|
|
return ret;
|
|
}
|
|
|
|
/* Register device with the networking stack. */
|
|
NET_DEVICE_DT_INST_OFFLOAD_DEFINE(0, modem_init, NULL, &mdata, NULL,
|
|
CONFIG_MODEM_SIMCOM_SIM7080_INIT_PRIORITY, &api_funcs,
|
|
MDM_MAX_DATA_LENGTH);
|
|
|
|
NET_SOCKET_OFFLOAD_REGISTER(simcom_sim7080, CONFIG_NET_SOCKETS_OFFLOAD_PRIORITY,
|
|
AF_UNSPEC, offload_is_supported, offload_socket);
|