1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/at/at.c

261 lines
6.0 KiB
C
Raw Normal View History

/*
* 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.
*/
#include <errno.h>
#include <string.h>
#include "at.h"
#include "fmt.h"
#include "isrpipe.h"
#include "periph/uart.h"
#include "xtimer.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
#ifndef AT_PRINT_INCOMING
#define AT_PRINT_INCOMING (0)
#endif
int at_dev_init(at_dev_t *dev, uart_t uart, uint32_t baudrate, char *buf, size_t bufsize)
{
dev->uart = uart;
isrpipe_init(&dev->isrpipe, buf, bufsize);
uart_init(uart, baudrate, (uart_rx_cb_t) isrpipe_write_one,
&dev->isrpipe);
return 0;
}
2017-08-28 11:01:07 +02:00
int at_expect_bytes(at_dev_t *dev, const char *bytes, uint32_t timeout)
{
2017-08-28 11:01:07 +02:00
while (*bytes) {
char c;
int res;
if ((res = isrpipe_read_timeout(&dev->isrpipe, &c, 1, timeout)) == 1) {
if (AT_PRINT_INCOMING) {
print(&c, 1);
}
2017-08-28 11:01:07 +02:00
if (c != *bytes++) {
return -1;
}
}
else {
return res;
}
}
return 0;
}
void at_send_bytes(at_dev_t *dev, const char *bytes, size_t len)
{
uart_write(dev->uart, (const uint8_t *)bytes, len);
}
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);
uart_write(dev->uart, (const uint8_t *)command, cmdlen);
2017-08-28 11:01:07 +02:00
uart_write(dev->uart, (const uint8_t *)AT_SEND_EOL, AT_SEND_EOL_LEN);
2017-08-28 11:01:07 +02:00
if (AT_SEND_ECHO) {
if (at_expect_bytes(dev, command, timeout)) {
return -1;
}
2018-05-24 15:15:33 +02:00
if (at_expect_bytes(dev, AT_SEND_EOL AT_RECV_EOL_1 AT_RECV_EOL_2, timeout)) {
2017-08-28 11:01:07 +02:00
return -2;
}
}
return 0;
}
void at_drain(at_dev_t *dev)
{
char _tmp[16];
int res;
do {
2017-08-28 11:01:07 +02:00
/* consider no character within 10ms "drained" */
res = isrpipe_read_timeout(&dev->isrpipe, _tmp, sizeof(_tmp), 10000U);
} while (res > 0);
}
2017-08-28 11:01:07 +02:00
ssize_t at_send_cmd_get_resp(at_dev_t *dev, const char *command,
char *resp_buf, size_t len, uint32_t timeout)
{
2017-08-28 11:01:07 +02:00
ssize_t res;
at_drain(dev);
res = at_send_cmd(dev, command, timeout);
if (res) {
goto out;
}
2018-05-23 10:37:42 +02:00
res = at_readline(dev, resp_buf, len, false, timeout);
if (res == 0) {
/* skip possible empty line */
2018-05-23 10:37:42 +02:00
res = at_readline(dev, resp_buf, len, false, timeout);
}
out:
return res;
}
2017-08-28 11:01:07 +02:00
ssize_t at_send_cmd_get_lines(at_dev_t *dev, const char *command,
2018-05-23 10:37:42 +02:00
char *resp_buf, size_t len, bool keep_eol, uint32_t timeout)
{
2018-05-24 15:15:33 +02:00
const char eol[] = AT_RECV_EOL_1 AT_RECV_EOL_2;
assert(sizeof(eol) > 1);
2017-08-28 11:01:07 +02:00
ssize_t res;
size_t bytes_left = len - 1;
char *pos = resp_buf;
memset(resp_buf, '\0', len);
at_drain(dev);
res = at_send_cmd(dev, command, timeout);
if (res) {
goto out;
}
2017-08-28 11:01:07 +02:00
while (1) {
2018-05-23 10:37:42 +02:00
res = at_readline(dev, pos, bytes_left, keep_eol, timeout);
if (res == 0) {
2018-05-23 10:37:42 +02:00
if (bytes_left) {
2018-05-24 15:15:33 +02:00
*pos++ = eol[sizeof(eol) - 2];
2018-05-23 10:37:42 +02:00
bytes_left--;
}
continue;
}
else if (res > 0) {
bytes_left -= res;
2018-05-23 10:37:42 +02:00
if ((res == (2 + keep_eol)) && (strncmp(pos, "OK", 2) == 0)) {
res = len - bytes_left;
break;
}
2018-05-23 10:37:42 +02:00
else if ((res == (5 + keep_eol)) && (strncmp(pos, "ERROR", 5) == 0)) {
return -1;
}
else if (strncmp(pos, "+CME ERROR:", 11) == 0) {
return -1;
}
2017-08-28 11:01:07 +02:00
else if (strncmp(pos, "+CMS ERROR:", 11) == 0) {
return -1;
}
else {
pos += res;
if (bytes_left) {
2018-05-24 15:15:33 +02:00
*pos++ = eol[sizeof(eol) - 2];
bytes_left--;
}
else {
return -1;
}
}
}
else {
break;
}
}
out:
return res;
}
int at_send_cmd_wait_prompt(at_dev_t *dev, const char *command, uint32_t timeout)
{
unsigned cmdlen = strlen(command);
at_drain(dev);
uart_write(dev->uart, (const uint8_t *)command, cmdlen);
2017-08-28 11:01:07 +02:00
uart_write(dev->uart, (const uint8_t *)AT_SEND_EOL, AT_SEND_EOL_LEN);
2017-08-28 11:01:07 +02:00
if (at_expect_bytes(dev, command, timeout)) {
return -1;
}
2018-05-24 15:15:33 +02:00
if (at_expect_bytes(dev, AT_SEND_EOL AT_RECV_EOL_2, timeout)) {
return -2;
}
2017-08-28 11:01:07 +02:00
if (at_expect_bytes(dev, ">", timeout)) {
return -3;
}
return 0;
}
int at_send_cmd_wait_ok(at_dev_t *dev, const char *command, uint32_t timeout)
{
int res;
2017-08-28 11:01:07 +02:00
char resp_buf[64];
res = at_send_cmd_get_resp(dev, command, resp_buf, sizeof(resp_buf), timeout);
if (res > 0) {
if (strcmp(resp_buf, "OK") == 0) {
res = 0;
}
else {
res = -1;
}
}
return res;
}
2018-05-23 10:37:42 +02:00
ssize_t at_readline(at_dev_t *dev, char *resp_buf, size_t len, bool keep_eol, uint32_t timeout)
{
2018-05-24 15:15:33 +02:00
const char eol[] = AT_RECV_EOL_1 AT_RECV_EOL_2;
assert(sizeof(eol) > 1);
ssize_t res = -1;
char *resp_pos = resp_buf;
memset(resp_buf, 0, len);
while (len) {
int read_res;
if ((read_res = isrpipe_read_timeout(&dev->isrpipe, resp_pos, 1, timeout)) == 1) {
if (AT_PRINT_INCOMING) {
print(resp_pos, read_res);
}
2018-05-24 15:15:33 +02:00
if (sizeof(eol) > 2 && *resp_pos == eol[0]) {
2018-05-23 10:37:42 +02:00
if (!keep_eol) {
continue;
}
}
2018-05-24 15:15:33 +02:00
if (*resp_pos == eol[sizeof(eol) - 2]) {
*resp_pos = '\0';
res = resp_pos - resp_buf;
goto out;
}
resp_pos += read_res;
len -= read_res;
}
else if (read_res == -ETIMEDOUT) {
res = -ETIMEDOUT;
break;
}
}
out:
2017-08-28 11:01:07 +02:00
if (res < 0) {
*resp_buf = '\0';
}
return res;
}