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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup drivers_at AT (Hayes) command set library
|
|
|
|
* @ingroup drivers
|
|
|
|
* @brief AT (Hayes) command set library
|
|
|
|
*
|
|
|
|
* This module provides functions to interact with devices using AT commands.
|
|
|
|
*
|
|
|
|
* Most functions compare the bytes echoed by the device with what they
|
|
|
|
* intended to send, and bail out if there's no match.
|
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* Furthermore, the library tries to cope with difficulties regarding different
|
2016-12-13 14:34:13 +01:00
|
|
|
* line endings. It usually sends "<command><CR>", but expects
|
|
|
|
* "<command>\LF\CR" as echo.
|
|
|
|
*
|
|
|
|
* As a debugging aid, when compiled with "-DAT_PRINT_INCOMING=1", every input
|
|
|
|
* byte gets printed.
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
*
|
|
|
|
* @brief AT (Hayes) library interface
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef AT_H
|
|
|
|
#define AT_H
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2017-08-28 11:01:07 +02:00
|
|
|
#include <unistd.h>
|
2018-05-23 10:37:42 +02:00
|
|
|
#include <stdbool.h>
|
2016-12-13 14:34:13 +01:00
|
|
|
|
|
|
|
#include "isrpipe.h"
|
|
|
|
#include "periph/uart.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2017-08-28 11:01:07 +02:00
|
|
|
#ifndef AT_SEND_EOL
|
2017-07-13 18:43:42 +02:00
|
|
|
/** End of line character to send after the AT command */
|
2017-08-28 11:01:07 +02:00
|
|
|
#define AT_SEND_EOL "\r"
|
2017-07-13 18:43:42 +02:00
|
|
|
#endif
|
|
|
|
|
2017-08-28 11:01:07 +02:00
|
|
|
#ifndef AT_SEND_ECHO
|
|
|
|
/** Enable/disable the expected echo after an AT command is sent */
|
|
|
|
#define AT_SEND_ECHO 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/** Shortcut for getting send end of line length */
|
|
|
|
#define AT_SEND_EOL_LEN (sizeof(AT_SEND_EOL) - 1)
|
|
|
|
|
2018-05-24 15:15:33 +02:00
|
|
|
#ifndef AT_RECV_EOL_1
|
|
|
|
/** 1st end of line character received (S3 aka CR character for a modem) */
|
|
|
|
#define AT_RECV_EOL_1 "\r"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AT_RECV_EOL_2
|
|
|
|
/** 1st end of line character received (S4 aka LF character for a modem) */
|
|
|
|
#define AT_RECV_EOL_2 "\n"
|
|
|
|
#endif
|
|
|
|
|
2018-06-04 17:05:11 +02:00
|
|
|
#ifndef AT_RECV_OK
|
|
|
|
/** default OK reply of an AT device */
|
|
|
|
#define AT_RECV_OK "OK"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AT_RECV_ERROR
|
|
|
|
/** default ERROR reply of an AT device */
|
|
|
|
#define AT_RECV_ERROR "ERROR"
|
|
|
|
#endif
|
|
|
|
|
2016-12-13 14:34:13 +01:00
|
|
|
/**
|
|
|
|
* @brief AT device structure
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
isrpipe_t isrpipe; /**< isrpipe used for getting data from uart */
|
|
|
|
uart_t uart; /**< UART device where the AT device is attached */
|
|
|
|
} at_dev_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize AT device struct
|
|
|
|
*
|
|
|
|
* @param[in] dev struct to initialize
|
|
|
|
* @param[in] uart UART the device is connected to
|
|
|
|
* @param[in] baudrate baudrate of the device
|
|
|
|
* @param[in] buf input buffer
|
|
|
|
* @param[in] bufsize size of @p buf
|
|
|
|
*
|
|
|
|
* @returns 0 on success
|
|
|
|
* @returns <0 otherwise
|
|
|
|
*/
|
|
|
|
int at_dev_init(at_dev_t *dev, uart_t uart, uint32_t baudrate, char *buf, size_t bufsize);
|
|
|
|
|
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Simple command helper
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
|
|
|
* This function sends an AT command to the device and waits for "OK".
|
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] command command string to send
|
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
|
|
|
* @returns 0 when device answers "OK"
|
|
|
|
* @returns <0 otherwise
|
|
|
|
*/
|
|
|
|
int at_send_cmd_wait_ok(at_dev_t *dev, const char *command, uint32_t timeout);
|
|
|
|
|
2017-07-13 18:44:28 +02:00
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Send AT command, wait for a prompt
|
2017-07-13 18:44:28 +02:00
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* This function sends the supplied @p command, then waits for the prompt (>)
|
|
|
|
* character and returns
|
2017-07-13 18:44:28 +02:00
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] command command string to send
|
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
|
|
|
* @return 0 when prompt is received
|
|
|
|
* @return <0 otherwise
|
|
|
|
*/
|
|
|
|
int at_send_cmd_wait_prompt(at_dev_t *dev, const char *command, uint32_t timeout);
|
|
|
|
|
2016-12-13 14:34:13 +01:00
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Send AT command, wait for response
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* This function sends the supplied @p command, then waits and returns one
|
2016-12-13 14:34:13 +01:00
|
|
|
* line of response.
|
|
|
|
*
|
|
|
|
* A possible empty line will be skipped.
|
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] command command to send
|
|
|
|
* @param[out] resp_buf buffer for storing response
|
|
|
|
* @param[in] len len of @p buffer
|
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* @returns length of response on success
|
2016-12-13 14:34:13 +01:00
|
|
|
* @returns <0 on error
|
|
|
|
*/
|
|
|
|
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
|
|
|
* @brief Send AT command, wait for multiline response
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* This function sends the supplied @p command, then returns all response
|
2016-12-13 14:34:13 +01:00
|
|
|
* lines until the device sends "OK".
|
|
|
|
*
|
|
|
|
* If a line starts with "ERROR" or "+CME ERROR:", or the buffer is full, the
|
|
|
|
* function returns -1.
|
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] command command to send
|
|
|
|
* @param[out] resp_buf buffer for storing response
|
2018-05-23 10:37:42 +02:00
|
|
|
* @param[in] len len of @p resp_buf
|
|
|
|
* @param[in] keep_eol true to keep the CR character in the response
|
2016-12-13 14:34:13 +01:00
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
2017-08-28 11:01:07 +02:00
|
|
|
* @returns length of response on success
|
2016-12-13 14:34:13 +01:00
|
|
|
* @returns <0 on error
|
|
|
|
*/
|
2018-05-23 10:37:42 +02:00
|
|
|
ssize_t at_send_cmd_get_lines(at_dev_t *dev, const char *command, char *resp_buf,
|
|
|
|
size_t len, bool keep_eol, uint32_t timeout);
|
2016-12-13 14:34:13 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Expect bytes from device
|
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
2017-08-28 11:01:07 +02:00
|
|
|
* @param[in] bytes buffer containing bytes to expect (NULL-terminated)
|
2016-12-13 14:34:13 +01:00
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
|
|
|
* @returns 0 on success
|
|
|
|
* @returns <0 otherwise
|
|
|
|
*/
|
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
|
|
|
|
2017-07-13 18:44:28 +02:00
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Send raw bytes to a device
|
2017-07-13 18:44:28 +02:00
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] bytes buffer containing bytes to send
|
|
|
|
* @param[in] len number of bytes to send
|
|
|
|
*/
|
|
|
|
void at_send_bytes(at_dev_t *dev, const char *bytes, size_t len);
|
|
|
|
|
2016-12-13 14:34:13 +01:00
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Send command to device
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] command command to send
|
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
|
|
|
* @returns 0 on success
|
|
|
|
* @returns <0 otherwise
|
|
|
|
*/
|
|
|
|
int at_send_cmd(at_dev_t *dev, const char *command, uint32_t timeout);
|
|
|
|
|
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Read a line from device
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
* @param[in] resp_buf buffer to store line
|
2018-05-23 10:37:42 +02:00
|
|
|
* @param[in] len size of @p resp_buf
|
|
|
|
* @param[in] keep_eol true to keep the CR character in the response
|
2016-12-13 14:34:13 +01:00
|
|
|
* @param[in] timeout timeout (in usec)
|
|
|
|
*
|
|
|
|
* @returns line length on success
|
|
|
|
* @returns <0 on error
|
|
|
|
*/
|
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);
|
2016-12-13 14:34:13 +01:00
|
|
|
|
|
|
|
/**
|
2017-08-28 11:01:07 +02:00
|
|
|
* @brief Drain device input buffer
|
2016-12-13 14:34:13 +01:00
|
|
|
*
|
|
|
|
* This function drains any possible bytes waiting in the device's input
|
|
|
|
* buffer.
|
|
|
|
*
|
|
|
|
* @param[in] dev device to operate on
|
|
|
|
*/
|
|
|
|
void at_drain(at_dev_t *dev);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* AT_H */
|
2017-08-28 11:01:07 +02:00
|
|
|
/** @} */
|