xs_encdec.h 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. /* copyright (c) 2022 - 2023 grunfink / MIT license */
  2. #ifndef _XS_ENCDEC_H
  3. #define _XS_ENCDEC_H
  4. xs_str *xs_base64_enc(const xs_val *data, int sz);
  5. xs_val *xs_base64_dec(const xs_str *data, int *size);
  6. int xs_is_base64(const char *str);
  7. #ifdef XS_IMPLEMENTATION
  8. /** base64 */
  9. static char *xs_b64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  10. "abcdefghijklmnopqrstuvwxyz"
  11. "0123456789+/=";
  12. xs_str *xs_base64_enc_tbl(const xs_val *data, int sz, const char *b64_tbl)
  13. /* encodes data to base64 using a table */
  14. {
  15. xs_str *s;
  16. unsigned char *p;
  17. char *i;
  18. int bsz, n;
  19. bsz = ((sz + 3 - 1) / 3) * 4;
  20. i = s = xs_realloc(NULL, _xs_blk_size(bsz + 1));
  21. p = (unsigned char *)data;
  22. for (n = 0; n < sz; n += 3) {
  23. int l = sz - n;
  24. if (l == 1) {
  25. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  26. *i++ = b64_tbl[(p[n] << 4) & 0x3f];
  27. *i++ = '=';
  28. *i++ = '=';
  29. }
  30. else
  31. if (l == 2) {
  32. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  33. *i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
  34. *i++ = b64_tbl[(p[n + 1] << 2) & 0x3f];
  35. *i++ = '=';
  36. }
  37. else {
  38. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  39. *i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
  40. *i++ = b64_tbl[(p[n + 1] << 2 | p[n + 2] >> 6) & 0x3f];
  41. *i++ = b64_tbl[(p[n + 2]) & 0x3f];
  42. }
  43. }
  44. *i = '\0';
  45. return s;
  46. }
  47. xs_str *xs_base64_enc(const xs_val *data, int sz)
  48. /* encodes data to base64 */
  49. {
  50. return xs_base64_enc_tbl(data, sz, xs_b64_tbl);
  51. }
  52. xs_val *xs_base64_dec_tbl(const xs_str *data, int *size, const char *b64_tbl)
  53. /* decodes data from base64 using a table */
  54. {
  55. xs_val *s = NULL;
  56. int sz = 0;
  57. char *p;
  58. p = (char *)data;
  59. /* size of data must be a multiple of 4 */
  60. if (strlen(p) % 4)
  61. return NULL;
  62. for (p = (char *)data; *p; p += 4) {
  63. int cs[4];
  64. int n;
  65. unsigned char tmp[3];
  66. for (n = 0; n < 4; n++) {
  67. char *ss = strchr(b64_tbl, p[n]);
  68. if (ss == NULL) {
  69. /* not a base64 char */
  70. return xs_free(s);
  71. }
  72. cs[n] = ss - b64_tbl;
  73. }
  74. n = 0;
  75. /* first byte */
  76. tmp[n++] = cs[0] << 2 | ((cs[1] >> 4) & 0x0f);
  77. /* second byte */
  78. if (cs[2] != 64)
  79. tmp[n++] = cs[1] << 4 | ((cs[2] >> 2) & 0x3f);
  80. /* third byte */
  81. if (cs[3] != 64)
  82. tmp[n++] = cs[2] << 6 | (cs[3] & 0x3f);
  83. /* must be done manually because data can be pure binary */
  84. s = xs_realloc(s, _xs_blk_size(sz + n));
  85. memcpy(s + sz, tmp, n);
  86. sz += n;
  87. }
  88. /* asciiz it to use it as a string */
  89. s = xs_realloc(s, _xs_blk_size(sz + 1));
  90. s[sz] = '\0';
  91. *size = sz;
  92. return s;
  93. }
  94. xs_val *xs_base64_dec(const xs_str *data, int *size)
  95. /* decodes data from base64 */
  96. {
  97. return xs_base64_dec_tbl(data, size, xs_b64_tbl);
  98. }
  99. int xs_is_base64_tbl(const char *str, const char *b64_tbl)
  100. /* returns 1 if str is a base64 string, with table */
  101. {
  102. while (*str) {
  103. if (strchr(b64_tbl, *str++) == NULL)
  104. return 0;
  105. }
  106. return 1;
  107. }
  108. int xs_is_base64(const char *str)
  109. /* returns 1 if str is a base64 string */
  110. {
  111. return xs_is_base64_tbl(str, xs_b64_tbl);
  112. }
  113. #endif /* XS_IMPLEMENTATION */
  114. #endif /* _XS_ENCDEC_H */