wchar.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Assertions.h>
  7. #include <AK/Format.h>
  8. #include <AK/UnicodeUtils.h>
  9. #include <errno.h>
  10. #include <string.h>
  11. #include <wchar.h>
  12. static unsigned int mbstate_expected_bytes(mbstate_t* state)
  13. {
  14. if (state->stored_bytes == 0) {
  15. return 0;
  16. }
  17. unsigned char first = state->bytes[0];
  18. // Single-byte sequences have their first bit unset
  19. if ((first & 0b10000000) == 0) {
  20. return 1;
  21. }
  22. // Two-byte sequences start with 0b110xxxxx
  23. if ((first & 0b11100000) == 0b11000000) {
  24. return 2;
  25. }
  26. // Three-byte sequences start with 0b1110xxxx
  27. if ((first & 0b11110000) == 0b11100000) {
  28. return 3;
  29. }
  30. // Four-byte sequences start with 0b11110xxx
  31. if ((first & 0b11111000) == 0b11110000) {
  32. return 4;
  33. }
  34. // Everything else is invalid
  35. return 0;
  36. }
  37. extern "C" {
  38. size_t wcslen(const wchar_t* str)
  39. {
  40. size_t len = 0;
  41. while (*(str++))
  42. ++len;
  43. return len;
  44. }
  45. wchar_t* wcscpy(wchar_t* dest, const wchar_t* src)
  46. {
  47. wchar_t* original_dest = dest;
  48. while ((*dest++ = *src++) != '\0')
  49. ;
  50. return original_dest;
  51. }
  52. wchar_t* wcsncpy(wchar_t* dest, const wchar_t* src, size_t num)
  53. {
  54. wchar_t* original_dest = dest;
  55. while (((*dest++ = *src++) != '\0') && ((size_t)(dest - original_dest) < num))
  56. ;
  57. return original_dest;
  58. }
  59. int wcscmp(const wchar_t* s1, const wchar_t* s2)
  60. {
  61. while (*s1 == *s2++)
  62. if (*s1++ == 0)
  63. return 0;
  64. return *(const wchar_t*)s1 - *(const wchar_t*)--s2;
  65. }
  66. int wcsncmp(const wchar_t* s1, const wchar_t* s2, size_t n)
  67. {
  68. if (!n)
  69. return 0;
  70. do {
  71. if (*s1 != *s2++)
  72. return *(const wchar_t*)s1 - *(const wchar_t*)--s2;
  73. if (*s1++ == 0)
  74. break;
  75. } while (--n);
  76. return 0;
  77. }
  78. wchar_t* wcschr(const wchar_t* str, int c)
  79. {
  80. wchar_t ch = c;
  81. for (;; ++str) {
  82. if (*str == ch)
  83. return const_cast<wchar_t*>(str);
  84. if (!*str)
  85. return nullptr;
  86. }
  87. }
  88. wchar_t* wcsrchr(const wchar_t* str, wchar_t wc)
  89. {
  90. wchar_t* last = nullptr;
  91. wchar_t c;
  92. for (; (c = *str); ++str) {
  93. if (c == wc)
  94. last = const_cast<wchar_t*>(str);
  95. }
  96. return last;
  97. }
  98. wchar_t* wcscat(wchar_t* dest, const wchar_t* src)
  99. {
  100. size_t dest_length = wcslen(dest);
  101. size_t i;
  102. for (i = 0; src[i] != '\0'; i++)
  103. dest[dest_length + i] = src[i];
  104. dest[dest_length + i] = '\0';
  105. return dest;
  106. }
  107. wchar_t* wcsncat(wchar_t* dest, const wchar_t* src, size_t n)
  108. {
  109. size_t dest_length = wcslen(dest);
  110. size_t i;
  111. for (i = 0; i < n && src[i] != '\0'; i++)
  112. dest[dest_length + i] = src[i];
  113. dest[dest_length + i] = '\0';
  114. return dest;
  115. }
  116. wchar_t* wcstok(wchar_t* str, const wchar_t* delim, wchar_t** ptr)
  117. {
  118. wchar_t* used_str = str;
  119. if (!used_str) {
  120. used_str = *ptr;
  121. }
  122. size_t token_start = 0;
  123. size_t token_end = 0;
  124. size_t str_len = wcslen(used_str);
  125. size_t delim_len = wcslen(delim);
  126. for (size_t i = 0; i < str_len; ++i) {
  127. bool is_proper_delim = false;
  128. for (size_t j = 0; j < delim_len; ++j) {
  129. if (used_str[i] == delim[j]) {
  130. // Skip beginning delimiters
  131. if (token_end - token_start == 0) {
  132. ++token_start;
  133. break;
  134. }
  135. is_proper_delim = true;
  136. }
  137. }
  138. ++token_end;
  139. if (is_proper_delim && token_end > 0) {
  140. --token_end;
  141. break;
  142. }
  143. }
  144. if (used_str[token_start] == '\0')
  145. return nullptr;
  146. if (token_end == 0) {
  147. return &used_str[token_start];
  148. }
  149. used_str[token_end] = '\0';
  150. return &used_str[token_start];
  151. }
  152. long wcstol(const wchar_t*, wchar_t**, int)
  153. {
  154. dbgln("FIXME: Implement wcstol()");
  155. TODO();
  156. }
  157. long long wcstoll(const wchar_t*, wchar_t**, int)
  158. {
  159. dbgln("FIXME: Implement wcstoll()");
  160. TODO();
  161. }
  162. wint_t btowc(int c)
  163. {
  164. if (c == EOF) {
  165. return WEOF;
  166. }
  167. // Multi-byte sequences in UTF-8 have their highest bit set
  168. if (c & (1 << 7)) {
  169. return WEOF;
  170. }
  171. return c;
  172. }
  173. size_t mbrtowc(wchar_t* pwc, const char* s, size_t n, mbstate_t* state)
  174. {
  175. static mbstate_t _anonymous_state = {};
  176. if (state == nullptr) {
  177. state = &_anonymous_state;
  178. }
  179. // s being a null pointer is a shorthand for reading a single null byte.
  180. if (s == nullptr) {
  181. pwc = nullptr;
  182. s = "";
  183. n = 1;
  184. }
  185. // Stop early if we can't read anything
  186. if (n == 0) {
  187. return 0;
  188. }
  189. size_t consumed_bytes = 0;
  190. // Fill the first byte if we haven't done that yet
  191. if (state->stored_bytes == 0) {
  192. state->bytes[state->stored_bytes++] = s[0];
  193. consumed_bytes++;
  194. }
  195. size_t expected_bytes = mbstate_expected_bytes(state);
  196. // Check if the first byte is invalid
  197. if (expected_bytes == 0) {
  198. *state = {};
  199. errno = EILSEQ;
  200. return -1;
  201. }
  202. while (state->stored_bytes < expected_bytes) {
  203. if (consumed_bytes == n) {
  204. // No complete multibyte character
  205. return -2;
  206. }
  207. unsigned char c = s[consumed_bytes];
  208. // Continuation bytes have to start with 0b10xxxxxx
  209. if ((c & 0b11000000) != 0b10000000) {
  210. // Invalid multibyte character
  211. *state = {};
  212. errno = EILSEQ;
  213. return -1;
  214. }
  215. state->bytes[state->stored_bytes++] = c;
  216. consumed_bytes++;
  217. }
  218. wchar_t codepoint = state->bytes[0];
  219. // Mask out the "length" bits if necessary
  220. if (expected_bytes > 1) {
  221. codepoint &= (1 << (7 - expected_bytes)) - 1;
  222. }
  223. for (unsigned int i = 1; i < expected_bytes; i++) {
  224. // Each continuation byte contains 6 bits of data
  225. codepoint = codepoint << 6;
  226. codepoint |= state->bytes[i] & 0b111111;
  227. }
  228. if (pwc) {
  229. *pwc = codepoint;
  230. }
  231. // We want to read the next multibyte character, but keep all other properties.
  232. state->stored_bytes = 0;
  233. if (codepoint == 0) {
  234. *state = {};
  235. return 0;
  236. }
  237. return consumed_bytes;
  238. }
  239. size_t mbrlen(const char*, size_t, mbstate_t*)
  240. {
  241. dbgln("FIXME: Implement mbrlen()");
  242. TODO();
  243. }
  244. size_t wcrtomb(char* s, wchar_t wc, mbstate_t*)
  245. {
  246. if (s == nullptr)
  247. wc = L'\0';
  248. auto nwritten = AK::UnicodeUtils::code_point_to_utf8(wc, [&s](char byte) {
  249. if (s != nullptr)
  250. *s++ = byte;
  251. });
  252. if (nwritten < 0) {
  253. errno = EILSEQ;
  254. return (size_t)-1;
  255. } else {
  256. return nwritten;
  257. }
  258. }
  259. int wcscoll(const wchar_t* ws1, const wchar_t* ws2)
  260. {
  261. // TODO: Actually implement a sensible sort order for this,
  262. // because right now we are doing what LC_COLLATE=C would do.
  263. return wcscmp(ws1, ws2);
  264. }
  265. int wctob(wint_t c)
  266. {
  267. if (c > 0x7f)
  268. return EOF;
  269. return static_cast<unsigned char>(c);
  270. }
  271. int mbsinit(const mbstate_t* state)
  272. {
  273. if (!state) {
  274. return 1;
  275. }
  276. if (state->stored_bytes != 0) {
  277. return 0;
  278. }
  279. return 1;
  280. }
  281. wchar_t* wcspbrk(const wchar_t* wcs, const wchar_t* accept)
  282. {
  283. for (const wchar_t* cur = accept; *cur; cur++) {
  284. wchar_t* res = wcschr(wcs, *cur);
  285. if (res)
  286. return res;
  287. }
  288. return nullptr;
  289. }
  290. wchar_t* wcsstr(const wchar_t* haystack, const wchar_t* needle)
  291. {
  292. size_t nlen = wcslen(needle);
  293. if (nlen == 0)
  294. return const_cast<wchar_t*>(haystack);
  295. size_t hlen = wcslen(haystack);
  296. while (hlen >= nlen) {
  297. if (wcsncmp(haystack, needle, nlen) == 0)
  298. return const_cast<wchar_t*>(haystack);
  299. haystack++;
  300. hlen--;
  301. }
  302. return nullptr;
  303. }
  304. wchar_t* wmemchr(const wchar_t* s, wchar_t c, size_t n)
  305. {
  306. for (size_t i = 0; i < n; i++) {
  307. if (s[i] == c)
  308. return const_cast<wchar_t*>(&s[i]);
  309. }
  310. return nullptr;
  311. }
  312. wchar_t* wmemcpy(wchar_t* dest, const wchar_t* src, size_t n)
  313. {
  314. for (size_t i = 0; i < n; i++)
  315. dest[i] = src[i];
  316. return dest;
  317. }
  318. wchar_t* wmemset(wchar_t* wcs, wchar_t wc, size_t n)
  319. {
  320. for (size_t i = 0; i < n; i++) {
  321. wcs[i] = wc;
  322. }
  323. return wcs;
  324. }
  325. wchar_t* wmemmove(wchar_t* dest, const wchar_t* src, size_t n)
  326. {
  327. if (dest > src) {
  328. for (size_t i = 1; i <= n; i++) {
  329. dest[n - i] = src[n - i];
  330. }
  331. } else if (dest < src) {
  332. for (size_t i = 0; i < n; i++) {
  333. dest[i] = src[i];
  334. }
  335. }
  336. return dest;
  337. }
  338. unsigned long wcstoul(const wchar_t*, wchar_t**, int)
  339. {
  340. dbgln("TODO: Implement wcstoul()");
  341. TODO();
  342. }
  343. unsigned long long wcstoull(const wchar_t*, wchar_t**, int)
  344. {
  345. dbgln("TODO: Implement wcstoull()");
  346. TODO();
  347. }
  348. float wcstof(const wchar_t*, wchar_t**)
  349. {
  350. dbgln("TODO: Implement wcstof()");
  351. TODO();
  352. }
  353. double wcstod(const wchar_t*, wchar_t**)
  354. {
  355. dbgln("TODO: Implement wcstod()");
  356. TODO();
  357. }
  358. long double wcstold(const wchar_t*, wchar_t**)
  359. {
  360. dbgln("TODO: Implement wcstold()");
  361. TODO();
  362. }
  363. int swprintf(wchar_t*, size_t, const wchar_t*, ...)
  364. {
  365. dbgln("TODO: Implement swprintf()");
  366. TODO();
  367. }
  368. int wcwidth(wchar_t wc)
  369. {
  370. if (wc == L'\0')
  371. return 0;
  372. // Printable ASCII.
  373. if (wc >= 0x20 && wc <= 0x7e)
  374. return 1;
  375. // Non-printable ASCII.
  376. if (wc <= 0x7f)
  377. return -1;
  378. // TODO: Implement wcwidth for non-ASCII characters.
  379. return 1;
  380. }
  381. size_t wcsrtombs(char* dest, const wchar_t** src, size_t len, mbstate_t* ps)
  382. {
  383. static mbstate_t _anonymous_state = {};
  384. if (ps == nullptr)
  385. ps = &_anonymous_state;
  386. size_t written = 0;
  387. while (true) {
  388. size_t ret = 0;
  389. char buf[MB_LEN_MAX];
  390. // Convert next wchar to multibyte.
  391. ret = wcrtomb(buf, **src, ps);
  392. // wchar can't be represented as multibyte.
  393. if (ret == (size_t)-1) {
  394. errno = EILSEQ;
  395. return (size_t)-1;
  396. }
  397. // New bytes don't fit the buffer.
  398. if (dest && len < written + ret) {
  399. return written;
  400. }
  401. if (dest) {
  402. memcpy(dest, buf, ret);
  403. dest += ret;
  404. }
  405. // Null character has been reached
  406. if (**src == L'\0') {
  407. *src = nullptr;
  408. return written;
  409. }
  410. *src += 1;
  411. written += ret;
  412. }
  413. }
  414. size_t mbsrtowcs(wchar_t* dst, const char** src, size_t len, mbstate_t* ps)
  415. {
  416. static mbstate_t _anonymous_state = {};
  417. if (ps == nullptr)
  418. ps = &_anonymous_state;
  419. size_t written = 0;
  420. while (written < len || !dst) {
  421. // Convert next multibyte to wchar.
  422. size_t ret = mbrtowc(dst, *src, MB_LEN_MAX, ps);
  423. // Multibyte sequence is invalid.
  424. if (ret == -1ul) {
  425. errno = EILSEQ;
  426. return (size_t)-1;
  427. }
  428. // Null byte has been reached.
  429. if (**src == '\0') {
  430. *src = nullptr;
  431. return written;
  432. }
  433. *src += ret;
  434. written += 1;
  435. if (dst)
  436. dst += 1;
  437. }
  438. // If we are here, we have written `len` wchars, but not reached the null byte.
  439. return written;
  440. }
  441. int wmemcmp(const wchar_t* s1, const wchar_t* s2, size_t n)
  442. {
  443. while (n-- > 0) {
  444. if (*s1++ != *s2++)
  445. return s1[-1] < s2[-1] ? -1 : 1;
  446. }
  447. return 0;
  448. }
  449. size_t wcsnrtombs(char*, const wchar_t**, size_t, size_t, mbstate_t*)
  450. {
  451. dbgln("FIXME: Implement wcsnrtombs()");
  452. TODO();
  453. }
  454. size_t mbsnrtowcs(wchar_t*, const char**, size_t, size_t, mbstate_t*)
  455. {
  456. dbgln("FIXME: Implement mbsnrtowcs()");
  457. TODO();
  458. }
  459. }