diff options
Diffstat (limited to 'usr.sbin/acme-client/base64.c')
-rw-r--r-- | usr.sbin/acme-client/base64.c | 130 |
1 files changed, 130 insertions, 0 deletions
diff --git a/usr.sbin/acme-client/base64.c b/usr.sbin/acme-client/base64.c new file mode 100644 index 00000000000..6f4364ddaa5 --- /dev/null +++ b/usr.sbin/acme-client/base64.c @@ -0,0 +1,130 @@ +/* $Id: base64.c,v 1.1 2016/08/31 22:01:42 florian Exp $ */ +/* + * Copyright (c) 2016 Kristaps Dzonsons <kristaps@bsd.lv> + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include <sys/types.h> + +#include <assert.h> +#include <stdarg.h> +#include <stdlib.h> + +#include "extern.h" + +static const char b64[] = + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz" + "0123456789+/"; + +/* + * Compute the maximum buffer required for a base64 encoded string of + * length "len". + */ +size_t +base64len(size_t len) +{ + + return(((len + 2) / 3 * 4) + 1); +} + +/* + * Base64 computation. + * This is heavily "assert"-d because Coverity complains. + */ +size_t +base64buf(char *enc, const char *str, size_t len) +{ + size_t i, val; + char *p; + + p = enc; + + for (i = 0; i < len - 2; i += 3) { + val = (str[i] >> 2) & 0x3F; + assert(val < sizeof(b64)); + *p++ = b64[val]; + + val = ((str[i] & 0x3) << 4) | + ((int)(str[i + 1] & 0xF0) >> 4); + assert(val < sizeof(b64)); + *p++ = b64[val]; + + val = ((str[i + 1] & 0xF) << 2) | + ((int)(str[i + 2] & 0xC0) >> 6); + assert(val < sizeof(b64)); + *p++ = b64[val]; + + val = str[i + 2] & 0x3F; + assert(val < sizeof(b64)); + *p++ = b64[val]; + } + + if (i < len) { + val = (str[i] >> 2) & 0x3F; + assert(val < sizeof(b64)); + *p++ = b64[val]; + + if (i == (len - 1)) { + val = ((str[i] & 0x3) << 4); + assert(val < sizeof(b64)); + *p++ = b64[val]; + *p++ = '='; + } else { + val = ((str[i] & 0x3) << 4) | + ((int)(str[i + 1] & 0xF0) >> 4); + assert(val < sizeof(b64)); + *p++ = b64[val]; + + val = ((str[i + 1] & 0xF) << 2); + assert(val < sizeof(b64)); + *p++ = b64[val]; + } + *p++ = '='; + } + + *p++ = '\0'; + return(p - enc); +} + +/* + * Pass a stream of bytes to be base64 encoded, then converted into + * base64url format. + * Returns NULL on allocation failure (not logged). + */ +char * +base64buf_url(const char *data, size_t len) +{ + size_t i, sz; + char *buf; + + sz = base64len(len); + if (NULL == (buf = malloc(sz))) + return(NULL); + + base64buf(buf, data, len); + + for (i = 0; i < sz; i++) + if ('+' == buf[i]) + buf[i] = '-'; + else if ('/' == buf[i]) + buf[i] = '_'; + else if ('=' == buf[i]) + buf[i] = '\0'; + + return(buf); +} |