sstream.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. #include "pocketpy/common/sstream.h"
  2. #include "pocketpy/common/str.h"
  3. #include "pocketpy/common/utils.h"
  4. #include "pocketpy/common/dmath.h"
  5. #include "pocketpy/pocketpy.h"
  6. #include <stdarg.h>
  7. #include <assert.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <ctype.h>
  11. void c11_sbuf__ctor(c11_sbuf* self) {
  12. c11_vector__ctor(&self->data, sizeof(char));
  13. c11_vector__reserve(&self->data, sizeof(c11_string) + 64);
  14. self->data.length = sizeof(c11_string);
  15. }
  16. void c11_sbuf__dtor(c11_sbuf* self) { c11_vector__dtor(&self->data); }
  17. void c11_sbuf__write_char(c11_sbuf* self, char c) { c11_vector__push(char, &self->data, c); }
  18. void c11_sbuf__write_pad(c11_sbuf* self, int count, char pad) {
  19. for(int i = 0; i < count; i++) {
  20. c11_sbuf__write_char(self, pad);
  21. }
  22. }
  23. void c11_sbuf__write_int(c11_sbuf* self, int i) {
  24. // len('-2147483648') == 11
  25. c11_vector__reserve(&self->data, self->data.length + 11 + 1);
  26. char* p = (char*)self->data.data + self->data.length;
  27. int n = snprintf(p, 11 + 1, "%d", i);
  28. self->data.length += n;
  29. }
  30. void c11_sbuf__write_i64(c11_sbuf* self, int64_t val) {
  31. // len('-9223372036854775808') == 20
  32. c11_vector__reserve(&self->data, self->data.length + 20 + 1);
  33. char* p = (char*)self->data.data + self->data.length;
  34. int n = snprintf(p, 20 + 1, "%lld", (long long)val);
  35. self->data.length += n;
  36. }
  37. void c11_sbuf__write_f64(c11_sbuf* self, double val, int precision) {
  38. if(dmath_isinf(val)) {
  39. c11_sbuf__write_cstr(self, val > 0 ? "inf" : "-inf");
  40. return;
  41. }
  42. if(dmath_isnan(val)) {
  43. c11_sbuf__write_cstr(self, "nan");
  44. return;
  45. }
  46. char b[32];
  47. int size;
  48. if(precision < 0) {
  49. for(int g = 15; g <= 17; g++) {
  50. size = snprintf(b, sizeof(b), "%.*g", g, val);
  51. if(strtod(b, NULL) == val) break;
  52. }
  53. c11_sbuf__write_cstr(self, b);
  54. bool all_is_digit = true;
  55. for(int i = 1; i < size; i++) {
  56. if(!isdigit(b[i])) {
  57. all_is_digit = false;
  58. break;
  59. }
  60. }
  61. if(all_is_digit) c11_sbuf__write_cstr(self, ".0");
  62. } else {
  63. size = snprintf(b, sizeof(b), "%.*f", precision, val);
  64. c11_sbuf__write_cstr(self, b);
  65. }
  66. }
  67. void c11_sbuf__write_sv(c11_sbuf* self, c11_sv sv) {
  68. c11_sbuf__write_cstrn(self, sv.data, sv.size);
  69. }
  70. void c11_sbuf__write_cstr(c11_sbuf* self, const char* str) {
  71. c11_sbuf__write_cstrn(self, str, strlen(str));
  72. }
  73. void c11_sbuf__write_cstrn(c11_sbuf* self, const char* str, int n) {
  74. c11_vector__extend(&self->data, str, n);
  75. }
  76. void c11_sbuf__write_quoted(c11_sbuf* self, c11_sv sv, char quote) {
  77. assert(quote == '"' || quote == '\'');
  78. c11_sbuf__write_char(self, quote);
  79. for(int i = 0; i < sv.size; i++) {
  80. char c = sv.data[i];
  81. switch(c) {
  82. case '"':
  83. case '\'':
  84. if(c == quote) c11_sbuf__write_char(self, '\\');
  85. c11_sbuf__write_char(self, c);
  86. break;
  87. case '\\': c11_sbuf__write_cstrn(self, "\\\\", 2); break;
  88. case '\n': c11_sbuf__write_cstrn(self, "\\n", 2); break;
  89. case '\r': c11_sbuf__write_cstrn(self, "\\r", 2); break;
  90. case '\t': c11_sbuf__write_cstrn(self, "\\t", 2); break;
  91. case '\b': c11_sbuf__write_cstrn(self, "\\b", 2); break;
  92. default: {
  93. int u8bytes = c11__u8_header(c, true);
  94. if(i + u8bytes > sv.size) u8bytes = 0; // invalid utf8
  95. if(u8bytes <= 1) {
  96. // not a valid utf8 char, or ascii
  97. if(!isprint((unsigned char)c)) {
  98. unsigned char uc = (unsigned char)c;
  99. c11_sbuf__write_cstrn(self, "\\x", 2);
  100. c11_sbuf__write_char(self, PK_HEX_TABLE[uc >> 4]);
  101. c11_sbuf__write_char(self, PK_HEX_TABLE[uc & 0xf]);
  102. } else {
  103. c11_sbuf__write_char(self, c);
  104. }
  105. } else {
  106. for(int j = 0; j < u8bytes; j++) {
  107. c11_sbuf__write_char(self, sv.data[i + j]);
  108. }
  109. i += u8bytes - 1;
  110. }
  111. break;
  112. }
  113. }
  114. }
  115. c11_sbuf__write_char(self, quote);
  116. }
  117. void c11_sbuf__write_hex(c11_sbuf* self, unsigned char c, bool non_zero) {
  118. unsigned char high = c >> 4;
  119. unsigned char low = c & 0xf;
  120. if(non_zero) {
  121. if(high) c11_sbuf__write_char(self, PK_HEX_TABLE[high]);
  122. if(high || low) c11_sbuf__write_char(self, PK_HEX_TABLE[low]);
  123. } else {
  124. c11_sbuf__write_char(self, PK_HEX_TABLE[high]);
  125. c11_sbuf__write_char(self, PK_HEX_TABLE[low]);
  126. }
  127. }
  128. void c11_sbuf__write_ptr(c11_sbuf* self, void* p) {
  129. if(p == NULL) {
  130. c11_sbuf__write_cstr(self, "0x0");
  131. return;
  132. }
  133. c11_sbuf__write_cstr(self, "0x");
  134. uintptr_t p_t = (uintptr_t)(p);
  135. bool non_zero = true;
  136. for(int i = sizeof(void*) - 1; i >= 0; i--) {
  137. unsigned char cpnt = (p_t >> (i * 8)) & 0xff;
  138. c11_sbuf__write_hex(self, cpnt, non_zero);
  139. if(cpnt != 0) non_zero = false;
  140. }
  141. }
  142. c11_string* c11_sbuf__submit(c11_sbuf* self) {
  143. c11_vector__push(char, &self->data, '\0');
  144. int arr_length;
  145. c11_string* retval = c11_vector__submit(&self->data, &arr_length);
  146. retval->size = arr_length - sizeof(c11_string) - 1;
  147. return retval;
  148. }
  149. void c11_sbuf__py_submit(c11_sbuf* self, py_Ref out) {
  150. c11_string* res = c11_sbuf__submit(self);
  151. py_newstrv(out, (c11_sv){res->data, res->size});
  152. c11_string__delete(res);
  153. }
  154. void pk_vsprintf(c11_sbuf* ss, const char* fmt, va_list args) {
  155. while(*fmt) {
  156. char c = *fmt;
  157. if(c != '%') {
  158. c11_sbuf__write_char(ss, c);
  159. fmt++;
  160. continue;
  161. }
  162. fmt++;
  163. c = *fmt;
  164. switch(c) {
  165. case 'd': {
  166. int i = va_arg(args, int);
  167. c11_sbuf__write_int(ss, i);
  168. break;
  169. }
  170. case 'i': {
  171. int64_t i = va_arg(args, int64_t);
  172. c11_sbuf__write_i64(ss, i);
  173. break;
  174. }
  175. case 'f': {
  176. double d = va_arg(args, double);
  177. c11_sbuf__write_f64(ss, d, -1);
  178. break;
  179. }
  180. case 's': {
  181. const char* s = va_arg(args, const char*);
  182. c11_sbuf__write_cstr(ss, s);
  183. break;
  184. }
  185. case 'q': {
  186. c11_sv sv = va_arg(args, c11_sv);
  187. c11_sbuf__write_quoted(ss, sv, '\'');
  188. break;
  189. }
  190. case 'Q': {
  191. c11_sv sv = va_arg(args, c11_sv);
  192. c11_sbuf__write_quoted(ss, sv, '"');
  193. break;
  194. }
  195. case 'v': {
  196. c11_sv sv = va_arg(args, c11_sv);
  197. c11_sbuf__write_sv(ss, sv);
  198. break;
  199. }
  200. case 'c': {
  201. char c = va_arg(args, int);
  202. c11_sbuf__write_char(ss, c);
  203. break;
  204. }
  205. case 'p': {
  206. void* p = va_arg(args, void*);
  207. c11_sbuf__write_ptr(ss, p);
  208. break;
  209. }
  210. case 't': {
  211. py_Type t = va_arg(args, int);
  212. c11_sbuf__write_cstr(ss, py_tpname(t));
  213. break;
  214. }
  215. case 'n': {
  216. py_Name n = va_arg(args, void*);
  217. c11_sbuf__write_cstr(ss, py_name2str(n));
  218. break;
  219. }
  220. case '%': c11_sbuf__write_char(ss, '%'); break;
  221. default:
  222. c11_sbuf__write_char(ss, c);
  223. assert(false); // invalid format
  224. break;
  225. }
  226. fmt++;
  227. }
  228. }
  229. void pk_sprintf(c11_sbuf* ss, const char* fmt, ...) {
  230. va_list args;
  231. va_start(args, fmt);
  232. pk_vsprintf(ss, fmt, args);
  233. va_end(args);
  234. }