Update copyright in libstdc++-v3.
[official-gcc.git] / libstdc++-v3 / config / locale / generic / codecvt_members.cc
blobd94cb1de3d907ba39481e313c2861baaa3a85b02
1 // std::codecvt implementation details, generic version -*- C++ -*-
3 // Copyright (C) 2002-2013 Free Software Foundation, Inc.
4 //
5 // This file is part of the GNU ISO C++ Library. This library is free
6 // software; you can redistribute it and/or modify it under the
7 // terms of the GNU General Public License as published by the
8 // Free Software Foundation; either version 3, or (at your option)
9 // any later version.
11 // This library is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 // GNU General Public License for more details.
16 // Under Section 7 of GPL version 3, you are granted additional
17 // permissions described in the GCC Runtime Library Exception, version
18 // 3.1, as published by the Free Software Foundation.
20 // You should have received a copy of the GNU General Public License and
21 // a copy of the GCC Runtime Library Exception along with this program;
22 // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
23 // <http://www.gnu.org/licenses/>.
26 // ISO C++ 14882: 22.2.1.5 - Template class codecvt
29 // Written by Benjamin Kosnik <bkoz@redhat.com>
31 #include <locale>
32 #include <cstdlib> // For MB_CUR_MAX
33 #include <climits> // For MB_LEN_MAX
34 #include <cstring>
36 namespace std _GLIBCXX_VISIBILITY(default)
38 _GLIBCXX_BEGIN_NAMESPACE_VERSION
40 // Specializations.
41 #ifdef _GLIBCXX_USE_WCHAR_T
42 codecvt_base::result
43 codecvt<wchar_t, char, mbstate_t>::
44 do_out(state_type& __state, const intern_type* __from,
45 const intern_type* __from_end, const intern_type*& __from_next,
46 extern_type* __to, extern_type* __to_end,
47 extern_type*& __to_next) const
49 result __ret = ok;
50 // The conversion must be done using a temporary destination buffer
51 // since it is not possible to pass the size of the buffer to wcrtomb
52 state_type __tmp_state(__state);
54 // The conversion must be done by calling wcrtomb in a loop rather
55 // than using wcsrtombs because wcsrtombs assumes that the input is
56 // zero-terminated.
58 // Either we can upper bound the total number of external characters to
59 // something smaller than __to_end - __to or the conversion must be done
60 // using a temporary destination buffer since it is not possible to
61 // pass the size of the buffer to wcrtomb
62 if (MB_CUR_MAX * (__from_end - __from) - (__to_end - __to) <= 0)
63 while (__from < __from_end)
65 const size_t __conv = wcrtomb(__to, *__from, &__tmp_state);
66 if (__conv == static_cast<size_t>(-1))
68 __ret = error;
69 break;
71 __state = __tmp_state;
72 __to += __conv;
73 __from++;
75 else
77 extern_type __buf[MB_LEN_MAX];
78 while (__from < __from_end && __to < __to_end)
80 const size_t __conv = wcrtomb(__buf, *__from, &__tmp_state);
81 if (__conv == static_cast<size_t>(-1))
83 __ret = error;
84 break;
86 else if (__conv > static_cast<size_t>(__to_end - __to))
88 __ret = partial;
89 break;
92 memcpy(__to, __buf, __conv);
93 __state = __tmp_state;
94 __to += __conv;
95 __from++;
99 if (__ret == ok && __from < __from_end)
100 __ret = partial;
102 __from_next = __from;
103 __to_next = __to;
104 return __ret;
107 codecvt_base::result
108 codecvt<wchar_t, char, mbstate_t>::
109 do_in(state_type& __state, const extern_type* __from,
110 const extern_type* __from_end, const extern_type*& __from_next,
111 intern_type* __to, intern_type* __to_end,
112 intern_type*& __to_next) const
114 result __ret = ok;
115 // This temporary state object is neccessary so __state won't be modified
116 // if [__from, __from_end) is a partial multibyte character.
117 state_type __tmp_state(__state);
119 // Conversion must be done by calling mbrtowc in a loop rather than
120 // by calling mbsrtowcs because mbsrtowcs assumes that the input
121 // sequence is zero-terminated.
122 while (__from < __from_end && __to < __to_end)
124 size_t __conv = mbrtowc(__to, __from, __from_end - __from,
125 &__tmp_state);
126 if (__conv == static_cast<size_t>(-1))
128 __ret = error;
129 break;
131 else if (__conv == static_cast<size_t>(-2))
133 // It is unclear what to return in this case (see DR 382).
134 __ret = partial;
135 break;
137 else if (__conv == 0)
139 // XXX Probably wrong for stateful encodings
140 __conv = 1;
141 *__to = L'\0';
144 __state = __tmp_state;
145 __to++;
146 __from += __conv;
149 // It is not clear that __from < __from_end implies __ret != ok
150 // (see DR 382).
151 if (__ret == ok && __from < __from_end)
152 __ret = partial;
154 __from_next = __from;
155 __to_next = __to;
156 return __ret;
159 int
160 codecvt<wchar_t, char, mbstate_t>::
161 do_encoding() const throw()
163 // XXX This implementation assumes that the encoding is
164 // stateless and is either single-byte or variable-width.
165 int __ret = 0;
166 if (MB_CUR_MAX == 1)
167 __ret = 1;
168 return __ret;
171 int
172 codecvt<wchar_t, char, mbstate_t>::
173 do_max_length() const throw()
175 // XXX Probably wrong for stateful encodings.
176 int __ret = MB_CUR_MAX;
177 return __ret;
180 int
181 codecvt<wchar_t, char, mbstate_t>::
182 do_length(state_type& __state, const extern_type* __from,
183 const extern_type* __end, size_t __max) const
185 int __ret = 0;
186 state_type __tmp_state(__state);
188 while (__from < __end && __max)
190 size_t __conv = mbrtowc(0, __from, __end - __from, &__tmp_state);
191 if (__conv == static_cast<size_t>(-1))
193 // Invalid source character
194 break;
196 else if (__conv == static_cast<size_t>(-2))
198 // Remainder of input does not form a complete destination
199 // character.
200 break;
202 else if (__conv == 0)
204 // XXX Probably wrong for stateful encodings
205 __conv = 1;
208 __state = __tmp_state;
209 __from += __conv;
210 __ret += __conv;
211 __max--;
214 return __ret;
216 #endif
218 _GLIBCXX_END_NAMESPACE_VERSION
219 } // namespace