| #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; |
| } |