2016-12-13 14:34:13 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:56:59 +02:00
|
|
|
#include <assert.h>
|
2016-12-13 14:34:13 +01:00
|
|
|
#include <errno.h>
|
2024-02-01 12:36:58 +01:00
|
|
|
#include <stdint.h>
|
2024-01-12 16:52:06 +01:00
|
|
|
#include <stdlib.h>
|
2016-12-13 14:34:13 +01:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "at.h"
|
|
|
|
#include "fmt.h"
|
|
|
|
#include "isrpipe.h"
|
2019-03-23 20:43:27 +01:00
|
|
|
#include "isrpipe/read_timeout.h"
|
2016-12-13 14:34:13 +01:00
|
|
|
#include "periph/uart.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2016-12-13 14:34:13 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#ifndef AT_PRINT_INCOMING
|
|
|
|
#define AT_PRINT_INCOMING (0)
|
|
|
|
#endif
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
|
2023-11-08 10:08:37 +01:00
|
|
|
#if defined(MODULE_AT_URC)
|
|
|
|
static int _check_urc(clist_node_t *node, void *arg);
|
|
|
|
#endif
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
static ssize_t at_readline_skip_empty_stop_at_str(at_dev_t *dev, char *resp_buf,
|
|
|
|
size_t len, bool keep_eol,
|
|
|
|
char const *substr, uint32_t timeout);
|
2024-02-01 12:36:58 +01:00
|
|
|
static ssize_t at_readline_stop_at_str(at_dev_t *dev, char *resp_buf, size_t len,
|
2024-01-12 16:52:06 +01:00
|
|
|
bool keep_eol, char const *substr,
|
|
|
|
uint32_t timeout);
|
2024-02-01 12:36:58 +01:00
|
|
|
static ssize_t read_line_or_echo(at_dev_t *dev, char const *cmd, char *resp_buf,
|
|
|
|
size_t len, uint32_t timeout);
|
2024-01-12 16:52:06 +01:00
|
|
|
|
|
|
|
static inline bool starts_with(char const *str, char const *prefix)
|
|
|
|
{
|
|
|
|
return strncmp(str, prefix, strlen(prefix)) == 0;
|
|
|
|
}
|
|
|
|
|
2020-06-22 10:04:09 +02:00
|
|
|
static void _isrpipe_write_one_wrapper(void *_dev, uint8_t data)
|
|
|
|
{
|
|
|
|
at_dev_t *dev = (at_dev_t *) _dev;
|
|
|
|
isrpipe_write_one(&dev->isrpipe, data);
|
2018-08-08 13:28:11 +02:00
|
|
|
}
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
int at_dev_init(at_dev_t *dev, at_dev_init_t const *init)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
assert(strlen(AT_RECV_EOL) >= 1);
|
2024-01-12 16:52:06 +01:00
|
|
|
assert(init->rp_buf_size >= 16);
|
2024-02-01 12:36:58 +01:00
|
|
|
|
|
|
|
dev->uart = init->uart;
|
2024-01-12 16:52:06 +01:00
|
|
|
dev->rp_buf = init->rp_buf;
|
|
|
|
dev->rp_buf_size = init->rp_buf_size;
|
2020-06-22 10:04:09 +02:00
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
isrpipe_init(&dev->isrpipe, (uint8_t *)init->rx_buf, init->rx_buf_size);
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
return uart_init(init->uart, init->baudrate, _isrpipe_write_one_wrapper, dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
int at_parse_resp(at_dev_t *dev, char const *resp)
|
|
|
|
{
|
|
|
|
if (*resp == '\0') {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if (starts_with(resp, CONFIG_AT_RECV_OK)) {
|
|
|
|
dev->rp_buf[0] = '\0';
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (starts_with(resp, CONFIG_AT_RECV_ERROR)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* A specific command may return either CME or CMS, we need not differentiate */
|
|
|
|
if (!starts_with(resp, "+CME ERROR: ") &&
|
|
|
|
!starts_with(resp, "+CMS ERROR: ")) {
|
|
|
|
/* neither `OK` nor error, must be a response or URC */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
resp += strlen("+CMx ERROR: ");
|
|
|
|
size_t resp_len = strlen(resp);
|
|
|
|
if (resp_len + 1 > dev->rp_buf_size) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
/* dev->rp_buf and resp may overlap */
|
|
|
|
memmove(dev->rp_buf, resp, resp_len + 1);
|
|
|
|
return -AT_ERR_EXTENDED;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
2017-08-28 11:01:07 +02:00
|
|
|
int at_expect_bytes(at_dev_t *dev, const char *bytes, uint32_t timeout)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res = 0;
|
2017-08-28 11:01:07 +02:00
|
|
|
while (*bytes) {
|
2016-12-13 14:34:13 +01:00
|
|
|
char c;
|
2019-06-05 18:13:42 +02:00
|
|
|
if ((res = isrpipe_read_timeout(&dev->isrpipe, (uint8_t *)&c, 1, timeout)) == 1) {
|
2016-12-13 14:34:13 +01:00
|
|
|
if (AT_PRINT_INCOMING) {
|
|
|
|
print(&c, 1);
|
|
|
|
}
|
2017-08-28 11:01:07 +02:00
|
|
|
if (c != *bytes++) {
|
2020-06-22 10:04:09 +02:00
|
|
|
res = -1;
|
|
|
|
goto out;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2020-06-22 10:04:09 +02:00
|
|
|
goto out;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
}
|
2020-06-22 10:04:09 +02:00
|
|
|
res = 0;
|
|
|
|
out:
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
2023-11-08 10:08:37 +01:00
|
|
|
int at_wait_bytes(at_dev_t *dev, const char *bytes, uint32_t timeout)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
do {
|
|
|
|
res = at_expect_bytes(dev, bytes, timeout);
|
|
|
|
} while (res != 0 && res != -ETIMEDOUT);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-07-13 18:44:28 +02:00
|
|
|
void at_send_bytes(at_dev_t *dev, const char *bytes, size_t len)
|
|
|
|
{
|
|
|
|
uart_write(dev->uart, (const uint8_t *)bytes, len);
|
|
|
|
}
|
|
|
|
|
2018-10-18 11:25:50 +02:00
|
|
|
ssize_t at_recv_bytes(at_dev_t *dev, char *bytes, size_t len, uint32_t timeout)
|
|
|
|
{
|
|
|
|
char *resp_pos = bytes;
|
|
|
|
while (len) {
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res;
|
|
|
|
if ((res = isrpipe_read_timeout(&dev->isrpipe, (uint8_t *)resp_pos,
|
|
|
|
1, timeout)) == 1) {
|
|
|
|
if (AT_PRINT_INCOMING) {
|
|
|
|
print(resp_pos, 1);
|
|
|
|
}
|
|
|
|
resp_pos += res;
|
|
|
|
len -= res;
|
2018-10-18 11:25:50 +02:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
else if (res == -ETIMEDOUT) {
|
2018-10-18 11:25:50 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (resp_pos - bytes);
|
|
|
|
}
|
|
|
|
|
2019-01-21 11:28:05 +01:00
|
|
|
int at_recv_bytes_until_string(at_dev_t *dev, const char *string,
|
|
|
|
char *bytes, size_t *bytes_len, uint32_t timeout)
|
|
|
|
{
|
|
|
|
size_t len = 0;
|
|
|
|
char *_string = (char *)string;
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res = 0;
|
2020-06-22 10:04:09 +02:00
|
|
|
|
2019-01-21 11:28:05 +01:00
|
|
|
while (*_string && len < *bytes_len) {
|
|
|
|
char c;
|
2019-06-05 18:13:42 +02:00
|
|
|
if ((res = isrpipe_read_timeout(&dev->isrpipe, (uint8_t *)&c, 1, timeout)) == 1) {
|
2019-01-21 11:28:05 +01:00
|
|
|
if (AT_PRINT_INCOMING) {
|
|
|
|
print(&c, 1);
|
|
|
|
}
|
|
|
|
if (c == *_string) {
|
|
|
|
_string++;
|
|
|
|
}
|
|
|
|
bytes[len] = c;
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
else {
|
2020-06-22 10:04:09 +02:00
|
|
|
break;
|
2019-01-21 11:28:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
*bytes_len = len;
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2019-01-21 11:28:05 +01:00
|
|
|
}
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static int wait_echo(at_dev_t *dev, char const *command, uint32_t timeout)
|
2024-02-13 09:10:50 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res;
|
|
|
|
while ((res = read_line_or_echo(dev, command, dev->rp_buf, dev->rp_buf_size, timeout)) > 0) {
|
|
|
|
/* keep reading until echo or some error happens. */
|
|
|
|
#ifdef MODULE_AT_URC
|
|
|
|
clist_foreach(&dev->urc_list, _check_urc, dev->rp_buf);
|
|
|
|
#endif
|
2024-02-13 09:10:50 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2024-02-13 09:10:50 +01:00
|
|
|
}
|
|
|
|
|
2016-12-13 14:34:13 +01:00
|
|
|
int at_send_cmd(at_dev_t *dev, const char *command, uint32_t timeout)
|
|
|
|
{
|
2017-08-28 11:01:07 +02:00
|
|
|
size_t cmdlen = strlen(command);
|
2016-12-13 14:34:13 +01:00
|
|
|
|
|
|
|
uart_write(dev->uart, (const uint8_t *)command, cmdlen);
|
2020-05-04 18:27:39 +02:00
|
|
|
uart_write(dev->uart, (const uint8_t *)CONFIG_AT_SEND_EOL, AT_SEND_EOL_LEN);
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
if (IS_ACTIVE(CONFIG_AT_SEND_SKIP_ECHO)) {
|
|
|
|
return 0;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
return wait_echo(dev, command, timeout);
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void at_drain(at_dev_t *dev)
|
|
|
|
{
|
2019-06-05 18:13:42 +02:00
|
|
|
uint8_t _tmp[16];
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res;
|
2016-12-13 14:34:13 +01:00
|
|
|
|
|
|
|
do {
|
2017-08-28 11:01:07 +02:00
|
|
|
/* consider no character within 10ms "drained" */
|
2016-12-13 14:34:13 +01:00
|
|
|
res = isrpipe_read_timeout(&dev->isrpipe, _tmp, sizeof(_tmp), 10000U);
|
|
|
|
} while (res > 0);
|
|
|
|
}
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static bool is_eol(char p)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
return p == '\r' || p == '\n';
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static char *skip_leading_eol(char *line)
|
|
|
|
{
|
|
|
|
while (is_eol(*line)) {
|
|
|
|
line++;
|
|
|
|
}
|
|
|
|
return line;
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static size_t trim_leading_eol(char *buf, size_t str_len)
|
|
|
|
{
|
|
|
|
char *p = skip_leading_eol(buf);
|
|
|
|
if (p == buf) {
|
|
|
|
/* not sure if memmove is a no-op in this case */
|
|
|
|
return str_len;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
size_t size_left = str_len - (size_t)(p - buf);
|
|
|
|
/* +1 for the terminating \0 */
|
|
|
|
memmove(buf, p, size_left + 1);
|
|
|
|
return size_left;
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static size_t trim_trailing_eol(char *line, size_t str_len)
|
|
|
|
{
|
|
|
|
while (str_len && is_eol(line[str_len - 1])) {
|
|
|
|
line[--str_len] = '\0';
|
|
|
|
}
|
|
|
|
return str_len;
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static size_t at_drain_n(at_dev_t *dev, size_t n)
|
|
|
|
{
|
|
|
|
unsigned char drain_buf[16];
|
|
|
|
while (n > 0) {
|
|
|
|
size_t const to_read = n > sizeof(drain_buf) ? sizeof(drain_buf) : n;
|
|
|
|
ssize_t res = isrpipe_read_timeout(&dev->isrpipe, drain_buf, to_read, 10000U);
|
|
|
|
if (res < 1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
n -= res;
|
|
|
|
}
|
|
|
|
return n;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
/**
|
|
|
|
* @retval 0 if an echo was received and flushed
|
|
|
|
* @retval length of the line if an URC was intercepted
|
|
|
|
* @retval <0 on error */
|
|
|
|
static ssize_t read_line_or_echo(at_dev_t *dev, char const *cmd, char *resp_buf,
|
2024-02-13 09:10:50 +01:00
|
|
|
size_t len, uint32_t timeout)
|
2022-09-04 20:58:27 +02:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
size_t const cmd_len = strlen(cmd);
|
|
|
|
if (cmd_len == 0 || len == 0) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len == 1) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
/* We keep EOL in case the echoed command contains binary data and, by
|
|
|
|
* chance, a EOL sequence. */
|
|
|
|
ssize_t res = at_readline_skip_empty_stop_at_str(dev, resp_buf, len, true,
|
|
|
|
cmd, timeout);
|
|
|
|
bool overflow = false;
|
|
|
|
if (res < 0) {
|
|
|
|
if (res != -ENOBUFS) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
/* fine to overflow when intercepting an echo */
|
|
|
|
overflow = true;
|
|
|
|
res = len - 1;
|
|
|
|
}
|
|
|
|
if ((unsigned)res > cmd_len) {
|
|
|
|
/* definitely a URC */
|
|
|
|
if (overflow) {
|
|
|
|
/* URC didn't fit into the buffer so it's garbage */
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
/* maybe a URC, but might also be:
|
|
|
|
* 1. the command contained binary data and, by chance, a newline sequence
|
|
|
|
* 2. the command overflowed the resp_buf */
|
|
|
|
if (strncmp(cmd, resp_buf, res)) {
|
|
|
|
/* no match, indeed a URC. */
|
|
|
|
return trim_trailing_eol(resp_buf, res);
|
|
|
|
}
|
|
|
|
/* very good chance this is a echo, flush the rest */
|
|
|
|
size_t const left_in_echo = cmd_len - res + AT_SEND_EOL_LEN;
|
|
|
|
res = at_drain_n(dev, left_in_echo);
|
|
|
|
if (res > 0) {
|
|
|
|
res = -ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
resp_buf[0] = '\0';
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t at_send_cmd_get_resp(at_dev_t *dev, const char *command,
|
|
|
|
char *resp_buf, size_t len, uint32_t timeout)
|
|
|
|
{
|
|
|
|
ssize_t res = at_send_cmd(dev, command, timeout);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
return at_readline_skip_empty(dev, resp_buf, len, false, timeout);
|
|
|
|
}
|
|
|
|
|
2024-05-28 09:48:11 +02:00
|
|
|
ssize_t at_get_resp_with_prefix(at_dev_t *dev, const char *resp_prefix,
|
2024-02-01 12:36:58 +01:00
|
|
|
char *resp_buf, size_t len, uint32_t timeout)
|
|
|
|
{
|
|
|
|
ssize_t res;
|
2023-11-08 10:08:37 +01:00
|
|
|
/* URCs may occur right after the command has been sent and before the
|
|
|
|
* expected response */
|
2024-01-12 16:52:06 +01:00
|
|
|
while ((res = at_readline_skip_empty(dev, resp_buf, len, false, timeout)) >= 0) {
|
|
|
|
if (!resp_prefix || *resp_prefix == '\0') {
|
|
|
|
break;
|
|
|
|
}
|
2023-11-08 10:08:37 +01:00
|
|
|
/* Strip the expected prefix */
|
2024-01-12 16:52:06 +01:00
|
|
|
size_t prefix_len = strlen(resp_prefix);
|
|
|
|
if (starts_with(resp_buf, resp_prefix)) {
|
|
|
|
size_t remaining_len = strlen(resp_buf) - prefix_len;
|
|
|
|
/* The one extra byte in the copy is the terminating nul byte */
|
|
|
|
memmove(resp_buf, resp_buf + prefix_len, remaining_len + 1);
|
|
|
|
res -= prefix_len;
|
|
|
|
break;
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
res = at_parse_resp(dev, resp_buf);
|
|
|
|
if (res == 0) {
|
|
|
|
/* empty response */
|
|
|
|
return 0;
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
#if IS_USED(MODULE_AT_URC)
|
2022-09-04 20:58:27 +02:00
|
|
|
else {
|
2024-01-12 16:52:06 +01:00
|
|
|
clist_foreach(&dev->urc_list, _check_urc, resp_buf);
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
#endif
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
2024-02-13 09:10:50 +01:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t at_send_cmd_get_resp_wait_ok(at_dev_t *dev, const char *command, const char *resp_prefix,
|
|
|
|
char *resp_buf, size_t len, uint32_t timeout)
|
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res = at_send_cmd(dev, command, timeout);
|
2024-02-13 09:10:50 +01:00
|
|
|
if (res) {
|
2024-01-12 16:52:06 +01:00
|
|
|
return res;
|
|
|
|
}
|
2024-05-28 09:48:11 +02:00
|
|
|
res = at_get_resp_with_prefix(dev, resp_prefix, resp_buf, len, timeout);
|
2024-02-13 09:10:50 +01:00
|
|
|
if (res < 1) {
|
2024-02-01 12:36:58 +01:00
|
|
|
/* error or OK (empty response) */
|
2024-01-12 16:52:06 +01:00
|
|
|
return res;
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
/* got response, wait for OK */
|
2024-02-13 09:10:50 +01:00
|
|
|
return at_wait_ok(dev, timeout);
|
2022-09-04 20:58:27 +02:00
|
|
|
}
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
#if IS_USED(MODULE_AT_URC)
|
|
|
|
static char *next_line(char *p)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
while (*p && *p != '\r' && *p != '\n') {
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
return skip_leading_eol(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_urc_lines(at_dev_t *dev, char *resp_buf)
|
|
|
|
{
|
|
|
|
char *p = resp_buf;
|
|
|
|
do {
|
|
|
|
char *next = next_line(p);
|
|
|
|
clist_foreach(&dev->urc_list, _check_urc, p);
|
|
|
|
p = next;
|
|
|
|
} while (*p);
|
|
|
|
}
|
|
|
|
#endif
|
2018-05-24 15:15:33 +02:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
static ssize_t get_lines(at_dev_t *dev, char *resp_buf, size_t len, uint32_t timeout)
|
|
|
|
{
|
2017-08-28 11:01:07 +02:00
|
|
|
ssize_t res;
|
2016-12-13 14:34:13 +01:00
|
|
|
char *pos = resp_buf;
|
2024-02-01 12:36:58 +01:00
|
|
|
while ((res = at_readline_skip_empty(dev, pos, len, true, timeout)) > 0) {
|
|
|
|
size_t const line_len = res;
|
|
|
|
res = at_parse_resp(dev, pos);
|
|
|
|
|
|
|
|
len -= line_len;
|
|
|
|
pos += line_len;
|
|
|
|
|
|
|
|
switch (res) {
|
|
|
|
case 0: /* OK */
|
|
|
|
return (size_t)(pos - resp_buf);
|
|
|
|
case 1: /* response or URC */
|
|
|
|
if (len == 0) {
|
|
|
|
return -ENOBUFS;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
continue;
|
|
|
|
default: /* <0 */
|
|
|
|
#if IS_USED(MODULE_AT_URC)
|
|
|
|
/* DCE responded with an error. If we got some lines before that,
|
|
|
|
* they must be URCs. */
|
|
|
|
handle_urc_lines(dev, resp_buf);
|
|
|
|
#endif
|
2024-01-12 16:52:06 +01:00
|
|
|
return res;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
return res;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
2024-02-13 09:10:50 +01:00
|
|
|
ssize_t at_send_cmd_get_lines(at_dev_t *dev, const char *command, char *resp_buf,
|
2024-02-01 12:36:58 +01:00
|
|
|
size_t len, uint32_t timeout)
|
2017-07-13 18:44:28 +02:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res = at_send_cmd(dev, command, timeout);
|
2024-01-12 16:52:06 +01:00
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
return get_lines(dev, resp_buf, len, timeout);
|
2024-02-13 09:10:50 +01:00
|
|
|
}
|
|
|
|
|
2024-05-28 09:48:11 +02:00
|
|
|
int at_wait_prompt(at_dev_t *dev, uint32_t timeout)
|
2024-02-13 09:10:50 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res;
|
2024-01-12 16:52:06 +01:00
|
|
|
do {
|
|
|
|
res = at_readline_skip_empty_stop_at_str(dev, dev->rp_buf, dev->rp_buf_size,
|
|
|
|
false, ">", timeout);
|
|
|
|
if (res < 0) {
|
|
|
|
break;
|
2023-11-08 10:08:37 +01:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
if (strstr(dev->rp_buf, ">")) {
|
|
|
|
return 0;
|
2023-11-08 10:08:37 +01:00
|
|
|
}
|
2024-01-12 16:52:06 +01:00
|
|
|
res = at_parse_resp(dev, dev->rp_buf);
|
|
|
|
#ifdef MODULE_AT_URC
|
|
|
|
if (res == 1) {
|
|
|
|
clist_foreach(&dev->urc_list, _check_urc, dev->rp_buf);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} while (res >= 0);
|
2017-07-13 18:44:28 +02:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2017-07-13 18:44:28 +02:00
|
|
|
}
|
|
|
|
|
2024-02-13 09:10:50 +01:00
|
|
|
int at_send_cmd_wait_prompt(at_dev_t *dev, const char *command, uint32_t timeout)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res = at_send_cmd(dev, command, timeout);
|
2024-02-13 09:10:50 +01:00
|
|
|
if (res) {
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
2024-05-28 09:48:11 +02:00
|
|
|
return at_wait_prompt(dev, timeout);
|
2024-02-13 09:10:50 +01:00
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-13 09:10:50 +01:00
|
|
|
int at_send_cmd_wait_ok(at_dev_t *dev, const char *command, uint32_t timeout)
|
|
|
|
{
|
|
|
|
int res = at_send_cmd(dev, command, timeout);
|
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
return at_wait_ok(dev, timeout);
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
/* Used to detect a substring that may happen before the EOL. For example,
|
|
|
|
* Ublox LTE modules don't add EOL after the prompt character `>`. */
|
2024-02-01 12:36:58 +01:00
|
|
|
static ssize_t at_readline_stop_at_str(at_dev_t *dev, char *resp_buf, size_t len,
|
2024-01-12 16:52:06 +01:00
|
|
|
bool keep_eol, char const *substr,
|
|
|
|
uint32_t timeout)
|
2016-12-13 14:34:13 +01:00
|
|
|
{
|
2024-01-12 16:52:06 +01:00
|
|
|
ssize_t res = 0;
|
|
|
|
size_t substr_len = 0;
|
2024-02-01 12:36:58 +01:00
|
|
|
if (len < 1) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
if (substr) {
|
|
|
|
substr_len = strlen(substr);
|
|
|
|
if (substr_len == 0) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
|
|
|
|
char *resp_pos = resp_buf;
|
2024-01-12 16:52:06 +01:00
|
|
|
char const *substr_p = resp_buf;
|
2024-02-01 12:36:58 +01:00
|
|
|
memset(resp_buf, 0, len);
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
while (len > 1) {
|
2024-02-01 12:36:58 +01:00
|
|
|
res = isrpipe_read_timeout(&dev->isrpipe, (uint8_t *)resp_pos, 1, timeout);
|
|
|
|
if (res < 1) {
|
|
|
|
res = -ETIMEDOUT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (AT_PRINT_INCOMING) {
|
|
|
|
print(resp_pos, 1);
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
resp_pos++;
|
|
|
|
len--;
|
2024-01-12 16:52:06 +01:00
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
if ((size_t)(resp_pos - resp_buf) >= strlen(AT_RECV_EOL)) {
|
|
|
|
char *const eol_begin = resp_pos - strlen(AT_RECV_EOL);
|
|
|
|
if (strcmp(eol_begin, AT_RECV_EOL) == 0) {
|
|
|
|
if (keep_eol) {
|
2024-01-12 16:52:06 +01:00
|
|
|
break;
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
*eol_begin = '\0';
|
|
|
|
resp_pos -= strlen(AT_RECV_EOL);
|
|
|
|
break;
|
2024-01-12 16:52:06 +01:00
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
|
|
|
|
if (substr && (size_t)(resp_pos - substr_p) >= substr_len) {
|
|
|
|
if (strncmp(substr_p, substr, substr_len) == 0) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
substr_p++;
|
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
}
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
if (len <= 1) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
2017-08-28 11:01:07 +02:00
|
|
|
if (res < 0) {
|
|
|
|
*resp_buf = '\0';
|
2024-01-12 16:52:06 +01:00
|
|
|
} else {
|
|
|
|
res = resp_pos - resp_buf;
|
2017-08-28 11:01:07 +02:00
|
|
|
}
|
2016-12-13 14:34:13 +01:00
|
|
|
return res;
|
|
|
|
}
|
2018-06-26 19:03:10 +02:00
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
ssize_t at_readline(at_dev_t *dev, char *resp_buf, size_t len, bool keep_eol,
|
|
|
|
uint32_t timeout)
|
|
|
|
{
|
|
|
|
return at_readline_stop_at_str(dev, resp_buf, len, keep_eol, NULL, timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t at_readline_skip_empty_stop_at_str(at_dev_t *dev, char *resp_buf,
|
|
|
|
size_t len, bool keep_eol,
|
|
|
|
char const *substr, uint32_t timeout)
|
2023-11-08 10:08:37 +01:00
|
|
|
{
|
2024-02-01 12:36:58 +01:00
|
|
|
ssize_t res;
|
|
|
|
if (len == 1) {
|
|
|
|
/* Reading in a buffer of length 1 will forever return an empty line */
|
|
|
|
return -ENOBUFS;
|
2023-11-08 10:08:37 +01:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
do {
|
|
|
|
res = at_readline_stop_at_str(dev, resp_buf, len, keep_eol, substr, timeout);
|
|
|
|
/* Trim any rogue EOL characters */
|
|
|
|
if (res > 0) {
|
|
|
|
res = trim_leading_eol(resp_buf, (size_t)res);
|
|
|
|
} else if (res == -ENOBUFS) {
|
|
|
|
res = trim_leading_eol(resp_buf, len - 1);
|
|
|
|
}
|
|
|
|
} while (res == 0);
|
2023-11-08 10:08:37 +01:00
|
|
|
return res;
|
2024-01-12 16:52:06 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
ssize_t at_readline_skip_empty(at_dev_t *dev, char *resp_buf, size_t len,
|
|
|
|
bool keep_eol, uint32_t timeout)
|
|
|
|
{
|
|
|
|
return at_readline_skip_empty_stop_at_str(dev, resp_buf, len, keep_eol, NULL, timeout);
|
2023-11-08 10:08:37 +01:00
|
|
|
}
|
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
int at_wait_ok(at_dev_t *dev, uint32_t timeout)
|
|
|
|
{
|
|
|
|
while (1) {
|
|
|
|
ssize_t res = at_readline_skip_empty(dev, dev->rp_buf, dev->rp_buf_size,
|
|
|
|
false, timeout);
|
|
|
|
if (res < 0) {
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2024-01-12 16:52:06 +01:00
|
|
|
}
|
|
|
|
res = at_parse_resp(dev, dev->rp_buf);
|
|
|
|
if (res < 1) {
|
2024-02-01 12:36:58 +01:00
|
|
|
return (int)res;
|
2024-01-12 16:52:06 +01:00
|
|
|
}
|
|
|
|
#ifdef MODULE_AT_URC
|
|
|
|
clist_foreach(&dev->urc_list, _check_urc, dev->rp_buf);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2018-06-26 19:03:10 +02:00
|
|
|
#ifdef MODULE_AT_URC
|
|
|
|
void at_add_urc(at_dev_t *dev, at_urc_t *urc)
|
|
|
|
{
|
|
|
|
assert(urc);
|
|
|
|
assert(urc->code);
|
|
|
|
assert(strlen(urc->code) != 0);
|
|
|
|
assert(urc->cb);
|
|
|
|
|
|
|
|
clist_rpush(&dev->urc_list, &urc->list_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
void at_remove_urc(at_dev_t *dev, at_urc_t *urc)
|
|
|
|
{
|
|
|
|
clist_remove(&dev->urc_list, &urc->list_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _check_urc(clist_node_t *node, void *arg)
|
|
|
|
{
|
|
|
|
const char *buf = arg;
|
|
|
|
at_urc_t *urc = container_of(node, at_urc_t, list_node);
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
DEBUG("Trying to match %s with %s\n", buf, urc->code);
|
2018-06-26 19:03:10 +02:00
|
|
|
|
2024-01-12 16:52:06 +01:00
|
|
|
if (starts_with(buf, urc->code)) {
|
2024-02-01 12:36:58 +01:00
|
|
|
DEBUG("Matched %s\n", urc->code);
|
2018-06-26 19:03:10 +02:00
|
|
|
urc->cb(urc->arg, buf);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void at_process_urc(at_dev_t *dev, uint32_t timeout)
|
|
|
|
{
|
|
|
|
DEBUG("Processing URC (timeout=%" PRIu32 "us)\n", timeout);
|
|
|
|
|
2024-02-01 12:36:58 +01:00
|
|
|
while (at_readline_skip_empty(dev, dev->rp_buf, dev->rp_buf_size, false, timeout) > 0) {
|
|
|
|
clist_foreach(&dev->urc_list, _check_urc, dev->rp_buf);
|
2018-06-26 19:03:10 +02:00
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void at_postprocess_urc(at_dev_t *dev, char *buf)
|
|
|
|
{
|
2018-06-26 19:03:10 +02:00
|
|
|
clist_foreach(&dev->urc_list, _check_urc, buf);
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
|
|
|
|
void at_postprocess_urc_all(at_dev_t *dev, char *buf)
|
|
|
|
{
|
|
|
|
handle_urc_lines(dev, buf);
|
|
|
|
}
|
2018-06-26 19:03:10 +02:00
|
|
|
#endif
|
2018-07-17 10:02:36 +02:00
|
|
|
|
|
|
|
void at_dev_poweron(at_dev_t *dev)
|
|
|
|
{
|
|
|
|
uart_poweron(dev->uart);
|
|
|
|
}
|
|
|
|
|
|
|
|
void at_dev_poweroff(at_dev_t *dev)
|
|
|
|
{
|
|
|
|
uart_poweroff(dev->uart);
|
|
|
|
}
|
2024-02-01 12:36:58 +01:00
|
|
|
|
|
|
|
#ifdef MODULE_EMBUNIT
|
|
|
|
/* Exports for unit tests */
|
|
|
|
__attribute__((alias("read_line_or_echo")))
|
|
|
|
ssize_t _emb_read_line_or_echo(at_dev_t *dev, char const *cmd, char *resp_buf,
|
|
|
|
size_t len, uint32_t timeout);
|
|
|
|
__attribute__((alias("get_lines")))
|
|
|
|
ssize_t _emb_get_lines(at_dev_t *dev, char *resp_buf, size_t len, uint32_t timeout);
|
|
|
|
__attribute__((alias("wait_echo")))
|
|
|
|
int _emb_wait_echo(at_dev_t *dev, char const *command, uint32_t timeout);
|
|
|
|
#endif
|