blob: f6dd8ecc9a2ad91dff086bf5768f90498ef5f583 [file] [log] [blame]
/***************************************************************************
*
* _num_get.h - definition of the std::num_get class template
*
* This is an internal header file used to implement the C++ Standard
* Library. It should never be #included directly by a program.
*
* $Id$
*
***************************************************************************
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied. See the License for the specific language governing
* permissions and limitations under the License.
*
* Copyright 2001-2008 Rogue Wave Software, Inc.
*
**************************************************************************/
#ifndef _RWSTD_LOC_NUM_GET_H_INCLUDED
#define _RWSTD_LOC_NUM_GET_H_INCLUDED
#if __GNUG__ >= 3
# pragma GCC system_header
#endif // gcc >= 3
#include <loc/_facet.h>
#include <rw/_ioiter.h>
#include <rw/_iosfwd.h>
#include <rw/_traits.h>
#include <rw/_defs.h>
_RWSTD_NAMESPACE (std) {
// 22.2.2.1
_EXPORT
template <class _CharT, class _InputIter = istreambuf_iterator<_CharT> >
struct num_get: _RW::__rw_facet
{
typedef _CharT char_type;
typedef _InputIter iter_type;
static _RW::__rw_facet_id id;
explicit num_get (_RWSTD_SIZE_T __refs = 0)
: _RW::__rw_facet (__refs) { }
#ifndef _RWSTD_NO_NATIVE_BOOL
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, bool &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#endif // _RWSTD_NO_NATIVE_BOOL
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned short &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#ifndef _RWSTD_NO_EXT_NUM_GET
// extension
iter_type
get (iter_type, iter_type, ios_base&, _RWSTD_IOSTATE&, short&) const;
// extension
iter_type
get (iter_type, iter_type, ios_base&, _RWSTD_IOSTATE&, int&) const;
#endif // _RWSTD_NO_EXT_NUM_GET
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned int &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, long &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned long &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, float &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, double &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#ifndef _RWSTD_LONG_DOUBLE
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, long double &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#endif // _RWSTD_LONG_DOUBLE
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, void* &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#ifdef _RWSTD_LONG_LONG
// extension
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, _RWSTD_LONG_LONG &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
// extension
iter_type get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err,
unsigned _RWSTD_LONG_LONG &__val) const {
_RWSTD_ASSERT_RANGE (__begin, __end);
return do_get (__begin, __end, __flags, __err, __val);
}
#endif // _RWSTD_LONG_LONG
protected:
#ifndef _RWSTD_NO_NATIVE_BOOL
virtual iter_type
do_get (iter_type, iter_type, ios_base&, _RWSTD_IOSTATE&, bool&) const;
#endif // _RWSTD_NO_NATIVE_BOOL
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned short &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_ushort, &__val);
}
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned int &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_uint, &__val);
}
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, long &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_long, &__val);
}
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned long &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_ulong, &__val);
}
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, float &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_float, &__val);
}
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, double &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_double, &__val);
}
#ifndef _RWSTD_LONG_DOUBLE
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, long double &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_ldouble, &__val);
}
#endif // _RWSTD_LONG_DOUBLE
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, void* &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_pvoid, &__val);
}
#ifdef _RWSTD_LONG_LONG
// extension
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, _RWSTD_LONG_LONG &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_llong, &__val);
}
// extension
virtual iter_type
do_get (iter_type __begin, iter_type __end, ios_base &__flags,
_RWSTD_IOSTATE &__err, unsigned _RWSTD_LONG_LONG &__val) const {
return _C_get (__begin, __end, __flags, __err, _C_ullong, &__val);
}
#endif // _RWSTD_LONG_LONG
private:
iter_type
_C_get (iter_type, iter_type, ios_base&,
_RWSTD_IOSTATE&, int, void*) const;
};
#ifndef _RWSTD_NO_SPECIALIZED_FACET_ID
_RWSTD_SPECIALIZED_CLASS
_RW::__rw_facet_id num_get<char, istreambuf_iterator<char> >::id;
# ifndef _RWSTD_NO_WCHAR_T
_RWSTD_SPECIALIZED_CLASS
_RW::__rw_facet_id num_get<wchar_t, istreambuf_iterator<wchar_t> >::id;
# endif // _RWSTD_NO_WCHAR_T
#endif // _RWSTD_NO_SPECIALIZED_FACET_ID
} // namespace std
_RWSTD_NAMESPACE (__rw) {
inline short
__rw_check_overflow_short (long __lval, _RWSTD_FMTFLAGS __flags,
_RWSTD_IOSTATE &__err)
{
#if _RWSTD_SHRT_MAX < _RWSTD_LONG_MAX
long __shrt_min;
long __shrt_max;
if ( __lval < 0
|| (__flags & _RW::__rw_basefield) == _RW::__rw_dec) {
// decimal parsing overflows outside the range below
__shrt_max = long (_RWSTD_SHRT_MAX);
__shrt_min = long (_RWSTD_SHRT_MIN);
}
else {
// other than decimal parsing overflows outside the range below
// (unreliable if basefield is 0 and the sequence is decimal)
__shrt_max = long (_RWSTD_USHRT_MAX);
__shrt_min = long (_RWSTD_USHRT_MIN);
}
// lwg issue 23: check for overflow
if (__lval < __shrt_min) {
__err |= _RW::__rw_failbit;
return short (_RWSTD_SHRT_MIN);
}
else if (__lval > __shrt_max) {
__err |= _RW::__rw_failbit;
return short (_RWSTD_SHRT_MAX);
}
#else // if LONG_MAX <= SHRT_MAX
_RWSTD_UNUSED (__flags);
_RWSTD_UNUSED (__err);
#endif // _RWSTD_SHRT_MAX < _RWSTD_LONG_MAX
return _RWSTD_STATIC_CAST (short, __lval);
}
inline int
__rw_check_overflow_int (long __lval, _RWSTD_FMTFLAGS __flags,
_RWSTD_IOSTATE &__err)
{
#if _RWSTD_INT_MAX < _RWSTD_LONG_MAX
long __int_min;
long __int_max;
if ( __lval < 0
|| (__flags & _RW::__rw_basefield) == _RW::__rw_dec) {
// decimal parsing overflows outside the range below
__int_max = long (_RWSTD_INT_MAX);
__int_min = long (_RWSTD_INT_MIN);
}
else {
// other than decimal parsing overflows outside the range below
// (unreliable if basefield is 0 and the sequence is decimal)
__int_max = long (_RWSTD_UINT_MAX);
__int_min = long (_RWSTD_UINT_MIN);
}
// lwg issue 23: check for overflow
if (__lval < __int_min) {
__err |= _RW::__rw_failbit;
return int (_RWSTD_INT_MIN);
}
else if (__lval > __int_max) {
__err |= _RW::__rw_failbit;
return int (_RWSTD_INT_MAX);
}
#else // if LONG_MAX <= INT_MAX
_RWSTD_UNUSED (__flags);
_RWSTD_UNUSED (__err);
#endif // _RWSTD_INT_MAX < _RWSTD_LONG_MAX
return _RWSTD_STATIC_CAST (int, __lval);
}
} // namespace __rw
#if _RWSTD_DEFINE_TEMPLATE_FIRST (_NUM_GET)
# include <loc/_num_get.cc>
#endif // _RWSTD_DEFINE_TEMPLATE_FIRST (_NUM_GET)
#ifdef _RWSTD_MSVC
# pragma warning (push)
# pragma warning (disable: 4231)
#endif // _RWSTD_MSVC
_RWSTD_NAMESPACE (std) {
#if _RWSTD_INSTANTIATE (_NUM_GET, _CHAR)
_RWSTD_INSTANTIATE_1 (struct _RWSTD_TI_EXPORT num_get<char>);
#endif // _RWSTD_INSTANTIATE (_NUM_GET, _CHAR)
#if _RWSTD_INSTANTIATE (_NUM_GET, _WCHAR_T)
_RWSTD_INSTANTIATE_1 (struct _RWSTD_TI_EXPORT num_get<wchar_t>);
#endif // _RWSTD_INSTANTIATE (_NUM_GET, _WCHAR_T)
} // namespace std
#ifdef _RWSTD_MSVC
# pragma warning (pop)
#endif // _RWSTD_MSVC
#if _RWSTD_DEFINE_TEMPLATE_LAST (_NUM_GET)
# include <loc/_num_get.cc>
#endif // _RWSTD_DEFINE_TEMPLATE_LAST (_NUM_GET)
#endif // _RWSTD_LOC_NUM_GET_H_INCLUDED