blob: 24003ada51b15d59841958244e1e588d5b8ab53f [file] [log] [blame]
<
/***************************************************************************
*
* 22.locale.num.put.cpp - tests exercising the std::num_put facet
*
* $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-2006 Rogue Wave Software.
*
**************************************************************************/
#ifdef __GNUG__
// prevent gcc -Wshadow warnings for convenience types
# define ushort __rw_ushort
# define uint __rw_uint
# define ulong __rw_ulong
# include <sys/types.h>
# undef ushort
# undef uint
# undef ulong
#endif // __GNUG__
#include <ios>
#include <locale>
#include <cerrno> // for ERANGE, errno
#include <cfloat> // for floating limits macros
#include <climits> // for integral limits macros
#include <clocale> // for LC_NUMERIC, setlocale()
#include <cstdio> // for sprintf()
#include <cstring> // for strcmp(), strlen()
#include <any.h> // for TOSTR()
#include <cmdopt.h> // for rw_enabled
#include <driver.h> // for rw_test
#include <rw_locale.h> // for rw_locales
#include <valcmp.h> // for rw_equal
/**************************************************************************/
// set by the command line option handler in response to:
static int rw_opt_enable_num_get = 0; // --enable-num_get
static int rw_opt_no_errno = 0; // --no-errno
static int rw_opt_no_grouping = 0; // --no-grouping
static int rw_opt_no_widen = 0; // --no-widen
/**************************************************************************/
// replacement ctype facet
template <class charT>
struct Ctype: std::ctype<charT>
{
typedef std::ctype<charT> Base;
typedef typename Base::char_type char_type;
static int n_widen_;
Ctype (): Base (0, 0, 1) { }
virtual char_type do_widen (char c) const {
++n_widen_;
switch (c) {
case '0': c = '9'; break;
case '1': c = '8'; break;
case '2': c = '7'; break;
case '3': c = '6'; break;
case '4': c = '5'; break;
case '5': c = '4'; break;
case '6': c = '3'; break;
case '7': c = '2'; break;
case '8': c = '1'; break;
case '9': c = '0'; break;
default: break;
}
return char_type (c);
}
virtual const char_type*
do_widen (const char *lo, const char *hi, char_type *dst) const {
return Base::do_widen (lo, hi, dst);
}
};
template <class charT>
int Ctype<charT>::n_widen_;
/**************************************************************************/
// replacement numpunct facet
template <class charT>
struct Punct: std::numpunct<charT>
{
typedef typename std::numpunct<charT>::char_type char_type;
typedef typename std::numpunct<charT>::string_type string_type;
static char_type decimal_point_;
static char_type thousands_sep_;
static const char *grouping_;
static const char_type *truename_;
static const char_type *falsename_;
static int n_objs_; // number of facet objects in existence
static int n_thousands_sep_; // number of calls to do_thousands_sep()
Punct (std::size_t ref = 0)
: std::numpunct<charT>(ref) {
++n_objs_;
}
~Punct () {
--n_objs_;
}
virtual char_type do_decimal_point () const {
return decimal_point_;
}
virtual std::string do_grouping () const {
return grouping_;
}
virtual char_type do_thousands_sep () const {
++n_thousands_sep_;
return thousands_sep_;
}
virtual string_type do_truename () const {
return truename_ ? string_type (truename_) : string_type ();
}
virtual string_type do_falsename () const {
return falsename_ ? string_type (falsename_) : string_type ();
}
};
template <class charT>
const char* Punct<charT>::grouping_ = "";
template <class charT>
typename Punct<charT>::char_type Punct<charT>::decimal_point_ = '.';
template <class charT>
typename Punct<charT>::char_type Punct<charT>::thousands_sep_ = ',';
template <class charT>
const typename Punct<charT>::char_type* Punct<charT>::truename_;
template <class charT>
const typename Punct<charT>::char_type* Punct<charT>::falsename_;
template <class charT>
int Punct<charT>::n_thousands_sep_;
template <class charT>
int Punct<charT>::n_objs_;
/**************************************************************************/
template <class charT>
struct Ios: std::basic_ios<charT>
{
Ios () { this->init (0); }
};
template <class charT>
struct NumGet: std::num_get<charT, const charT*> { NumGet () { } };
template <class charT>
struct NumPut: std::num_put<charT, charT*>
{
typedef std::num_put<charT, charT*> Base;
typedef typename Base::char_type char_type;
typedef typename Base::iter_type iter_type;
enum {
test_bool, test_short, test_ushort, test_int, test_uint,
test_long, test_ulong, test_llong, test_ullong,
test_pvoid, test_float, test_double, test_ldouble
};
static int ncalls_ [13];
NumPut (std::size_t ref = 0): Base (ref) { }
#ifndef _RWSTD_NO_BOOL
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill, bool v) const {
++ncalls_ [test_bool];
return Base::do_put (it, f, fill, v);
}
#endif // _RWSTD_NO_BOOL
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill, long v) const {
++ncalls_ [test_long];
return Base::do_put (it, f, fill, v);
}
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill,
unsigned long v) const {
++ncalls_ [test_ulong];
return Base::do_put (it, f, fill, v);
}
#ifdef _RWSTD_LONG_LONG
// provided as an extension unless _RWSTD_NO_LONG_LONG is #defined
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill,
unsigned _RWSTD_LONG_LONG v) const {
++ncalls_ [test_ullong];
return Base::do_put (it, f, fill, v);
}
// provided as an extension unless _RWSTD_NO_LONG_LONG is #defined
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill,
_RWSTD_LONG_LONG v) const {
++ncalls_ [test_llong];
return Base::do_put (it, f, fill, v);
}
#endif // _RWSTD_LONG_LONG
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill, double v) const {
++ncalls_ [test_double];
return Base::do_put (it, f, fill, v);
}
#ifndef _RWSTD_NO_LONG_DOUBLE
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill,
long double v) const {
++ncalls_ [test_ldouble];
return Base::do_put (it, f, fill, v);
}
#endif // _RWSTD_NO_LONG_DOUBLE
virtual iter_type
do_put (iter_type it, std::ios_base &f, char_type fill,
const void *v) const {
++ncalls_ [test_pvoid];
return Base::do_put (it, f, fill, v);
}
};
// explicit initialization of the static array below used
// to work around a SunPro 5.4 bug (PR #27293)
template <class charT>
/* static */ int NumPut<charT>::ncalls_ [13] = { 0 };
/**************************************************************************/
template <class charT, class T>
void do_test (charT /* dummy */,
const char *cname, // charT name
const char *tname, // T name
int lineno, // line number
T val, // value to format
int flags, // ios flags
std::streamsize prec, // precision
std::streamsize width, // field width
char fill, // fill character
const char *grouping, // grouping string
const char *str, // expected output
int err_expect = -1, // expected iostate
T val_expect = T () /* expected num_get result */)
{
if (!rw_enabled (lineno)) {
rw_note (0, __FILE__, __LINE__, "test on line %d disabled", lineno);
return;
}
// create a distinct punctuation facet for each iteration to make sure
// any data cached in between successive calls to the facet's public
// member functions are flushed
const Punct<charT> pun (1);
Ios<charT> ios;
NumPut<charT> np;
ios.imbue (std::locale (ios.getloc (), (const std::numpunct<charT>*)&pun));
ios.flags (std::ios_base::fmtflags (flags));
ios.precision (prec);
ios.width (width);
pun.grouping_ = grouping;
#if defined (_RWSTD_LDBL_MAX_10_EXP)
charT buf [_RWSTD_LDBL_MAX_10_EXP + 2] = { 0 };
#else // if !defined (_RWSTD_LDBL_MAX_10_EXP)
charT buf [4096] = { 0 };
#endif // _RWSTD_LDBL_MAX_10_EXP
const charT* const bufend = np.put (buf, ios, fill, val);
// verify 22.2.2.2.2, p21
if (ios.width ()) {
static int width_fail = 0;
// fail at most once for each specialization
rw_assert (!width_fail++, 0, lineno,
"%line d: num_put<%s>::put (..., %s = %s) "
"failed to reset width from %d; width() = %d",
__LINE__, cname, tname, TOSTR (val),
TOSTR (buf), width, ios.width ());
}
/******************************************************************/
if (str) {
char cbuf [sizeof buf / sizeof *buf] = { '\0' };
if ('%' == *str) {
std::sprintf (cbuf, str, val);
str = cbuf;
#if defined (_WIN32) || defined (_WIN64)
std::size_t len = std::strlen (str);
if ( ('e' == cbuf [len - 5] || 'E' == cbuf [len - 5])
&& ('-' == cbuf [len - 4] || '+' == cbuf [len - 4])
&& ('0' == cbuf [len - 3])) {
cbuf [len - 3] = cbuf [len - 2];
cbuf [len - 2] = cbuf [len - 1];
cbuf [len - 1] = cbuf [len];
}
#endif // _WIN{32,64}
}
// compare output produced by num_put with that produced by printf()
rw_assert (0 == rw_strncmp (buf, str), 0, lineno,
"line %d: num_put<%s>::put (..., %s = %s) "
"wrote %{*Ac}, expected %{*Ac}, "
"flags = %{If}, precision = %d"
"%{?}, grouping = \"%#s\"%{;}",
__LINE__, cname, tname, TOSTR (val),
sizeof *buf, buf, sizeof *str, str,
flags, prec,
grouping && *grouping, grouping);
}
/******************************************************************/
#ifndef NO_NUM_GET
if (!rw_opt_enable_num_get)
return;
// skip negative precision (extension requiring special treatment)
if (prec < 0)
return;
const charT *next = buf;
// skip leading fill characters (if any)
for ( ; *next == fill; ++next);
// find the first non-fill character if it exists
const charT *last = next;
for ( ; *last && *last != fill; ++last);
for ( ; *last && *last == fill; ++last);
// do not perform extraction if there are any fill characters
// in the middle of output (they serve as punctuators and break
// up the input sequence)
if (next != last && *last)
return;
// do not perform extraction if the fill character is the minus
// sign, the plus sign, the thousands separator, the decimal point,
// or a digit
if ( '-' == fill
|| '+' == fill
|| pun.thousands_sep_ == fill
|| pun.decimal_point_ == fill
|| fill >= '0' && fill <= '9')
return;
// do not perform extraction if there is no data to extract
// (unless this is an extraction-only test)
if (!*next && str)
return;
NumGet<charT> ng;
T x = T ();
std::ios_base::iostate err = std::ios_base::goodbit;
if (-1 == err_expect) {
// lwg issue 17: special treatment for bool:
// The in iterator is always left pointing one position beyond
// the last character successfully matched. If val is set, then
// err is set to str.goodbit; or to str.eofbit if, when seeking
// another character to match, it is found that (in==end). If
// val is not set, then err is set to str.failbit; or to
// (str.failbit|str.eofbit) if the reason for the failure was
// that (in==end). [Example: for targets true:"a" and false:"abb",
// the input sequence "a" yields val==true and err==str.eofbit;
// the input sequence "abc" yields err=str.failbit, with in ending
// at the 'c' element. For targets true:"1" and false:"0", the
// input sequence "1" yields val==true and err=str.goodbit. For
// empty targets (""), any input sequence yields err==str.failbit.
// --end example]
err_expect = T (1) == T (2) && flags & std::ios::boolalpha
? std::ios::goodbit
: last == bufend && last [-1] != fill
? std::ios::eofbit : std::ios::goodbit;
val_expect = val;
}
last = ng.get (next, bufend, ios, err, x);
// verify the state
rw_assert (err == err_expect, 0, lineno,
"line %d: num_get<%s>::get (%s, ..., %s&) "
"flags = %{If}, "
"%{?}grouping = \"%#s\", %{;}"
"iostate = %{Is}, expected %{Is}",
__LINE__, cname,
TOSTR (next), tname,
flags,
grouping && *grouping, grouping,
err, err_expect);
// verify the parsed value
rw_assert (rw_equal (x, val_expect), 0, lineno,
"line %d: num_get<%s>::get (%s, ..., %s&) got %s, "
"expected %s, flags = %{If}, "
"%{?}grouping = \"%#s\", %{;}",
__LINE__, cname,
TOSTR (next), tname,
TOSTR (x), TOSTR (val_expect),
flags,
grouping && *grouping, grouping);
#else
_RWSTD_UNUSED (bufend);
_RWSTD_UNUSED (err_expect);
_RWSTD_UNUSED (val_expect);
#endif // NO_NUM_GET
}
/**************************************************************************/
template <class charT>
struct Streambuf: std::basic_streambuf<charT, std::char_traits<charT> > { };
template <class charT>
void direct_use_test (charT, const char *cname)
{
_RWSTD_UNUSED (cname);
// verify that num_put objects can be used directly w/o first
// having been installed in a locale object; the behavior of
// such facets is unspecified (but not undefined)
Ios<charT> ios;
const std::num_put<charT> np;
Streambuf<charT> sb;
#define DIRECT_USE_TEST(T) \
np.put (std::ostreambuf_iterator<charT>(&sb), ios, charT (), (T)0)
#ifndef _RWSTD_NO_BOOL
DIRECT_USE_TEST (bool);
#endif // _RWSTD_NO_BOOL
DIRECT_USE_TEST (unsigned long);
DIRECT_USE_TEST (long);
DIRECT_USE_TEST (double);
#ifndef _RWSTD_NO_LONG_DOUBLE
DIRECT_USE_TEST (long double);
#endif
DIRECT_USE_TEST (void*);
}
/**************************************************************************/
// for convenience
#define boolalpha std::ios_base::boolalpha
#define dec std::ios_base::dec
#define fixed std::ios_base::fixed
#define hex std::ios_base::hex
#define internal std::ios_base::internal
#define left std::ios_base::left
#define oct std::ios_base::oct
#define right std::ios_base::right
#define scientific std::ios_base::scientific
#define showbase std::ios_base::showbase
#define showpoint std::ios_base::showpoint
#define showpos std::ios_base::showpos
#define skipws std::ios_base::skipws
#define unitbuf std::ios_base::unitbuf
#define uppercase std::ios_base::uppercase
#define bin std::ios_base::bin
#define adjustfield std::ios_base::adjustfield
#define basefield std::ios_base::basefield
#define floatfield std::ios_base::floatfield
#define nolock std::ios_base::nolock
#define nolockbuf std::ios_base::nolockbuf
#define Bad std::ios_base::badbit
#define Eof std::ios_base::eofbit
#define Fail std::ios_base::failbit
#define Good std::ios_base::goodbit
template <class charT>
void bool_test (charT, const char *cname)
{
#ifndef _RWSTD_NO_BOOL
rw_info (0, 0, __LINE__, "std::num_put<%s>::put (..., bool)", cname);
const char* const tname = "bool";
static const charT boolnames[][4] = {
{ 'y', 'e', 's', '\0' }, { 'n', 'o', '.', '\0' },
// unusual strings to try to trip up the formatting algorithm
{ '+', '1', '\0' }, { '-', '2', '\0' },
{ '-', '+', '1', '\0' }, { '+', '-', '0', '\0' },
{ '\001', '\0' }, { '\0' }
};
Punct<charT>::decimal_point_ = '*';
Punct<charT>::truename_ = boolnames [0];
Punct<charT>::falsename_ = boolnames [1];
// set line number to __LINE__ and perform test so that
// assertion output will point to the failed TEST line
#define TEST do_test
#define T charT (), cname, tname, __LINE__
TEST (T, false, 0, 0, 0, ' ', "", "0");
TEST (T, true, 0, 0, 0, ' ', "", "1");
TEST (T, false, 0, 0, 0, ' ', "", "%d");
TEST (T, true, 0, 0, 0, ' ', "", "%d");
TEST (T, false, showpos, 0, 0, ' ', "", "%+d");
TEST (T, true, showpos, 0, 0, ' ', "", "%+d");
TEST (T, false, oct, 0, 0, ' ', "", "%o");
TEST (T, true, oct, 0, 0, ' ', "", "%o");
TEST (T, false, dec, 0, 0, ' ', "", "%d");
TEST (T, true, dec, 0, 0, ' ', "", "%d");
TEST (T, false, hex, 0, 0, ' ', "", "%x");
TEST (T, true, hex, 0, 0, ' ', "", "%x");
rw_info (0, 0, __LINE__, "std::ios::boolalpha");
TEST (T, false, boolalpha, 0, 0, ' ', "", "no.");
TEST (T, true, boolalpha, 0, 0, ' ', "", "yes");
// left justified boolalpha
TEST (T, false, boolalpha | left, 0, 6, ' ', "", "no. ");
TEST (T, true, boolalpha | left, 0, 6, ' ', "", "yes ");
// right justified boolalpha
TEST (T, false, boolalpha | right, 0, 6, ' ', "", " no.");
TEST (T, true, boolalpha | right, 0, 6, ' ', "", " yes");
// implied right justified boolalpha (internal bit set)
TEST (T, false, boolalpha | internal, 0, 4, ' ', "", " no.");
TEST (T, true, boolalpha | internal, 0, 4, ' ', "", " yes");
Punct<charT>::truename_ = boolnames [2];
Punct<charT>::falsename_ = boolnames [3];
TEST (T, false, boolalpha, 0, 1, ' ', "", "-2");
TEST (T, true, boolalpha, 0, 1, ' ', "", "+1");
TEST (T, false, boolalpha | internal, 0, 4, ' ', "", " -2");
TEST (T, true, boolalpha | internal, 0, 4, ' ', "", " +1");
Punct<charT>::truename_ = boolnames [4];
Punct<charT>::falsename_ = boolnames [5];
TEST (T, false, boolalpha, 0, 1, ' ', "", "+-0");
TEST (T, true, boolalpha, 0, 1, ' ', "", "-+1");
TEST (T, false, boolalpha | internal, 0, 4, ' ', "", " +-0");
TEST (T, true, boolalpha | internal, 0, 4, ' ', "", " -+1");
Punct<charT>::truename_ = boolnames [6];
Punct<charT>::falsename_ = boolnames [7];
TEST (T, false, boolalpha, 0, 0, ' ', "", "");
TEST (T, true, boolalpha, 0, 0, ' ', "", "\001");
TEST (T, false, boolalpha, 0, 1, ' ', "", " ");
TEST (T, true, boolalpha, 0, 1, ' ', "", "\001");
TEST (T, false, boolalpha, 0, 1, ',', "", ",");
TEST (T, true, boolalpha, 0, 1, ',', "", "\001");
TEST (T, false, boolalpha | internal, 0,4, ',', "", ",,,,");
TEST (T, true, boolalpha | internal, 0,4, ',', "", ",,,\001");
#endif // _RWSTD_NO_BOOL
}
/**************************************************************************/
template <class charT>
void long_test (charT, const char *cname)
{
const char* const tname = "long";
rw_info (0, 0, __LINE__, "std::num_put<%s>::put (..., %s)", cname, tname);
// working around a VAC++/AIX bug where LONG_{MIN,MAX} are
// of type int rather than long as required (see PR #28798)
#undef LONG_MIN
#define LONG_MIN _RWSTD_LONG_MIN
#undef LONG_MAX
#define LONG_MAX _RWSTD_LONG_MAX
#undef GET_FAIL
#define GET_FAIL (Eof | Fail), LONG_MAX
//////////////////////////////////////////////////////////////////
// implicit decimal output
rw_info (0, 0, __LINE__, "std::ios::fmtflags ()");
TEST (T, 0L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 1L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 2L, 0, 0, 0, ' ', "", "%ld");
TEST (T, -3L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 12L, 0, 0, 0, ' ', "", "%ld");
TEST (T, -13L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 345L, 0, 0, 0, ' ', "", "%ld");
TEST (T, -456L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 6789L, 0, 0, 0, ' ', "", "%ld");
TEST (T, -7890L, 0, 0, 0, ' ', "", "%ld");
TEST (T, 98765L, 0, 0, 0, ' ', "", "%ld");
TEST (T, -98766L, 0, 0, 0, ' ', "", "%ld");
TEST (T, LONG_MAX, 0, 0, 0, ' ', "", "%ld");
TEST (T, LONG_MIN, 0, 0, 0, ' ', "", "%ld");
//////////////////////////////////////////////////////////////////
// explicit decimal ouptut
rw_info (0, 0, __LINE__, "std::ios::dec");
TEST (T, 0L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 1L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 2L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 12L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 345L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 6789L, dec, 0, 0, ' ', "", "%ld");
TEST (T, 98765L, dec, 0, 0, ' ', "", "%ld");
TEST (T, LONG_MAX, dec, 0, 0, ' ', "", "%ld");
TEST (T, ~0L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -1L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -2L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -12L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -345L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -6789L, dec, 0, 0, ' ', "", "%ld");
TEST (T, -98765L, dec, 0, 0, ' ', "", "%ld");
TEST (T, LONG_MIN, dec, 0, 0, ' ', "", "%ld");
rw_info (0, 0, __LINE__, "std::ios::dec | std::ios::showbase");
TEST (T, 0L, dec | showbase, 0, 0, ' ', "", "%#ld");
TEST (T, 1234L, dec | showbase, 0, 0, ' ', "", "%#ld");
TEST (T, -1235L, dec | showbase, 0, 0, ' ', "", "%#ld");
TEST (T, LONG_MAX, dec | showbase, 0, 0, ' ', "", "%#ld");
TEST (T, LONG_MIN, dec | showbase, 0, 0, ' ', "", "%#ld");
rw_info (0, 0, __LINE__, "std::ios::dec | std::ios::showpos");
TEST (T, 0L, dec | showpos, 0, 0, ' ', "", "%+ld");
TEST (T, 1236L, dec | showpos, 0, 0, ' ', "", "%+ld");
TEST (T, -1237L, dec | showpos, 0, 0, ' ', "", "%+ld");
TEST (T, LONG_MAX, dec | showpos, 0, 0, ' ', "", "%+ld");
TEST (T, LONG_MIN, dec | showpos, 0, 0, ' ', "", "%+ld");
rw_info (0, 0, __LINE__,
"std::ios::dec | std::ios::showbase | std::ios::showpos");
TEST (T, 0L, dec | showbase | showpos, 0, 0, ' ', "", "%#+ld");
TEST (T, 1238L, dec | showbase | showpos, 0, 0, ' ', "", "%#+ld");
TEST (T, -1239L, dec | showbase | showpos, 0, 0, ' ', "", "%#+ld");
TEST (T, LONG_MAX, dec | showbase | showpos, 0, 0, ' ', "", "%#+ld");
TEST (T, LONG_MIN, dec | showbase | showpos, 0, 0, ' ', "", "%#+ld");
// left justfication
rw_info (0, 0, __LINE__, "std::ios::dec | std::ios::left");
TEST (T, 0L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 1L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 12L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 123L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 1234L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 12345L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 123456L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 1234567L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 12345678L, dec | left, 0, 10, ' ', "", "%-10ld");
TEST (T, 123456789L, dec | left, 0, 10, ' ', "", "%-10ld");
// left justfication, signed dec
TEST (T, -1L, dec | left, 0, 10, ' ', "", "-1 ");
TEST (T, -12L, dec | left, 0, 10, ' ', "", "-12 ");
TEST (T, -123L, dec | left, 0, 10, ' ', "", "-123 ");
TEST (T, -1234L, dec | left, 0, 10, ' ', "", "-1234 ");
TEST (T, -12345L, dec | left, 0, 10, ' ', "", "-12345 ");
TEST (T, -123456L, dec | left, 0, 10, ' ', "", "-123456 ");
TEST (T, -1234567L, dec | left, 0, 10, ' ', "", "-1234567 ");
TEST (T, -12345678L, dec | left, 0, 10, ' ', "", "-12345678 ");
TEST (T, -123456789L, dec | left, 0, 10, ' ', "", "-123456789");
// left justification with fill char
TEST (T, -1L, dec | left, 0, 10, '#', "", "-1########");
TEST (T, -12L, dec | left, 0, 10, '@', "", "-12@@@@@@@");
TEST (T, -123L, dec | left, 0, 10, '*', "", "-123******");
TEST (T, -1234L, dec | left, 0, 10, '=', "", "-1234=====");
TEST (T, -12345L, dec | left, 0, 10, '.', "", "-12345....");
TEST (T, -123456L, dec | left, 0, 10, ',', "", "-123456,,,");
TEST (T, -1234567L, dec | left, 0, 10, '-', "", "-1234567--");
TEST (T, -12345678L, dec | left, 0, 10, '+', "", "-12345678+");
TEST (T, -123456789L, dec | left, 0, 10, ';', "", "-123456789");
// left justfication with grouping
TEST (T, 1L, dec | left, 0, 14, ' ', "\2", "1 ");
TEST (T, 12L, dec | left, 0, 14, ' ', "\2", "12 ");
TEST (T, 123L, dec | left, 0, 14, ' ', "\2", "1,23 ");
TEST (T, 1234L, dec | left, 0, 14, ' ', "\2", "12,34 ");
TEST (T, 12345L, dec | left, 0, 14, ' ', "\2", "1,23,45 ");
TEST (T, 123456L, dec | left, 0, 14, ' ', "\2", "12,34,56 ");
TEST (T, 1234567L, dec | left, 0, 14, ' ', "\2", "1,23,45,67 ");
TEST (T, 12345678L, dec | left, 0, 14, ' ', "\2", "12,34,56,78 ");
TEST (T, 123456789L, dec | left, 0, 14, ' ', "\2", "1,23,45,67,89 ");
TEST (T, 123456789L, dec | left, 0, 14, ',', "\2", "1,23,45,67,89,");
TEST (T, 1L, dec | left, 0, 14, ' ', "\2\1\3", "1 ");
TEST (T, 12L, dec | left, 0, 14, ' ', "\2\1\3", "12 ");
TEST (T, 123L, dec | left, 0, 14, ' ', "\2\1\3", "1,23 ");
TEST (T, 1234L, dec | left, 0, 14, ' ', "\2\1\3", "1,2,34 ");
TEST (T, 12345L, dec | left, 0, 14, ' ', "\2\1\3", "12,3,45 ");
TEST (T, 123456L, dec | left, 0, 14, ' ', "\2\1\3", "123,4,56 ");
TEST (T, 1234567L, dec | left, 0, 14, ' ', "\2\1\3", "1,234,5,67 ");
TEST (T, 12345678L, dec | left, 0, 14, ' ', "\2\1\3", "12,345,6,78 ");
TEST (T, 123456789L, dec | left, 0, 14, ' ', "\2\1\3", "123,456,7,89 ");
TEST (T, 123456789L, dec | left, 0, 14, '0', "\2\1\3", "123,456,7,8900");
TEST (T, -1L, dec | left, 0, 14, ' ', "\1\2\3", "-1 ");
TEST (T, -12L, dec | left, 0, 14, ' ', "\1\2\3", "-1,2 ");
TEST (T, -123L, dec | left, 0, 14, ' ', "\1\2\3", "-12,3 ");
TEST (T, -1234L, dec | left, 0, 14, ' ', "\1\2\3", "-1,23,4 ");
TEST (T, -12345L, dec | left, 0, 14, ' ', "\1\2\3", "-12,34,5 ");
TEST (T, -123456L, dec | left, 0, 14, ' ', "\1\2\3", "-123,45,6 ");
TEST (T, -1234567L, dec | left, 0, 14, ' ', "\1\2\3", "-1,234,56,7 ");
TEST (T, -12345678L, dec | left, 0, 14, ' ', "\1\2\3", "-12,345,67,8 ");
TEST (T, -123456789L, dec | left, 0, 14, ' ', "\1\2\3", "-123,456,78,9 ");
TEST (T, -123456789L, dec | left, 0, 14, ' ', "\3\1\2", "-1,23,45,6,789");
TEST (T, -123456780L, dec | left, 0, 14, ' ', "\x1", "-1,2,3,4,5,6,7,8,0");
TEST (T, -123456781L, dec | left, 0, 14, ' ', "\x2", "-1,23,45,67,81");
TEST (T, -123456782L, dec | left, 0, 14, ' ', "\x3", "-123,456,782 ");
TEST (T, -123456783L, dec | left, 0, 14, ' ', "\x4", "-1,2345,6783 ");
TEST (T, -123456784L, dec | left, 0, 14, ' ', "\x5", "-1234,56784 ");
TEST (T, -123456785L, dec | left, 0, 14, ' ', "\x6", "-123,456785 ");
TEST (T, -123456786L, dec | left, 0, 14, ' ', "\x7", "-12,3456786 ");
TEST (T, -123456787L, dec | left, 0, 14, ' ', "\x8", "-1,23456787 ");
TEST (T, -123456788L, dec | left, 0, 14, ' ', "\x9", "-123456788 ");
#ifndef _RWSTD_NO_EXT_BIN_IO
rw_info (0, 0, __LINE__, "std::ios::dec | std::ios::bin [extension]");
TEST (T, 33333333L, bin | dec, 0, 0, ' ', "", "%ld");
#endif // _RWSTD_NO_EXT_BIN_IO
//////////////////////////////////////////////////////////////////
// explicit octal ouptut
rw_info (0, 0, __LINE__, "std::ios::oct");
TEST (T, 0L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 1L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 2L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 12L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 345L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 6789L, oct, 0, 0, ' ', "", "%lo");
TEST (T, 98765L, oct, 0, 0, ' ', "", "%lo");
TEST (T, LONG_MAX, oct, 0, 0, ' ', "", "%lo");
TEST (T, LONG_MAX - 1, oct, 0, 0, ' ', "", "%lo");
// negative values formatted as oct cause overflow on input (lwg issue 23)
TEST (T, ~0L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -1L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -2L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -12L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -345L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -6789L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, -98765L, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
TEST (T, LONG_MIN, oct, 0, 0, ' ', "", "%lo", GET_FAIL);
rw_info (0, 0, __LINE__, "std::ios::oct | std::ios::dec");
TEST (T, 13579L, oct | dec, 0, 0, ' ', "", "%ld");
rw_info (0, 0, __LINE__, "std::ios::oct | std::ios::showbase");
TEST (T, 0L, oct | showbase, 0, 0, ' ', "", "%#lo");
TEST (T, 2345L, oct | showbase, 0, 0, ' ', "", "%#lo");
TEST (T, -2346L, oct | showbase, 0, 0, ' ', "", "%#lo", GET_FAIL);
TEST (T, LONG_MAX, oct | showbase, 0, 0, ' ', "", "%#+lo");
TEST (T, LONG_MIN, oct | showbase, 0, 0, ' ', "", "%#+lo", GET_FAIL);
rw_info (0, 0, __LINE__, "std::ios::oct | std::ios::showpos");
TEST (T, 0L, oct | showpos, 0, 0, ' ', "", "%+lo");
TEST (T, 2347L, oct | showpos, 0, 0, ' ', "", "%+lo");
TEST (T, -2348L, oct | showpos, 0, 0, ' ', "", "%+lo", GET_FAIL);
TEST (T, LONG_MAX, oct | showpos, 0, 0, ' ', "", "%+lo");
TEST (T, LONG_MIN, oct | showpos, 0, 0, ' ', "", "%+lo", GET_FAIL);
rw_info (0, 0, __LINE__,
"std::ios::oct | std::ios::showbase | std::ios::showpos");
TEST (T, 0L, oct | showbase | showpos, 0, 0, ' ', "", "%#+lo");
TEST (T, 2349L, oct | showbase | showpos, 0, 0, ' ', "", "%#+lo");
TEST (T, -2350L, oct | showbase | showpos, 0, 0, ' ', "", "%#+lo",
GET_FAIL);
TEST (T, LONG_MAX, oct | showbase | showpos, 0, 0, ' ', "", "%#+lo");
TEST (T, LONG_MIN, oct | showbase | showpos, 0, 0, ' ', "", "%#+lo",
GET_FAIL);
#ifndef _RWSTD_NO_EXT_BIN_IO
rw_info (0, 0, __LINE__, "std::ios::oct | std::ios::bin [extension]");
TEST (T, 22222222L, bin | oct, 0, 0, ' ', "", "%ld");
#endif // _RWSTD_NO_EXT_BIN_IO
//////////////////////////////////////////////////////////////////
// explicit hexadecimal ouptut
rw_info (0, 0, __LINE__, "std::ios::hex");
TEST (T, 0L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 1L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 2L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 12L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 345L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 6789L, hex, 0, 0, ' ', "", "%lx");
TEST (T, 98765L, hex, 0, 0, ' ', "", "%lx");
TEST (T, LONG_MAX, hex, 0, 0, ' ', "", "%lx");
TEST (T, LONG_MAX - 1, hex, 0, 0, ' ', "", "%lx");
TEST (T, LONG_MAX - 2, hex, 0, 0, ' ', "", "%lx");
// negative values formatted as hex cause overflow on input (lwg issue 23)
TEST (T, ~0L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -1L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -2L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -12L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -345L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -6789L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, -98765L, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
TEST (T, LONG_MIN, hex, 0, 0, ' ', "", "%lx", GET_FAIL);
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::uppercase");
TEST (T, 0L, hex | uppercase, 0, 0, ' ', "", "%lX");
TEST (T, 1L, hex | uppercase, 0, 0, ' ', "", "%lX");
TEST (T, 0XABCDL, hex | uppercase, 0, 0, ' ', "", "%lX");
TEST (T, 0XBCDEL, hex | uppercase, 0, 0, ' ', "", "%lX");
TEST (T, 0XCDEFL, hex | uppercase, 0, 0, ' ', "", "%lX");
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::showbase");
TEST (T, 0L, hex | showbase, 0, 0, ' ', "", "%#lx");
TEST (T, 1L, hex | showbase, 0, 0, ' ', "", "%#lx");
TEST (T, 0xdef1L, hex | showbase, 0, 0, ' ', "", "%#lx");
rw_info (0, 0, __LINE__,
"std::ios::hex | std::ios::uppercase | std::ios::showbase");
TEST (T, 0L, hex | uppercase | showbase, 0, 0, ' ', "", "%#lX");
TEST (T, 1L, hex | uppercase | showbase, 0, 0, ' ', "", "%#lX");
TEST (T, 0XEF02L, hex | uppercase | showbase, 0, 0, ' ', "", "%#lX");
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::oct");
TEST (T, 0L, oct | hex, 0, 0, ' ', "", "%ld");
TEST (T, 97531L, oct | hex, 0, 0, ' ', "", "%ld");
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::dec");
TEST (T, 86420L, dec | hex, 0, 0, ' ', "", "%ld");
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::dec | std::ios::oct");
TEST (T, 11111111L, oct | dec | hex, 0, 0, ' ', "", "%ld");
#ifndef _RWSTD_NO_EXT_BIN_IO
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::bin [extension]");
TEST (T, 44444444L, bin | hex, 0, 0, ' ', "", "%ld");
rw_info (0, 0, __LINE__, "std::ios::hex | std::ios::dec | "
"std::ios::oct | std::ios::bin [extension]");
TEST (T, 55555555L, bin | oct | dec | hex, 0, 0, ' ', "", "%ld");
#endif // _RWSTD_NO_EXT_BIN_IO
//////////////////////////////////////////////////////////////////
// extension: fixed and negative precision
rw_info (0, 0, __LINE__,
"std::ios::fixed with negative precision [extension]");
TEST (T, 987654321L, 0, -1, 0, ' ', "", "987654321");
TEST (T, 987654322L, fixed, 0, 0, ' ', "", "987654322");
TEST (T, 987654323L, fixed, -1, 0, ' ', "", "98765432*3");
TEST (T, -987654323L, fixed, -1, 0, ' ', "", "-98765432*3");
TEST (T, 987654324L, fixed, -2, 0, ' ', "", "9876543*24");
TEST (T, -987654324L, fixed, -2, 0, ' ', "", "-9876543*24");
TEST (T, 987654325L, fixed, -3, 0, ' ', "", "987654*325");
TEST (T, -987654325L, fixed, -3, 0, ' ', "", "-987654*325");
TEST (T, 987654326L, fixed, -4, 0, ' ', "", "98765*4326");
TEST (T, -987654326L, fixed, -4, 0, ' ', "", "-98765*4326");
TEST (T, 987654327L, fixed, -5, 0, ' ', "", "9876*54327");
TEST (T, -987654327L, fixed, -5, 0, ' ', "", "-9876*54327");
TEST (T, 987654328L, fixed, -6, 0, ' ', "", "987*654328");
TEST (T, -987654328L, fixed, -6, 0, ' ', "", "-987*654328");
TEST (T, 987654329L, fixed, -7, 0, ' ', "", "98*7654329");
TEST (T, -987654329L, fixed, -7, 0, ' ', "", "-98*7654329");
TEST (T, 987654330L, fixed, -8, 0, ' ', "", "9*87654330");
TEST (T, -987654330L, fixed, -8, 0, ' ', "", "-9*87654330");
TEST (T, 987654331L, fixed, -9, 0, ' ', "", "0*987654331");
TEST (T, -987654331L, fixed, -9, 0, ' ', "", "-0*987654331");
TEST (T, 987654332L, fixed, -10, 0, ' ', "", "0*0987654332");
TEST (T, -987654332L, fixed, -10, 0, ' ', "", "-0*0987654332");
TEST (T, 987654333L, fixed, -11, 0, ' ', "", "0*00987654333");
TEST (T, -987654333L, fixed, -11, 0, ' ', "", "-0*00987654333");
TEST (T, 0L, fixed, -12, 0, ' ', "", "0*000000000000");
//////////////////////////////////////////////////////////////////
// second group is the last group (i.e., prevent repetition)
#if CHAR_MAX == UCHAR_MAX
# define GROUPING "\2\1\xff"
#else
# define GROUPING "\2\1\x7f"
#endif
TEST (T, 123456789L, dec | left, 0, 0, ' ', GROUPING, "123456,7,89");
TEST (T, -123456789L, dec | left, 0, 0, ' ', GROUPING, "-123456,7,89");
#undef GROUPING
#if CHAR_MAX == UCHAR_MAX
# define GROUPING "\2\3\xff"
#else
# define GROUPING "\2\3\x7f"
#endif
TEST (T, 123456789L, dec | left, 0, 0, ' ', GROUPING, "1234,567,89");
TEST (T, -123456789L, dec | left, 0, 0, ' ', GROUPING, "-1234,567,89");
TEST (T, 0x12345678L, hex | showbase, 0, 0, ' ', GROUPING, "0x123,456,78");
//////////////////////////////////////////////////////////////////
// right justfication
rw_info (0, 0, __LINE__, "std::ios::right");
TEST (T, 1L, dec | right, 0, 10, ' ', "", " 1");
TEST (T, 12L, dec | right, 0, 10, ' ', "", " 12");
TEST (T, 123L, dec | right, 0, 10, ' ', "", " 123");
TEST (T, 1234L, dec | right, 0, 10, ' ', "", " 1234");
TEST (T, 12345L, dec | right, 0, 10, ' ', "", " 12345");
TEST (T, 123456L, dec | right, 0, 10, ' ', "", " 123456");
TEST (T, 1234567L, dec | right, 0, 10, ' ', "", " 1234567");
TEST (T, 12345678L, dec | right, 0, 10, ' ', "", " 12345678");
TEST (T, 123456789L, dec | right, 0, 10, ' ', "", " 123456789");
TEST (T, 123456789L, dec | right, 0, 10, '0', "", "0123456789");
TEST (T, 123456789L, dec | right, 0, 10, '+', "", "+123456789");
TEST (T, 123456789L, dec | right, 0, 10, '-', "", "-123456789");
// right justification, oct
TEST (T, 0L, oct | right, 0, 10, ' ', "", " 0");
TEST (T, 01L, oct | right, 0, 10, ' ', "", " 1");
TEST (T, 012L, oct | right, 0, 10, ' ', "", " 12");
TEST (T, 0123L, oct | right, 0, 10, ' ', "", " 123");
TEST (T, 01234L, oct | right, 0, 10, ' ', "", " 1234");
TEST (T, 012345L, oct | right, 0, 10, ' ', "", " 12345");
TEST (T, 0123456L, oct | right, 0, 10, ' ', "", " 123456");
TEST (T, 01234567L, oct | right, 0, 10, ' ', "", " 1234567");
TEST (T, 012345670L, oct | right, 0, 10, ' ', "", " 12345670");
TEST (T, 012345670L, oct | right, 0, 10, '1', "", "1112345670");
// right justification with grouping
TEST (T, 0L, oct | right, 0, 10, ' ', "\2", " 0");
TEST (T, 01L, oct | right, 0, 10, ' ', "\2", " 1");
TEST (T, 012L, oct | right, 0, 10, ' ', "\2", " 12");
TEST (T, 0123L, oct | right, 0, 10, ' ', "\2", " 1,23");
TEST (T, 01234L, oct | right, 0, 10, ' ', "\2", " 12,34");
TEST (T, 012345L, oct | right, 0, 10, ' ', "\2", " 1,23,45");
TEST (T, 0123456L, oct | right, 0, 10, ' ', "\2", " 12,34,56");
TEST (T, 01234567L, oct | right, 0, 10, ' ', "\2", "1,23,45,67");
TEST (T, 012345670L, oct | right, 0, 10, ' ', "\2", "12,34,56,70");
#undef FLAGS
#define flags hex | showbase | internal
#define FLAGS flags | uppercase
// internal justfication, hex
rw_info (0, 0, __LINE__, "std::ios::internal");
TEST (T, 0X1L, FLAGS, 0, 10, ' ', "", "0X 1");
TEST (T, 0x12L, flags, 0, 10, ' ', "", "0x 12");
TEST (T, 0X123L, FLAGS, 0, 10, ' ', "", "0X 123");
TEST (T, 0x1234L, flags, 0, 10, ' ', "", "0x 1234");
TEST (T, 0X12345L, FLAGS, 0, 10, ' ', "", "0X 12345");
TEST (T, 0x123abcL, flags, 0, 10, ' ', "", "0x 123abc");
TEST (T, 0x123abcL, flags, 0, 10, '0', "", "0x00123abc");
TEST (T, 0X1234ABCL, FLAGS, 0, 10, ' ', "", "0X 1234ABC");
TEST (T, 0X12345678L, FLAGS, 0, 10, ' ', "", "0X12345678");
TEST (T, 0X12345678L, FLAGS, 0, 10, '0', "", "0X12345678");
// internal justfication, hex, grouping
TEST (T, 0X1L, FLAGS, 0, 10, ' ', "\1\2", "0X 1");
TEST (T, 0x12L, flags, 0, 10, ' ', "\1\2", "0x 1,2");
TEST (T, 0X123L, FLAGS, 0, 10, ' ', "\1\2", "0X 12,3");
TEST (T, 0x1234L, flags, 0, 10, ' ', "\1\2", "0x 1,23,4");
TEST (T, 0X12345L, FLAGS, 0, 10, ' ', "\1\2", "0X 12,34,5");
TEST (T, 0X12345L, FLAGS, 0, 10, ',', "\1\2", "0X,12,34,5");
TEST (T, 0x123abcL, flags, 0, 10, ' ', "\1\2", "0x1,23,ab,c");
TEST (T, 0X1234ABCL, FLAGS, 0, 10, ' ', "\1\2", "0X12,34,AB,C");
TEST (T, 0X12345678L, FLAGS, 0, 10, ' ', "\1\2", "0X1,23,45,67,8");
TEST (T, 0X12345678L, FLAGS, 0, 10, '0', "\1\2", "0X1,23,45,67,8");
#undef flags
#undef FLAGS
#define FLAGS dec | showpos | internal
// internal justfication, signed dec
TEST (T, 0L, FLAGS, 0, 10, ' ', "", "+ 0");
TEST (T, -1L, FLAGS, 0, 10, ' ', "", "- 1");
TEST (T, +12L, FLAGS, 0, 10, ' ', "", "+ 12");
TEST (T, -123L, FLAGS, 0, 10, ' ', "", "- 123");
TEST (T, +1234L, FLAGS, 0, 10, ' ', "", "+ 1234");
TEST (T, -12345L, FLAGS, 0, 10, ' ', "", "- 12345");
TEST (T, +123456L, FLAGS, 0, 10, ' ', "", "+ 123456");
TEST (T, -1234567L, FLAGS, 0, 10, ' ', "", "- 1234567");
TEST (T, +12345678L, FLAGS, 0, 10, ' ', "", "+ 12345678");
TEST (T, -123456789L, FLAGS, 0, 10, '0', "", "-123456789");
TEST (T, +1L, FLAGS, 0, 10, '-', "", "+--------1");
TEST (T, -12L, FLAGS, 0, 10, '-', "", "--------12");
TEST (T, +123L, FLAGS, 0, 10, '+', "", "+++++++123");
TEST (T, -1234L, FLAGS, 0, 10, '+', "", "-+++++1234");
TEST (T, +12345L, FLAGS, 0, 10, '0', "", "+000012345");
TEST (T, -123456L, FLAGS, 0, 10, '1', "", "-111123456");
TEST (T, +1234567L, FLAGS, 0, 10, '2', "", "+221234567");
TEST (T, -12345678L, FLAGS, 0, 10, '3', "", "-312345678");
TEST (T, +123456789L, FLAGS, 0, 10, '4', "", "+123456789");
#ifndef _RWSTD_NO_EXT_BIN_IO
// bin
rw_info (0, 0, __LINE__, "std::ios::bin [extension]");
TEST (T, 0L, bin, 0, 16, '.', "\4", "...............0");
TEST (T, 1L, bin, 0, 16, '.', "\4", "...............1");
TEST (T, 2L, bin, 0, 16, '.', "\4", "..............10");
TEST (T, 3L, bin, 0, 16, '.', "\4", "..............11");
TEST (T, 4L, bin, 0, 16, '.', "\4", ".............100");
TEST (T, 5L, bin, 0, 16, '.', "\4", ".............101");
TEST (T, 6L, bin, 0, 16, '.', "\4", ".............110");
TEST (T, 7L, bin, 0, 16, '.', "\4", ".............111");
TEST (T, 8L, bin, 0, 16, '.', "\4", "............1000");
TEST (T, 9L, bin, 0, 16, '.', "\4", "............1001");
TEST (T, 0x0aL, bin, 0, 16, '.', "\4", "............1010");
TEST (T, 0x0bL, bin, 0, 16, '.', "\4", "............1011");
TEST (T, 0x0cL, bin, 0, 16, '.', "\4", "............1100");
TEST (T, 0x0dL, bin, 0, 16, '.', "\4", "............1101");
TEST (T, 0x0eL, bin, 0, 16, '.', "\4", "............1110");
TEST (T, 0x0fL, bin, 0, 16, '.', "\4", "............1111");
TEST (T, 0xf0L, bin, 0, 16, '.', "\4", ".......1111,0000");
TEST (T, 0xf1L, bin, 0, 16, '.', "\4", ".......1111,0001");
TEST (T, 0xf2L, bin, 0, 16, '.', "\4", ".......1111,0010");
TEST (T, 0xf3L, bin, 0, 16, '.', "\4", ".......1111,0011");
TEST (T, 0xf4L, bin, 0, 16, '.', "\4", ".......1111,0100");
TEST (T, 0xf5L, bin, 0, 16, '.', "\4", ".......1111,0101");
TEST (T, 0x12345678L, bin, 0, 0, '.', "\4",
"1,0010,0011,0100,0101,0110,0111,1000");
TEST (T, 0xfedcba98L, bin, 0, 0, '\0', "\010",
"11111110,11011100,10111010,10011000");
#endif // _RWSTD_NO_EXT_BIN_IO
// locale 3.0 extension
#define BASE(n) ((n) << _RWSTD_IOS_BASEOFF)
// bases 0 and 10 are both base 10
// base 1 is roman (values 1 through 4999)
// bases 2 through 36 are what they are
// anything else is unspecified
rw_info (0, 0, __LINE__, "base 1 (Roman), and 2 through 36 [extension]");
TEST (T, 1234L, BASE ( 0), 0, 0, '\0', "", "1234");
TEST (T, 1234L, BASE ( 1), 0, 0, '\0', "", "mccxxxiv");
TEST (T, 0x1234L, BASE ( 2), 0, 0, '\0', "", "1001000110100");
TEST (T, 01234L, oct | BASE ( 8), 0, 0, '\0', "", "1234");
TEST (T, 1234L, dec | BASE (10), 0, 0, '\0', "", "1234");
TEST (T, 0x1234L, hex | BASE (16), 0, 0, '\0', "", "1234");
TEST (T, 1234L, BASE ( 2), 0, 0, '\0', "", "10011010010");
TEST (T, 1234L, BASE ( 3), 0, 0, '\0', "", "1200201");
TEST (T, 1234L, BASE ( 4), 0, 0, '\0', "", "103102");
TEST (T, 1234L, BASE ( 5), 0, 0, '\0', "", "14414");
TEST (T, 1234L, BASE ( 6), 0, 0, '\0', "", "5414");
TEST (T, 1234L, BASE ( 7), 0, 0, '\0', "", "3412");
TEST (T, 1234L, BASE ( 9), 0, 0, '\0', "", "1621");
TEST (T, 1234L, dec | BASE (10), 0, 0, '\0', "", "1234");
TEST (T, 1234L, BASE (11), 0, 0, '\0', "", "a22");
TEST (T, 1234L, BASE (12), 0, 0, '\0', "", "86a");
TEST (T, 1234L, BASE (13), 0, 0, '\0', "", "73c");
TEST (T, 1234L, BASE (14), 0, 0, '\0', "", "642");
TEST (T, 1234L, BASE (15), 0, 0, '\0', "", "574");
TEST (T, 1234L, hex | BASE (16), 0, 0, '\0', "", "4d2");
TEST (T, 1234L, BASE (17), 0, 0, '\0', "", "44a");
TEST (T, 1234L, BASE (18), 0, 0, '\0', "", "3ea");
TEST (T, 1234L, BASE (19), 0, 0, '\0', "", "37i");
TEST (T, 1234L, BASE (20), 0, 0, '\0', "", "31e");
TEST (T, 1234L, BASE (21), 0, 0, '\0', "", "2gg");
TEST (T, 1234L, BASE (22), 0, 0, '\0', "", "2c2");
TEST (T, 1234L, BASE (23), 0, 0, '\0', "", "27f");
TEST (T, 1234L, BASE (24), 0, 0, '\0', "", "23a");
TEST (T, 1234L, BASE (25), 0, 0, '\0', "", "1o9");
TEST (T, 1234L, BASE (26), 0, 0, '\0', "", "1lc");
TEST (T, 1234L, BASE (27), 0, 0, '\0', "", "1ij");
TEST (T, 1234L, BASE (28), 0, 0, '\0', "", "1g2");
TEST (T, 1234L, BASE (29), 0, 0, '\0', "", "1dg");
TEST (T, 1234L, BASE (30), 0, 0, '\0', "", "1b4");
TEST (T, 1234L, BASE (31), 0, 0, '\0', "", "18p");
TEST (T, 1234L, BASE (32), 0, 0, '\0', "", "16i");
TEST (T, 1234L, BASE (33), 0, 0, '\0', "", "14d");
TEST (T, 1234L, BASE (34), 0, 0, '\0', "", "12a");
TEST (T, 1234L, BASE (35), 0, 0, '\0', "", "109");
TEST (T, 1234L, BASE (36), 0, 0, '\0', "", "ya");
// effect of non-empty grouping is unspecified
TEST (T, 0L, BASE (1), 0, 0, '\0', "", "0");
TEST (T, 1L, BASE (1), 0, 0, '\0', "", "i");
TEST (T, 2L, BASE (1), 0, 0, '\0', "", "ii");
TEST (T, 3L, BASE (1), 0, 0, '\0', "", "iii");
TEST (T, 4L, BASE (1), 0, 0, '\0', "", "iv");
TEST (T, 5L, BASE (1), 0, 0, '\0', "", "v");
TEST (T, 6L, BASE (1), 0, 0, '\0', "", "vi");
TEST (T, 7L, BASE (1), 0, 0, '\0', "", "vii");
TEST (T, 8L, BASE (1), 0, 0, '\0', "", "viii");
TEST (T, 9L, BASE (1), 0, 0, '\0', "", "ix");
TEST (T, 10L, BASE (1), 0, 0, '\0', "", "x");
TEST (T, 50L, BASE (1), 0, 0, '\0', "", "l");
TEST (T, 100L, BASE (1), 0, 0, '\0', "", "c");
TEST (T, 500L, BASE (1), 0, 0, '\0', "", "d");
TEST (T, 1000L, BASE (1), 0, 0, '\0', "", "m");
TEST (T, 49L, BASE (1), 0, 0, '\0', "", "xlix");
TEST (T, 88L, BASE (1), 0, 0, '\0', "", "lxxxviii");
TEST (T, 99L, BASE (1), 0, 0, '\0', "", "xcix");
TEST (T, 1999L, BASE (1), 0, 0, '\0', "", "mcmxcix");
TEST (T, 2000L, BASE (1), 0, 0, '\0', "", "mm");
TEST (T, 2001L, BASE (1), 0, 0, '\0', "", "mmi");
TEST (T, 4999L, BASE (1), 0, 0, '\0', "", "mmmmcmxcix");
TEST (T, 5000L, BASE (1), 0, 0, '\0', "", "5000");
TEST (T, 1492L, BASE (1), 0, 10, '*', "", "***mcdxcii");
TEST (T, 1776L, BASE (1) | uppercase, 0, 0, '\0', "", "MDCCLXXVI");
}
/**************************************************************************/
template <class charT>
void ulong_test (charT, const char *cname)
{
const char* const tname = "unsigned long";
rw_info (0, 0, __LINE__, "std::num_put<%s>::put (..., %s)", cname, tname);
typedef unsigned long ULong;
//////////////////////////////////////////////////////////////////
rw_info (0, 0, __LINE__, "std::ios::dec");
TEST (T, 0UL, dec, 0, 0, ' ', "", "%lu");
TEST (T, ULong (LONG_MAX), dec, 0, 0, ' ', "", "%lu");
rw_info (0, 0, __LINE__, "std::ios::dec | std::ios::shopos");
TEST (T, 0UL, dec | showpos, 0, 0, ' ', "", "%+lu");
TEST (T, 1UL, dec | showpos, 0, 0, ' ', "", "%+lu");
TEST (T, ULong (LONG_MAX), dec | showpos, 0, 0, ' ', "", "%+lu");
//////////////////////////////////////////////////////////////////
rw_info (0, 0, __LINE__, "std::ios::oct");
TEST (T, ULong (CHAR_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (UCHAR_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (SCHAR_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (SHRT_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (USHRT_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (INT_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (UINT_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (LONG_MAX), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (CHAR_MIN), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (SCHAR_MIN), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (SHRT_MIN), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (INT_MIN), oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (LONG_MIN), oct, 0, 0, ' ', "", "%lo");
// no overflow
TEST (T, 1UL, oct, 0, 0, ' ', "", "%lo");
TEST (T, ~0UL, oct, 0, 0, ' ', "", "%lo");
TEST (T, ULong (ULONG_MAX), oct, 0, 0, ' ', "", "%lo");
//////////////////////////////////////////////////////////////////
rw_info (0, 0, __LINE__, "std::ios::hex");
TEST (T, ULong (CHAR_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (UCHAR_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (SCHAR_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (SHRT_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (USHRT_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (INT_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (UINT_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (LONG_MAX), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (CHAR_MIN), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (SCHAR_MIN), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (SHRT_MIN), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (INT_MIN), hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (LONG_MIN), hex, 0, 0, ' ', "", "%lx");
TEST (T, 1UL, hex, 0, 0, ' ', "", "%lx");
TEST (T, ~0UL, hex, 0, 0, ' ', "", "%lx");
TEST (T, ULong (ULONG_MAX), hex, 0, 0, ' ', "", "%lx");
}
/**************************************************************************/
template <class charT>
void llong_test (charT, const char *cname)
{
const char* const tname = "long long";
rw_info (0, 0, 0, "std::num_put<%s>::put (..., %s)", cname, tname);
#ifndef _RWSTD_NO_LONG_LONG
# define STDIO_FMAT "%" _RWSTD_LLONG_PRINTF_PREFIX "d"
# ifndef _MSC_VER
# define LL(number) number ## LL
# else // if defined (_MSC_VER)
// MSVC 7.0 doesn't recognize the LL suffix
# define LL(number) number ## I64
# endif // _MSC_VER
TEST (T, LL (0), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (1), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (21), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (4321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (54321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (7654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (87654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, LL (987654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, ~LL (0), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (1), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (21), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (4321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (54321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (7654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (87654321), dec, 0, 0, ' ', "", STDIO_FMAT);
TEST (T, -LL (987654321), dec, 0, 0, ' ', "", STDIO_FMAT);
# undef FLAGS
# define FLAGS hex | showbase | internal
# undef OPTS
# define OPTS FLAGS, 0, 20, ' ', "\1\2\3\4\1\2\3\4"
// internal justfication, hex format, grouping
TEST (T, LL (0), OPTS, " 0");
TEST (T, LL (0x1), OPTS, "0x 1");
TEST (T, LL (0x12), OPTS, "0x 1,2");
TEST (T, LL (0x123), OPTS, "0x 12,3");
TEST (T, LL (0x1234), OPTS, "0x 1,23,4");
TEST (T, LL (0x12345), OPTS, "0x 12,34,5");
TEST (T, LL (0x123456), OPTS, "0x 123,45,6");
TEST (T, LL (0x1234567), OPTS, "0x 1,234,56,7");
TEST (T, LL (0x12345678), OPTS, "0x 12,345,67,8");
TEST (T, LL (0x123456789), OPTS, "0x 123,456,78,9");
TEST (T, LL (0x123456789a), OPTS, "0x 1234,567,89,a");
TEST (T, LL (0x123456789ab), OPTS, "0x 1,2345,678,9a,b");
TEST (T, LL (0x123456789abc), OPTS, "0x 1,2,3456,789,ab,c");
TEST (T, LL (0x123456789abcd), OPTS, "0x12,3,4567,89a,bc,d");
TEST (T, LL (0x123456789abcde), OPTS, "0x1,23,4,5678,9ab,cd,e");
TEST (T, LL (0x123456789abcdef), OPTS, "0x12,34,5,6789,abc,de,f");
TEST (T, LL (0x123456789abcdef0), OPTS, "0x123,45,6,789a,bcd,ef,0");
#else // if defined (_RWSTD_NO_LONG_LONG)
rw_note (0, 0, __LINE__, "num_put<%s>::put (..., %s) not exercised, "
"macro _RWSTD_NO_LONG_LONG #defined",
cname, tname);
#endif // _RWSTD_LONG_LONG
}
/**************************************************************************/
template <class charT>
void ullong_test (charT, const char *cname)
{
const char* const tname = "unsigned long long";
rw_info (0, 0, 0, "std::num_put<%s>::put (..., %s)", cname, tname);
#ifndef _RWSTD_NO_LONG_LONG
rw_warn (0, 0, __LINE__, "num_put<%s>::put (..., %s) not exercised",
cname, tname);
#else // if defined (_RWSTD_NO_LONG_LONG)
rw_note (0, 0, __LINE__, "num_put<%s>::put (..., %s) not exercised, "
"macro _RWSTD_NO_LONG_LONG #defined",
cname, tname);
#endif // _RWSTD_NO_LONG_LONG
}
/**************************************************************************/
template <class charT>
void dbl_test (charT, const char *cname)
{
const char* const tname = "double";
rw_info (0, 0, 0, "std::num_put<%s>::put (..., %s)", cname, tname);
Punct<charT>::decimal_point_ = '.';
TEST (T, 0.0, 0, 0, 0, ' ', "", "%g");
TEST (T, -0.0, 0, 0, 0, ' ', "", "%g");
TEST (T, 1.0, 0, 0, 0, ' ', "", "%g");
TEST (T, 1.0, 0, 0, 0, ' ', "", "1");
TEST (T, -1.0, 0, 0, 0, ' ', "", "%g");
TEST (T, -1.0, 0, 0, 0, ' ', "", "-1");
TEST (T, 1.1, 0, 0, 0, ' ', "", "%g");
TEST (T, 1.1, 0, 0, 0, ' ', "", "1.1");
TEST (T, -1.1, 0, 0, 0, ' ', "", "%g");
TEST (T, -1.1, 0, 0, 0, ' ', "", "-1.1");
// exercise formatting of very large numbers in a fixed notation
// using '+' as the fill character to disable num_get tests
TEST (T, 1.0e+128, fixed, 0, 0, '+', "", "%.0f", Eof);
TEST (T, 1.0e+256, fixed, 0, 0, '+', "", "%.0f", Eof);
#undef CAT
#undef CONCAT
#define CAT(a, b) CONCAT (a, b)
#define CONCAT(a, b) a ## b
const double d = CAT (1.0e+, _RWSTD_DBL_MAX_10_EXP);
TEST (T, d, fixed, 0, 0, '+', "", "%.0f", Eof);
{
char stdiofmt [8];
const std::streamsize prec = std::streamsize (DBL_DIG + 3);
std::sprintf (stdiofmt, "%%.%dg", int (prec));
TEST (T, DBL_MIN, 0, prec, 0, ' ', "", stdiofmt);
TEST (T, DBL_MAX, 0, prec, 0, ' ', "", stdiofmt);
}
if (1) {
// verify that the global LC_NUMERIC setting has no impact on the facet
for (const char *name = rw_locales (LC_NUMERIC, 0); name && *name;
name += std::strlen (name) + 1) {
// find the first locale whose decimal_point character
// is different than in the classic C locale (i.e., than '.')
if (0 == std::setlocale (LC_NUMERIC, name))
continue;
const std::lconv* const conv = std::localeconv ();
if (!conv)
continue;
if (conv->decimal_point && '.' != *conv->decimal_point)
break;
}
Punct<charT>::decimal_point_ = '*';
Punct<charT>::thousands_sep_ = '/';
TEST (T, 12345.678900, fixed, 6, 0, ' ', "\000", "12345*678900");
TEST (T, 123456.78900, fixed, 5, 0, ' ', "\002", "12/34/56*78900");
TEST (T, 1234567.8900, fixed, 4, 0, ' ', "\003", "1/234/567*8900");
Punct<charT>::decimal_point_ = '.';
TEST (T, 12345678.900, fixed, 3, 0, ' ', "\004", "1234/5678.900");
// reset the global locale
std::setlocale (LC_NUMERIC, "C");
}
}
/**************************************************************************/
template <class charT>
void ldbl_test (charT, const char *cname)
{
#ifndef _RWSTD_NO_LONG_DOUBLE
const char* const tname = "long double";
rw_info (0, 0, 0, "std::num_put<%s>::put (..., %s) "
"[sizeof (long double) = %u]",
cname, tname, sizeof (long double));
typedef long double LDbl;
Punct<charT>::decimal_point_ = '.';
TEST (T, 0.0L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 1.0L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 2.1L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, -3.2L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, -4.3L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 1.0e+10L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 2.0e+20L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 4.0e+30L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 1.0e-10L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 2.0e-20L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, 4.0e-30L, 0, 0, 0, ' ', "", "%Lg");
TEST (T, LDbl (CHAR_MAX), 0, 0, 0, ' ', "", "%Lg");
TEST (T, LDbl (UCHAR_MAX), 0, 0, 0, ' ', "", "%Lg");
TEST (T, LDbl (SCHAR_MAX), 0, 0, 0, ' ', "", "%Lg");
TEST (T, LDbl (SHRT_MAX), 0, 0, 0, ' ', "", "%Lg");
TEST (T, LDbl (USHRT_MAX), 0, 0, 0, ' ', "", "%Lg");
// specify greater precision than the default 6 for large numbers
TEST (T, LDbl (INT_MAX), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (UINT_MAX), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (LONG_MAX), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (ULONG_MAX), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (FLT_MIN), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (FLT_MAX), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (DBL_MIN), 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, LDbl (DBL_MAX), 0, 32, 0, ' ', "", "%.32Lg");
{
char stdiofmt [8];
const std::streamsize prec = std::streamsize (LDBL_DIG + 3);
std::sprintf (stdiofmt, "%%.%d" _RWSTD_LDBL_PRINTF_PREFIX "g",
int (prec));
TEST (T, LDbl (LDBL_MIN), 0, prec, 0, ' ', "", stdiofmt);
TEST (T, LDbl (LDBL_MAX), 0, prec, 0, ' ', "", stdiofmt);
}
const long double Pi = 3.14159265358979323846L;
// some test cases below that use precision in conjuction with
// scientific in the libc format specifier in order to exercise
// the proposed resolution of lwg issue 231
TEST (T, Pi, 0, 32, 0, ' ', "", "%.32Lg");
TEST (T, Pi, fixed, 0, 0, ' ', "", "%.0Lf", Eof, 3.0L);
TEST (T, Pi, scientific, 0, 0, ' ', "", "%.0Le", Eof, 3.0L);
TEST (T, Pi, fixed, 0, 0, ' ', "", "3", Eof, 3.0L);
TEST (T, Pi, scientific, 0, 0, ' ', "", "3e+00", Eof, 3.0L);
TEST (T, Pi, uppercase, 32, 0, ' ', "", "%.32LG");
TEST (T, Pi, uppercase | fixed, 0, 0, ' ', "", "%.0Lf", Eof, 3.0L);
TEST (T, Pi, uppercase | scientific, 0, 0, ' ', "", "%.0LE", Eof, 3.0L);
TEST (T, Pi, uppercase | scientific, 0, 0, ' ', "", "3E+00", Eof, 3.0L);
#define Showpos(f) (showpos | f)
TEST (T, Pi, Showpos (0), 32, 0, ' ', "", "%+.32Lg");
TEST (T, Pi, Showpos (fixed), 0, 0, ' ', "", "%+.0Lf", Eof, 3.0L);
TEST (T, Pi, Showpos (scientific), 0, 0, ' ', "", "%+.0Le", Eof, 3.0L);
TEST (T, Pi, Showpos (fixed), 0, 0, ' ', "", "+3", Eof, 3.0L);
TEST (T, Pi, Showpos (scientific), 0, 0, ' ', "", "+3e+00", Eof, 3.0L);
#define SHOWPOS(f) (showpos | uppercase | f)
TEST (T, Pi, SHOWPOS (0), 32, 0, ' ', "", "%+.32LG");
TEST (T, Pi, SHOWPOS (fixed), 0, 0, ' ', "", "%+.0Lf", Eof, 3.0L);
TEST (T, Pi, SHOWPOS (scientific), 0, 0, ' ', "", "%+.0LE", Eof, 3.0L);
TEST (T, Pi, SHOWPOS (fixed), 0, 0, ' ', "", "+3", Eof, 3.0L);
TEST (T, Pi, SHOWPOS (scientific), 0, 0, ' ', "", "+3E+00", Eof, 3.0L);
#define Showpoint(f) (showpoint | f)
TEST (T, Pi, Showpoint (0), 32, 0, ' ', "", "%#.32Lg");
TEST (T, Pi, Showpoint (fixed), 0, 0, ' ', "", "%#.0Lf", Eof, 3.0L);
TEST (T, Pi, Showpoint (scientific), 0, 0, ' ', "", "%#.0Le", Eof, 3.0L);
TEST (T, Pi, Showpoint (fixed), 0, 0, ' ', "", "3.", Eof, 3.0L);
TEST (T, Pi, Showpoint (scientific), 0, 0, ' ', "", "3.e+00", Eof, 3.0L);
#define SHOWPOINT(f) (showpoint | uppercase | f)
TEST (T, Pi, SHOWPOINT (0), 32, 0, ' ', "", "%#.32LG");
TEST (T, Pi, SHOWPOINT (fixed), 0, 0, ' ', "", "%#.0Lf", Eof, 3.0L);
TEST (T, Pi, SHOWPOINT (scientific), 0, 0, ' ', "", "%#.0LE", Eof, 3.0L);
TEST (T, Pi, SHOWPOINT (scientific), 0, 0, ' ', "", "3.E+00", Eof, 3.0L);
#define Showall(f) (showpoint | showpos | f)
TEST (T, Pi, Showall (0), 32, 0, ' ', "", "%#+.32Lg");
TEST (T, Pi, Showall (fixed), 0, 0, ' ', "", "%#+.0Lf", Eof, 3.0L);
TEST (T, Pi, Showall (scientific), 0, 0, ' ', "", "%#+.0Le", Eof, 3.0L);
TEST (T, Pi, Showall (fixed), 0, 0, ' ', "", "+3.", Eof, 3.0L);
TEST (T, Pi, Showall (scientific), 0, 0, ' ', "", "+3.e+00", Eof, 3.0L);
#define SHOWALL(f) (showpoint | showpos | uppercase | f)
TEST (T, Pi, SHOWALL (0), 32, 0, ' ', "", "%#+.32LG");
TEST (T, Pi, SHOWALL (fixed), 0, 0, ' ', "", "%#+.0Lf", Eof, 3.0L);
TEST (T, Pi, SHOWALL (scientific), 0, 0, ' ', "", "%#+.0LE", Eof, 3.0L);
TEST (T, Pi, SHOWALL (scientific), 0, 0, ' ', "", "+3.E+00", Eof, 3.0L);
// with {g,G}, precision indicates the number of significant digits
TEST (T, Pi, 0, 1, 0, ' ', "", "%.1Lg", Eof, 3.0L);
TEST (T, Pi, 0, 2, 0, ' ', "", "%.2Lg", Eof, 3.1L);
TEST (T, Pi, 0, 3, 0, ' ', "", "%.3Lg", Eof, 3.14L);
TEST (T, Pi, 0, 10, 0, ' ', "", "%.10Lg", Eof, 3.141592654L);
// C89 and C99 both specify that (only if specified using the asterisk)
// negative precision is treated the same as no precision at all; verify
// that num_put handles negative precision gracefully (i.e., ignores it)
TEST (T, Pi, 0, -1, 0, ' ', "", "%Lg");
TEST (T, Pi, 0, -9, 0, ' ', "", "%Lg");
// with {e,E,f,F}, precision indicates the number of fractional digits
TEST (T, Pi, fixed, 1, 0, ' ', "", "%.1Lf", Eof, 3.1L);
TEST (T, Pi, fixed, 2, 0, ' ', "", "%.2Lf", Eof, 3.14L);
TEST (T, Pi, fixed, 3, 0, ' ', "", "%.3Lf", Eof, 3.142L);
TEST (T, Pi, fixed, 10, 0, ' ', "", "%.10Lf", Eof, 3.1415926536L);
// exercise formatting of very large numbers in a fixed notation
// using '-' as the fill character to disable num_get tests
TEST (T, 2.0e+128L, fixed, 0, 0, '-', "", "%.0Lf", Eof);
TEST (T, 3.0e+256L, fixed, 0, 0, '-', "", "%.0Lf", Eof);
TEST (T, 4.0e+300L, fixed, 0, 0, '-', "", "%.0Lf", Eof);
// extension: fixed format and negative precision
TEST (T, Pi, fixed, -2, 0, ' ', "", "0.03");
TEST (T, Pi, fixed, -8, 0, ' ', "", "0.00000003");
#undef CAT
#undef CONCAT
#define CAT(a, b) CONCAT (a, b)
#define CONCAT(a, b) a ## b
const long double ld = CAT (CAT (1.0e+, _RWSTD_LDBL_MAX_10_EXP), L);
TEST (T, ld, fixed, 0, 0, '-', "", "%.0Lf", Eof);
TEST (T, Pi, scientific, 2, 0, ' ', "", "%.2Le", Eof, 3.14L);
TEST (T, Pi, scientific, 4, 0, ' ', "", "%.4Le", Eof, 3.1416L);
TEST (T, Pi, scientific, 6, 0, ' ', "", "%.6Le", Eof, 3.141593L);
TEST (T, Pi, scientific, 12, 0, ' ', "", "%.12Le", Eof, 3.14159265359L);
TEST (T, Pi, scientific, -3, 0, ' ', "", "%Le");
TEST (T, Pi, scientific, -7, 0, ' ', "", "%Le");
#endif // _RWSTD_NO_LONG_DOUBLE
}
/**************************************************************************/
template <class charT>
void pvoid_test (charT, const char *cname)
{
const char* const tname = "const void*";
rw_info (0, 0, 0, "std::num_put<%s>::put (..., %s)", cname, tname);
typedef void* PVoid;
int foo = 0;
int bar = 0;
// use "0" rather than "%p" for null pointers to avoid
// having to deal with GNU libc's "(nil)" format
TEST (T, PVoid ( 0), 0, 0, 0, ' ', "", "0");
TEST (T, PVoid ( 1), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( &foo), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( &bar), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( ~0L), 0, 0, 0, ' ', "", "%p");
// 22.2.2.2.2, p16 and lwg issue 282: grouping and thousands_sep
// is only used in arithmetic types
TEST (T, PVoid ( 0), 0, 0, 0, ' ', "\1", "0");
TEST (T, PVoid ( 1), 0, 0, 0, ' ', "\1", "%p");
TEST (T, PVoid ( &foo), 0, 0, 0, ' ', "\1\1", "%p");
TEST (T, PVoid ( &bar), 0, 0, 0, ' ', "\1\2", "%p");
TEST (T, PVoid ( ~0L), 0, 0, 0, ' ', "\1\3", "%p");
TEST (T, PVoid ( SHRT_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( SHRT_MIN), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid (USHRT_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( INT_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( INT_MIN), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( UINT_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( LONG_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid ( LONG_MIN), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid (ULONG_MAX), 0, 0, 0, ' ', "", "%p");
#ifdef _RWSTD_LLONG_MAX
# if _RWSTD_LLONG_SIZE <= _RWSTD_PTR_SIZE
TEST (T, PVoid (_RWSTD_LLONG_MAX), 0, 0, 0, ' ', "", "%p");
TEST (T, PVoid (_RWSTD_ULLONG_MAX), 0, 0, 0, ' ', "", "%p");
# endif // LLONG_MAX
#endif // _RWSTD_LLONG_MAX
}
/**************************************************************************/
template <class charT>
void errno_test (charT, const char *cname)
{
const char* const tname = "long";
rw_info (0, 0, 0,
"std::num_put<%s>::put (..., %s) and errno", cname, tname);
// verify that errno doesn't change after, or adversely affect
// successful insertion; and that the insertion doesn't change
// the value of errno
errno = -1;
TEST (T, 12345L, 0, 0, 0, ' ', "", "12345");
rw_assert (-1 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", -1);
errno = 0;
TEST (T, 12346L, 0, 0, 0, ' ', "", "12346");
rw_assert (0 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", -0);
errno = 1;
TEST (T, 12347L, 0, 0, 0, ' ', "", "12347");
rw_assert (1 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", 1);
errno = 2;
TEST (T, 12348L, 0, 0, 0, ' ', "", "12348");
rw_assert (2 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", 2);
errno = 3;
TEST (T, 12349L, 0, 0, 0, ' ', "", "12349");
rw_assert (3 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", 3);
errno = 4;
TEST (T, 12350L, 0, 0, 0, ' ', "", "12350");
rw_assert (4 == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", 4);
errno = ERANGE;
TEST (T, 12351L, 0, 0, 0, ' ', "", "12351");
rw_assert (ERANGE == errno, 0, __LINE__,
"errno unexpectedly changed from %{#m} to %{#m} (%m)", ERANGE);
errno = Bad;
TEST (T, 12352L, 0, 0, 0, ' ', "", "12352");
rw_assert (Bad == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", Bad);
errno = Eof;
TEST (T, 12353L, 0, 0, 0, ' ', "", "12353");
rw_assert (Eof == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", Eof);
errno = Fail;
TEST (T, 12354L, 0, 0, 0, ' ', "", "12354");
rw_assert (Fail == errno, 0, __LINE__,
"errno unexpectedly changed from %d to %{#m} (%m)", Fail);
errno = 0;
}
/**************************************************************************/
// verify that std::ctype<>::widen() is called at least once
// for every distinct narow character (not for fill characters)
// note that an implementation is allowed to cache the results
// of facet virtual functions() called with the same arguments
// (including the implicit this pointer), so the numbers here
// are minimum required values but not exact or maximums
void widen_test ()
{
rw_info (0, 0, 0,
"std::num_put<char>::put (..., long) calls ctype<char>::widen()");
static const struct {
long val;
std::streamsize width;
int flags;
int n_widen;
const char *str;
} data[] = {
{ 0, 0, 0, 1, "9" },
{ 1, 0, 0, 1, "8" },
{ 11, 0, 0, 1, "88" },
{ 12, 0, 0, 2, "87" },
{ 123, 0, 0, 3, "876" },
{ 111, 0, 0, 1, "888" },
{ -1234, 0, 0, 5, "-8765" },
{ 12121, 0, 0, 2, "87878" },
{ 12345, 0, showpos, 6, "+87654" },
{ 0123456, 0, oct | showbase, 7, "9876543" },
{ 0x234567, 0, hex | showbase, 8, "9x765432" },
{ 0x2345678, 10, hex | showbase, 9, " 9x7654321" },
{ 9, 5, showpos | internal, 2, "+ 0" }
};
for (unsigned i = 0; i != sizeof data / sizeof *data; ++i) {
const Ctype<char> ctp;
// construct and initialize a basic_ios-derived object
Ios<char> ios;
// construct a num_put-derived facet to exercise
const NumPut<char> np;
// imbue a stream object with a custom locale
// containing the replacement ctype facet
ios.imbue (std::locale (ios.getloc (), (const std::ctype<char>*)&ctp));
ios.flags (std::ios_base::fmtflags (data [i].flags));
ios.width (data [i].width);
char buf [40];
// reset function call counter
Ctype<char>::n_widen_ = 0;
*np.put (buf, ios, ' ', data [i].val) = '\0';
rw_assert (data [i].n_widen <= Ctype<char>::n_widen_, 0, __LINE__,
"%d. num_put<char>::do_put (..., %d) called "
"ctype<char>::do_widen() %d times, %d expected; "
"flags = %{If}, width = %d", i, data [i].val,
Ctype<char>::n_widen_, data [i].n_widen,
data [i].flags, data [i].width);
rw_assert (0 == rw_strncmp (buf, data [i].str), 0, __LINE__,
"%d. num_put<char>::do_put (..., %d) produced %s, "
"expected \"%{#*s}\"; "
"flags = %{If}, width = %d",
i, data [i].val, buf,
int (sizeof data [i].str [0]), data [i].str,
data [i].flags, data [i].width);
}
}
/**************************************************************************/
// verify that std::numpunct<>::grouping() is called and used correctly
void grouping_test ()
{
// construct a "replacement" numpunct-derived facet
const Punct<char> pun (1);
// construct and initialize a basic_ios-derived object
Ios<char> ios;
// construct a num_put-derived facet to exercise
const NumPut<char> np;
// imbue a stream object with a custom locale
// containing the replacement punctuation facet
ios.imbue (std::locale (ios.getloc (), (const std::numpunct<char>*)&pun));
rw_assert (1 == Punct<char>::n_objs_, 0, __LINE__,
"%d facets exist, 1 expected", Punct<char>::n_objs_);
// decimal output, no special formatting
ios.setf (std::ios::fmtflags ());
char buf [40];
// group after every digit
Punct<char>::grouping_ = "\1";
// reset the do_thousands_sep()-call counter
Punct<char>::n_thousands_sep_ = 0;
*np.put (buf, ios, '\0', 123456789L) = '\0';
// verify that the number was formatted correctly
rw_assert (0 == std::strcmp (buf, "1,2,3,4,5,6,7,8,9"), 0, __LINE__,
"num_put<char>::do_put (..., 123456789) produced \"%s\", "
"expected \"%s\"", buf, "1,2,3,4,5,6,7,8,9");
// verify that do_thousands_sep() was called at least once
rw_assert (0 < Punct<char>::n_thousands_sep_, 0, __LINE__,
"num_put<char>::do_put (..., 123456789) failed to call "
"numpunct<char>::do_thousands_sep()");
// repeat test to verify that distinct instances
// of the same facet do not cache each other's data
{
// nested scope works around a SunPro bug (PR #27543)
ios.imbue (std::locale (ios.getloc (),
(const std::numpunct<char>*)new Punct<char>(0)));
}
rw_assert (2 == Punct<char>::n_objs_, 0, __LINE__,
"%d facets exist, 2 expected", Punct<char>::n_objs_);
// now group after every other digit
Punct<char>::grouping_ = "\2";
// reset the do_thousands_sep()-call counter
Punct<char>::n_thousands_sep_ = 0;
*np.put (buf, ios, '\0', 123456790L) = '\0';
// again verify that the number was formatted correctly
rw_assert (0 == std::strcmp (buf, "1,23,45,67,90"), 0, __LINE__,
"num_put<char>::do_put (..., 123456790) produced \"%s\", "
"expected \"%s\"", buf, "1,23,45,67,90");
// verify that do_thousands_sep() was called at least once
// (i.e., that there is no unwarranted caching going on)
rw_assert (0 < Punct<char>::n_thousands_sep_, 0, __LINE__,
"num_put<char>::do_put (..., 123456790) failed to call "
"numpunct<char>::do_thousands_sep()");
// replace imbued locale with a copy of the classic locale
// and remove the previously installed locale containing
// the user-defined facet; this must destroy the dynamically
// created facet leaving only the local facet on the stack
{
// nested scope works around a SunPro bug (PR #27543)
ios.imbue (std::locale::classic ());
}
rw_assert (1 == Punct<char>::n_objs_, 0, __LINE__,
"%d facets exist, 1 expected", Punct<char>::n_objs_);
}
/**************************************************************************/
template <class charT>
void numput_virtuals_test (charT, const char *cname)
{
const NumPut<charT> put (1);
const std::num_put<charT, charT*> &np = put;
Ios<charT> ios;
ios.flags (std::ios_base::fmtflags ());
#define Abs(i) ((i) < 0 ? (-i) : (i))
#define ASSERT(T, N, T2, tname, t2name) \
do { \
/* number of expected calls to the do_put() member */ \
const int expect = N + put.ncalls_ [put.test_ ## T]; \
const int expect_2 = !N + put.ncalls_ [put.test_ ## T2]; \
charT buf [80]; \
const charT* const end = np.put (buf, ios, charT (), (T)0); \
rw_assert (end == buf + 1, 0, __LINE__, \
"line %d: num_put<%s>::put (..., %s = 0) " \
"return value", __LINE__, cname, tname); \
const int ncalls = put.ncalls_ [put.test_ ## T]; \
const int ncalls_2 = put.ncalls_ [put.test_ ## T2]; \
rw_assert (expect == ncalls, 0, __LINE__, \
"line %d: num_put<%s>::put (..., %s) called " \
"do_put(..., %s) %d times, expected %d", \
__LINE__, cname, tname, \
tname, Abs (ncalls - expect - N), N); \
/* make sure extensions, if any, are implemented in */ \
/* terms of and call the standard virtual functions */ \
/* bogus addressof operator prevents warnings about */ \
/* unreachable code when T is the same as T2 */ \
if (put.test_ ## T != put.test_ ## T2) \
rw_assert (expect_2 == ncalls_2, 0, __LINE__, \
"line %d: num_put<%s>::put (..., %s) called " \
"do_put(..., %s) %d times, expected %d", \
__LINE__, cname, tname, \
t2name, Abs (expect_2 - ncalls_2 - !N), !N); \
} while (0)
typedef unsigned short ushort;
typedef unsigned int uint;
typedef unsigned long ulong;
// verify that each public member function calls each corresponding
// implementation do_xxx() member functions exactly once
// in the case of extensions, verify that each extended public member
// function calls the appropriate implementation standard do_xxx()
// member function
#ifndef _RWSTD_NO_BOOL
ASSERT (bool, 1, bool, "bool", "bool");
#endif // _RWSTD_NO_BOOL
ASSERT (long, 1, long, "long", "long");
ASSERT (ulong, 1, ulong, "unsigned long", "unsigned long");
#if defined (_RWSTD_LONG_LONG) && !defined (_RWSTD_NO_EXT_NUM_PUT)
typedef _RWSTD_LONG_LONG llong;
typedef unsigned _RWSTD_LONG_LONG ullong;
ASSERT (llong, 1, llong, "long long", "long long");
ASSERT (ullong, 1, ullong, "unsigned long long", "unsigned long long");
#endif // _RWSTD_LONG_LONG && !_RWSTD_NO_EXT_NUM_PUT
ASSERT (double, 1, double, "double", "double");
#ifndef _RWSTD_NO_LONG_DOUBLE
typedef long double ldouble;
ASSERT (ldouble, 1, ldouble, "long double", "long double");
#endif // _RWSTD_NO_LONG_DOUBLE
typedef const void* pvoid;
ASSERT (pvoid, 1, pvoid, "const void*", "const void*");
}
/**************************************************************************/
template <class charT>
void run_tests (charT, const char *cname)
{
numput_virtuals_test (charT (), cname);
direct_use_test (charT (), cname);
#undef TEST
#define TEST(T, tname) \
if (rw_enabled (#T)) \
T ## _test (charT (), cname); \