123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606 |
- #ifndef B64_DOCUMENTATION_SKIP_SECTION
- # define B64_VER_C_B64_MAJOR 1
- # define B64_VER_C_B64_MINOR 2
- # define B64_VER_C_B64_REVISION 2
- # define B64_VER_C_B64_EDIT 16
- #endif
- #include "b64.h"
- #include <assert.h>
- #include <string.h>
- #ifndef B64_DOCUMENTATION_SKIP_SECTION
- # define NUM_PLAIN_DATA_BYTES (3)
- # define NUM_ENCODED_DATA_BYTES (4)
- #endif
- #ifndef NUM_ELEMENTS
- # define NUM_ELEMENTS(x) (sizeof(x) / sizeof(x[0]))
- #endif
- #if defined(_MSC_VER) && \
- _MSC_VER < 1000
- # pragma warning(disable : 4705)
- #endif
- static const char b64_chars[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
- static const signed char b64_indexes[] =
- {
-
- -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
-
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, 62, -1, -1, -1, 63
- , 52, 53, 54, 55, 56, 57, 58, 59
- , 60, 61, -1, -1, -1, -1, -1, -1
-
- , -1, 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, -1, -1, -1, -1, -1
-
- , -1, 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, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- , -1, -1, -1, -1, -1, -1, -1, -1
- };
- static size_t b64_encode_( unsigned char const *src
- , size_t srcSize
- , char *const dest
- , size_t destLen
- , unsigned lineLen
- , B64_RC *rc)
- {
- size_t total = ((srcSize + (NUM_PLAIN_DATA_BYTES - 1)) / NUM_PLAIN_DATA_BYTES) * NUM_ENCODED_DATA_BYTES;
- assert(NULL != rc);
- *rc = B64_RC_OK;
- if(lineLen > 0)
- {
- unsigned numLines = (unsigned int)(total + (lineLen - 1)) / lineLen;
- total += 2 * (numLines - 1);
- }
- if(NULL == dest)
- {
- return total;
- }
- else if(destLen < total)
- {
- *rc = B64_RC_INSUFFICIENT_BUFFER;
- return 0;
- }
- else
- {
- char *p = dest;
- char *end = dest + destLen;
- size_t len = 0;
- for(; NUM_PLAIN_DATA_BYTES <= srcSize; srcSize -= NUM_PLAIN_DATA_BYTES)
- {
- char characters[NUM_ENCODED_DATA_BYTES] = {0};
-
-
- characters[0] = (char)((src[0] & 0xfc) >> 2);
-
- characters[1] = (char)(((src[0] & 0x03) << 4) + ((src[1] & 0xf0) >> 4));
-
- characters[2] = (char)(((src[1] & 0x0f) << 2) + ((src[2] & 0xc0) >> 6));
-
- characters[3] = (char)(src[2] & 0x3f);
- #ifndef __WATCOMC__
- assert(characters[0] >= 0 && characters[0] < 64);
- assert(characters[1] >= 0 && characters[1] < 64);
- assert(characters[2] >= 0 && characters[2] < 64);
- assert(characters[3] >= 0 && characters[3] < 64);
- #endif
- src += NUM_PLAIN_DATA_BYTES;
- *p++ = b64_chars[(unsigned char)characters[0]];
- assert(NULL != strchr(b64_chars, *(p-1)));
- ++len;
- assert(len != lineLen);
- *p++ = b64_chars[(unsigned char)characters[1]];
- assert(NULL != strchr(b64_chars, *(p-1)));
- ++len;
- assert(len != lineLen);
- *p++ = b64_chars[(unsigned char)characters[2]];
- assert(NULL != strchr(b64_chars, *(p-1)));
- ++len;
- assert(len != lineLen);
- *p++ = b64_chars[(unsigned char)characters[3]];
- assert(NULL != strchr(b64_chars, *(p-1)));
- if( ++len == lineLen &&
- p != end)
- {
- *p++ = '\r';
- *p++ = '\n';
- len = 0;
- }
- }
- if(0 != srcSize)
- {
-
- unsigned char dummy[NUM_PLAIN_DATA_BYTES] = {0};
- size_t i;
- for(i = 0; i < srcSize; ++i)
- {
- dummy[i] = *src++;
- }
- for(; i < NUM_PLAIN_DATA_BYTES; ++i)
- {
- dummy[i] = '\0';
- }
- b64_encode_(&dummy[0], NUM_PLAIN_DATA_BYTES, p, NUM_ENCODED_DATA_BYTES * (1 + 2), 0, rc);
- for(p += 1 + srcSize; srcSize++ < NUM_PLAIN_DATA_BYTES; )
- {
- *p++ = '=';
- }
- }
- return total;
- }
- }
- static size_t b64_decode_( char const *src
- , size_t srcLen
- , unsigned char *dest
- , size_t destSize
- , unsigned flags
- , char const **badChar
- , B64_RC *rc)
- {
- const size_t wholeChunks = (srcLen / NUM_ENCODED_DATA_BYTES);
- const size_t remainderBytes = (srcLen % NUM_ENCODED_DATA_BYTES);
- size_t maxTotal = (wholeChunks + (0 != remainderBytes)) * NUM_PLAIN_DATA_BYTES;
- unsigned char *dest_ = dest;
- ((void)remainderBytes);
- assert(NULL != badChar);
- assert(NULL != rc);
- *badChar = NULL;
- *rc = B64_RC_OK;
- if(NULL == dest)
- {
- return maxTotal;
- }
- else if(destSize < maxTotal)
- {
- *rc = B64_RC_INSUFFICIENT_BUFFER;
- return 0;
- }
- else
- {
-
- char const *begin = src;
- char const *const end = begin + srcLen;
- size_t currIndex = 0;
- size_t numPads = 0;
- signed char indexes[NUM_ENCODED_DATA_BYTES] = {0};
- for(; begin != end; ++begin)
- {
- const char ch = *begin;
- if('=' == ch)
- {
- assert(currIndex < NUM_ENCODED_DATA_BYTES);
- indexes[currIndex++] = '\0';
- ++numPads;
- }
- else
- {
-
- signed char ix = b64_indexes[(unsigned char)ch];
- if(-1 == ix)
- {
- switch(ch)
- {
- case ' ':
- case '\t':
- case '\b':
- case '\v':
- if(B64_F_STOP_ON_UNEXPECTED_WS & flags)
- {
- *rc = B64_RC_DATA_ERROR;
- *badChar = begin;
- return 0;
- }
- else
- {
-
- }
- case '\r':
- case '\n':
- continue;
- default:
- if(B64_F_STOP_ON_UNKNOWN_CHAR & flags)
- {
- *rc = B64_RC_DATA_ERROR;
- *badChar = begin;
- return 0;
- }
- else
- {
- continue;
- }
- }
- }
- else
- {
- numPads = 0;
- assert(currIndex < NUM_ENCODED_DATA_BYTES);
- indexes[currIndex++] = ix;
- }
- }
- if(NUM_ENCODED_DATA_BYTES == currIndex)
- {
- unsigned char bytes[NUM_PLAIN_DATA_BYTES] = {0};
- bytes[0] = (unsigned char)((indexes[0] << 2) + ((indexes[1] & 0x30) >> 4));
- currIndex = 0;
- *dest++ = bytes[0];
- if(2 != numPads)
- {
- bytes[1] = (unsigned char)(((indexes[1] & 0xf) << 4) + ((indexes[2] & 0x3c) >> 2));
- *dest++ = bytes[1];
- if(1 != numPads)
- {
- bytes[2] = (unsigned char)(((indexes[2] & 0x3) << 6) + indexes[3]);
- *dest++ = bytes[2];
- }
- }
- if(0 != numPads)
- {
- break;
- }
- }
- }
- return (size_t)(dest - dest_);
- }
- }
- size_t b64_encode(void const *src, size_t srcSize, char *dest, size_t destLen)
- {
-
- B64_RC rc_;
- return b64_encode_((unsigned char const*)src, srcSize, dest, destLen, 0, &rc_);
- }
- size_t b64_encode2( void const *src
- , size_t srcSize
- , char *dest
- , size_t destLen
- , unsigned flags
- , int lineLen
- , B64_RC *rc )
- {
-
- B64_RC rc_;
- if(NULL == rc)
- {
- rc = &rc_;
- }
- switch(B64_F_LINE_LEN_MASK & flags)
- {
- case B64_F_LINE_LEN_USE_PARAM:
- if(lineLen >= 0)
- {
- break;
- }
-
- case B64_F_LINE_LEN_64:
- lineLen = 64;
- break;
- case B64_F_LINE_LEN_76:
- lineLen = 76;
- break;
- default:
- assert(!"Bad line length flag specified to b64_encode2()");
- case B64_F_LINE_LEN_INFINITE:
- lineLen = 0;
- break;
- }
- assert(0 == (lineLen % 4));
- return b64_encode_((unsigned char const*)src, srcSize, dest, destLen, (unsigned)lineLen, rc);
- }
- size_t b64_decode(char const *src, size_t srcLen, void *dest, size_t destSize)
- {
-
- char const *badChar_;
- B64_RC rc_;
- return b64_decode_(src, srcLen, (unsigned char*)dest, destSize, B64_F_STOP_ON_NOTHING, &badChar_, &rc_);
- }
- size_t b64_decode2( char const *src
- , size_t srcLen
- , void *dest
- , size_t destSize
- , unsigned flags
- , char const **badChar
- , B64_RC *rc )
- {
- char const *badChar_;
- B64_RC rc_;
-
- if(NULL == badChar)
- {
- badChar = &badChar_;
- }
- if(NULL == rc)
- {
- rc = &rc_;
- }
- return b64_decode_(src, srcLen, (unsigned char*)dest, destSize, flags, badChar, rc);
- }
- #ifdef B64_DOCUMENTATION_SKIP_SECTION
- struct b64ErrorString_t_
- #else /* !B64_DOCUMENTATION_SKIP_SECTION */
- typedef struct b64ErrorString_t_ b64ErrorString_t_;
- struct b64ErrorString_t_
- #endif /* !B64_DOCUMENTATION_SKIP_SECTION */
- {
- int code;
- char const *str;
- size_t len;
- };
- #define SEVERITY_STR_DECL(rc, desc) \
- \
- static const char s_str##rc[] = desc; \
- static const b64ErrorString_t_ s_rct##rc = { rc, s_str##rc, NUM_ELEMENTS(s_str##rc) - 1 }
- #define SEVERITY_STR_ENTRY(rc) \
- \
- &s_rct##rc
- static char const *b64_LookupCodeA_(int code, b64ErrorString_t_ const **mappings, size_t cMappings, size_t *len)
- {
-
- size_t len_;
- if(NULL == len)
- {
- len = &len_;
- }
-
- if( code >= 0 &&
- code < B64_max_RC_value)
- {
- if(code == mappings[code]->code)
- {
- return (*len = mappings[code]->len, mappings[code]->str);
- }
- }
-
- { size_t i; for(i = 0; i < cMappings; ++i)
- {
- if(code == mappings[i]->code)
- {
- return (*len = mappings[i]->len, mappings[i]->str);
- }
- }}
- return (*len = 0, "");
- }
- static char const *b64_LookupErrorStringA_(int error, size_t *len)
- {
- SEVERITY_STR_DECL(B64_RC_OK , "Operation was successful" );
- SEVERITY_STR_DECL(B64_RC_INSUFFICIENT_BUFFER , "The given translation buffer was not of sufficient size" );
- SEVERITY_STR_DECL(B64_RC_TRUNCATED_INPUT , "The input did not represent a fully formed stream of octet couplings" );
- SEVERITY_STR_DECL(B64_RC_DATA_ERROR , "Invalid data" );
- static const b64ErrorString_t_ *s_strings[] =
- {
- SEVERITY_STR_ENTRY(B64_RC_OK),
- SEVERITY_STR_ENTRY(B64_RC_INSUFFICIENT_BUFFER),
- SEVERITY_STR_ENTRY(B64_RC_TRUNCATED_INPUT),
- SEVERITY_STR_ENTRY(B64_RC_DATA_ERROR),
- };
- return b64_LookupCodeA_(error, s_strings, NUM_ELEMENTS(s_strings), len);
- }
- char const *b64_getErrorString(B64_RC code)
- {
- return b64_LookupErrorStringA_((int)code, NULL);
- }
- size_t b64_getErrorStringLength(B64_RC code)
- {
- size_t len;
- return (b64_LookupErrorStringA_((int)code, &len), len);
- }
|