2017-05-23 17:07:24 +00:00
|
|
|
/* Copyright (c) 2017 ARM Limited
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*
|
|
|
|
* @section DESCRIPTION
|
|
|
|
*
|
|
|
|
* Parser for the AT command syntax
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#ifndef MBED_ATCMDPARSER_H
|
|
|
|
#define MBED_ATCMDPARSER_H
|
|
|
|
|
|
|
|
#include "mbed.h"
|
|
|
|
#include <cstdarg>
|
|
|
|
#include "Callback.h"
|
|
|
|
|
2017-10-24 15:05:45 +00:00
|
|
|
namespace mbed {
|
|
|
|
|
|
|
|
/** \addtogroup platform */
|
|
|
|
/** @{*/
|
|
|
|
/**
|
|
|
|
* \defgroup platform_ATCmdParser ATCmdParser class
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
/**
|
|
|
|
* Parser class for parsing AT commands
|
|
|
|
*
|
|
|
|
* Here are some examples:
|
|
|
|
* @code
|
2017-07-07 21:45:01 +00:00
|
|
|
* UARTSerial serial = UARTSerial(D1, D0);
|
|
|
|
* ATCmdParser at = ATCmdParser(&serial, "\r\n");
|
2017-05-23 17:07:24 +00:00
|
|
|
* int value;
|
|
|
|
* char buffer[100];
|
|
|
|
*
|
|
|
|
* at.send("AT") && at.recv("OK");
|
|
|
|
* at.send("AT+CWMODE=%d", 3) && at.recv("OK");
|
|
|
|
* at.send("AT+CWMODE?") && at.recv("+CWMODE:%d\r\nOK", &value);
|
|
|
|
* at.recv("+IPD,%d:", &value);
|
|
|
|
* at.read(buffer, value);
|
|
|
|
* at.recv("OK");
|
|
|
|
* @endcode
|
|
|
|
*/
|
|
|
|
|
2018-06-27 14:09:15 +00:00
|
|
|
class ATCmdParser : private NonCopyable<ATCmdParser> {
|
2017-05-23 17:07:24 +00:00
|
|
|
private:
|
|
|
|
// File handle
|
|
|
|
// Not owned by ATCmdParser
|
|
|
|
FileHandle *_fh;
|
|
|
|
|
|
|
|
int _buffer_size;
|
|
|
|
char *_buffer;
|
|
|
|
int _timeout;
|
|
|
|
|
|
|
|
// Parsing information
|
|
|
|
const char *_output_delimiter;
|
|
|
|
int _output_delim_size;
|
|
|
|
char _in_prev;
|
|
|
|
bool _dbg_on;
|
|
|
|
bool _aborted;
|
|
|
|
|
|
|
|
struct oob {
|
|
|
|
unsigned len;
|
|
|
|
const char *prefix;
|
|
|
|
mbed::Callback<void()> cb;
|
|
|
|
oob *next;
|
|
|
|
};
|
|
|
|
oob *_oobs;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param fh A FileHandle to the digital interface, used for AT commands
|
2018-10-12 22:16:09 +00:00
|
|
|
* @param output_delimiter End of command-line termination
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param buffer_size Size of internal buffer for transaction
|
|
|
|
* @param timeout Timeout of the connection
|
|
|
|
* @param debug Turns on/off debug output for AT commands
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
ATCmdParser(FileHandle *fh, const char *output_delimiter = "\r",
|
2018-06-27 14:09:15 +00:00
|
|
|
int buffer_size = 256, int timeout = 8000, bool debug = false)
|
|
|
|
: _fh(fh), _buffer_size(buffer_size), _in_prev(0), _oobs(NULL)
|
2017-05-23 17:07:24 +00:00
|
|
|
{
|
|
|
|
_buffer = new char[buffer_size];
|
|
|
|
set_timeout(timeout);
|
|
|
|
set_delimiter(output_delimiter);
|
|
|
|
debug_on(debug);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Destructor
|
|
|
|
*/
|
|
|
|
~ATCmdParser()
|
|
|
|
{
|
|
|
|
while (_oobs) {
|
|
|
|
struct oob *oob = _oobs;
|
|
|
|
_oobs = oob->next;
|
|
|
|
delete oob;
|
|
|
|
}
|
|
|
|
delete[] _buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allows timeout to be changed between commands
|
|
|
|
*
|
2018-10-12 22:16:09 +00:00
|
|
|
* @param timeout ATCmdParser APIs (read/write/send/recv ..etc) throw an
|
2018-10-12 22:09:31 +00:00
|
|
|
* error if no response is received in `timeout` duration
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
void set_timeout(int timeout)
|
|
|
|
{
|
|
|
|
_timeout = timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-10-12 22:16:09 +00:00
|
|
|
* For backward compatibility.
|
2018-04-03 19:07:03 +00:00
|
|
|
* @deprecated Do not use this function. This function has been replaced with set_timeout for consistency.
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
|
|
|
* Please use set_timeout(int) API only from now on.
|
|
|
|
* Allows timeout to be changed between commands
|
|
|
|
*
|
2018-10-12 22:16:09 +00:00
|
|
|
* @param timeout ATCmdParser APIs (read/write/send/recv ..etc) throw an
|
2018-10-12 22:09:31 +00:00
|
|
|
* error if no response is received in `timeout` duration
|
|
|
|
*
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
MBED_DEPRECATED_SINCE("mbed-os-5.5.0", "Replaced with set_timeout for consistency")
|
|
|
|
void setTimeout(int timeout)
|
|
|
|
{
|
|
|
|
set_timeout(timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets string of characters to use as line delimiters
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param output_delimiter String of characters to use as line delimiters
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
void set_delimiter(const char *output_delimiter)
|
|
|
|
{
|
|
|
|
_output_delimiter = output_delimiter;
|
|
|
|
_output_delim_size = strlen(output_delimiter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For backwards compatibility.
|
2018-04-03 19:07:03 +00:00
|
|
|
* @deprecated Do not use this function. This function has been replaced with set_delimiter for consistency.
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
|
|
|
* Please use set_delimiter(const char *) API only from now on.
|
|
|
|
* Sets string of characters to use as line delimiters
|
|
|
|
*
|
2017-06-01 18:14:51 +00:00
|
|
|
* @param output_delimiter string of characters to use as line delimiters
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
MBED_DEPRECATED_SINCE("mbed-os-5.5.0", "Replaced with set_delimiter for consistency")
|
|
|
|
void setDelimiter(const char *output_delimiter)
|
|
|
|
{
|
|
|
|
set_delimiter(output_delimiter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Allows traces from modem to be turned on or off
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param on Set as 1 to turn on traces and vice versa.
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
void debug_on(uint8_t on)
|
|
|
|
{
|
|
|
|
_dbg_on = (on) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-10-12 22:16:09 +00:00
|
|
|
* For backward compatibility.
|
2018-04-03 19:07:03 +00:00
|
|
|
* @deprecated Do not use this function. This function has been replaced with debug_on for consistency.
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
|
|
|
* Allows traces from modem to be turned on or off
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param on Set as 1 to turn on traces and vice versa.
|
2017-05-23 17:07:24 +00:00
|
|
|
*/
|
|
|
|
MBED_DEPRECATED_SINCE("mbed-os-5.5.0", "Replaced with debug_on for consistency")
|
|
|
|
void debugOn(uint8_t on)
|
|
|
|
{
|
|
|
|
debug_on(on);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends an AT command
|
|
|
|
*
|
|
|
|
* Sends a formatted command using printf style formatting
|
2017-06-01 18:14:51 +00:00
|
|
|
* @see printf
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
|
|
|
* @param command printf-like format string of command to send which
|
|
|
|
* is appended with a newline
|
|
|
|
* @param ... all printf-like arguments to insert into command
|
|
|
|
* @return true only if command is successfully sent
|
|
|
|
*/
|
2018-06-27 14:09:15 +00:00
|
|
|
bool send(const char *command, ...) MBED_PRINTF_METHOD(1, 2);
|
2017-05-23 17:07:24 +00:00
|
|
|
|
|
|
|
bool vsend(const char *command, va_list args);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Receive an AT response
|
|
|
|
*
|
|
|
|
* Receives a formatted response using scanf style formatting
|
2017-06-01 18:14:51 +00:00
|
|
|
* @see scanf
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
|
|
|
* Responses are parsed line at a time.
|
|
|
|
* Any received data that does not match the response is ignored until
|
|
|
|
* a timeout occurs.
|
|
|
|
*
|
|
|
|
* @param response scanf-like format string of response to expect
|
|
|
|
* @param ... all scanf-like arguments to extract from response
|
|
|
|
* @return true only if response is successfully matched
|
|
|
|
*/
|
2018-06-27 14:09:15 +00:00
|
|
|
bool recv(const char *response, ...) MBED_SCANF_METHOD(1, 2);
|
2017-05-23 17:07:24 +00:00
|
|
|
|
|
|
|
bool vrecv(const char *response, va_list args);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write a single byte to the underlying stream
|
|
|
|
*
|
|
|
|
* @param c The byte to write
|
|
|
|
* @return The byte that was written or -1 during a timeout
|
|
|
|
*/
|
|
|
|
int putc(char c);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a single byte from the underlying stream
|
|
|
|
*
|
|
|
|
* @return The byte that was read or -1 during a timeout
|
|
|
|
*/
|
|
|
|
int getc();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write an array of bytes to the underlying stream
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param data The array of bytes to write
|
|
|
|
* @param size Number of bytes to write
|
2017-05-23 17:07:24 +00:00
|
|
|
* @return number of bytes written or -1 on failure
|
|
|
|
*/
|
|
|
|
int write(const char *data, int size);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Read an array of bytes from the underlying stream
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param data The buffer for filling the read bytes
|
|
|
|
* @param size Number of bytes to read
|
2017-05-23 17:07:24 +00:00
|
|
|
* @return number of bytes read or -1 on failure
|
|
|
|
*/
|
|
|
|
int read(char *data, int size);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Direct printf to underlying stream
|
2017-06-01 18:14:51 +00:00
|
|
|
* @see printf
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param format Format string to pass to printf
|
|
|
|
* @param ... Variable arguments to printf
|
2017-05-23 17:07:24 +00:00
|
|
|
* @return number of bytes written or -1 on failure
|
|
|
|
*/
|
2018-06-27 14:09:15 +00:00
|
|
|
int printf(const char *format, ...) MBED_PRINTF_METHOD(1, 2);
|
2017-05-23 17:07:24 +00:00
|
|
|
|
|
|
|
int vprintf(const char *format, va_list args);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Direct scanf on underlying stream
|
2017-06-01 18:14:51 +00:00
|
|
|
* @see scanf
|
2017-05-23 17:07:24 +00:00
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param format Format string to pass to scanf
|
|
|
|
* @param ... Variable arguments to scanf
|
2017-05-23 17:07:24 +00:00
|
|
|
* @return number of bytes read or -1 on failure
|
|
|
|
*/
|
2018-06-27 14:09:15 +00:00
|
|
|
int scanf(const char *format, ...) MBED_SCANF_METHOD(1, 2);
|
2017-05-23 17:07:24 +00:00
|
|
|
|
|
|
|
int vscanf(const char *format, va_list args);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Attach a callback for out-of-band data
|
|
|
|
*
|
2018-10-12 22:09:31 +00:00
|
|
|
* @param prefix String on when to initiate callback
|
|
|
|
* @param func Callback to call when string is read
|
2017-05-23 17:07:24 +00:00
|
|
|
* @note out-of-band data is only processed during a scanf call
|
|
|
|
*/
|
|
|
|
void oob(const char *prefix, mbed::Callback<void()> func);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Flushes the underlying stream
|
|
|
|
*/
|
|
|
|
void flush();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Abort current recv
|
|
|
|
*
|
2018-10-12 22:16:09 +00:00
|
|
|
* Can be called from out-of-band handler to interrupt the current
|
2017-05-23 17:07:24 +00:00
|
|
|
* recv operation.
|
|
|
|
*/
|
|
|
|
void abort();
|
2018-06-27 14:09:15 +00:00
|
|
|
|
2017-09-27 15:46:01 +00:00
|
|
|
/**
|
|
|
|
* Process out-of-band data
|
|
|
|
*
|
|
|
|
* Process out-of-band data in the receive buffer. This function
|
|
|
|
* returns immediately if there is no data to process.
|
|
|
|
*
|
2018-10-12 22:16:09 +00:00
|
|
|
* @return true if out-of-band data processed, false otherwise
|
2017-09-27 15:46:01 +00:00
|
|
|
*/
|
|
|
|
bool process_oob(void);
|
2017-05-23 17:07:24 +00:00
|
|
|
};
|
2017-10-24 15:05:45 +00:00
|
|
|
|
|
|
|
/**@}*/
|
|
|
|
|
|
|
|
/**@}*/
|
|
|
|
|
2017-05-23 17:07:24 +00:00
|
|
|
} //namespace mbed
|
|
|
|
|
|
|
|
#endif //MBED_ATCMDPARSER_H
|