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