2018-05-22 12:52:27 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 OTA keys S.A.
|
2018-05-22 15:49:26 +02:00
|
|
|
* 2018 Inria
|
2018-05-22 12:52:27 +02:00
|
|
|
*
|
|
|
|
* 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>
|
2018-05-22 15:49:26 +02:00
|
|
|
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
|
2018-05-22 12:52:27 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2018-05-22 13:22:55 +02:00
|
|
|
#include <stdio.h>
|
2018-05-22 15:49:26 +02:00
|
|
|
#include <stdlib.h>
|
2018-06-23 00:20:45 +02:00
|
|
|
#include <string.h>
|
2018-05-22 13:22:55 +02:00
|
|
|
|
2018-05-22 12:52:27 +02:00
|
|
|
#include "at.h"
|
|
|
|
#include "shell.h"
|
|
|
|
#include "timex.h"
|
|
|
|
|
2018-05-22 15:49:26 +02:00
|
|
|
#include "periph/uart.h"
|
|
|
|
|
2018-05-22 12:52:27 +02:00
|
|
|
static at_dev_t at_dev;
|
|
|
|
static char buf[256];
|
|
|
|
static char resp[1024];
|
|
|
|
|
|
|
|
static int init(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
2018-05-22 15:49:26 +02:00
|
|
|
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]);
|
|
|
|
|
2019-01-19 23:16:53 +01:00
|
|
|
int res = at_dev_init(&at_dev, UART_DEV(uart), baudrate, buf, sizeof(buf));
|
2018-05-22 12:52:27 +02:00
|
|
|
|
2019-01-19 23:16:53 +01:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2018-05-22 12:52:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int send(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc < 2) {
|
2018-05-22 15:49:26 +02:00
|
|
|
printf("Usage: %s <command>\n", argv[0]);
|
2018-05-22 12:52:27 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 10:38:05 +02:00
|
|
|
ssize_t len;
|
|
|
|
if ((len = at_send_cmd_get_resp(&at_dev, argv[1], resp, sizeof(resp), 10 * US_PER_SEC)) < 0) {
|
2018-05-22 12:52:27 +02:00
|
|
|
puts("Error");
|
2018-05-22 15:49:26 +02:00
|
|
|
return 1;
|
2018-05-22 12:52:27 +02:00
|
|
|
}
|
|
|
|
|
2018-05-23 10:38:05 +02:00
|
|
|
printf("Response (len=%d): %s\n", (int)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 (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;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t len;
|
2018-05-23 10:38:05 +02:00
|
|
|
if ((len = at_send_cmd_get_lines(&at_dev, argv[1], resp, sizeof(resp), true, 10 * US_PER_SEC)) < 0) {
|
2018-05-23 10:38:05 +02:00
|
|
|
puts("Error");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("Response (len=%d): %s\n", (int)len, resp);
|
2018-05-22 15:49:26 +02:00
|
|
|
|
2018-05-22 12:52:27 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-18 11:25:50 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
sprintf(buffer, "%s%s", argv[1], 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=%d): %s\n", (int)len, buffer);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-22 12:52:27 +02:00
|
|
|
static int drain(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
|
|
|
at_drain(&at_dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-07-17 10:15:35 +02:00
|
|
|
static int power_on(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
|
|
|
at_dev_poweron(&at_dev);
|
|
|
|
|
|
|
|
puts("Powered on");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int power_off(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
|
|
|
|
at_dev_poweroff(&at_dev);
|
|
|
|
|
|
|
|
puts("Powered off");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-23 00:20:45 +02:00
|
|
|
#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
|
|
|
|
|
2018-05-22 12:52:27 +02:00
|
|
|
static const shell_command_t shell_commands[] = {
|
|
|
|
{ "init", "Initialize AT device", init },
|
|
|
|
{ "send", "Send a command and wait response", send },
|
2018-05-23 10:38:05 +02:00
|
|
|
{ "send_ok", "Send a command and wait OK", send_ok },
|
|
|
|
{ "send_lines", "Send a command and wait lines", send_lines },
|
2018-10-18 11:25:50 +02:00
|
|
|
{ "send_recv_bytes", "Send a command and wait response as raw bytes", send_recv_bytes },
|
2018-05-22 12:52:27 +02:00
|
|
|
{ "drain", "Drain AT device", drain },
|
2018-07-17 10:15:35 +02:00
|
|
|
{ "power_on", "Power on AT device", power_on },
|
|
|
|
{ "power_off", "Power off AT device", power_off },
|
2018-06-23 00:20:45 +02:00
|
|
|
#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
|
2018-05-22 12:52:27 +02:00
|
|
|
{ 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;
|
|
|
|
}
|