Franck Roudet / CmdMessenger

CmdMessenger.cpp

Committer:
francxk
Date:
2015-11-26
Revision:
0:44e65e1ce5ee
Child:
1:798b310a5240

File content as of revision 0:44e65e1ce5ee:

/*
  CmdMessenger - library that provides command based messaging
  Permission is hereby granted, free of charge, to any person obtaining
  a copy of this software and associated documentation files (the
  "Software"), to deal in the Software without restriction, including
  without limitation the rights to use, copy, modify, merge, publish,
  distribute, sublicense, and/or sell copies of the Software, and to
  permit persons to whom the Software is furnished to do so, subject to
  the following conditions:
  The above copyright notice and this permission notice shall be
  included in all copies or substantial portions of the Software.
  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  Initial Messenger Library - Thomas Ouellet Fredericks.
  CmdMessenger Version 1    - Neil Dudman.
  CmdMessenger Version 2    - Dreamcat4.
  CmdMessenger Version 3    - Thijs Elenbaas.
  3.6  - Fixes
  - Better compatibility between platforms
  - Unit tests
  3.5  - Fixes, speed improvements for Teensy
  3.4  - Internal update
  3.3  - Fixed warnings
  - Some code optimization
  3.2  - Small fixes and sending long argument support
  3.1  - Added examples
  3.0  - Bugfixes on 2.2
  - Wait for acknowlegde
  - Sending of common type arguments (float, int, char)
  - Multi-argument commands
  - Escaping of special characters
  - Sending of binary data of any type (uses escaping)
  */

extern "C" {
#include <stdlib.h>
#include <stdarg.h>
}
#include <stdio.h>
#include <CmdMessenger.h>

#define _CMDMESSENGER_VERSION 3_6 // software version of this library

#ifdef __MBED__
#define BYTEAVAILLABLE(comms) comms->readable()
#define READONECHAR(comms) comms->getc()
#define PRINTONECHAR(comms,c) comms->putc(c)
#define PRINTSTRING(comms,s) comms->puts(s)
#endif
#ifdef ARDUINO
#define BYTEAVAILLABLE(comms) comms->available()
#define READONECHAR(comms) comms->read()
#define PRINTONECHAR(comms,c) comms->print(c)
#define PRINTSTRING(comms,s) comms->print(s)
#endif

// **** Initialization **** 

/**
 * CmdMessenger constructor
 */
CmdMessenger::CmdMessenger(__DEVICESTREAMTYPE &ccomms, const char fld_separator, const char cmd_separator, const char esc_character)
{
    init(ccomms, fld_separator, cmd_separator, esc_character);
}

/**
 * Enables printing newline after a sent command
 */
void CmdMessenger::init(__DEVICESTREAMTYPE &ccomms, const char fld_separator, const char cmd_separator, const char esc_character)
{
    default_callback = NULL;
    comms = &ccomms;
    print_newlines = false;
    field_separator = fld_separator;
    command_separator = cmd_separator;
    escape_character = esc_character;
    bufferLength = MESSENGERBUFFERSIZE;
    bufferLastIndex = MESSENGERBUFFERSIZE - 1;
    reset();

    default_callback = NULL;
    for (int i = 0; i < MAXCALLBACKS; i++)
        callbackList[i] = NULL;

    pauseProcessing = false;
}

/**
 * Resets the command buffer and message state
 */
void CmdMessenger::reset()
{
    bufferIndex = 0;
    current = NULL;
    last = NULL;
    dumped = true;
}

/**
 * Enables printing newline after a sent command
 */
void CmdMessenger::printLfCr(bool addNewLine)
{
    print_newlines = addNewLine;
}

/**
 * Attaches an default function for commands that are not explicitly attached
 */
void CmdMessenger::attach(messengerCallbackFunction newFunction)
{
    default_callback = newFunction;
}

/**
 * Attaches a function to a command ID
 */
void CmdMessenger::attach(byte msgId, messengerCallbackFunction newFunction)
{
    if (msgId >= 0 && msgId < MAXCALLBACKS)
        callbackList[msgId] = newFunction;
}

// **** Command processing ****

/**
 * Feeds serial data in CmdMessenger
 */
void CmdMessenger::feedinSerialData()
{
    while (!pauseProcessing && BYTEAVAILLABLE(comms))
    {
        // The Stream class has a readBytes() function that reads many bytes at once. On Teensy 2.0 and 3.0, readBytes() is optimized. 
        // Benchmarks about the incredible difference it makes: http://www.pjrc.com/teensy/benchmark_usb_serial_receive.html
#ifdef ARDUINO
        size_t bytesAvailable = min(comms->available(), MAXSTREAMBUFFERSIZE);
        comms->readBytes(streamBuffer, bytesAvailable);
#endif
#ifdef __MBED__
        size_t bytesAvailable = 0;
        while (BYTEAVAILLABLE(comms) && bytesAvailable < MAXSTREAMBUFFERSIZE) {
            streamBuffer[bytesAvailable++]=comms->getc();
        }
#endif
        // Process the bytes in the stream buffer, and handles dispatches callbacks, if commands are received
        for (size_t byteNo = 0; byteNo < bytesAvailable; byteNo++)
        {
            int messageState = processLine(streamBuffer[byteNo]);

            // If waiting for acknowledge command
            if (messageState == kEndOfMessage)
            {
                handleMessage();
            }
        }
    }
}

/**
 * Processes bytes and determines message state
 */
uint8_t CmdMessenger::processLine(char serialChar)
{
    messageState = kProccesingMessage;
    //char serialChar = (char)serialByte;
    bool escaped = isEscaped(&serialChar, escape_character, &CmdlastChar);
    if ((serialChar == command_separator) && !escaped) {
        commandBuffer[bufferIndex] = 0;
        if (bufferIndex > 0) {
            messageState = kEndOfMessage;
            current = commandBuffer;
            CmdlastChar = '\0';
        }
        reset();
    }
    else {
        commandBuffer[bufferIndex] = serialChar;
        bufferIndex++;
        if (bufferIndex >= bufferLastIndex) reset();
    }
    return messageState;
}

/**
 * Dispatches attached callbacks based on command
 */
void CmdMessenger::handleMessage()
{
    lastCommandId = readInt16Arg();
    // if command attached, we will call it
    if (lastCommandId >= 0 && lastCommandId < MAXCALLBACKS && ArgOk && callbackList[lastCommandId] != NULL)
        (*callbackList[lastCommandId])();
    else // If command not attached, call default callback (if attached)
        if (default_callback != NULL) (*default_callback)();
}

/**
 * Waits for reply from sender or timeout before continuing
 */
bool CmdMessenger::blockedTillReply(unsigned int timeout, byte ackCmdId)
{
    /** TODO:ggf
    unsigned long time = millis();
    unsigned long start = time;
    bool receivedAck = false;
    while ((time - start) < timeout && !receivedAck) {
        time = millis();
        receivedAck = checkForAck(ackCmdId);
    }
    return receivedAck;
    **/
    return true;
}

/**
 *   Loops as long data is available to determine if acknowledge has come in
 */
bool CmdMessenger::checkForAck(byte ackCommand)
{
    while (BYTEAVAILLABLE(comms)) {
        //Processes a byte and determines if an acknowlegde has come in
        int messageState = processLine(READONECHAR(comms));
        if (messageState == kEndOfMessage) {
            int id = readInt16Arg();
            if (ackCommand == id && ArgOk) {
                return true;
            }
            else {
                return false;
            }
        }
        return false;
    }
    return false;
}

/**
 * Gets next argument. Returns true if an argument is available
 */
bool CmdMessenger::next()
{
    char * temppointer = NULL;
    // Currently, cmd messenger only supports 1 char for the field seperator
    switch (messageState) {
    case kProccesingMessage:
        return false;
    case kEndOfMessage:
        temppointer = commandBuffer;
        messageState = kProcessingArguments;
    default:
        if (dumped)
            current = split_r(temppointer, field_separator, &last);
        if (current != NULL) {
            dumped = true;
            return true;
        }
    }
    return false;
}

/**
 * Returns if an argument is available. Alias for next()
 */
bool CmdMessenger::available()
{
    return next();
}

/**
 * Returns if the latest argument is well formed.
 */
bool CmdMessenger::isArgOk()
{
    return ArgOk;
}

/**
 * Returns the commandID of the current command
 */
uint8_t CmdMessenger::commandID()
{
    return lastCommandId;
}

// ****  Command sending ****

/**
 * Send start of command. This makes it easy to send multiple arguments per command
 */
void CmdMessenger::sendCmdStart(byte cmdId)
{
    if (!startCommand) {
        startCommand = true;
        pauseProcessing = true;
#ifdef __MBED__
        comms->printf("%i",cmdId);
#else
        comms->print(cmdId);
#endif
    }
}

/**
 * Send an escaped command argument
 */
void CmdMessenger::sendCmdEscArg(char* arg)
{
    if (startCommand) {
        PRINTONECHAR(comms,field_separator);
        printEsc(arg);
    }
}

/**
 * Send formatted argument.
 *  Note that floating points are not supported and resulting string is limited to 128 chars
 */
void CmdMessenger::sendCmdfArg(char *fmt, ...)
{
    const int maxMessageSize = 128;
    if (startCommand) {
        char msg[maxMessageSize];
        va_list args;
        va_start(args, fmt);
        vsnprintf(msg, maxMessageSize, fmt, args);
        va_end(args);

        PRINTONECHAR(comms,field_separator);
        PRINTSTRING(comms,msg);
    }
}

/**
 * Send double argument in scientific format.
 *  This will overcome the boundary of normal float sending which is limited to abs(f) <= MAXLONG
 */
void CmdMessenger::sendCmdSciArg(double arg, unsigned int n)
{
    if (startCommand)
    {
        PRINTONECHAR(comms,field_separator);
        printSci(arg, n);
    }
}

/**
 * Send end of command
 */
bool CmdMessenger::sendCmdEnd(bool reqAc, byte ackCmdId, unsigned int timeout)
{
    bool ackReply = false;
    if (startCommand) {
        PRINTONECHAR(comms,command_separator);
        if (print_newlines)
#ifdef __MBED__
            comms->puts("\r\n");
#else
            comms->println(); // should append BOTH \r\n
#endif
        if (reqAc) {
            ackReply = blockedTillReply(timeout, ackCmdId);
        }
    }
    pauseProcessing = false;
    startCommand = false;
    return ackReply;
}

/**
 * Send a command without arguments, with acknowledge
 */
bool CmdMessenger::sendCmd(byte cmdId, bool reqAc, byte ackCmdId)
{
    if (!startCommand) {
        sendCmdStart(cmdId);
        return sendCmdEnd(reqAc, ackCmdId, DEFAULT_TIMEOUT);
    }
    return false;
}

/**
 * Send a command without arguments, without acknowledge
 */
bool CmdMessenger::sendCmd(byte cmdId)
{
    if (!startCommand) {
        sendCmdStart(cmdId);
        return sendCmdEnd(false, 1, DEFAULT_TIMEOUT);
    }
    return false;
}

// **** Command receiving ****

/**
 * Find next argument in command
 */
int CmdMessenger::findNext(char *str, char delim)
{
    int pos = 0;
    bool escaped = false;
    bool EOL = false;
    ArglastChar = '\0';
    while (true) {
        escaped = isEscaped(str, escape_character, &ArglastChar);
        EOL = (*str == '\0' && !escaped);
        if (EOL) {
            return pos;
        }
        if (*str == field_separator && !escaped) {
            return pos;
        }
        else {
            str++;
            pos++;
        }
    }
    return pos;
}

/**
 * Read the next argument as int
 */
int16_t CmdMessenger::readInt16Arg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        return atoi(current);
    }
    ArgOk = false;
    return 0;
}

/**
 * Read the next argument as int
 */
int32_t CmdMessenger::readInt32Arg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        return atol(current);
    }
    ArgOk = false;
    return 0L;
}

/**
 * Read the next argument as bool
 */
bool CmdMessenger::readBoolArg()
{
    return (readInt16Arg() != 0) ? true : false;
}

/**
 * Read the next argument as char
 */
char CmdMessenger::readCharArg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        return current[0];
    }
    ArgOk = false;
    return 0;
}

/**
 * Read the next argument as float
 */
float CmdMessenger::readFloatArg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        //return atof(current);
        return strtod(current, NULL);
    }
    ArgOk = false;
    return 0;
}

/**
 * Read the next argument as double
 */
double CmdMessenger::readDoubleArg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        return strtod(current, NULL);
    }
    ArgOk = false;
    return 0;
}

/**
 * Read next argument as string.
 * Note that the String is valid until the current command is replaced
 */
char* CmdMessenger::readStringArg()
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        return current;
    }
    ArgOk = false;
    return '\0';
}

/**
 * Return next argument as a new string
 * Note that this is useful if the string needs to be persisted
 */
void CmdMessenger::copyStringArg(char *string, uint8_t size)
{
    if (next()) {
        dumped = true;
        ArgOk = true;
        strlcpy(string, current, size);
    }
    else {
        ArgOk = false;
        if (size) string[0] = '\0';
    }
}

/**
 * Compare the next argument with a string
 */
uint8_t CmdMessenger::compareStringArg(char *string)
{
    if (next()) {
        if (strcmp(string, current) == 0) {
            dumped = true;
            ArgOk = true;
            return 1;
        }
        else {
            ArgOk = false;
            return 0;
        }
    }
    return 0;
}

// **** Escaping tools ****

/**
 * Unescapes a string
 * Note that this is done inline
 */
void CmdMessenger::unescape(char *fromChar)
{
    // Move unescaped characters right
    char *toChar = fromChar;
    while (*fromChar != '\0') {
        if (*fromChar == escape_character) {
            fromChar++;
        }
        *toChar++ = *fromChar++;
    }
    // Pad string with \0 if string was shortened
    for (; toChar < fromChar; toChar++) {
        *toChar = '\0';
    }
}

/**
 * Split string in different tokens, based on delimiter
 * Note that this is basically strtok_r, but with support for an escape character
 */
char* CmdMessenger::split_r(char *str, const char delim, char **nextp)
{
    char *ret;
    // if input null, this is not the first call, use the nextp pointer instead
    if (str == NULL) {
        str = *nextp;
    }
    // Strip leading delimiters
    while (findNext(str, delim) == 0 && *str) {
        str++;
    }
    // If this is a \0 char, return null
    if (*str == '\0') {
        return NULL;
    }
    // Set start of return pointer to this position
    ret = str;
    // Find next delimiter
    str += findNext(str, delim);
    // and exchange this for a a \0 char. This will terminate the char
    if (*str) {
        *str++ = '\0';
    }
    // Set the next pointer to this char
    *nextp = str;
    // return current pointer
    return ret;
}

/**
 * Indicates if the current character is escaped
 */
bool CmdMessenger::isEscaped(char *currChar, const char escapeChar, char *lastChar)
{
    bool escaped;
    escaped = (*lastChar == escapeChar);
    *lastChar = *currChar;

    // special case: the escape char has been escaped:
    if (*lastChar == escape_character && escaped) {
        *lastChar = '\0';
    }
    return escaped;
}

/**
 * Escape and print a string
 */
void CmdMessenger::printEsc(char *str)
{
    while (*str != '\0') {
        printEsc(*str++);
    }
}

/**
 * Escape and print a character
 */
void CmdMessenger::printEsc(char str)
{
    if (str == field_separator || str == command_separator || str == escape_character || str == '\0') {
        PRINTONECHAR(comms,escape_character);
    }
    PRINTONECHAR(comms,str);
}

/**
 * Print float and double in scientific format
 */
void CmdMessenger::printSci(double f, unsigned int digits)
{
    // handle sign
    if (f < 0.0)
    {
        PRINTONECHAR(comms,'-');
        f = -f;
    }

    // handle infinite values
    if (isinf(f))
    {
        PRINTSTRING(comms,"INF");
        return;
    }
    // handle Not a Number
    if (isnan(f))
    {
        PRINTSTRING(comms,"NaN");
        return;
    }

    // max digits
    if (digits > 6) digits = 6;
    long multiplier = pow(10.0, double(digits));     // fix int => long

    int exponent;
    if (abs(f) < 10.0) {
        exponent = 0;
    }
    else {
        exponent = int(log10(f));
    }
    float g = f / pow(10, double(exponent));
    if ((g < 1.0) && (g != 0.0))
    {
        g *= 10;
        exponent--;
    }

    long whole = long(g);                     // single digit
    long part = long((g - whole)*multiplier + 0.5);  // # digits
    // Check for rounding above .99:
    if (part == 100) {
        whole++;
        part = 0;
    }
    char format[16];
    sprintf(format, "%%ld.%%0%dldE%%+d", digits);
    char output[16];
    sprintf(output, format, whole, part, exponent);
    PRINTSTRING(comms,output);
}