1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/drivers/at/main.c
2024-04-15 20:59:44 +02:00

498 lines
12 KiB
C

/*
* Copyright (C) 2018 OTA keys S.A.
* 2018 Inria
*
* 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.
*/
/**
* @ingroup tests
* @{
*
* @file
* @brief AT module test application
*
* @author Vincent Dupont <vincent@otakeys.com>
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
*
* @}
*/
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "at.h"
#include "shell.h"
#include "timex.h"
#include "periph/uart.h"
static at_dev_t at_dev;
static char buf[256];
static char resp[1024];
static char rp_buf[256];
static bool initialized = false;
static bool is_power_on = false;
static int init(int argc, char **argv)
{
(void)argc;
(void)argv;
if (argc < 3) {
printf("Usage: %s <uart> <baudrate>\n", argv[0]);
return 1;
}
uint8_t uart = atoi(argv[1]);
uint32_t baudrate = atoi(argv[2]);
if (uart >= UART_NUMOF) {
printf("Wrong UART device number - should be in range 0-%d.\n", UART_NUMOF - 1);
return 1;
}
at_dev_init_t at_init_params = {
.baudrate = baudrate,
.rp_buf = rp_buf,
.rp_buf_size = sizeof(rp_buf),
.rx_buf = buf,
.rx_buf_size = sizeof(buf),
.uart = UART_DEV(uart),
};
int res = at_dev_init(&at_dev, &at_init_params);
/* check the UART initialization return value and respond as needed */
if (res == UART_NODEV) {
puts("Invalid UART device given!");
return 1;
}
else if (res == UART_NOBAUD) {
puts("Baudrate is not applicable!");
return 1;
}
initialized = true;
return res;
}
static int send(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <command>\n", argv[0]);
return 1;
}
if (initialized == false) {
puts("Error: AT device not initialized.\n");
puts("Execute init function first.\n");
return 1;
}
if (is_power_on == false) {
puts("Error: AT device not power on.\n");
puts("Execute power_on function first.\n");
return 1;
}
ssize_t len;
if ((len = at_send_cmd_get_resp(&at_dev, argv[1], resp, sizeof(resp), 10 * US_PER_SEC)) < 0) {
puts("Error");
return 1;
}
printf("Response (len=%" PRIdSIZE "): %s\n", len, resp);
return 0;
}
static int send_ok(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <command>\n", argv[0]);
return 1;
}
if (initialized == false) {
puts("Error: AT device not initialized.\n");
puts("Execute init function first.\n");
return 1;
}
if (is_power_on == false) {
puts("Error: AT device not power on.\n");
puts("Execute power_on function first.\n");
return 1;
}
if (at_send_cmd_wait_ok(&at_dev, argv[1], 10 * US_PER_SEC) < 0) {
puts("Error");
return 1;
}
puts("OK");
return 0;
}
static int send_lines(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <command>\n", argv[0]);
return 1;
}
if (initialized == false) {
puts("Error: AT device not initialized.\n");
puts("Execute init function first.\n");
return 1;
}
if (is_power_on == false) {
puts("Error: AT device not power on.\n");
puts("Execute power_on function first.\n");
return 1;
}
ssize_t len;
if ((len = at_send_cmd_get_lines(&at_dev, argv[1], resp, sizeof(resp),
10 * US_PER_SEC)) < 0) {
puts("Error");
return 1;
}
printf("Response (len=%" PRIdSIZE "): %s\n", len, resp);
return 0;
}
static int send_recv_bytes(int argc, char **argv)
{
char buffer[64];
if (argc < 3) {
printf("Usage: %s <command> <number of bytes>\n", argv[0]);
return 1;
}
if (initialized == false) {
puts("Error: AT device not initialized.\n");
puts("Execute init function first.\n");
return 1;
}
if (is_power_on == false) {
puts("Error: AT device not power on.\n");
puts("Execute power_on function first.\n");
return 1;
}
sprintf(buffer, "%s%s", argv[1], CONFIG_AT_SEND_EOL);
at_send_bytes(&at_dev, buffer, strlen(buffer));
ssize_t len = at_recv_bytes(&at_dev, buffer, atoi(argv[2]), 10 * US_PER_SEC);
printf("Response (len=%" PRIdSIZE "): %s\n", len, buffer);
return 0;
}
static int send_recv_bytes_until_string(int argc, char **argv)
{
char buffer[128];
size_t len = sizeof(buffer);
memset(buffer, 0, sizeof(buffer));
if (argc < 3) {
printf("Usage: %s <command> <string to expect>\n", argv[0]);
return 1;
}
if (initialized == false) {
puts("Error: AT device not initialized.\n");
puts("Execute init function first.\n");
return 1;
}
if (is_power_on == false) {
puts("Error: AT device not power on.\n");
puts("Execute power_on function first.\n");
return 1;
}
sprintf(buffer, "%s%s", argv[1], CONFIG_AT_SEND_EOL);
at_send_bytes(&at_dev, buffer, strlen(buffer));
memset(buffer, 0, sizeof(buffer));
int res = at_recv_bytes_until_string(&at_dev, argv[2], buffer, &len,
10 * US_PER_SEC);
if (res) {
puts("Error");
return 1;
}
printf("Response (len=%" PRIdSIZE "): %s\n", len, buffer);
return 0;
}
static int drain(int argc, char **argv)
{
(void)argc;
(void)argv;
at_drain(&at_dev);
return 0;
}
static int power_on(int argc, char **argv)
{
(void)argc;
(void)argv;
at_dev_poweron(&at_dev);
is_power_on = true;
puts("Powered on");
return 0;
}
static int power_off(int argc, char **argv)
{
(void)argc;
(void)argv;
at_dev_poweroff(&at_dev);
is_power_on = false;
puts("Powered off");
return 0;
}
#ifdef MODULE_AT_URC
#ifndef MAX_URC_NB
#define MAX_URC_NB 5
#endif
#ifndef MAX_URC_LEN
#define MAX_URC_LEN 32
#endif
static at_urc_t urc_list[MAX_URC_NB];
static char urc_str[MAX_URC_NB][MAX_URC_LEN];
static bool urc_used[MAX_URC_NB];
static void _urc_cb(void *arg, const char *urc)
{
(void)arg;
printf("urc received: %s\n", urc);
}
static int add_urc(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <urc>\n", argv[0]);
return 1;
}
if (strlen(argv[1]) > MAX_URC_LEN - 1) {
puts("urc is too long");
return 1;
}
for (size_t i = 0; i < MAX_URC_NB; i++) {
if (!urc_used[i]) {
strcpy(urc_str[i], argv[1]);
urc_list[i].code = urc_str[i];
urc_list[i].arg = NULL;
urc_list[i].cb = _urc_cb;
urc_used[i] = true;
at_add_urc(&at_dev, &urc_list[i]);
puts("urc registered");
return 0;
}
}
puts("Not enough memory, urc is not registered");
return 1;
}
static int process_urc(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <timeout>\n", argv[0]);
return 1;
}
uint32_t timeout = strtoul(argv[1], NULL, 0);
at_process_urc(&at_dev, timeout);
puts("urc processed");
return 0;
}
static int remove_urc(int argc, char **argv)
{
if (argc < 2) {
printf("Usage: %s <urc>\n", argv[0]);
return 1;
}
for (size_t i = 0; i < MAX_URC_NB; i++) {
if (urc_used[i] && strcmp(urc_list[i].code, argv[1]) == 0) {
at_remove_urc(&at_dev, &urc_list[i]);
urc_used[i] = false;
puts("urc removed");
return 0;
}
}
puts("urc not found");
return 1;
}
#endif
static int emulate_dce(int argc, char **argv)
{
(void)argc;
(void)argv;
int res = 0;
char resp_buf[128];
#ifdef MODULE_AT_URC
at_urc_t urc = {.cb = _urc_cb, .code = "+CSCON"};
at_add_urc(&at_dev, &urc);
#endif
res = at_send_cmd_wait_ok(&at_dev, "AT+CFUN=1", US_PER_SEC);
if (res) {
printf("%u: Error AT+CFUN=1: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd(&at_dev, "AT+CFUN=1", US_PER_SEC);
if (res) {
printf("%u: Error AT+CFUN=1: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_wait_ok(&at_dev, US_PER_SEC);
if (res) {
printf("%u: Error AT+CFUN=1: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_get_resp_wait_ok(&at_dev, "AT+CEREG?",
"+CEREG:", resp_buf,
sizeof(resp_buf), US_PER_SEC);
if (res < 0) {
printf("%u: Error AT+CEREG?: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_wait_prompt(&at_dev, "AT+USECMNG=0,0,\"cert\",128", US_PER_SEC);
if (res) {
printf("%u: Error AT+USECMNG: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_wait_prompt(&at_dev, "AT+PROMPTERROR", US_PER_SEC);
if (res != -AT_ERR_EXTENDED) {
printf("%u: Error AT+PROMPTERROR: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = atol(at_get_err_info(&at_dev));
if (res != 1984) {
printf("%u: Error AT+PROMPTERROR: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_wait_ok(&at_dev, "AT+CFUN=8", US_PER_SEC);
if (res != -1) {
printf("%u: Error AT+CFUN=8: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_wait_ok(&at_dev, "AT+CFUN=9", US_PER_SEC);
if (res != -AT_ERR_EXTENDED) {
printf("%u: Error AT+CFUN=9: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = atol(at_get_err_info(&at_dev));
if (res != 666) {
printf("%u: Error AT+CFUN=9: %d\n", __LINE__, res);
res = 1;
goto exit;
}
res = at_send_cmd_get_lines(&at_dev, "AT+GETTWOLINES", resp_buf,
sizeof(resp_buf), US_PER_SEC);
if (res < 0) {
printf("%u: Error AT+GETTWOLINES: %d\n", __LINE__, res);
res = 1;
goto exit;
}
if (strcmp(resp_buf, "first_line\nsecond_line\nOK")) {
printf("%u: Error AT+GETTWOLINES: response not matching\n", __LINE__);
res = 1;
goto exit;
}
res = 0;
exit:
#ifdef MODULE_AT_URC
at_remove_urc(&at_dev, &urc);
#endif
printf("%s finished with %d\n", __func__, res);
return res;
}
static const shell_command_t shell_commands[] = {
{ "init", "Initialize AT device", init },
{ "send", "Send a command and wait response", send },
{ "send_ok", "Send a command and wait OK", send_ok },
{ "send_lines", "Send a command and wait lines", send_lines },
{ "send_recv_bytes", "Send a command and wait response as raw bytes", send_recv_bytes },
{ "send_recv_until_string", "Send a command and receive response until "
"the expected pattern arrives", send_recv_bytes_until_string },
{ "drain", "Drain AT device", drain },
{ "power_on", "Power on AT device", power_on },
{ "power_off", "Power off AT device", power_off },
{ "emulate_dce", "Test against the DCE emulation script.", emulate_dce},
#ifdef MODULE_AT_URC
{ "add_urc", "Register an URC", add_urc },
{ "remove_urc", "De-register an URC", remove_urc },
{ "process_urc", "Process the URCs", process_urc },
#endif
{ NULL, NULL, NULL },
};
int main(void)
{
puts("AT command test app");
/* run the shell */
char line_buf[SHELL_DEFAULT_BUFSIZE];
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
return 0;
}