/* * FIFO circular buffer * * @file CO_fifo.c * @ingroup CO_CANopen_309_fifo * @author Janez Paternoster * @copyright 2020 Janez Paternoster * * This file is part of , 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 #include "301/CO_fifo.h" #if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ENABLE) != 0 #include #include #include "crc16-ccitt.h" #if ((CO_CONFIG_FIFO)&CO_CONFIG_FIFO_ASCII_COMMANDS) != 0 #include #include /* 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 */