X-Git-Url: https://diplodocus.org/git/nmh/blobdiff_plain/b705240a0933857f7248170bb1419391f651cff6..ec173fd2c:/sbr/base64.c?ds=sidebyside diff --git a/sbr/base64.c b/sbr/base64.c index 8426fb2f..ef3aa32a 100644 --- a/sbr/base64.c +++ b/sbr/base64.c @@ -5,21 +5,33 @@ * complete copyright information. */ -#include -#include -#include +#include "h/mh.h" +#include "error.h" +#include "h/mime.h" +#include "h/md5.h" #include static const char nib2b64[0x40+1] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; +/* + * Copy data from one file to another, converting to base64-encoding. + * + * Arguments include: + * + * in - Input filehandle (unencoded data) + * out - Output filename (base64-encoded data) + * crlf - If set, output encoded CRLF for every LF on input. + * + * Returns OK on success, NOTOK otherwise. + */ int writeBase64aux (FILE *in, FILE *out, int crlf) { unsigned int cc, n; unsigned char inbuf[3]; - int skipnl = 0; + bool skipnl = false; n = BPERLIN; while ((cc = fread (inbuf, sizeof(*inbuf), sizeof(inbuf), in)) > 0) { unsigned long bits; @@ -58,7 +70,7 @@ writeBase64aux (FILE *in, FILE *out, int crlf) inbuf[cc++] = '\n'; else ungetc('\n', in); - skipnl = 1; + skipnl = true; } else { /* This only works as long as sizeof(inbuf) == 3 */ ungetc(inbuf[cc - 1], in); @@ -67,7 +79,7 @@ writeBase64aux (FILE *in, FILE *out, int crlf) inbuf[++i] = '\n'; } } else { - skipnl = 0; + skipnl = false; } } } @@ -227,13 +239,19 @@ static const unsigned char b642nib[0x80] = { /* * Decode a base64 string. The result, decoded, must be freed by the caller. - * See description of arguments with declaration in h/prototypes.h. + * + * encoded - the string to be decoded + * decoded - the decoded bytes + * len - number of decoded bytes + * skip-crs - non-zero for text content, and for which CR's should be + * skipped + * digest - for an MD5 digest, it can be null */ int decodeBase64 (const char *encoded, unsigned char **decoded, size_t *len, - int skip_crs, unsigned char *digest) { + int skip_crs, unsigned char *digest) +{ const char *cp = encoded; - int self_delimiting = 0; int bitno, skip; uint32_t bits; /* Size the decoded string very conservatively. */ @@ -247,6 +265,7 @@ decodeBase64 (const char *encoded, unsigned char **decoded, size_t *len, bits = 0L; skip = 0; + bool self_delimiting = false; for (; *cp; ++cp) { switch (*cp) { unsigned char value; @@ -257,7 +276,8 @@ decodeBase64 (const char *encoded, unsigned char **decoded, size_t *len, } if (skip || (((unsigned char) *cp) & 0x80) || (value = b642nib[((unsigned char) *cp) & 0x7f]) > 0x3f) { - inform("invalid BASE64 encoding in %s", cp); + inform("invalid base64 byte %#x: %.42s", + *(unsigned char *)cp, cp); charstring_free (decoded_c); *decoded = NULL; @@ -300,18 +320,15 @@ test_end: case '=': if (++skip <= 3) goto test_end; - self_delimiting = 1; + self_delimiting = true; break; } } if (! self_delimiting && bitno != 18) { - int i; - /* Show some context for the error. */ - for (i = 0; i < 20 && cp > encoded; ++i, --cp) {} - inform("premature ending (bitno %d) near %s", bitno, - cp ? cp : encoded); + cp -= min(cp - encoded, 20); + inform("premature ending (bitno %d) near %s", bitno, cp); charstring_free (decoded_c); *decoded = NULL; @@ -336,7 +353,8 @@ test_end: * is allocated by the function and must be freed by the caller. */ void -hexify (const unsigned char *input, size_t len, char **output) { +hexify (const unsigned char *input, size_t len, char **output) +{ /* Start with a charstring capacity that's arbitrarily larger than len. */ const charstring_t tmp = charstring_create (2 * len); const unsigned char *cp = input;