1494 lines
46 KiB
C
1494 lines
46 KiB
C
|
|
/*
|
||
|
|
* FIFO circular buffer
|
||
|
|
*
|
||
|
|
* @file CO_fifo.c
|
||
|
|
* @ingroup CO_CANopen_309_fifo
|
||
|
|
* @author Janez Paternoster
|
||
|
|
* @copyright 2020 Janez Paternoster
|
||
|
|
*
|
||
|
|
* This file is part of <https://github.com/CANopenNode/CANopenNode>, a CANopen Stack.
|
||
|
|
*
|
||
|
|
* 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.
|
||
|
|
*/
|
||
|
|
|
||
|
|
#include <string.h>
|
||
|
|
|
||
|
|
#include "301/CO_fifo.h"
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ENABLE) != 0
|
||
|
|
|
||
|
|
#include <ctype.h>
|
||
|
|
#include <stdlib.h>
|
||
|
|
#include "crc16-ccitt.h"
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ASCII_COMMANDS) != 0
|
||
|
|
#include <stdio.h>
|
||
|
|
#include <inttypes.h>
|
||
|
|
|
||
|
|
/* Non-graphical character for command delimiter */
|
||
|
|
#define DELIM_COMMAND ((uint8_t)'\n')
|
||
|
|
/* Graphical character for comment delimiter */
|
||
|
|
#define DELIM_COMMENT ((uint8_t)'#')
|
||
|
|
/* Graphical character for double quotes */
|
||
|
|
#define DELIM_DQUOTE ((uint8_t)'"')
|
||
|
|
#endif /* (CO_CONFIG_FIFO) & CO_CONFIG_FIFO_ASCII_COMMANDS */
|
||
|
|
|
||
|
|
/* verify configuration */
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_CRC16_CCITT) != 0
|
||
|
|
#if ((CO_CONFIG_CRC16)&CO_CONFIG_CRC16_ENABLE) == 0
|
||
|
|
#error CO_CONFIG_CRC16_ENABLE must be enabled.
|
||
|
|
#endif
|
||
|
|
#endif
|
||
|
|
|
||
|
|
void
|
||
|
|
CO_fifo_init(CO_fifo_t* fifo, uint8_t* buf, size_t bufSize) {
|
||
|
|
|
||
|
|
if ((fifo == NULL) || (buf == NULL) || (bufSize < 2U)) {
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
fifo->readPtr = 0;
|
||
|
|
fifo->writePtr = 0;
|
||
|
|
fifo->buf = buf;
|
||
|
|
fifo->bufSize = bufSize;
|
||
|
|
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* Circular FIFO buffer example for fifo->bufSize = 7 (usable size = 6): ******
|
||
|
|
* *
|
||
|
|
* 0 * * * * *
|
||
|
|
* 1 rp==wp readPtr writePtr * *
|
||
|
|
* 2 * * * * *
|
||
|
|
* 3 * * * writePtr *
|
||
|
|
* 4 * writePtr readPtr readPtr *
|
||
|
|
* 5 * * * * *
|
||
|
|
* 6 * * * * *
|
||
|
|
* *
|
||
|
|
* empty 3 bytes 4 bytes buffer *
|
||
|
|
* buffer in buff in buff full *
|
||
|
|
******************************************************************************/
|
||
|
|
size_t
|
||
|
|
CO_fifo_write(CO_fifo_t* fifo, const uint8_t* buf, size_t count, uint16_t* crc) {
|
||
|
|
size_t i;
|
||
|
|
uint8_t* bufDest;
|
||
|
|
|
||
|
|
if ((fifo == NULL) || (fifo->buf == NULL) || (buf == NULL)) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
bufDest = &fifo->buf[fifo->writePtr];
|
||
|
|
for (i = count; i > 0U; i--) {
|
||
|
|
size_t writePtrNext = fifo->writePtr + 1U;
|
||
|
|
|
||
|
|
/* is circular buffer full */
|
||
|
|
if ((writePtrNext == fifo->readPtr) || ((writePtrNext == fifo->bufSize) && (fifo->readPtr == 0U))) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
*bufDest = *buf;
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_CRC16_CCITT) != 0
|
||
|
|
if (crc != NULL) {
|
||
|
|
crc16_ccitt_single(crc, *buf);
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
/* increment variables */
|
||
|
|
if (writePtrNext == fifo->bufSize) {
|
||
|
|
fifo->writePtr = 0;
|
||
|
|
bufDest = &fifo->buf[0];
|
||
|
|
} else {
|
||
|
|
fifo->writePtr++;
|
||
|
|
bufDest++;
|
||
|
|
}
|
||
|
|
buf++;
|
||
|
|
}
|
||
|
|
|
||
|
|
return count - i;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_read(CO_fifo_t* fifo, uint8_t* buf, size_t count, bool_t* eof) {
|
||
|
|
size_t i;
|
||
|
|
const uint8_t* bufSrc;
|
||
|
|
bool_t alive_cycle = true;
|
||
|
|
|
||
|
|
if (eof != NULL) {
|
||
|
|
*eof = false;
|
||
|
|
}
|
||
|
|
if ((fifo == NULL) || (buf == NULL) || (fifo->readPtr == fifo->writePtr)) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
bufSrc = &fifo->buf[fifo->readPtr];
|
||
|
|
for (i = count; (i > 0U) && alive_cycle;) {
|
||
|
|
const uint8_t c = *bufSrc;
|
||
|
|
|
||
|
|
/* is circular buffer empty */
|
||
|
|
if (fifo->readPtr == fifo->writePtr) {
|
||
|
|
alive_cycle = false;
|
||
|
|
} else {
|
||
|
|
*buf = c;
|
||
|
|
buf++;
|
||
|
|
|
||
|
|
/* increment variables */
|
||
|
|
if (++fifo->readPtr == fifo->bufSize) {
|
||
|
|
fifo->readPtr = 0;
|
||
|
|
bufSrc = &fifo->buf[0];
|
||
|
|
} else {
|
||
|
|
bufSrc++;
|
||
|
|
}
|
||
|
|
i--;
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ASCII_COMMANDS) != 0
|
||
|
|
/* is delimiter? */
|
||
|
|
if ((eof != NULL) && (c == DELIM_COMMAND)) {
|
||
|
|
*eof = true;
|
||
|
|
alive_cycle = false;
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return count - i;
|
||
|
|
}
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ALT_READ) != 0
|
||
|
|
size_t
|
||
|
|
CO_fifo_altBegin(CO_fifo_t* fifo, size_t offset) {
|
||
|
|
size_t i;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
fifo->altReadPtr = fifo->readPtr;
|
||
|
|
for (i = offset; i > 0U; i--) {
|
||
|
|
/* is circular buffer empty */
|
||
|
|
if (fifo->altReadPtr == fifo->writePtr) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* increment variable */
|
||
|
|
if (++fifo->altReadPtr == fifo->bufSize) {
|
||
|
|
fifo->altReadPtr = 0;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return offset - i;
|
||
|
|
}
|
||
|
|
|
||
|
|
void
|
||
|
|
CO_fifo_altFinish(CO_fifo_t* fifo, uint16_t* crc) {
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (crc == NULL) {
|
||
|
|
fifo->readPtr = fifo->altReadPtr;
|
||
|
|
} else {
|
||
|
|
const uint8_t* bufSrc = &fifo->buf[fifo->readPtr];
|
||
|
|
while (fifo->readPtr != fifo->altReadPtr) {
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_CRC16_CCITT) != 0
|
||
|
|
crc16_ccitt_single(crc, *bufSrc);
|
||
|
|
#endif
|
||
|
|
/* increment variable */
|
||
|
|
if (++fifo->readPtr == fifo->bufSize) {
|
||
|
|
fifo->readPtr = 0;
|
||
|
|
bufSrc = &fifo->buf[0];
|
||
|
|
} else {
|
||
|
|
bufSrc++;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_altRead(CO_fifo_t* fifo, uint8_t* buf, size_t count) {
|
||
|
|
size_t i;
|
||
|
|
const uint8_t* bufSrc;
|
||
|
|
|
||
|
|
bufSrc = &fifo->buf[fifo->altReadPtr];
|
||
|
|
for (i = count; i > 0U; i--) {
|
||
|
|
const uint8_t c = *bufSrc;
|
||
|
|
|
||
|
|
/* is there no more data */
|
||
|
|
if (fifo->altReadPtr == fifo->writePtr) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
*buf = c;
|
||
|
|
buf++;
|
||
|
|
|
||
|
|
/* increment variables */
|
||
|
|
if (++fifo->altReadPtr == fifo->bufSize) {
|
||
|
|
fifo->altReadPtr = 0;
|
||
|
|
bufSrc = &fifo->buf[0];
|
||
|
|
} else {
|
||
|
|
bufSrc++;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return count - i;
|
||
|
|
}
|
||
|
|
#endif /* (CO_CONFIG_FIFO) & CO_CONFIG_FIFO_ALT_READ */
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ASCII_COMMANDS) != 0
|
||
|
|
bool_t
|
||
|
|
CO_fifo_CommSearch(CO_fifo_t* fifo, bool_t clear) {
|
||
|
|
bool_t newCommand = false;
|
||
|
|
size_t count;
|
||
|
|
uint8_t* commandEnd;
|
||
|
|
|
||
|
|
if ((fifo == NULL) || (fifo->readPtr == fifo->writePtr)) {
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* search delimiter until writePtr or until end of buffer */
|
||
|
|
if (fifo->readPtr < fifo->writePtr) {
|
||
|
|
count = fifo->writePtr - fifo->readPtr;
|
||
|
|
} else {
|
||
|
|
count = fifo->bufSize - fifo->readPtr;
|
||
|
|
}
|
||
|
|
commandEnd = (uint8_t*)memchr((const void*)&fifo->buf[fifo->readPtr], (int32_t)DELIM_COMMAND, count);
|
||
|
|
if (commandEnd != NULL) {
|
||
|
|
newCommand = true;
|
||
|
|
} else if (fifo->readPtr > fifo->writePtr) {
|
||
|
|
/* not found, search in the beginning of the circular buffer */
|
||
|
|
commandEnd = (uint8_t*)memchr((const void*)&fifo->buf[0], (int32_t)DELIM_COMMAND, fifo->writePtr);
|
||
|
|
if ((commandEnd != NULL) || (fifo->readPtr == (fifo->writePtr + 1U))) {
|
||
|
|
/* command delimiter found or buffer full */
|
||
|
|
newCommand = true;
|
||
|
|
}
|
||
|
|
} else if ((fifo->readPtr == 0U) && (fifo->writePtr == (fifo->bufSize - 1U))) {
|
||
|
|
/* buffer full */
|
||
|
|
newCommand = true;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
|
||
|
|
/* Clear buffer if set so */
|
||
|
|
if (clear) {
|
||
|
|
if (commandEnd != NULL) {
|
||
|
|
fifo->readPtr = ((size_t)commandEnd - (size_t)fifo->buf) + 1U;
|
||
|
|
if (fifo->readPtr == fifo->bufSize) {
|
||
|
|
fifo->readPtr = 0;
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
fifo->readPtr = fifo->writePtr;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return newCommand;
|
||
|
|
}
|
||
|
|
|
||
|
|
bool_t
|
||
|
|
CO_fifo_trimSpaces(CO_fifo_t* fifo, bool_t* insideComment) {
|
||
|
|
bool_t delimCommandFound = false;
|
||
|
|
bool_t alive_cycle = true;
|
||
|
|
|
||
|
|
if ((fifo != NULL) && (insideComment != NULL)) {
|
||
|
|
while ((fifo->readPtr != fifo->writePtr) && alive_cycle) {
|
||
|
|
uint8_t c = fifo->buf[fifo->readPtr];
|
||
|
|
|
||
|
|
if (c == DELIM_COMMENT) {
|
||
|
|
*insideComment = true;
|
||
|
|
} else if ((isgraph((int)c) != 0) && !(*insideComment)) {
|
||
|
|
alive_cycle = false;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
|
||
|
|
if (alive_cycle) {
|
||
|
|
if (++fifo->readPtr == fifo->bufSize) {
|
||
|
|
fifo->readPtr = 0;
|
||
|
|
}
|
||
|
|
if (c == DELIM_COMMAND) {
|
||
|
|
delimCommandFound = true;
|
||
|
|
*insideComment = false;
|
||
|
|
alive_cycle = false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
return delimCommandFound;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readToken(CO_fifo_t* fifo, char* buf, size_t count, uint8_t* closed, bool_t* err) {
|
||
|
|
bool_t delimCommandFound = false;
|
||
|
|
bool_t delimCommentFound = false;
|
||
|
|
size_t tokenSize = 0;
|
||
|
|
|
||
|
|
if ((fifo != NULL) && (buf != NULL) && (count > 1U) && ((err == NULL) || (*err == false))
|
||
|
|
&& (fifo->readPtr != fifo->writePtr)) {
|
||
|
|
bool_t finished = false;
|
||
|
|
uint8_t step = 0;
|
||
|
|
size_t ptr = fifo->readPtr; /* current pointer (integer, 0 based) */
|
||
|
|
uint8_t* c = &fifo->buf[ptr]; /* current character */
|
||
|
|
do {
|
||
|
|
switch (step) {
|
||
|
|
case 0: /* skip leading empty characters, stop on delimiter */
|
||
|
|
if (isgraph((int)*c) != 0) {
|
||
|
|
if (*c == DELIM_COMMENT) {
|
||
|
|
delimCommentFound = true;
|
||
|
|
} else {
|
||
|
|
buf[tokenSize] = (char)*c;
|
||
|
|
tokenSize++;
|
||
|
|
step++;
|
||
|
|
}
|
||
|
|
} else if (*c == DELIM_COMMAND) {
|
||
|
|
delimCommandFound = true;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
case 1: /* search for end of the token */
|
||
|
|
if (isgraph((int)*c) != 0) {
|
||
|
|
if (*c == DELIM_COMMENT) {
|
||
|
|
delimCommentFound = true;
|
||
|
|
} else if (tokenSize < count) {
|
||
|
|
buf[tokenSize] = (char)*c;
|
||
|
|
tokenSize++;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
if (*c == DELIM_COMMAND) {
|
||
|
|
delimCommandFound = true;
|
||
|
|
}
|
||
|
|
step++;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
case 2: /* skip trailing empty characters */
|
||
|
|
if (isgraph((int)*c) != 0) {
|
||
|
|
if (*c == DELIM_COMMENT) {
|
||
|
|
delimCommentFound = true;
|
||
|
|
} else {
|
||
|
|
fifo->readPtr = ptr;
|
||
|
|
finished = true;
|
||
|
|
}
|
||
|
|
} else if (*c == DELIM_COMMAND) {
|
||
|
|
delimCommandFound = true;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
default:
|
||
|
|
/* MISRA C 2004 15.3 */
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
if (delimCommentFound == true) {
|
||
|
|
/* Comment delimiter found, clear all till end of the line. */
|
||
|
|
fifo->readPtr = ptr;
|
||
|
|
delimCommandFound = CO_fifo_CommSearch(fifo, true);
|
||
|
|
finished = true;
|
||
|
|
} else if (delimCommandFound) {
|
||
|
|
/* command delimiter found, set readPtr behind it. */
|
||
|
|
if (++ptr == fifo->bufSize) {
|
||
|
|
ptr = 0;
|
||
|
|
}
|
||
|
|
fifo->readPtr = ptr;
|
||
|
|
finished = true;
|
||
|
|
} else if (!finished) {
|
||
|
|
/* find next character in the circular buffer */
|
||
|
|
if (++ptr == fifo->bufSize) {
|
||
|
|
ptr = 0;
|
||
|
|
c = &fifo->buf[ptr];
|
||
|
|
} else {
|
||
|
|
c++;
|
||
|
|
}
|
||
|
|
/* end, if buffer is now empty */
|
||
|
|
if (ptr == fifo->writePtr) {
|
||
|
|
if (step == 2U) {
|
||
|
|
fifo->readPtr = ptr;
|
||
|
|
} else {
|
||
|
|
tokenSize = 0;
|
||
|
|
}
|
||
|
|
finished = true;
|
||
|
|
}
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
} while (!finished);
|
||
|
|
}
|
||
|
|
|
||
|
|
/* set 'err' return value */
|
||
|
|
if ((err != NULL) && (*err == false)) {
|
||
|
|
if ((tokenSize == count)
|
||
|
|
|| ((closed != NULL)
|
||
|
|
&& (((*closed == 1U) && (!delimCommandFound || (tokenSize == 0U)))
|
||
|
|
|| ((*closed == 0U) && (delimCommandFound || (tokenSize == 0U)))))) {
|
||
|
|
*err = true;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
/* set 'closed' return value */
|
||
|
|
if (closed != NULL) {
|
||
|
|
*closed = delimCommandFound ? 1U : 0U;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* token was larger then size of the buffer, all was cleaned, return empty */
|
||
|
|
if (tokenSize == count) {
|
||
|
|
tokenSize = 0;
|
||
|
|
}
|
||
|
|
/* write string terminator character */
|
||
|
|
if ((buf != NULL) && (count > tokenSize)) {
|
||
|
|
buf[tokenSize] = '\0';
|
||
|
|
}
|
||
|
|
|
||
|
|
return tokenSize;
|
||
|
|
}
|
||
|
|
#endif /* (CO_CONFIG_FIFO) & CO_CONFIG_FIFO_ASCII_COMMANDS */
|
||
|
|
|
||
|
|
#if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ASCII_DATATYPES) != 0
|
||
|
|
/* Tables for mime-base64 encoding, as specified in RFC 2045, (without CR-LF, but one long string).
|
||
|
|
* Base64 is used for encoding binary data into easy transferable printable characters. In general,
|
||
|
|
* each three bytes of binary data are translated into four characters, where characters are
|
||
|
|
* selected from 64 characters long table. See https://en.wikipedia.org/wiki/Base64 */
|
||
|
|
static const char base64EncTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
||
|
|
|
||
|
|
static const uint8_t base64DecTable[] = {
|
||
|
|
255, 255, 255, 255, 255, 255, 255, 255, 255, 103, 101, 255, 255, 102, 255, 255, 255, 255, 255, 255, 255, 255,
|
||
|
|
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 103, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 62,
|
||
|
|
255, 255, 255, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 255, 255, 255, 100, 255, 255, 255, 0,
|
||
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22,
|
||
|
|
23, 24, 25, 255, 255, 255, 255, 255, 255, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38,
|
||
|
|
39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 255, 255, 255, 255, 255};
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readU82a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint8_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 6U)) {
|
||
|
|
(void)CO_fifo_read(fifo, &n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRIu8, n);
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readU162a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint16_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 8U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRIu16, CO_SWAP_16(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readU322a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint32_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 12U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRIu32, CO_SWAP_32(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readU642a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint64_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 20U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRIu64, CO_SWAP_64(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readX82a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint8_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 6U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "0x%02" PRIX8, (uint32_t)n);
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readX162a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint16_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 8U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "0x%04" PRIX16, (uint32_t)CO_SWAP_16(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readX322a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint32_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 12U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "0x%08" PRIX32, CO_SWAP_32(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readX642a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
uint64_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 20U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "0x%016" PRIX64, CO_SWAP_64(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readI82a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
int8_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 6U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRId8, n);
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readI162a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
int16_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 8U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRId16, CO_SWAP_16(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readI322a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
int32_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 13U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRId32, CO_SWAP_32(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readI642a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
int64_t n = 0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 23U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%" PRId64, CO_SWAP_64(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readR322a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
float32_t n = (float32_t)0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 20U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%g", (float32_t)CO_SWAP_32(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readR642a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
float64_t n = (float64_t)0;
|
||
|
|
|
||
|
|
if (fifo == NULL) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((CO_fifo_getOccupied(fifo) == sizeof(n)) && (count >= 30U)) {
|
||
|
|
(void)CO_fifo_read(fifo, (uint8_t*)&n, sizeof(n), NULL);
|
||
|
|
return (size_t)sprintf(buf, "%g", (float64_t)CO_SWAP_64(n));
|
||
|
|
} else {
|
||
|
|
return CO_fifo_readHex2a(fifo, buf, count, end);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readHex2a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
(void)end; /* unused */
|
||
|
|
|
||
|
|
size_t len = 0;
|
||
|
|
|
||
|
|
if ((fifo != NULL) && (count > 3U)) {
|
||
|
|
/* Very first write is without leading space */
|
||
|
|
if (!fifo->started) {
|
||
|
|
uint8_t c;
|
||
|
|
if (CO_fifo_getc(fifo, &c)) {
|
||
|
|
len = (size_t)sprintf(&buf[0], "%02" PRIX8, (uint32_t)c);
|
||
|
|
fifo->started = true;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
while ((len + 3U) < count) {
|
||
|
|
uint8_t c;
|
||
|
|
if (!CO_fifo_getc(fifo, &c)) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
len += (size_t)sprintf(&buf[len], " %02" PRIX8, (uint32_t)c);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return len;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readVs2a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
size_t len = 0;
|
||
|
|
|
||
|
|
if ((fifo != NULL) && (count > 3U)) {
|
||
|
|
/* Start with '"' */
|
||
|
|
if (!fifo->started) {
|
||
|
|
buf[len] = '"';
|
||
|
|
len++;
|
||
|
|
fifo->started = true;
|
||
|
|
}
|
||
|
|
|
||
|
|
while ((len + 2U) < count) {
|
||
|
|
uint8_t c;
|
||
|
|
if (!CO_fifo_getc(fifo, &c)) {
|
||
|
|
if (end) {
|
||
|
|
buf[len] = '"';
|
||
|
|
len++;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
} else if ((c != 0U) && (c != (uint8_t)'\r')) {
|
||
|
|
/* skip null and CR inside string */
|
||
|
|
buf[len] = (char)c;
|
||
|
|
len++;
|
||
|
|
if (c == DELIM_DQUOTE) {
|
||
|
|
buf[len] = '"';
|
||
|
|
len++;
|
||
|
|
}
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
return len;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_readB642a(CO_fifo_t* fifo, char* buf, size_t count, bool_t end) {
|
||
|
|
/* mime-base64 encoding, see description above base64EncTable */
|
||
|
|
|
||
|
|
size_t len = 0;
|
||
|
|
|
||
|
|
if ((fifo != NULL) && (count >= 4U)) {
|
||
|
|
uint8_t step;
|
||
|
|
uint16_t word;
|
||
|
|
|
||
|
|
if (!fifo->started) {
|
||
|
|
fifo->started = true;
|
||
|
|
step = 0;
|
||
|
|
word = 0;
|
||
|
|
} else {
|
||
|
|
/* get memorized variables from previous function calls */
|
||
|
|
step = (uint8_t)(fifo->aux >> 16);
|
||
|
|
word = (uint16_t)fifo->aux;
|
||
|
|
}
|
||
|
|
|
||
|
|
while ((len + 3U) <= count) {
|
||
|
|
uint8_t c;
|
||
|
|
|
||
|
|
if (!CO_fifo_getc(fifo, &c)) {
|
||
|
|
/* buffer is empty, is also SDO communication finished? */
|
||
|
|
if (end) {
|
||
|
|
/* add padding if necessary */
|
||
|
|
switch (step) {
|
||
|
|
case 1:
|
||
|
|
buf[len] = base64EncTable[(word >> 4) & 0x3FU];
|
||
|
|
len++;
|
||
|
|
buf[len] = '=';
|
||
|
|
len++;
|
||
|
|
buf[len] = '=';
|
||
|
|
len++;
|
||
|
|
break;
|
||
|
|
case 2:
|
||
|
|
buf[len] = base64EncTable[(word >> 6) & 0x3FU];
|
||
|
|
len++;
|
||
|
|
buf[len] = '=';
|
||
|
|
len++;
|
||
|
|
break;
|
||
|
|
default:
|
||
|
|
/* MISRA C 2004 15.3 */
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
word |= c;
|
||
|
|
|
||
|
|
switch (step++) {
|
||
|
|
case 0:
|
||
|
|
buf[len] = base64EncTable[(word >> 2) & 0x3FU];
|
||
|
|
len++;
|
||
|
|
break;
|
||
|
|
case 1:
|
||
|
|
buf[len] = base64EncTable[(word >> 4) & 0x3FU];
|
||
|
|
len++;
|
||
|
|
break;
|
||
|
|
default:
|
||
|
|
buf[len] = base64EncTable[(word >> 6) & 0x3FU];
|
||
|
|
len++;
|
||
|
|
buf[len] = base64EncTable[word & 0x3FU];
|
||
|
|
len++;
|
||
|
|
step = 0;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
word <<= 8;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* memorize variables for next iteration */
|
||
|
|
fifo->aux = ((uint32_t)step << 16) | word;
|
||
|
|
}
|
||
|
|
|
||
|
|
return len;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2U8(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
uint32_t u32 = strtoul(buf, &sRet, 0);
|
||
|
|
if ((sRet != strchr(buf, (int32_t)('\0'))) || (u32 > (uint32_t)UINT8_MAX)) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
uint8_t num = (uint8_t)u32;
|
||
|
|
nWr = CO_fifo_write(dest, &num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2U16(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
uint32_t u32 = strtoul(buf, &sRet, 0);
|
||
|
|
if ((sRet != strchr(buf, (int32_t)('\0'))) || (u32 > (uint32_t)UINT16_MAX)) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
uint16_t num = CO_SWAP_16((uint16_t)u32);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2U32(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
uint32_t u32 = strtoul(buf, &sRet, 0);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
uint32_t num = CO_SWAP_32(u32);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2U64(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[25];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
uint64_t u64 = strtoull(buf, &sRet, 0);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
uint64_t num = CO_SWAP_64(u64);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = (uint8_t)st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2I8(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
int32_t i32 = strtol(buf, &sRet, 0);
|
||
|
|
if ((sRet != strchr(buf, (int32_t)('\0'))) || (i32 < INT8_MIN) || (i32 > INT8_MAX)) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
int8_t num = (int8_t)i32;
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2I16(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
int32_t i32 = strtol(buf, &sRet, 0);
|
||
|
|
if ((sRet != strchr(buf, (int32_t)('\0'))) || (i32 < INT16_MIN) || (i32 > INT16_MAX)) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
int16_t num = CO_SWAP_16((int16_t)i32);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2I32(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[15];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
int32_t i32 = strtol(buf, &sRet, 0);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
int32_t num = CO_SWAP_32(i32);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2I64(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[25];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
int64_t i64 = strtoll(buf, &sRet, 0);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
int64_t num = CO_SWAP_64(i64);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = (uint8_t)st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2R32(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[30];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
float32_t f32 = strtof(buf, &sRet);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
float32_t num = CO_SWAP_32(f32);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2R64(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
char buf[40];
|
||
|
|
uint8_t closed = 0xFFU;
|
||
|
|
bool_t err = false;
|
||
|
|
size_t nWr = 0;
|
||
|
|
size_t nRd = CO_fifo_readToken(src, buf, sizeof(buf), &closed, &err);
|
||
|
|
uint8_t st = closed;
|
||
|
|
if ((nRd == 0U) || err) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
char* sRet;
|
||
|
|
float64_t f64 = strtof(buf, &sRet);
|
||
|
|
if (sRet != strchr(buf, (int32_t)('\0'))) {
|
||
|
|
st |= CO_fifo_st_errVal;
|
||
|
|
} else {
|
||
|
|
float64_t num = CO_SWAP_64(f64);
|
||
|
|
nWr = CO_fifo_write(dest, (uint8_t*)&num, sizeof(num), NULL);
|
||
|
|
if (nWr != sizeof(num)) {
|
||
|
|
st |= CO_fifo_st_errBuf;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
return nWr;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2Hex(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
size_t destSpace, destSpaceStart;
|
||
|
|
bool_t finished = false;
|
||
|
|
uint8_t step;
|
||
|
|
uint8_t firstChar;
|
||
|
|
uint8_t st = 0;
|
||
|
|
|
||
|
|
if ((dest == NULL) || (src == NULL)) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* get free space of the dest fifo */
|
||
|
|
destSpaceStart = CO_fifo_getSpace(dest);
|
||
|
|
destSpace = destSpaceStart;
|
||
|
|
|
||
|
|
/* is this the first write into dest? */
|
||
|
|
if (!dest->started) {
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || insideComment) {
|
||
|
|
/* command delimiter found without string, this is an error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
dest->started = true;
|
||
|
|
step = 0;
|
||
|
|
firstChar = 0;
|
||
|
|
} else {
|
||
|
|
/* get memorized variables from previous function calls */
|
||
|
|
step = (uint8_t)(dest->aux >> 8);
|
||
|
|
firstChar = (uint8_t)(dest->aux & 0xFFU);
|
||
|
|
}
|
||
|
|
|
||
|
|
/* repeat until destination space available and no error and not finished
|
||
|
|
* and source characters available */
|
||
|
|
while ((destSpace > 0U) && ((st & CO_fifo_st_errMask) == 0U) && !finished) {
|
||
|
|
uint8_t c;
|
||
|
|
if (!CO_fifo_getc(src, &c)) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (step == 6U) {
|
||
|
|
/* command is inside comment, waiting for command delimiter */
|
||
|
|
bool_t insideComment = true;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
}
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
if ((int32_t)(isxdigit((int32_t)c)) != 0) {
|
||
|
|
/* first or second hex digit */
|
||
|
|
if (step == 0U) {
|
||
|
|
firstChar = c;
|
||
|
|
step = 1;
|
||
|
|
} else {
|
||
|
|
/* write the byte */
|
||
|
|
uint8_t s[3];
|
||
|
|
int32_t num;
|
||
|
|
s[0] = firstChar;
|
||
|
|
s[1] = c;
|
||
|
|
s[2] = 0;
|
||
|
|
num = strtol((char*)&s[0], NULL, 16);
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)num);
|
||
|
|
destSpace--;
|
||
|
|
step = 0;
|
||
|
|
}
|
||
|
|
} else if ((int32_t)(isgraph((int32_t)c)) != 0) {
|
||
|
|
/* printable character, not hex digit */
|
||
|
|
if (c == DELIM_COMMENT) { /* comment start */
|
||
|
|
step = 6;
|
||
|
|
} else { /* syntax error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
/* this is space or delimiter */
|
||
|
|
if (step == 1U) {
|
||
|
|
/* write the byte */
|
||
|
|
uint8_t s[2];
|
||
|
|
int32_t num;
|
||
|
|
s[0] = firstChar;
|
||
|
|
s[1] = 0;
|
||
|
|
num = strtol((char*)&s[0], NULL, 16);
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)num);
|
||
|
|
destSpace--;
|
||
|
|
step = 0;
|
||
|
|
}
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
/* newline found, finish */
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else if (insideComment) {
|
||
|
|
step = 6;
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
}
|
||
|
|
} /* while ... */
|
||
|
|
|
||
|
|
if (!finished) {
|
||
|
|
st |= CO_fifo_st_partial;
|
||
|
|
/* memorize variables for next iteration */
|
||
|
|
dest->aux = ((uint32_t)step << 8) | firstChar;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
|
||
|
|
return destSpaceStart - destSpace;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2Vs(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
size_t destSpace, destSpaceStart;
|
||
|
|
bool_t finished = false;
|
||
|
|
uint8_t step;
|
||
|
|
uint8_t st = 0;
|
||
|
|
|
||
|
|
if ((dest == NULL) || (src == NULL)) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* get free space of the dest fifo */
|
||
|
|
destSpaceStart = CO_fifo_getSpace(dest);
|
||
|
|
destSpace = destSpaceStart;
|
||
|
|
|
||
|
|
/* is this the first write into dest? */
|
||
|
|
if (!dest->started) {
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || insideComment) {
|
||
|
|
/* command delimiter found without string, this is an error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
dest->started = true;
|
||
|
|
step = 0;
|
||
|
|
} else {
|
||
|
|
/* get memorized variables from previous function calls */
|
||
|
|
step = (uint8_t)dest->aux;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* repeat until destination space available and no error and not finished and source characters available */
|
||
|
|
while ((destSpace > 0U) && ((st & CO_fifo_st_errMask) == 0U) && !finished) {
|
||
|
|
uint8_t c;
|
||
|
|
if (!CO_fifo_getc(src, &c)) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
switch (step) {
|
||
|
|
case 0: /* beginning of the string, first write into dest */
|
||
|
|
if (c == DELIM_DQUOTE) {
|
||
|
|
/* Indicated beginning of the string, skip this character. */
|
||
|
|
step = 1;
|
||
|
|
} else {
|
||
|
|
/* this must be a single word string without '"' */
|
||
|
|
/* copy the character */
|
||
|
|
(void)CO_fifo_putc(dest, c);
|
||
|
|
destSpace--;
|
||
|
|
step = 2;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 1: /* inside string, quoted string */
|
||
|
|
case 2: /* inside string, single word, no quotes */
|
||
|
|
if (c == DELIM_DQUOTE) {
|
||
|
|
/* double quote found, this may be end of the string or escaped
|
||
|
|
* double quote (with two double quotes) */
|
||
|
|
step += 2U;
|
||
|
|
} else if ((isgraph((int)c) == 0) && (step == 2U)) {
|
||
|
|
/* end of single word string */
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else {
|
||
|
|
step = insideComment ? 6U : 5U;
|
||
|
|
}
|
||
|
|
} else if (c == DELIM_COMMAND) {
|
||
|
|
/* no closing quote, error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
/* copy the character */
|
||
|
|
(void)CO_fifo_putc(dest, c);
|
||
|
|
destSpace--;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 3: /* previous was double quote, parsing quoted string */
|
||
|
|
case 4: /* previous was double quote, parsing no quoted word */
|
||
|
|
if (c == DELIM_DQUOTE) {
|
||
|
|
/* escaped double quote, copy the character and continue */
|
||
|
|
(void)CO_fifo_putc(dest, c);
|
||
|
|
destSpace--;
|
||
|
|
step -= 2U;
|
||
|
|
} else {
|
||
|
|
/* previous character was closing double quote */
|
||
|
|
if (step == 4U) {
|
||
|
|
/* no opening double quote, syntax error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else {
|
||
|
|
if (isgraph((int)c) == 0) {
|
||
|
|
/* end of quoted string */
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else {
|
||
|
|
step = insideComment ? 6U : 5U;
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
/* space must follow closing double quote, error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 5: { /* String token is finished, waiting for command delimiter */
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else if (insideComment) {
|
||
|
|
step = 6;
|
||
|
|
} else if (isgraph((int)c) != 0) {
|
||
|
|
if (c == DELIM_COMMENT) { /* comment start */
|
||
|
|
step = 6;
|
||
|
|
} else { /* syntax error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
case 6: { /* String token is finished, waiting for command delimiter */
|
||
|
|
bool_t insideComment = true;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
default: /* internal error */ st |= CO_fifo_st_errInt; break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!finished) {
|
||
|
|
st |= CO_fifo_st_partial;
|
||
|
|
/* memorize variables for next iteration */
|
||
|
|
dest->aux = step;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
|
||
|
|
return destSpaceStart - destSpace;
|
||
|
|
}
|
||
|
|
|
||
|
|
size_t
|
||
|
|
CO_fifo_cpyTok2B64(CO_fifo_t* dest, CO_fifo_t* src, uint8_t* status) {
|
||
|
|
/* mime-base64 decoding, see description above base64EncTable */
|
||
|
|
|
||
|
|
size_t destSpace, destSpaceStart;
|
||
|
|
bool_t finished = false;
|
||
|
|
uint8_t step;
|
||
|
|
uint32_t dword;
|
||
|
|
uint8_t st = 0;
|
||
|
|
|
||
|
|
if ((dest == NULL) || (src == NULL)) {
|
||
|
|
return 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* get free space of the dest fifo */
|
||
|
|
destSpaceStart = CO_fifo_getSpace(dest);
|
||
|
|
destSpace = destSpaceStart;
|
||
|
|
|
||
|
|
/* is this the first write into dest? */
|
||
|
|
if (!dest->started) {
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || insideComment) {
|
||
|
|
/* command delimiter found without string, this is an error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
dest->started = true;
|
||
|
|
step = 0;
|
||
|
|
dword = 0;
|
||
|
|
} else {
|
||
|
|
/* get memorized variables from previous function calls */
|
||
|
|
step = (uint8_t)(dest->aux >> 24);
|
||
|
|
dword = dest->aux & 0xFFFFFFU;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* repeat until destination space available and no error and not finished and source characters available */
|
||
|
|
while ((destSpace >= 3U) && ((st & CO_fifo_st_errMask) == 0U) && !finished) {
|
||
|
|
uint8_t c;
|
||
|
|
if (!CO_fifo_getc(src, &c)) {
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (step >= 5U) {
|
||
|
|
/* String token is finished, waiting for command delimiter */
|
||
|
|
bool_t insideComment = step > 5U;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else if (insideComment) {
|
||
|
|
step = 6;
|
||
|
|
} else if ((isgraph((int)c) != 0) && (c != (uint8_t)'=')) {
|
||
|
|
if (c == DELIM_COMMENT) { /* comment start */
|
||
|
|
step = 6;
|
||
|
|
} else { /* syntax error */
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
}
|
||
|
|
} else { /* MISRA C 2004 14.10 */
|
||
|
|
}
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
uint8_t code = base64DecTable[c & 0x7FU];
|
||
|
|
|
||
|
|
if (((c & 0x80U) != 0U) || ((code & 0x80U) != 0U)) {
|
||
|
|
st |= CO_fifo_st_errTok;
|
||
|
|
} else if (code >= 64U /* '=' (pad) or DELIM_COMMAND or space */) {
|
||
|
|
/* base64 string finished, write remaining bytes */
|
||
|
|
switch (step) {
|
||
|
|
case 2:
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)(dword >> 4));
|
||
|
|
destSpace--;
|
||
|
|
break;
|
||
|
|
case 3:
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)(dword >> 10));
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)(dword >> 2));
|
||
|
|
destSpace -= 2U;
|
||
|
|
break;
|
||
|
|
default:
|
||
|
|
/* MISRA C 2004 15.3 */
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
bool_t insideComment = false;
|
||
|
|
if (CO_fifo_trimSpaces(src, &insideComment) || (c == DELIM_COMMAND)) {
|
||
|
|
st |= CO_fifo_st_closed;
|
||
|
|
finished = true;
|
||
|
|
} else {
|
||
|
|
step = insideComment ? 6U : 5U;
|
||
|
|
}
|
||
|
|
} else {
|
||
|
|
dword = (dword << 6) | code;
|
||
|
|
if (step++ == 3U) {
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)((dword >> 16) & 0xFFU));
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)((dword >> 8) & 0xFFU));
|
||
|
|
(void)CO_fifo_putc(dest, (uint8_t)(dword & 0xFFU));
|
||
|
|
destSpace -= 3U;
|
||
|
|
dword = 0;
|
||
|
|
step = 0;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
} /* while ... */
|
||
|
|
|
||
|
|
if (!finished) {
|
||
|
|
st |= CO_fifo_st_partial;
|
||
|
|
/* memorize variables for next iteration */
|
||
|
|
dest->aux = ((uint32_t)step << 24) | (dword & 0xFFFFFFU);
|
||
|
|
}
|
||
|
|
|
||
|
|
if (status != NULL) {
|
||
|
|
*status = st;
|
||
|
|
}
|
||
|
|
|
||
|
|
return destSpaceStart - destSpace;
|
||
|
|
}
|
||
|
|
|
||
|
|
#endif /* (CO_CONFIG_FIFO) & CO_CONFIG_FIFO_ASCII_DATATYPES */
|
||
|
|
|
||
|
|
#endif /* (CO_CONFIG_FIFO) & CO_CONFIG_FIFO_ENABLE */
|