xs_encdec.h 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. /* copyright (c) 2022 grunfink - MIT license */
  2. #ifndef _XS_ENCDEC_H
  3. #define _XS_ENCDEC_H
  4. d_char *xs_hex_enc(const char *data, int size);
  5. d_char *xs_hex_dec(const char *hex, int *size);
  6. d_char *xs_base64_enc(const char *data, int sz);
  7. d_char *xs_base64_dec(const char *data, int *size);
  8. d_char *xs_utf8_enc(d_char *str, unsigned int cpoint);
  9. #ifdef XS_IMPLEMENTATION
  10. d_char *xs_hex_enc(const char *data, int size)
  11. /* returns an hexdump of data */
  12. {
  13. d_char *s;
  14. char *p;
  15. int n;
  16. p = s = xs_realloc(NULL, _xs_blk_size(size * 2 + 1));
  17. for (n = 0; n < size; n++) {
  18. sprintf(p, "%02x", (unsigned char)data[n]);
  19. p += 2;
  20. }
  21. *p = '\0';
  22. return s;
  23. }
  24. d_char *xs_hex_dec(const char *hex, int *size)
  25. /* decodes an hexdump into data */
  26. {
  27. int sz = strlen(hex);
  28. d_char *s = NULL;
  29. char *p;
  30. int n;
  31. if (sz % 2)
  32. return NULL;
  33. p = s = xs_realloc(NULL, _xs_blk_size(sz / 2 + 1));
  34. for (n = 0; n < sz; n += 2) {
  35. int i;
  36. if (sscanf(&hex[n], "%02x", &i) == 0) {
  37. /* decoding error */
  38. free(s);
  39. return NULL;
  40. }
  41. else
  42. *p = i;
  43. p++;
  44. }
  45. *p = '\0';
  46. *size = sz / 2;
  47. return s;
  48. }
  49. d_char *xs_base64_enc(const char *data, int sz)
  50. /* encodes data to base64 */
  51. {
  52. d_char *s;
  53. unsigned char *p;
  54. char *i;
  55. int bsz, n;
  56. static char *b64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  57. "abcdefghijklmnopqrstuvwxyz"
  58. "0123456789+/";
  59. bsz = ((sz + 3 - 1) / 3) * 4;
  60. i = s = xs_realloc(NULL, _xs_blk_size(bsz + 1));
  61. p = (unsigned char *)data;
  62. for (n = 0; n < sz; n += 3) {
  63. int l = sz - n;
  64. if (l == 1) {
  65. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  66. *i++ = b64_tbl[(p[n] << 4) & 0x3f];
  67. *i++ = '=';
  68. *i++ = '=';
  69. }
  70. else
  71. if (l == 2) {
  72. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  73. *i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
  74. *i++ = b64_tbl[(p[n + 1] << 2) & 0x3f];
  75. *i++ = '=';
  76. }
  77. else {
  78. *i++ = b64_tbl[(p[n] >> 2) & 0x3f];
  79. *i++ = b64_tbl[(p[n] << 4 | p[n + 1] >> 4) & 0x3f];
  80. *i++ = b64_tbl[(p[n + 1] << 2 | p[n + 2] >> 6) & 0x3f];
  81. *i++ = b64_tbl[(p[n + 2]) & 0x3f];
  82. }
  83. }
  84. *i = '\0';
  85. return s;
  86. }
  87. d_char *xs_base64_dec(const char *data, int *size)
  88. /* decodes data from base64 */
  89. {
  90. d_char *s = NULL;
  91. int sz = 0;
  92. char *p;
  93. static char *b64_tbl = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  94. "abcdefghijklmnopqrstuvwxyz"
  95. "0123456789+/=";
  96. p = (char *)data;
  97. /* size of data must be a multiple of 4 */
  98. if (strlen(p) % 4)
  99. return NULL;
  100. for (p = (char *)data; *p; p += 4) {
  101. int cs[4];
  102. int n;
  103. unsigned char tmp[3];
  104. for (n = 0; n < 4; n++) {
  105. char *ss = strchr(b64_tbl, p[n]);
  106. if (ss == NULL) {
  107. /* not a base64 char */
  108. free(s);
  109. return NULL;
  110. }
  111. cs[n] = ss - b64_tbl;
  112. }
  113. n = 0;
  114. /* first byte */
  115. tmp[n++] = cs[0] << 2 | ((cs[1] >> 4) & 0x0f);
  116. /* second byte */
  117. if (cs[2] != 64)
  118. tmp[n++] = cs[1] << 4 | ((cs[2] >> 2) & 0x3f);
  119. /* third byte */
  120. if (cs[3] != 64)
  121. tmp[n++] = cs[2] << 6 | (cs[3] & 0x3f);
  122. /* must be done manually because data can be pure binary */
  123. s = xs_realloc(s, _xs_blk_size(sz + n));
  124. memcpy(s + sz, tmp, n);
  125. sz += n;
  126. }
  127. /* asciiz it to use it as a string */
  128. s = xs_realloc(s, _xs_blk_size(sz + 1));
  129. s[sz] = '\0';
  130. *size = sz;
  131. return s;
  132. }
  133. d_char *xs_utf8_enc(d_char *str, unsigned int cpoint)
  134. /* encodes an Unicode codepoint to utf8 */
  135. {
  136. unsigned char tmp[4];
  137. int n = 0;
  138. if (cpoint < 0x80)
  139. tmp[n++] = cpoint & 0xff;
  140. else
  141. if (cpoint < 0x800) {
  142. tmp[n++] = 0xc0 | (cpoint >> 6);
  143. tmp[n++] = 0x80 | (cpoint & 0x3f);
  144. }
  145. else
  146. if (cpoint < 0x10000) {
  147. tmp[n++] = 0xe0 | (cpoint >> 12);
  148. tmp[n++] = 0x80 | ((cpoint >> 6) & 0x3f);
  149. tmp[n++] = 0x80 | (cpoint & 0x3f);
  150. }
  151. else
  152. if (cpoint < 0x200000) {
  153. tmp[n++] = 0xf0 | (cpoint >> 18);
  154. tmp[n++] = 0x80 | ((cpoint >> 12) & 0x3f);
  155. tmp[n++] = 0x80 | ((cpoint >> 6) & 0x3f);
  156. tmp[n++] = 0x80 | (cpoint & 0x3f);
  157. }
  158. return xs_append_m(str, (char *)tmp, n);
  159. }
  160. #endif /* XS_IMPLEMENTATION */
  161. #endif /* _XS_ENCDEC_H */