aboutsummaryrefslogtreecommitdiff
path: root/winsup/mingw
diff options
context:
space:
mode:
authorDanny Smith <dannysmith@users.sourceforge.net>2005-04-24 11:30:27 +0000
committerDanny Smith <dannysmith@users.sourceforge.net>2005-04-24 11:30:27 +0000
commit6fbeb6a3f07e9c5c62966e8bf2a4ecfd367f23a9 (patch)
treeed637eb4a0126508546e5c4100176eb1123ec931 /winsup/mingw
parent4fc953d6a1448be843847bfbd29e022d78134eee (diff)
downloadnewlib-6fbeb6a3f07e9c5c62966e8bf2a4ecfd367f23a9.zip
newlib-6fbeb6a3f07e9c5c62966e8bf2a4ecfd367f23a9.tar.gz
newlib-6fbeb6a3f07e9c5c62966e8bf2a4ecfd367f23a9.tar.bz2
* mingwex/mbrtowc.c: New file.
* mingwex/wcrtomb.c: New file. * mingwex/btowc.c: New file. * mingwex/wctob.c: New file. * mingwex/mb_wc_common.h: New file. * mingwex/Makefile.in (DISTFILES): Add new files. (Q8_OBJS): Add new objects. * include/wchar.h: Adjust comment about mbrtowc() and related funcions. Add __restrict__ to pointer params in prototypes. (wmemset. wmemchr, wmemcpy, wmemmove, wcstoll, wcstoull): Remove arg names from protototypes.
Diffstat (limited to 'winsup/mingw')
-rw-r--r--winsup/mingw/ChangeLog19
-rw-r--r--winsup/mingw/include/wchar.h48
-rw-r--r--winsup/mingw/mingwex/Makefile.in7
-rwxr-xr-xwinsup/mingw/mingwex/btowc.c19
-rwxr-xr-xwinsup/mingw/mingwex/mb_wc_common.h18
-rwxr-xr-xwinsup/mingw/mingwex/mbrtowc.c164
-rwxr-xr-xwinsup/mingw/mingwex/wcrtomb.c94
-rwxr-xr-xwinsup/mingw/mingwex/wctob.c21
8 files changed, 364 insertions, 26 deletions
diff --git a/winsup/mingw/ChangeLog b/winsup/mingw/ChangeLog
index 093b09d..7a55658 100644
--- a/winsup/mingw/ChangeLog
+++ b/winsup/mingw/ChangeLog
@@ -1,8 +1,21 @@
-2005-04-23 Wu Yongwei <adah@sh163.net>
+2005-04-23 Danny Smith <dannysmith@users.sourceforge.net>
- mingwex/dirent.c: Formatting changes.
+ * mingwex/mbrtowc.c: New file.
+ * mingwex/wcrtomb.c: New file.
+ * mingwex/btowc.c: New file.
+ * mingwex/wctob.c: New file.
+ * mingwex/mb_wc_common.h: New file.
+ * mingwex/Makefile.in (DISTFILES): Add new files.
+ (Q8_OBJS): Add new objects.
+ * include/wchar.h: Adjust comment about mbrtowc() and related
+ funcions. Add __restrict__ to pointer params in prototypes.
+ (wmemset. wmemchr, wmemcpy, wmemmove, wcstoll, wcstoull): Remove
+ arg names from protototypes.
+
+2005-04-23 Wu Yongwei <adah@sh163.net>
- mingwex/dirent.c (_topendir): Make the end-of-path slash check
+ * mingwex/dirent.c: Formatting changes.
+ * mingwex/dirent.c (_topendir): Make the end-of-path slash check
MBCS-safe.
2005-03-31 Danny Smith <dannysmith@users.sourceforge.net>
diff --git a/winsup/mingw/include/wchar.h b/winsup/mingw/include/wchar.h
index 125160a..54dc17f 100644
--- a/winsup/mingw/include/wchar.h
+++ b/winsup/mingw/include/wchar.h
@@ -254,9 +254,11 @@ _CRTIMP wchar_t* __cdecl wcsupr (wchar_t*);
#define _WSTRING_DEFINED
#endif /* _WSTRING_DEFINED */
-/* These are resolved by -lmsvcp60 */
-/* If you don't have msvcp60.dll in your windows system directory, you can
- easily obtain it with a search from your favorite search engine. */
+/* These are resolved by -lmingwex. Alternatively, they can be resolved by
+ adding -lmsvcp60 to your command line, which will give you the VC++
+ versions of these functions. If you want the latter and don't have
+ msvcp60.dll in your windows system directory, you can easily obtain
+ it with a search from your favorite search engine. */
#ifndef __STRICT_ANSI__
typedef wchar_t _Wint_t;
#endif
@@ -264,29 +266,35 @@ typedef wchar_t _Wint_t;
typedef int mbstate_t;
wint_t __cdecl btowc(int);
-size_t __cdecl mbrlen(const char *, size_t, mbstate_t *);
-size_t __cdecl mbrtowc(wchar_t *, const char *, size_t, mbstate_t *);
-size_t __cdecl mbsrtowcs(wchar_t *, const char **, size_t, mbstate_t *);
-
-size_t __cdecl wcrtomb(char *, wchar_t, mbstate_t *);
-size_t __cdecl wcsrtombs(char *, const wchar_t **, size_t, mbstate_t *);
+size_t __cdecl mbrlen(const char * __restrict__, size_t,
+ mbstate_t * __restrict__);
+size_t __cdecl mbrtowc(wchar_t * __restrict__, const char * __restrict__,
+ size_t, mbstate_t * __restrict__);
+size_t __cdecl mbsrtowcs(wchar_t * __restrict__, const char ** __restrict__,
+ size_t, mbstate_t * __restrict__);
+size_t __cdecl wcrtomb(char * __restrict__, wchar_t,
+ mbstate_t * __restrict__);
+size_t __cdecl wcsrtombs(char * __restrict__, const wchar_t ** __restrict__,
+ size_t, mbstate_t * __restrict__);
int __cdecl wctob(wint_t);
#ifndef __NO_ISOCEXT /* these need static lib libmingwex.a */
-__CRT_INLINE int __cdecl fwide(FILE* __UNUSED_PARAM(stream), int __UNUSED_PARAM(mode))
+__CRT_INLINE int __cdecl fwide(FILE* __UNUSED_PARAM(stream),
+ int __UNUSED_PARAM(mode))
{return -1;} /* limited to byte orientation */
__CRT_INLINE int __cdecl mbsinit(const mbstate_t* __UNUSED_PARAM(ps))
{return 1;}
-wchar_t* __cdecl wmemset(wchar_t* s, wchar_t c, size_t n);
-wchar_t* __cdecl wmemchr(const wchar_t* s, wchar_t c, size_t n);
-int wmemcmp(const wchar_t* s1, const wchar_t * s2, size_t n);
-wchar_t* __cdecl wmemcpy(wchar_t* __restrict__ s1, const wchar_t* __restrict__ s2,
- size_t n);
-wchar_t* __cdecl wmemmove(wchar_t* s1, const wchar_t* s2, size_t n);
-long long __cdecl wcstoll(const wchar_t* __restrict__ nptr,
- wchar_t** __restrict__ endptr, int base);
-unsigned long long __cdecl wcstoull(const wchar_t* __restrict__ nptr,
- wchar_t ** __restrict__ endptr, int base);
+wchar_t* __cdecl wmemset(wchar_t *, wchar_t, size_t);
+wchar_t* __cdecl wmemchr(const wchar_t*, wchar_t, size_t);
+int wmemcmp(const wchar_t*, const wchar_t *, size_t);
+wchar_t* __cdecl wmemcpy(wchar_t* __restrict__,
+ const wchar_t* __restrict__,
+ size_t);
+wchar_t* __cdecl wmemmove(wchar_t* s1, const wchar_t *, size_t);
+long long __cdecl wcstoll(const wchar_t * __restrict__,
+ wchar_t** __restrict__, int);
+unsigned long long __cdecl wcstoull(const wchar_t * __restrict__,
+ wchar_t ** __restrict__, int);
#endif /* __NO_ISOCEXT */
#ifndef __STRICT_ANSI__
diff --git a/winsup/mingw/mingwex/Makefile.in b/winsup/mingw/mingwex/Makefile.in
index 5cfde76..8876cf9 100644
--- a/winsup/mingw/mingwex/Makefile.in
+++ b/winsup/mingw/mingwex/Makefile.in
@@ -34,7 +34,8 @@ DISTFILES = Makefile.in configure configure.in \
mingw-fseek.c sitest.c strtof.c strtoimax.c strtold.c strtoumax.c \
testwmem.c tst-aligned-malloc.c ulltoa.c ulltow.c wcstof.c \
wcstoimax.c wcstold.c wcstoumax.c wctrans.c wctype.c \
- wdirent.c wmemchr.c wmemcmp.c wmemcpy.c wmemmove.c wmemset.c wtoll.c
+ wdirent.c wmemchr.c wmemcmp.c wmemcpy.c wmemmove.c wmemset.c wtoll.c \
+ wcrtomb.c wctob.c mbrtowc.c btowc.c mb_wc_common.h
MATH_DISTFILES = \
acosf.c acosl.c asinf.c asinl.c atan2f.c atan2l.c \
atanf.c atanl.c cbrt.c cbrtf.c cbrtl.c ceilf.S ceill.S \
@@ -114,7 +115,7 @@ Q8_OBJS = \
fwide.o imaxabs.o imaxdiv.o mbsinit.o \
strtoimax.o strtoumax.o wcstoimax.o wcstoumax.o \
wmemchr.o wmemcmp.o wmemcpy.o wmemmove.o wmemset.o \
- wctrans.o wctype.o
+ wctrans.o wctype.o wcrtomb.o wctob.o mbrtowc.o btowc.o
STDLIB_OBJS = \
strtold.o wcstold.o
STDLIB_STUB_OBJS = \
@@ -232,7 +233,7 @@ wcstold.o: $(srcdir)/wcstold.c $(srcdir)/math/cephes_emath.h
acosh.o acoshf.o acoshl.o \
asinh.o asinhf.o asinhl.o \
atanh.o atanhf.o atanhl.o: fastmath.h
-
+mbrtowc.o wcrtomb.o: mb_wc_common.h
dist:
mkdir $(distdir)/mingwex
diff --git a/winsup/mingw/mingwex/btowc.c b/winsup/mingw/mingwex/btowc.c
new file mode 100755
index 0000000..b369190
--- /dev/null
+++ b/winsup/mingw/mingwex/btowc.c
@@ -0,0 +1,19 @@
+#include "mb_wc_common.h"
+#include <wchar.h>
+#include <stdio.h>
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+
+wint_t btowc (int c)
+{
+ if (c == EOF)
+ return (WEOF);
+ else
+ {
+ unsigned char ch = c;
+ wchar_t wc = WEOF;
+ MultiByteToWideChar (get_cp_from_locale(), MB_ERR_INVALID_CHARS,
+ (char*)&ch, 1, &wc, 1);
+ return wc;
+ }
+}
diff --git a/winsup/mingw/mingwex/mb_wc_common.h b/winsup/mingw/mingwex/mb_wc_common.h
new file mode 100755
index 0000000..a4ea81c
--- /dev/null
+++ b/winsup/mingw/mingwex/mb_wc_common.h
@@ -0,0 +1,18 @@
+#include <locale.h>
+#include <string.h>
+#include <stdlib.h>
+
+static inline
+unsigned int get_cp_from_locale (void)
+{
+ char* cp_string;
+ /*
+ locale :: "lang[_country[.code_page]]"
+ | ".code_page"
+
+ */
+
+ if ((cp_string = strchr(setlocale(LC_CTYPE, NULL), '.')))
+ return ((unsigned) atoi (cp_string + 1));
+ return 0;
+}
diff --git a/winsup/mingw/mingwex/mbrtowc.c b/winsup/mingw/mingwex/mbrtowc.c
new file mode 100755
index 0000000..e337417
--- /dev/null
+++ b/winsup/mingw/mingwex/mbrtowc.c
@@ -0,0 +1,164 @@
+#include "mb_wc_common.h"
+#include <wchar.h>
+#include <stdlib.h>
+#include <errno.h>
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+
+#define MBTOWC_FLAGS (MB_PRECOMPOSED | MB_ERR_INVALID_CHARS)
+/* Attribute `nonnull' was valid as of gcc 3.3. */
+#ifndef ATTRIBUTE_NONNULL
+# if (__GNUC__ > 3 ||( __GNUC__ == 3 && __GNUC_MINOR__ >= 3))
+# define ATTRIBUTE_NONNULL(m...) __attribute__ ((__nonnull__ (m)))
+# else
+# define ATTRIBUTE_NONNULL(m...)
+# endif /* GNUC >= 3.3 */
+#endif /* ATTRIBUTE_NONNULL */
+
+static int ATTRIBUTE_NONNULL(1, 4)
+__mbrtowc_cp (wchar_t * __restrict__ pwc, const char * __restrict__ s,
+ size_t n, mbstate_t* __restrict__ ps,
+ const unsigned int cp, const unsigned int mb_max)
+{
+ union {
+ mbstate_t val;
+ char mbcs[4];
+ } shift_state;
+
+
+ /* Do the prelim checks */
+ if (s == NULL)
+ return 0;
+
+ if (n == 0)
+ /* The standard doesn't mention this case explicitly. Tell
+ caller that the conversion from a non-null s is incomplete. */
+ return -2;
+
+ /* Save the current shift state, in case we need it in DBCS case. */
+ shift_state.val = *ps;
+ *ps = 0;
+
+ if (!*s)
+ {
+ *pwc = 0;
+ return 0;
+ }
+
+ if (mb_max > 1)
+ {
+ if (shift_state.mbcs[0] != 0)
+ {
+ /* Complete the mb char with the trailing byte. */
+ shift_state.mbcs[1] = *s; /* the second byte */
+ if (MultiByteToWideChar(cp, MB_ERR_INVALID_CHARS,
+ shift_state.mbcs, 2, pwc, 1)
+ == 0)
+ {
+ /* An invalid trailing byte */
+ errno = EILSEQ;
+ return -1;
+ }
+ return 2;
+ }
+ else if (IsDBCSLeadByteEx (cp, *s))
+ {
+ /* If told to translate one byte, just save the leadbyte
+ in *ps. */
+ if (n < 2)
+ {
+ ((char*) ps)[0] = *s;
+ return -2;
+ }
+ /* Else translate the first two bytes */
+ else if (MultiByteToWideChar (cp, MB_ERR_INVALID_CHARS,
+ s, 2, pwc, 1)
+ == 0)
+ {
+ errno = EILSEQ;
+ return -1;
+ }
+ return 2;
+ }
+ }
+
+ /* Fall through to single byte char */
+ if (cp == 0)
+ *pwc = (wchar_t)(unsigned char)*s;
+
+ else if (MultiByteToWideChar (cp, MB_ERR_INVALID_CHARS, s, 1, pwc, 1)
+ == 0)
+ {
+ errno = EILSEQ;
+ return -1;
+ }
+ return 1;
+}
+
+size_t
+mbrtowc (wchar_t * __restrict__ pwc, const char * __restrict__ s,
+ size_t n, mbstate_t* __restrict__ ps)
+{
+ static mbstate_t internal_mbstate = 0;
+ wchar_t byte_bucket = 0;
+ wchar_t* dst = pwc ? pwc : &byte_bucket;
+
+ return (size_t) __mbrtowc_cp (dst, s, n, ps ? ps : &internal_mbstate,
+ get_cp_from_locale(), MB_CUR_MAX);
+}
+
+
+size_t
+mbsrtowcs (wchar_t* __restrict__ dst, const char ** __restrict__ src,
+ size_t len, mbstate_t* __restrict__ ps)
+{
+ int ret =0 ;
+ size_t n = 0;
+ static mbstate_t internal_mbstate = 0;
+ mbstate_t* internal_ps = ps ? ps : &internal_mbstate;
+ const unsigned int cp = get_cp_from_locale();;
+ const unsigned int mb_max = MB_CUR_MAX;
+
+ if ( src == NULL || *src == NULL ) /* undefined behavior */
+ return 0;
+
+ if (dst != NULL)
+ {
+ while (n < len
+ && (ret = __mbrtowc_cp(dst, *src, len - n,
+ internal_ps, cp, mb_max))
+ > 0)
+ {
+ ++dst;
+ *src += ret;
+ n += ret;
+ }
+
+ if (n < len && ret == 0)
+ *src = (char *)NULL;
+ }
+
+ else
+ {
+ wchar_t byte_bucket = 0;
+ while (n < len
+ && (ret = __mbrtowc_cp (&byte_bucket, *src, mb_max,
+ internal_ps, cp, mb_max))
+ > 0)
+ {
+ *src += ret;
+ n += ret;
+ }
+ }
+ return n;
+}
+
+size_t
+mbrlen (const char * __restrict__ s, size_t n,
+ mbstate_t * __restrict__ ps)
+{
+ static mbstate_t s_mbstate = 0;
+ wchar_t byte_bucket = 0;
+ return __mbrtowc_cp (&byte_bucket, s, n, (ps) ? ps : &s_mbstate,
+ get_cp_from_locale(), MB_CUR_MAX);
+}
diff --git a/winsup/mingw/mingwex/wcrtomb.c b/winsup/mingw/mingwex/wcrtomb.c
new file mode 100755
index 0000000..08a5fcc
--- /dev/null
+++ b/winsup/mingw/mingwex/wcrtomb.c
@@ -0,0 +1,94 @@
+#include "mb_wc_common.h"
+#include <wchar.h>
+#include <stdlib.h>
+#include <errno.h>
+#include <limits.h>
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+
+
+static int __MINGW_ATTRIB_NONNULL(1)
+ __wcrtomb_cp (char *dst, wchar_t wc, const unsigned int cp,
+ const unsigned int mb_max)
+{
+ if (wc > 255)
+ {
+ errno = EILSEQ;
+ return -1;
+ }
+
+ if (cp == 0)
+ {
+ *dst = (char) wc;
+ return 1;
+ }
+ else
+ {
+ int invalid_char = 0;
+
+ int size = WideCharToMultiByte(get_cp_from_locale(),
+ 0 /* Is this correct flag? */,
+ &wc, 1, dst, mb_max,
+ NULL, &invalid_char);
+ if (size == 0 || invalid_char)
+ {
+ errno = EILSEQ;
+ return -1;
+ }
+ return size;
+ }
+}
+
+size_t
+wcrtomb (char *dst, wchar_t wc, mbstate_t * __UNUSED_PARAM (ps))
+{
+ char byte_bucket [MB_LEN_MAX];
+ char* tmp_dst = dst ? dst : byte_bucket;
+ return (size_t)__wcrtomb_cp (tmp_dst, wc, get_cp_from_locale (),
+ MB_CUR_MAX);
+}
+
+size_t wcsrtombs (char *dst, const wchar_t **src, size_t len,
+ mbstate_t * __UNUSED_PARAM (ps))
+{
+ int ret = 0;
+ size_t n = 0;
+ const unsigned int cp = get_cp_from_locale();
+ const unsigned int mb_max = MB_CUR_MAX;
+
+ if (src == NULL || *src == NULL) /* undefined behavior */
+ return 0;
+
+ if (dst != NULL)
+ {
+ const wchar_t ** saved_src = src;
+ while (n < len)
+ {
+ if ((ret = __wcrtomb_cp (dst, **src, cp, mb_max)) <= 0)
+ return (size_t) -1;
+ n += ret;
+ dst += ret;
+ if (*(dst - 1) == '\0')
+ {
+ *saved_src = (wchar_t) NULL;;
+ return (n - 1);
+ }
+ *src++;
+ }
+ }
+ else
+ {
+ char byte_bucket [MB_LEN_MAX];
+ while (n < len)
+ {
+ if ((ret = __wcrtomb_cp (byte_bucket, **src, cp, mb_max))
+ <= 0)
+ return (size_t) -1;
+ n += ret;
+ if (byte_bucket [ret - 1] == '\0')
+ return (n - 1);
+ *src++;
+ }
+ }
+ return n;
+}
diff --git a/winsup/mingw/mingwex/wctob.c b/winsup/mingw/mingwex/wctob.c
new file mode 100755
index 0000000..2b40fc1
--- /dev/null
+++ b/winsup/mingw/mingwex/wctob.c
@@ -0,0 +1,21 @@
+#include "mb_wc_common.h"
+#include <wchar.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <errno.h>
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+
+/* Return just the first byte after translating to multibyte. */
+int wctob (wint_t wc )
+{
+ wchar_t w = wc;
+ char c;
+ int invalid_char = 0;
+ if (!WideCharToMultiByte (get_cp_from_locale(),
+ 0 /* Is this correct flag? */,
+ &w, 1, &c, 1, NULL, &invalid_char)
+ || invalid_char)
+ return EOF;
+ return (int) c;
+}