implement a private state for the uchar.h functions
authorJens Gustedt <Jens.Gustedt@inria.fr>
Sun, 9 Nov 2014 10:18:08 +0000 (11:18 +0100)
committerRich Felker <dalias@aerifal.cx>
Sat, 15 Nov 2014 20:08:03 +0000 (15:08 -0500)
The C standard is imperative on that:

  7.28.1 ... If ps is a null pointer, each function uses its own internal
  mbstate_t object instead, which is initialized at program startup to
  the initial conversion state;

and these functions are also not supposed to implicitly use the state of
the wchar.h functions:

  7.29.6.3 ... The implementation behaves as if no library function calls
  these functions with a null pointer for ps.

Previously this resulted in two bugs.

 - The functions c16rtomb and mbrtoc16 would crash when called with ps
   set to null.

 - The function mbrtoc32 used the private state of mbrtowc, which it
   is not allowed to do.

src/multibyte/c16rtomb.c
src/multibyte/mbrtoc16.c
src/multibyte/mbrtoc32.c

index 2e8ec97015d70153e67b5ba05e70ed95bf34be11..39ca3758fa16eb16d12f468046045f0f425e5d1c 100644 (file)
@@ -4,6 +4,8 @@
 
 size_t c16rtomb(char *restrict s, char16_t c16, mbstate_t *restrict ps)
 {
+       static unsigned internal_state;
+       if (!ps) ps = (void *)&internal_state;
        unsigned *x = (unsigned *)ps;
        wchar_t wc;
 
index 74b7d77eee1361db07e70bc857ce165f08230083..765ff9037cd3554db14e275650adb7c807632777 100644 (file)
@@ -3,6 +3,8 @@
 
 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);
index c6d2082419e452db5acb64dbc27426c2550e5baf..9b6b2367398e426815dc8f0390daae3c6957779d 100644 (file)
@@ -3,6 +3,8 @@
 
 size_t mbrtoc32(char32_t *restrict pc32, const char *restrict s, size_t n, mbstate_t *restrict ps)
 {
+       static unsigned internal_state;
+       if (!ps) ps = (void *)&internal_state;
        if (!s) return mbrtoc32(0, "", 1, ps);
        wchar_t wc;
        size_t ret = mbrtowc(&wc, s, n, ps);