diff options
author | paolo <paolo@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-02-15 23:29:52 +0000 |
---|---|---|
committer | paolo <paolo@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-02-15 23:29:52 +0000 |
commit | 0fcc981fa6eb41869dce802e289b012f4c3726be (patch) | |
tree | 36ecbbfafa6758d85faee24364c6147bffcc899a /libstdc++-v3 | |
parent | 3a662a81d9cc11e32f547ab9da65617c6aaac53c (diff) | |
download | gcc-0fcc981fa6eb41869dce802e289b012f4c3726be.tar.gz |
2005-02-15 Paolo Carlini <pcarlini@suse.de>
Jon Grimm <jgrimm2@us.ibm.com>
PR libstdc++/19955
* include/bits/locale_facets.h (ctype<char>::_M_narrow_init()):
Fix the logic setting _M_narrow_ok: first check whether the
transformation is trivial with a dflt == 0, then deal with the
special case of zero.
* testsuite/22_locale/ctype/narrow/char/19955.cc: New.
* include/bits/locale_facets.h (ctype<char>::_M_widen_init()):
Tweak consistently to use memcmp; minor formatting fixes.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@95082 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'libstdc++-v3')
-rw-r--r-- | libstdc++-v3/ChangeLog | 13 | ||||
-rw-r--r-- | libstdc++-v3/include/bits/locale_facets.h | 54 | ||||
-rw-r--r-- | libstdc++-v3/testsuite/22_locale/ctype/narrow/char/19955.cc | 105 |
3 files changed, 143 insertions, 29 deletions
diff --git a/libstdc++-v3/ChangeLog b/libstdc++-v3/ChangeLog index afdac3002dd..0d5155bc367 100644 --- a/libstdc++-v3/ChangeLog +++ b/libstdc++-v3/ChangeLog @@ -1,3 +1,16 @@ +2005-02-15 Paolo Carlini <pcarlini@suse.de> + Jon Grimm <jgrimm2@us.ibm.com> + + PR libstdc++/19955 + * include/bits/locale_facets.h (ctype<char>::_M_narrow_init()): + Fix the logic setting _M_narrow_ok: first check whether the + transformation is trivial with a dflt == 0, then deal with the + special case of zero. + * testsuite/22_locale/ctype/narrow/char/19955.cc: New. + + * include/bits/locale_facets.h (ctype<char>::_M_widen_init()): + Tweak consistently to use memcmp; minor formatting fixes. + 2005-02-15 Jakub Jelinek <jakub@redhat.com> PR libstdc++/19946 diff --git a/libstdc++-v3/include/bits/locale_facets.h b/libstdc++-v3/include/bits/locale_facets.h index 922ee21084a..152170eb0a6 100644 --- a/libstdc++-v3/include/bits/locale_facets.h +++ b/libstdc++-v3/include/bits/locale_facets.h @@ -1,6 +1,6 @@ // Locale support -*- C++ -*- -// Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004 +// Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005 // Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free @@ -690,7 +690,7 @@ namespace std mutable char _M_widen[1 + static_cast<unsigned char>(-1)]; mutable char _M_narrow[1 + static_cast<unsigned char>(-1)]; mutable char _M_narrow_ok; // 0 uninitialized, 1 init, - // 2 non-consecutive + // 2 memcpy can't be used public: /// The facet id for ctype<char> @@ -865,7 +865,8 @@ namespace std char_type widen(char __c) const { - if (_M_widen_ok) return _M_widen[static_cast<unsigned char>(__c)]; + if (_M_widen_ok) + return _M_widen[static_cast<unsigned char>(__c)]; this->_M_widen_init(); return this->do_widen(__c); } @@ -896,7 +897,8 @@ namespace std memcpy(__to, __lo, __hi - __lo); return __hi; } - if (!_M_widen_ok) _M_widen_init(); + if (!_M_widen_ok) + _M_widen_init(); return this->do_widen(__lo, __hi, __to); } @@ -924,7 +926,8 @@ namespace std if (_M_narrow[static_cast<unsigned char>(__c)]) return _M_narrow[static_cast<unsigned char>(__c)]; const char __t = do_narrow(__c, __dfault); - if (__t != __dfault) _M_narrow[static_cast<unsigned char>(__c)] = __t; + if (__t != __dfault) + _M_narrow[static_cast<unsigned char>(__c)] = __t; return __t; } @@ -954,7 +957,7 @@ namespace std narrow(const char_type* __lo, const char_type* __hi, char __dfault, char *__to) const { - if (__builtin_expect(_M_narrow_ok == 1,true)) + if (__builtin_expect(_M_narrow_ok == 1, true)) { memcpy(__to, __lo, __hi - __lo); return __hi; @@ -1161,17 +1164,13 @@ namespace std _M_widen_ok = 1; // Set _M_widen_ok to 2 if memcpy can't be used. - for (size_t __j = 0; __j < sizeof(_M_widen); ++__j) - if (__tmp[__j] != _M_widen[__j]) - { - _M_widen_ok = 2; - break; - } + if (memcmp(__tmp, _M_widen, sizeof(_M_widen))) + _M_widen_ok = 2; } // Fill in the narrowing cache and flag whether all values are - // valid or not. _M_narrow_ok is set to 1 if the whole table is - // narrowed, 2 if only some values could be narrowed. + // valid or not. _M_narrow_ok is set to 2 if memcpy can't + // be used. void _M_narrow_init() const { char __tmp[sizeof(_M_narrow)]; @@ -1179,21 +1178,18 @@ namespace std __tmp[__i] = __i; do_narrow(__tmp, __tmp + sizeof(__tmp), 0, _M_narrow); - // Check if any default values were created. Do this by - // renarrowing with a different default value and comparing. - bool __consecutive = true; - for (size_t __j = 0; __j < sizeof(_M_narrow); ++__j) - if (!_M_narrow[__j]) - { - char __c; - do_narrow(__tmp + __j, __tmp + __j + 1, 1, &__c); - if (__c == 1) - { - __consecutive = false; - break; - } - } - _M_narrow_ok = __consecutive ? 1 : 2; + _M_narrow_ok = 1; + if (memcmp(__tmp, _M_narrow, sizeof(_M_narrow))) + _M_narrow_ok = 2; + else + { + // Deal with the special case of zero: renarrow with a + // different default and compare. + char __c; + do_narrow(__tmp, __tmp + 1, 1, &__c); + if (__c == 1) + _M_narrow_ok = 2; + } } }; diff --git a/libstdc++-v3/testsuite/22_locale/ctype/narrow/char/19955.cc b/libstdc++-v3/testsuite/22_locale/ctype/narrow/char/19955.cc new file mode 100644 index 00000000000..4ce1da33f01 --- /dev/null +++ b/libstdc++-v3/testsuite/22_locale/ctype/narrow/char/19955.cc @@ -0,0 +1,105 @@ +// Copyright (C) 2005 Free Software Foundation, Inc. +// +// This file is part of the GNU ISO C++ Library. This library is free +// software; you can redistribute it and/or modify it under the +// terms of the GNU General Public License as published by the +// Free Software Foundation; either version 2, or (at your option) +// any later version. + +// This library is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. + +// You should have received a copy of the GNU General Public License along +// with this library; see the file COPYING. If not, write to the Free +// Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, +// USA. + +// As a special exception, you may use this file as part of a free software +// library without restriction. Specifically, if other files instantiate +// templates or use macros or inline functions from this file, or you compile +// this file and link it with other files to produce an executable, this +// file does not by itself cause the resulting executable to be covered by +// the GNU General Public License. This exception does not however +// invalidate any other reasons why the executable file might be covered by +// the GNU General Public License. + +// 22.2.1.3.2 ctype<char> members + +#include <locale> +#include <testsuite_hooks.h> + +class Ctype1 +: public std::ctype<char> +{ +protected: + const char* + do_narrow(const char* lo, const char* hi, + char dflt, char* to) const + { + for (int i = 0; lo != hi; ++lo, ++to, ++i) + *to = *lo + i; + return hi; + } +}; + +class Ctype2 +: public std::ctype<char> +{ +protected: + const char* + do_narrow(const char* lo, const char* hi, + char dflt, char* to) const + { + for (int i = 0; lo != hi; ++lo, ++to, ++i) + if (*lo == '\000') + *to = dflt; + else + *to = *lo; + return hi; + } +}; + +// libstdc++/19955 +void test01() +{ + using namespace std; + bool test __attribute__((unused)) = true; + + const char src[] = "abcd"; + + locale mylocale1(locale::classic(), new Ctype1); + const ctype<char>& mc1 = use_facet<ctype<char> >(mylocale1); + + char dst1[sizeof(src)]; + memset(dst1, 0, sizeof(src)); + char dst2[sizeof(src)]; + memset(dst2, 0, sizeof(src)); + + mc1.narrow(src, src + sizeof(src), '*', dst1); + mc1.narrow(src, src + sizeof(src), '*', dst2); + + VERIFY( !memcmp(dst1, "aceg\004", 5) ); + VERIFY( !memcmp(dst1, dst2, 5) ); + + locale mylocale2(locale::classic(), new Ctype2); + const ctype<char>& mc2 = use_facet<ctype<char> >(mylocale2); + + char dst3[sizeof(src)]; + memset(dst3, 0, sizeof(src)); + char dst4[sizeof(src)]; + memset(dst4, 0, sizeof(src)); + + mc2.narrow(src, src + sizeof(src), '*', dst3); + mc2.narrow(src, src + sizeof(src), '*', dst4); + + VERIFY( !memcmp(dst3, "abcd*", 5) ); + VERIFY( !memcmp(dst3, dst4, 5) ); +} + +int main() +{ + test01(); + return 0; +} |