| #include <uchar.h> |
| #include <wchar.h> |
| |
| size_t mbrtoc16(char16_t *restrict pc16, const char *restrict s, size_t n, mbstate_t *restrict ps) |
| { |
| static unsigned internal_state; |
| if (!ps) ps = (void *)&internal_state; |
| unsigned *pending = (unsigned *)ps; |
| |
| if (!s) return mbrtoc16(0, "", 1, ps); |
| |
| /* mbrtowc states for partial UTF-8 characters have the high bit set; |
| * we use nonzero states without high bit for pending surrogates. */ |
| if ((int)*pending > 0) { |
| if (pc16) *pc16 = *pending; |
| *pending = 0; |
| return -3; |
| } |
| |
| wchar_t wc; |
| size_t ret = mbrtowc(&wc, s, n, ps); |
| if (ret <= 4) { |
| if (wc >= 0x10000) { |
| *pending = (wc & 0x3ff) + 0xdc00; |
| wc = 0xd7c0 + (wc >> 10); |
| } |
| if (pc16) *pc16 = wc; |
| } |
| return ret; |
| } |