From a76400f4e68030bf91bc20b39ed585e596e18ea5 Mon Sep 17 00:00:00 2001 From: Jonathan Wakely Date: Sat, 19 May 2018 03:03:42 +0100 Subject: [PATCH] Fix std::codecvt_utf8 for Mingw * src/c++11/codecvt.cc (__codecvt_utf8_base::do_in) [__SIZEOF_WCHAR_T__==2 && __BYTE_ORDER__!=__ORDER_BIG_ENDIAN__]: Set little_endian element in bitmask. * testsuite/22_locale/codecvt/codecvt_utf8/69703.cc: Run all tests. * testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc: New. From-SVN: r260389 --- libstdc++-v3/ChangeLog | 8 +++ libstdc++-v3/src/c++11/codecvt.cc | 7 ++- .../22_locale/codecvt/codecvt_utf8/69703.cc | 5 +- .../codecvt/codecvt_utf8/wchar_t/1.cc | 52 +++++++++++++++++++ 4 files changed, 68 insertions(+), 4 deletions(-) create mode 100644 libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc diff --git a/libstdc++-v3/ChangeLog b/libstdc++-v3/ChangeLog index 092fa7550871..b0d5adf2d1a5 100644 --- a/libstdc++-v3/ChangeLog +++ b/libstdc++-v3/ChangeLog @@ -1,3 +1,11 @@ +2018-05-19 Jonathan Wakely + + * src/c++11/codecvt.cc (__codecvt_utf8_base::do_in) + [__SIZEOF_WCHAR_T__==2 && __BYTE_ORDER__!=__ORDER_BIG_ENDIAN__]: Set + little_endian element in bitmask. + * testsuite/22_locale/codecvt/codecvt_utf8/69703.cc: Run all tests. + * testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc: New. + 2018-05-18 François Dumont * include/bits/stl_tree.h diff --git a/libstdc++-v3/src/c++11/codecvt.cc b/libstdc++-v3/src/c++11/codecvt.cc index 259de8077584..3a1a825070cf 100644 --- a/libstdc++-v3/src/c++11/codecvt.cc +++ b/libstdc++-v3/src/c++11/codecvt.cc @@ -1086,7 +1086,12 @@ do_in(state_type&, const extern_type* __from, const extern_type* __from_end, reinterpret_cast(__to), reinterpret_cast(__to_end) }; - auto res = ucs2_in(from, to, _M_maxcode, _M_mode); +#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__ + codecvt_mode mode = {}; +#else + codecvt_mode mode = little_endian; +#endif + auto res = ucs2_in(from, to, _M_maxcode, mode); #elif __SIZEOF_WCHAR_T__ == 4 range to{ reinterpret_cast(__to), diff --git a/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/69703.cc b/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/69703.cc index 36f18a59947b..56872267d1b9 100644 --- a/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/69703.cc +++ b/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/69703.cc @@ -68,7 +68,6 @@ test03() VERIFY( in[2] == U'c' ); } - void test04() { @@ -90,6 +89,6 @@ main() { test01(); test02(); - test01(); - test02(); + test03(); + test04(); } diff --git a/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc b/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc new file mode 100644 index 000000000000..c44f91f357eb --- /dev/null +++ b/libstdc++-v3/testsuite/22_locale/codecvt/codecvt_utf8/wchar_t/1.cc @@ -0,0 +1,52 @@ +// Copyright (C) 2018 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 3, 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 COPYING3. If not see +// . + +// { dg-do run { target c++11 } } + +#include +#include +#include + +void +test01() +{ + const char out[] = u8"\u00A33.50"; + wchar_t in[8] = {}; + std::codecvt_utf8 cvt; + std::mbstate_t st; + const char* no; + wchar_t* ni; + auto res = cvt.in(st, out, out+6, no, in, in+8, ni); + VERIFY( res == std::codecvt_base::ok ); + VERIFY( in[1] == L'3' ); + VERIFY( in[2] == L'.' ); + VERIFY( in[3] == L'5' ); + VERIFY( in[4] == L'0' ); + + char out2[8] = {}; + char* no2; + const wchar_t* ni2; + res = cvt.out(st, in, ni, ni2, out2, out2+8, no2); + VERIFY( res == std::codecvt_base::ok ); + VERIFY( out2 == std::string(out) ); +} + +int +main() +{ + test01(); +}