snac2/xs_encdec.h

220 lines
4.8 KiB
C
Raw Normal View History

2023-01-17 08:50:16 +00:00
/* copyright (c) 2022 - 2023 grunfink / MIT license */
2022-09-19 18:41:11 +00:00
#ifndef _XS_ENCDEC_H
#define _XS_ENCDEC_H
2023-01-28 16:49:02 +00:00
xs_str *xs_hex_enc(const xs_val *data, int size);
xs_val *xs_hex_dec(const xs_str *hex, int *size);
2023-04-10 07:14:44 +00:00
int xs_is_hex(const char *str);
2023-01-28 16:49:02 +00:00
xs_str *xs_base64_enc(const xs_val *data, int sz);
xs_val *xs_base64_dec(const xs_str *data, int *size);
xs_str *xs_utf8_enc(xs_str *str, unsigned int cpoint);
2022-09-19 18:41:11 +00:00
#ifdef XS_IMPLEMENTATION
2023-01-28 16:49:02 +00:00
xs_str *xs_hex_enc(const xs_val *data, int size)
2022-09-19 18:41:11 +00:00
/* returns an hexdump of data */
{
2023-01-28 16:49:02 +00:00
xs_str *s;
2022-09-19 18:41:11 +00:00
char *p;
int n;
2022-10-18 09:21:37 +00:00
p = s = xs_realloc(NULL, _xs_blk_size(size * 2 + 1));
2022-09-19 18:41:11 +00:00
for (n = 0; n < size; n++) {
2023-04-19 20:53:07 +00:00
snprintf(p, 3, "%02x", (unsigned char)data[n]);
2022-09-19 18:41:11 +00:00
p += 2;
}
2022-10-18 09:21:37 +00:00
*p = '\0';
2022-09-19 18:41:11 +00:00
return s;
}
2023-01-28 16:49:02 +00:00
xs_val *xs_hex_dec(const xs_str *hex, int *size)
2022-09-19 18:41:11 +00:00
/* decodes an hexdump into data */
{
int sz = strlen(hex);
2023-01-28 16:49:02 +00:00
xs_val *s = NULL;
2022-09-19 18:41:11 +00:00
char *p;
int n;
2022-09-30 08:03:35 +00:00
if (sz % 2)
2022-10-18 09:21:37 +00:00
return NULL;
2022-09-30 08:03:35 +00:00
2022-10-18 09:21:37 +00:00
p = s = xs_realloc(NULL, _xs_blk_size(sz / 2 + 1));
2022-09-19 18:41:11 +00:00
for (n = 0; n < sz; n += 2) {
int i;
if (sscanf(&hex[n], "%02x", &i) == 0) {
/* decoding error */
2022-10-25 07:32:41 +00:00
return xs_free(s);
2022-09-19 18:41:11 +00:00
}
else
*p = i;
p++;
}
2022-10-18 09:21:37 +00:00
*p = '\0';
*size = sz / 2;
2022-09-19 18:41:11 +00:00
return s;
}
2023-04-10 07:14:44 +00:00
int xs_is_hex(const char *str)
/* returns 1 if str is an hex string */
{
while (*str) {
if (strchr("0123456789abcdefABCDEF", *str++) == NULL)
return 0;
}
return 1;
}
2023-01-28 16:49:02 +00:00
xs_str *xs_base64_enc(const xs_val *data, int sz)
2022-09-19 18:41:11 +00:00
/* encodes data to base64 */
{
2023-01-28 16:49:02 +00:00
xs_str *s;
2022-09-19 18:41:11 +00:00
unsigned char *p;
2022-09-30 08:03:35 +00:00
char *i;
int bsz, n;
2022-09-19 18:41:11 +00:00
static char *b64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz"
"0123456789+/";
2022-09-30 08:03:35 +00:00
bsz = ((sz + 3 - 1) / 3) * 4;
2022-10-18 09:21:37 +00:00
i = s = xs_realloc(NULL, _xs_blk_size(bsz + 1));
2022-09-19 18:41:11 +00:00
p = (unsigned char *)data;
for (n = 0; n < sz; n += 3) {
int l = sz - n;
if (l == 1) {
2022-09-30 08:03:35 +00:00
*i++ = b64_tbl[(p[n] >> 2) & 0x3f];
*i++ = b64_tbl[(p[n] << 4) & 0x3f];
*i++ = '=';
*i++ = '=';
2022-09-19 18:41:11 +00:00
}
else
if (l == 2) {
2022-09-30 08:03:35 +00:00
*i++ = b64_tbl[(p[n] >> 2) & 0x3f];
*i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
*i++ = b64_tbl[(p[n + 1] << 2) & 0x3f];
*i++ = '=';
2022-09-19 18:41:11 +00:00
}
else {
2022-09-30 08:03:35 +00:00
*i++ = b64_tbl[(p[n] >> 2) & 0x3f];
*i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
*i++ = b64_tbl[(p[n + 1] << 2 | p[n + 2] >> 6) & 0x3f];
*i++ = b64_tbl[(p[n + 2]) & 0x3f];
2022-09-19 18:41:11 +00:00
}
}
2022-10-18 09:21:37 +00:00
*i = '\0';
2022-09-19 18:41:11 +00:00
return s;
}
2023-01-28 16:49:02 +00:00
xs_val *xs_base64_dec(const xs_str *data, int *size)
2022-09-19 18:41:11 +00:00
/* decodes data from base64 */
{
2023-01-28 16:49:02 +00:00
xs_val *s = NULL;
2022-09-19 18:41:11 +00:00
int sz = 0;
char *p;
static char *b64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
"abcdefghijklmnopqrstuvwxyz"
"0123456789+/=";
p = (char *)data;
2022-09-30 08:03:35 +00:00
/* size of data must be a multiple of 4 */
if (strlen(p) % 4)
2022-10-18 09:21:37 +00:00
return NULL;
2022-09-30 08:03:35 +00:00
2022-09-19 18:41:11 +00:00
for (p = (char *)data; *p; p += 4) {
int cs[4];
int n;
unsigned char tmp[3];
for (n = 0; n < 4; n++) {
char *ss = strchr(b64_tbl, p[n]);
if (ss == NULL) {
/* not a base64 char */
2022-10-25 07:32:41 +00:00
return xs_free(s);
2022-09-19 18:41:11 +00:00
}
cs[n] = ss - b64_tbl;
}
n = 0;
/* first byte */
tmp[n++] = cs[0] << 2 | ((cs[1] >> 4) & 0x0f);
/* second byte */
if (cs[2] != 64)
tmp[n++] = cs[1] << 4 | ((cs[2] >> 2) & 0x3f);
/* third byte */
if (cs[3] != 64)
tmp[n++] = cs[2] << 6 | (cs[3] & 0x3f);
/* must be done manually because data can be pure binary */
2022-10-18 09:21:37 +00:00
s = xs_realloc(s, _xs_blk_size(sz + n));
2022-09-19 18:41:11 +00:00
memcpy(s + sz, tmp, n);
sz += n;
}
2022-10-18 09:21:37 +00:00
/* asciiz it to use it as a string */
s = xs_realloc(s, _xs_blk_size(sz + 1));
s[sz] = '\0';
2022-09-19 18:41:11 +00:00
*size = sz;
return s;
}
2023-01-28 16:49:02 +00:00
xs_str *xs_utf8_enc(xs_str *str, unsigned int cpoint)
2022-09-19 18:41:11 +00:00
/* encodes an Unicode codepoint to utf8 */
{
unsigned char tmp[4];
int n = 0;
if (cpoint < 0x80)
tmp[n++] = cpoint & 0xff;
else
if (cpoint < 0x800) {
tmp[n++] = 0xc0 | (cpoint >> 6);
tmp[n++] = 0x80 | (cpoint & 0x3f);
}
else
if (cpoint < 0x10000) {
tmp[n++] = 0xe0 | (cpoint >> 12);
tmp[n++] = 0x80 | ((cpoint >> 6) & 0x3f);
tmp[n++] = 0x80 | (cpoint & 0x3f);
}
else
if (cpoint < 0x200000) {
tmp[n++] = 0xf0 | (cpoint >> 18);
tmp[n++] = 0x80 | ((cpoint >> 12) & 0x3f);
tmp[n++] = 0x80 | ((cpoint >> 6) & 0x3f);
tmp[n++] = 0x80 | (cpoint & 0x3f);
}
2022-09-19 18:41:11 +00:00
return xs_append_m(str, (char *)tmp, n);
}
#endif /* XS_IMPLEMENTATION */
#endif /* _XS_ENCDEC_H */