fmtlegacy/format.cc

1271 lines
36 KiB
C++
Raw Normal View History

2012-12-07 16:31:09 +00:00
/*
Formatting library for C++
2012-12-12 15:44:41 +00:00
Copyright (c) 2012 - 2014, Victor Zverovich
2012-12-12 15:44:41 +00:00
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2012-12-07 16:31:09 +00:00
*/
// Disable useless MSVC warnings.
#undef _CRT_SECURE_NO_WARNINGS
#define _CRT_SECURE_NO_WARNINGS
2012-12-21 23:02:25 +00:00
#undef _SCL_SECURE_NO_WARNINGS
#define _SCL_SECURE_NO_WARNINGS
2013-01-14 23:16:20 +00:00
#include "format.h"
2014-04-30 13:55:21 +00:00
#include <string.h>
2012-12-12 17:17:28 +00:00
#include <cctype>
2014-07-01 23:23:50 +00:00
#include <cerrno>
2014-04-24 19:37:06 +00:00
#include <climits>
2013-09-07 17:15:08 +00:00
#include <cmath>
#include <cstdarg>
2013-09-07 17:15:08 +00:00
2014-04-30 13:55:21 +00:00
#ifdef _WIN32
2014-05-01 14:09:42 +00:00
# define WIN32_LEAN_AND_MEAN
2014-06-10 22:38:57 +00:00
# ifdef __MINGW32__
# include <cstring>
# endif
2014-04-30 13:55:21 +00:00
# include <windows.h>
2014-04-30 14:41:54 +00:00
# undef ERROR
2014-04-30 13:55:21 +00:00
#endif
2014-07-14 13:55:29 +00:00
using fmt::LongLong;
2014-02-15 18:48:34 +00:00
using fmt::ULongLong;
2014-07-02 13:33:25 +00:00
using fmt::internal::Arg;
2014-02-15 18:48:34 +00:00
2014-03-11 18:56:24 +00:00
#if _MSC_VER
# pragma warning(push)
# pragma warning(disable: 4127) // conditional expression is constant
#endif
2013-09-07 17:15:08 +00:00
namespace {
#ifndef _MSC_VER
2014-07-29 16:14:07 +00:00
// Portable version of signbit.
// When compiled in C++11 mode signbit is no longer a macro but a function
// defined in namespace std and the macro is undefined.
2014-07-29 16:14:07 +00:00
inline int getsign(double x) {
#ifdef signbit
return signbit(x);
#else
return std::signbit(x);
#endif
2014-07-29 16:14:07 +00:00
}
2013-09-07 17:15:08 +00:00
2014-07-29 14:49:34 +00:00
// Portable version of isinf.
#ifdef isinf
2014-08-13 15:01:51 +00:00
inline int isinfinity(double x) { return isinf(x); }
inline int isinfinity(long double x) { return isinf(x); }
2014-07-29 14:49:34 +00:00
#else
2014-08-13 15:01:51 +00:00
inline int isinfinity(double x) { return std::isinf(x); }
inline int isinfinity(long double x) { return std::isinf(x); }
2013-09-07 17:15:08 +00:00
#endif
2014-08-13 14:51:02 +00:00
2013-09-07 17:15:08 +00:00
#define FMT_SNPRINTF snprintf
#else // _MSC_VER
2013-09-07 17:15:08 +00:00
2014-07-29 16:14:07 +00:00
inline int getsign(double value) {
2013-09-07 17:15:08 +00:00
if (value < 0) return 1;
if (value == value) return 0;
int dec = 0, sign = 0;
char buffer[2]; // The buffer size must be >= 2 or _ecvt_s will fail.
_ecvt_s(buffer, sizeof(buffer), value, 0, &dec, &sign);
return sign;
}
2014-07-29 15:22:52 +00:00
inline int isinfinity(double x) { return !_finite(x); }
2013-09-07 17:15:08 +00:00
2014-08-19 15:47:38 +00:00
inline int fmt_snprintf(char *buffer, size_t size, const char *format, ...) {
va_list args;
va_start(args, format);
int result = vsnprintf_s(buffer, size, _TRUNCATE, format, args);
va_end(args);
return result;
}
2014-08-19 15:47:38 +00:00
#define FMT_SNPRINTF fmt_snprintf
2013-09-07 17:15:08 +00:00
#endif // _MSC_VER
template <typename T>
struct IsLongDouble { enum {VALUE = 0}; };
template <>
struct IsLongDouble<long double> { enum {VALUE = 1}; };
2014-08-17 14:53:55 +00:00
// Checks if a value fits in int - used to avoid warnings about comparing
// signed and unsigned integers.
template <bool IsSigned>
struct IntChecker {
template <typename T>
static bool fits_in_int(T value) {
unsigned max = INT_MAX;
return value <= max;
}
};
template <>
struct IntChecker<true> {
template <typename T>
static bool fits_in_int(T value) {
return value >= INT_MIN && value <= INT_MAX;
}
};
const char RESET_COLOR[] = "\x1b[0m";
typedef void (*FormatFunc)(fmt::Writer &, int , fmt::StringRef);
void report_error(FormatFunc func,
int error_code, fmt::StringRef message) FMT_NOEXCEPT(true) {
try {
fmt::Writer full_message;
2014-06-21 15:32:00 +00:00
func(full_message, error_code, message); // TODO: make sure this doesn't throw
std::fwrite(full_message.c_str(), full_message.size(), 1, stderr);
std::fputc('\n', stderr);
} catch (...) {}
}
// IsZeroInt::visit(arg) returns true iff arg is a zero integer.
class IsZeroInt : public fmt::internal::ArgVisitor<IsZeroInt, bool> {
public:
template <typename T>
bool visit_any_int(T value) { return value == 0; }
};
2014-07-02 13:33:25 +00:00
// Parses an unsigned integer advancing s to the end of the parsed input.
// This function assumes that the first character of s is a digit.
template <typename Char>
int parse_nonnegative_int(const Char *&s) {
2014-07-02 13:33:25 +00:00
assert('0' <= *s && *s <= '9');
unsigned value = 0;
do {
unsigned new_value = value * 10 + (*s++ - '0');
// Check if value wrapped around.
if (new_value < value) {
value = UINT_MAX;
break;
}
value = new_value;
2014-07-02 13:33:25 +00:00
} while ('0' <= *s && *s <= '9');
if (value > INT_MAX)
throw fmt::FormatError("number is too big");
2014-07-02 13:33:25 +00:00
return value;
}
template <typename Char>
const Char *find_closing_brace(const Char *s, int num_open_braces = 1) {
for (int n = num_open_braces; *s; ++s) {
if (*s == '{') {
++n;
} else if (*s == '}') {
if (--n == 0)
return s;
}
}
throw fmt::FormatError("unmatched '{' in format");
}
2014-07-14 13:55:29 +00:00
template <typename Char>
void check_sign(const Char *&s, const Arg &arg) {
char sign = static_cast<char>(*s);
if (arg.type > Arg::LAST_NUMERIC_TYPE) {
throw fmt::FormatError(fmt::format(
"format specifier '{}' requires numeric argument", sign));
}
if (arg.type == Arg::UINT || arg.type == Arg::ULONG_LONG) {
throw fmt::FormatError(fmt::format(
"format specifier '{}' requires signed argument", sign));
}
++s;
}
// Checks if an argument is a valid printf width specifier and sets
// left alignment if it is negative.
class WidthHandler : public fmt::internal::ArgVisitor<WidthHandler, unsigned> {
2014-07-14 13:55:29 +00:00
private:
fmt::FormatSpec &spec_;
2014-07-14 13:55:29 +00:00
public:
explicit WidthHandler(fmt::FormatSpec &spec) : spec_(spec) {}
unsigned visit_unhandled_arg() {
2014-07-14 13:55:29 +00:00
throw fmt::FormatError("width is not integer");
}
template <typename T>
unsigned visit_any_int(T value) {
typedef typename fmt::internal::IntTraits<T>::MainType UnsignedType;
UnsignedType width = value;
if (fmt::internal::is_negative(value)) {
2014-07-14 13:55:29 +00:00
spec_.align_ = fmt::ALIGN_LEFT;
width = 0 - width;
}
if (width > INT_MAX)
throw fmt::FormatError("number is too big");
return static_cast<unsigned>(width);
2014-07-14 13:55:29 +00:00
}
};
class PrecisionHandler :
public fmt::internal::ArgVisitor<PrecisionHandler, int> {
public:
unsigned visit_unhandled_arg() {
throw fmt::FormatError("precision is not integer");
}
template <typename T>
int visit_any_int(T value) {
2014-08-17 14:53:55 +00:00
if (!IntChecker<std::numeric_limits<T>::is_signed>::fits_in_int(value))
throw fmt::FormatError("number is too big");
return static_cast<int>(value);
}
};
2014-08-28 15:11:21 +00:00
// Converts an integer argument to an integral type T for printf.
2014-07-30 14:37:16 +00:00
template <typename T>
class ArgConverter : public fmt::internal::ArgVisitor<ArgConverter<T>, void> {
private:
fmt::internal::Arg &arg_;
2014-08-08 13:51:09 +00:00
wchar_t type_;
2014-07-30 14:37:16 +00:00
public:
2014-08-08 13:51:09 +00:00
ArgConverter(fmt::internal::Arg &arg, wchar_t type)
: arg_(arg), type_(type) {}
2014-07-30 14:37:16 +00:00
template <typename U>
void visit_any_int(U value) {
2014-07-31 14:43:14 +00:00
bool is_signed = type_ == 'd' || type_ == 'i';
using fmt::internal::Arg;
if (sizeof(T) <= sizeof(int)) {
2014-08-28 15:11:21 +00:00
// Extra casts are used to silence warnings.
2014-07-31 14:43:14 +00:00
if (is_signed) {
arg_.type = Arg::INT;
2014-08-15 16:03:59 +00:00
arg_.int_value = static_cast<int>(static_cast<T>(value));
2014-07-31 14:43:14 +00:00
} else {
arg_.type = Arg::UINT;
2014-08-18 14:03:12 +00:00
arg_.uint_value = static_cast<unsigned>(
static_cast<typename fmt::internal::MakeUnsigned<T>::Type>(value));
2014-07-31 14:43:14 +00:00
}
2014-07-30 14:37:16 +00:00
} else {
2014-07-31 14:43:14 +00:00
if (is_signed) {
arg_.type = Arg::LONG_LONG;
2014-08-01 14:15:27 +00:00
arg_.long_long_value =
static_cast<typename fmt::internal::MakeUnsigned<U>::Type>(value);
2014-07-31 14:43:14 +00:00
} else {
arg_.type = Arg::ULONG_LONG;
arg_.ulong_long_value =
2014-08-01 14:15:27 +00:00
static_cast<typename fmt::internal::MakeUnsigned<U>::Type>(value);
2014-07-31 14:43:14 +00:00
}
2014-07-30 14:37:16 +00:00
}
}
};
2014-08-28 15:11:21 +00:00
// Converts an integer argument to char for printf.
2014-08-19 15:14:21 +00:00
class CharConverter : public fmt::internal::ArgVisitor<CharConverter, void> {
private:
fmt::internal::Arg &arg_;
public:
explicit CharConverter(fmt::internal::Arg &arg) : arg_(arg) {}
template <typename T>
void visit_any_int(T value) {
arg_.type = Arg::CHAR;
arg_.int_value = static_cast<char>(value);
}
};
// This function template is used to prevent compile errors when handling
// incompatible string arguments, e.g. handling a wide string in a narrow
// string formatter.
template <typename Char>
Arg::StringValue<Char> ignore_incompatible_str(Arg::StringValue<wchar_t>);
template <>
inline Arg::StringValue<char> ignore_incompatible_str(
Arg::StringValue<wchar_t>) { return Arg::StringValue<char>(); }
template <>
inline Arg::StringValue<wchar_t> ignore_incompatible_str(
Arg::StringValue<wchar_t> s) { return s; }
} // namespace
2013-09-07 17:15:08 +00:00
2014-06-30 21:26:29 +00:00
void fmt::SystemError::init(
int error_code, StringRef format_str, const ArgList &args) {
error_code_ = error_code;
Writer w;
internal::format_system_error(w, error_code, format(format_str, args));
2014-06-30 21:26:29 +00:00
std::runtime_error &base = *this;
base = std::runtime_error(w.str());
}
template <typename T>
int fmt::internal::CharTraits<char>::format_float(
char *buffer, std::size_t size, const char *format,
unsigned width, int precision, T value) {
if (width == 0) {
2013-09-07 17:15:08 +00:00
return precision < 0 ?
FMT_SNPRINTF(buffer, size, format, value) :
FMT_SNPRINTF(buffer, size, format, precision, value);
2013-09-07 17:15:08 +00:00
}
return precision < 0 ?
FMT_SNPRINTF(buffer, size, format, width, value) :
FMT_SNPRINTF(buffer, size, format, width, precision, value);
}
template <typename T>
int fmt::internal::CharTraits<wchar_t>::format_float(
wchar_t *buffer, std::size_t size, const wchar_t *format,
unsigned width, int precision, T value) {
if (width == 0) {
2013-09-07 17:15:08 +00:00
return precision < 0 ?
swprintf(buffer, size, format, value) :
swprintf(buffer, size, format, precision, value);
2013-09-07 17:15:08 +00:00
}
return precision < 0 ?
swprintf(buffer, size, format, width, value) :
swprintf(buffer, size, format, width, precision, value);
}
const char fmt::internal::DIGITS[] =
"0001020304050607080910111213141516171819"
"2021222324252627282930313233343536373839"
"4041424344454647484950515253545556575859"
"6061626364656667686970717273747576777879"
"8081828384858687888990919293949596979899";
2014-02-19 20:43:55 +00:00
2014-02-19 21:02:22 +00:00
#define FMT_POWERS_OF_10(factor) \
factor * 10, \
factor * 100, \
factor * 1000, \
factor * 10000, \
factor * 100000, \
factor * 1000000, \
factor * 10000000, \
factor * 100000000, \
factor * 1000000000
const uint32_t fmt::internal::POWERS_OF_10_32[] = {0, FMT_POWERS_OF_10(1)};
2014-02-19 20:43:55 +00:00
const uint64_t fmt::internal::POWERS_OF_10_64[] = {
0,
2014-02-19 21:02:22 +00:00
FMT_POWERS_OF_10(1),
FMT_POWERS_OF_10(ULongLong(1000000000)),
2014-08-19 15:47:38 +00:00
// Multiply several constants instead of using a single long long constant
2014-02-19 21:02:22 +00:00
// to avoid warnings about C++98 not supporting long long.
2014-02-15 19:16:44 +00:00
ULongLong(1000000000) * ULongLong(1000000000) * 10
};
void fmt::internal::report_unknown_type(char code, const char *type) {
if (std::isprint(static_cast<unsigned char>(code))) {
2014-06-30 13:43:53 +00:00
throw fmt::FormatError(
fmt::format("unknown format code '{}' for {}", code, type));
}
throw fmt::FormatError(
2014-06-30 13:43:53 +00:00
fmt::format("unknown format code '\\x{:02x}' for {}",
static_cast<unsigned>(code), type));
}
2013-09-07 03:23:42 +00:00
2014-04-30 14:23:43 +00:00
#ifdef _WIN32
fmt::internal::UTF8ToUTF16::UTF8ToUTF16(fmt::StringRef s) {
int length = MultiByteToWideChar(
CP_UTF8, MB_ERR_INVALID_CHARS, s.c_str(), -1, 0, 0);
static const char ERROR[] = "cannot convert string from UTF-8 to UTF-16";
if (length == 0)
throw WindowsError(GetLastError(), ERROR);
2014-04-30 14:23:43 +00:00
buffer_.resize(length);
length = MultiByteToWideChar(
CP_UTF8, MB_ERR_INVALID_CHARS, s.c_str(), -1, &buffer_[0], length);
if (length == 0)
throw WindowsError(GetLastError(), ERROR);
2014-04-30 14:23:43 +00:00
}
fmt::internal::UTF16ToUTF8::UTF16ToUTF8(fmt::WStringRef s) {
if (int error_code = convert(s)) {
2014-07-09 15:39:01 +00:00
throw WindowsError(error_code,
2014-04-30 14:23:43 +00:00
"cannot convert string from UTF-16 to UTF-8");
}
}
int fmt::internal::UTF16ToUTF8::convert(fmt::WStringRef s) {
2014-04-30 14:23:43 +00:00
int length = WideCharToMultiByte(CP_UTF8, 0, s.c_str(), -1, 0, 0, 0, 0);
if (length == 0)
return GetLastError();
buffer_.resize(length);
length = WideCharToMultiByte(
CP_UTF8, 0, s.c_str(), -1, &buffer_[0], length, 0, 0);
if (length == 0)
return GetLastError();
return 0;
}
2014-06-30 21:26:29 +00:00
void fmt::WindowsError::init(
int error_code, StringRef format_str, const ArgList &args) {
error_code_ = error_code;
Writer w;
internal::format_windows_error(w, error_code, format(format_str, args));
2014-06-30 21:26:29 +00:00
std::runtime_error &base = *this;
base = std::runtime_error(w.str());
}
2014-04-30 14:23:43 +00:00
#endif
int fmt::internal::safe_strerror(
int error_code, char *&buffer, std::size_t buffer_size) FMT_NOEXCEPT(true) {
assert(buffer != 0 && buffer_size != 0);
int result = 0;
#ifdef _GNU_SOURCE
char *message = strerror_r(error_code, buffer, buffer_size);
// If the buffer is full then the message is probably truncated.
if (message == buffer && strlen(buffer) == buffer_size - 1)
result = ERANGE;
buffer = message;
2014-07-09 16:45:18 +00:00
#elif __MINGW32__
errno = 0;
(void)buffer_size;
buffer = strerror(error_code);
result = errno;
#elif _WIN32
result = strerror_s(buffer, buffer_size, error_code);
// If the buffer is full then the message is probably truncated.
if (result == 0 && std::strlen(buffer) == buffer_size - 1)
result = ERANGE;
#else
result = strerror_r(error_code, buffer, buffer_size);
if (result == -1)
result = errno; // glibc versions before 2.13 return result in errno.
2014-04-30 13:55:21 +00:00
#endif
return result;
2014-04-30 13:55:21 +00:00
}
2013-09-07 03:23:42 +00:00
void fmt::internal::format_system_error(
fmt::Writer &out, int error_code, fmt::StringRef message) {
Array<char, INLINE_BUFFER_SIZE> buffer;
buffer.resize(INLINE_BUFFER_SIZE);
char *system_message = 0;
for (;;) {
system_message = &buffer[0];
int result = safe_strerror(error_code, system_message, buffer.size());
if (result == 0)
break;
if (result != ERANGE) {
// Can't get error message, report error code instead.
out << message << ": error code = " << error_code;
return;
}
buffer.resize(buffer.size() * 2);
}
out << message << ": " << system_message;
}
#ifdef _WIN32
void fmt::internal::format_windows_error(
fmt::Writer &out, int error_code, fmt::StringRef message) {
class String {
private:
LPWSTR str_;
public:
String() : str_() {}
~String() { LocalFree(str_); }
LPWSTR *ptr() { return &str_; }
LPCWSTR c_str() const { return str_; }
};
String system_message;
if (FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, 0,
error_code, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
reinterpret_cast<LPWSTR>(system_message.ptr()), 0, 0)) {
UTF16ToUTF8 utf8_message;
if (!utf8_message.convert(system_message.c_str())) {
2014-07-01 00:40:53 +00:00
out << message << ": " << utf8_message;
return;
}
}
// Can't get error message, report error code instead.
out << message << ": error code = " << error_code;
}
#endif
// An argument formatter.
template <typename Char>
class fmt::internal::ArgFormatter :
public fmt::internal::ArgVisitor<fmt::internal::ArgFormatter<Char>, void> {
private:
2014-07-16 15:38:15 +00:00
fmt::BasicFormatter<Char> &formatter_;
fmt::BasicWriter<Char> &writer_;
fmt::FormatSpec &spec_;
const Char *format_;
public:
2014-07-26 16:45:03 +00:00
ArgFormatter(
fmt::BasicFormatter<Char> &f,fmt::FormatSpec &s, const Char *fmt)
2014-07-16 15:38:15 +00:00
: formatter_(f), writer_(f.writer()), spec_(s), format_(fmt) {}
template <typename T>
void visit_any_int(T value) { writer_.write_int(value, spec_); }
template <typename T>
void visit_any_double(T value) { writer_.write_double(value, spec_); }
void visit_char(int value) {
if (spec_.type_ && spec_.type_ != 'c') {
spec_.flags_ |= CHAR_FLAG;
writer_.write_int(value, spec_);
return;
}
if (spec_.align_ == ALIGN_NUMERIC || spec_.flags_ != 0)
throw FormatError("invalid format specifier for char");
typedef typename fmt::BasicWriter<Char>::CharPtr CharPtr;
CharPtr out = CharPtr();
if (spec_.width_ > 1) {
Char fill = static_cast<Char>(spec_.fill());
out = writer_.grow_buffer(spec_.width_);
if (spec_.align_ == fmt::ALIGN_RIGHT) {
std::fill_n(out, spec_.width_ - 1, fill);
out += spec_.width_ - 1;
} else if (spec_.align_ == fmt::ALIGN_CENTER) {
out = writer_.fill_padding(out, spec_.width_, 1, fill);
} else {
std::fill_n(out + 1, spec_.width_ - 1, fill);
}
} else {
out = writer_.grow_buffer(1);
}
*out = static_cast<Char>(value);
}
void visit_string(Arg::StringValue<char> value) {
writer_.write_str(value, spec_);
}
void visit_wstring(Arg::StringValue<wchar_t> value) {
writer_.write_str(ignore_incompatible_str<Char>(value), spec_);
}
void visit_pointer(const void *value) {
if (spec_.type_ && spec_.type_ != 'p')
fmt::internal::report_unknown_type(spec_.type_, "pointer");
spec_.flags_ = fmt::HASH_FLAG;
spec_.type_ = 'x';
writer_.write_int(reinterpret_cast<uintptr_t>(value), spec_);
}
void visit_custom(Arg::CustomValue c) {
2014-07-16 15:38:15 +00:00
c.format(&formatter_, c.value, format_);
}
};
2013-09-07 03:23:42 +00:00
// Fills the padding around the content and returns the pointer to the
// content area.
template <typename Char>
typename fmt::BasicWriter<Char>::CharPtr
fmt::BasicWriter<Char>::fill_padding(CharPtr buffer,
unsigned total_size, std::size_t content_size, wchar_t fill) {
2013-09-07 03:23:42 +00:00
std::size_t padding = total_size - content_size;
std::size_t left_padding = padding / 2;
2013-09-08 20:47:06 +00:00
Char fill_char = static_cast<Char>(fill);
std::fill_n(buffer, left_padding, fill_char);
2013-09-07 03:23:42 +00:00
buffer += left_padding;
CharPtr content = buffer;
2013-09-08 20:47:06 +00:00
std::fill_n(buffer + content_size, padding - left_padding, fill_char);
2013-09-07 03:23:42 +00:00
return content;
}
template <typename Char>
template <typename T>
void fmt::BasicWriter<Char>::write_double(T value, const FormatSpec &spec) {
2013-09-07 03:23:42 +00:00
// Check type.
char type = spec.type();
bool upper = false;
switch (type) {
case 0:
type = 'g';
break;
case 'e': case 'f': case 'g': case 'a':
2013-09-07 03:23:42 +00:00
break;
case 'F':
#ifdef _MSC_VER
// MSVC's printf doesn't support 'F'.
type = 'f';
#endif
// Fall through.
case 'E': case 'G': case 'A':
2013-09-07 03:23:42 +00:00
upper = true;
break;
default:
internal::report_unknown_type(type, "double");
2013-09-07 03:23:42 +00:00
break;
}
char sign = 0;
2014-07-29 16:14:07 +00:00
// Use getsign instead of value < 0 because the latter is always
2013-09-07 03:23:42 +00:00
// false for NaN.
2014-07-29 16:14:07 +00:00
if (getsign(static_cast<double>(value))) {
2013-09-07 03:23:42 +00:00
sign = '-';
value = -value;
2014-07-25 15:29:06 +00:00
} else if (spec.flag(SIGN_FLAG)) {
sign = spec.flag(PLUS_FLAG) ? '+' : ' ';
2013-09-07 03:23:42 +00:00
}
if (value != value) {
// Format NaN ourselves because sprintf's output is not consistent
// across platforms.
std::size_t size = 4;
const char *nan = upper ? " NAN" : " nan";
if (!sign) {
--size;
++nan;
}
2014-06-30 21:26:29 +00:00
CharPtr out = write_str(nan, size, spec);
2013-09-07 03:23:42 +00:00
if (sign)
*out = sign;
return;
}
2014-08-13 14:51:02 +00:00
if (isinfinity(value)) {
2013-09-07 03:23:42 +00:00
// Format infinity ourselves because sprintf's output is not consistent
// across platforms.
std::size_t size = 4;
const char *inf = upper ? " INF" : " inf";
if (!sign) {
--size;
++inf;
}
2014-06-30 21:26:29 +00:00
CharPtr out = write_str(inf, size, spec);
2013-09-07 03:23:42 +00:00
if (sign)
*out = sign;
return;
}
std::size_t offset = buffer_.size();
unsigned width = spec.width();
if (sign) {
buffer_.reserve(buffer_.size() + (std::max)(width, 1u));
if (width > 0)
--width;
++offset;
}
// Build format string.
enum { MAX_FORMAT_SIZE = 10}; // longest format: %#-*.*Lg
Char format[MAX_FORMAT_SIZE];
Char *format_ptr = format;
*format_ptr++ = '%';
unsigned width_for_sprintf = width;
2014-07-25 15:29:06 +00:00
if (spec.flag(HASH_FLAG))
2013-09-07 03:23:42 +00:00
*format_ptr++ = '#';
if (spec.align() == ALIGN_CENTER) {
width_for_sprintf = 0;
} else {
if (spec.align() == ALIGN_LEFT)
*format_ptr++ = '-';
if (width != 0)
*format_ptr++ = '*';
}
2014-06-21 15:32:00 +00:00
if (spec.precision() >= 0) {
2013-09-07 03:23:42 +00:00
*format_ptr++ = '.';
*format_ptr++ = '*';
}
if (IsLongDouble<T>::VALUE)
2013-09-07 03:23:42 +00:00
*format_ptr++ = 'L';
*format_ptr++ = type;
*format_ptr = '\0';
// Format using snprintf.
2013-09-08 20:30:14 +00:00
Char fill = static_cast<Char>(spec.fill());
2013-09-07 03:23:42 +00:00
for (;;) {
std::size_t size = buffer_.capacity() - offset;
#if _MSC_VER
// MSVC's vsnprintf_s doesn't work with zero size, so reserve
// space for at least one extra character to make the size non-zero.
// Note that the buffer's capacity will increase by more than 1.
if (size == 0) {
buffer_.reserve(offset + 1);
size = buffer_.capacity() - offset;
}
#endif
2013-09-07 03:23:42 +00:00
Char *start = &buffer_[offset];
int n = internal::CharTraits<Char>::format_float(
2014-06-21 15:32:00 +00:00
start, size, format, width_for_sprintf, spec.precision(), value);
2013-09-07 03:23:42 +00:00
if (n >= 0 && offset + n < buffer_.capacity()) {
if (sign) {
if ((spec.align() != ALIGN_RIGHT && spec.align() != ALIGN_DEFAULT) ||
*start != ' ') {
*(start - 1) = sign;
sign = 0;
} else {
2013-09-08 20:30:14 +00:00
*(start - 1) = fill;
2013-09-07 03:23:42 +00:00
}
++n;
}
if (spec.align() == ALIGN_CENTER &&
spec.width() > static_cast<unsigned>(n)) {
unsigned width = spec.width();
CharPtr p = grow_buffer(width);
2013-09-07 03:23:42 +00:00
std::copy(p, p + n, p + (width - n) / 2);
fill_padding(p, spec.width(), n, fill);
2013-09-07 03:23:42 +00:00
return;
}
if (spec.fill() != ' ' || sign) {
while (*start == ' ')
2013-09-08 20:30:14 +00:00
*start++ = fill;
2013-09-07 03:23:42 +00:00
if (sign)
*(start - 1) = sign;
}
grow_buffer(n);
2013-09-07 03:23:42 +00:00
return;
}
2014-04-24 01:37:49 +00:00
// If n is negative we ask to increase the capacity by at least 1,
// but as std::vector, the buffer grows exponentially.
buffer_.reserve(n >= 0 ? offset + n + 1 : buffer_.capacity() + 1);
2013-09-07 03:23:42 +00:00
}
}
template <typename Char>
2014-07-26 16:45:03 +00:00
template <typename StrChar>
2014-06-30 21:26:29 +00:00
void fmt::BasicWriter<Char>::write_str(
2014-07-26 16:45:03 +00:00
const Arg::StringValue<StrChar> &str, const FormatSpec &spec) {
// Check if StrChar is convertible to Char.
internal::CharTraits<Char>::convert(StrChar());
if (spec.type_ && spec.type_ != 's')
internal::report_unknown_type(spec.type_, "string");
2014-07-26 16:45:03 +00:00
const StrChar *s = str.value;
std::size_t size = str.size;
if (size == 0) {
if (!s)
throw FormatError("string pointer is null");
if (*s)
2014-07-26 16:45:03 +00:00
size = std::char_traits<StrChar>::length(s);
}
2014-06-30 21:26:29 +00:00
write_str(s, size, spec);
}
template <typename Char>
inline const Arg &fmt::BasicFormatter<Char>::parse_arg_index(const Char *&s) {
2014-08-27 16:13:42 +00:00
const char *error = 0;
const Arg *arg = *s < '0' || *s > '9' ?
next_arg(error) : get_arg(parse_nonnegative_int(s), error);
2014-08-27 16:13:42 +00:00
if (error)
throw FormatError(*s != '}' && *s != ':' ? "invalid format string" : error);
return *arg;
2013-09-07 03:23:42 +00:00
}
const Arg *fmt::internal::FormatterBase::next_arg(const char *&error) {
2014-07-16 15:38:15 +00:00
if (next_arg_index_ < 0) {
2014-08-27 16:13:42 +00:00
error = "cannot switch from manual to automatic argument indexing";
return 0;
2014-07-16 15:38:15 +00:00
}
unsigned arg_index = next_arg_index_++;
if (arg_index < args_.size())
return &args_[arg_index];
2014-08-27 16:13:42 +00:00
error = "argument index is out of range in format";
return 0;
2014-07-16 15:38:15 +00:00
}
const Arg *fmt::internal::FormatterBase::get_arg(
unsigned arg_index, const char *&error) {
if (next_arg_index_ > 0) {
error = "cannot switch from automatic to manual argument indexing";
return 0;
2014-07-16 15:38:15 +00:00
}
next_arg_index_ = -1;
if (arg_index < args_.size())
return &args_[arg_index];
error = "argument index is out of range in format";
return 0;
2014-07-16 15:38:15 +00:00
}
template <typename Char>
void fmt::internal::PrintfFormatter<Char>::parse_flags(
2014-06-19 14:40:35 +00:00
FormatSpec &spec, const Char *&s) {
for (;;) {
2014-06-07 15:57:55 +00:00
switch (*s++) {
case '-':
spec.align_ = ALIGN_LEFT;
break;
case '+':
spec.flags_ |= SIGN_FLAG | PLUS_FLAG;
break;
case '0':
spec.fill_ = '0';
2014-06-07 15:57:55 +00:00
break;
case ' ':
2014-06-07 15:57:55 +00:00
spec.flags_ |= SIGN_FLAG;
break;
case '#':
2014-06-19 14:40:35 +00:00
spec.flags_ |= HASH_FLAG;
break;
default:
2014-06-07 15:57:55 +00:00
--s;
return;
}
}
}
template <typename Char>
const Arg &fmt::internal::PrintfFormatter<Char>::get_arg(
const Char *s, unsigned arg_index) {
const char *error = 0;
const Arg *arg = arg_index == UINT_MAX ?
next_arg(error) : FormatterBase::get_arg(arg_index - 1, error);
if (error)
throw FormatError(!*s ? "invalid format string" : error);
return *arg;
}
2014-06-19 14:40:35 +00:00
template <typename Char>
unsigned fmt::internal::PrintfFormatter<Char>::parse_header(
const Char *&s, FormatSpec &spec) {
2014-06-19 14:40:35 +00:00
unsigned arg_index = UINT_MAX;
Char c = *s;
if (c >= '0' && c <= '9') {
// Parse an argument index (if followed by '$') or a width possibly
2014-06-20 14:34:02 +00:00
// preceded with '0' flag(s).
unsigned value = parse_nonnegative_int(s);
2014-06-19 14:40:35 +00:00
if (*s == '$') { // value is an argument index
++s;
arg_index = value;
} else {
if (c == '0')
spec.fill_ = '0';
if (value != 0) {
// Nonzero value means that we parsed width and don't need to
// parse it or flags again, so return now.
spec.width_ = value;
return arg_index;
}
}
}
parse_flags(spec, s);
2014-06-23 15:10:50 +00:00
// Parse width.
2014-06-19 14:40:35 +00:00
if (*s >= '0' && *s <= '9') {
spec.width_ = parse_nonnegative_int(s);
2014-06-19 14:40:35 +00:00
} else if (*s == '*') {
++s;
2014-08-28 14:48:55 +00:00
spec.width_ = WidthHandler(spec).visit(get_arg(s));
2014-06-19 14:40:35 +00:00
}
return arg_index;
}
template <typename Char>
void fmt::internal::PrintfFormatter<Char>::format(
BasicWriter<Char> &writer, BasicStringRef<Char> format,
const ArgList &args) {
const Char *start = format.c_str();
args_ = args;
next_arg_index_ = 0;
const Char *s = start;
while (*s) {
Char c = *s++;
if (c != '%') continue;
if (*s == c) {
write(writer, start, s);
start = ++s;
continue;
}
write(writer, start, s - 1);
2014-06-19 14:40:35 +00:00
FormatSpec spec;
spec.align_ = ALIGN_RIGHT;
2014-06-23 14:16:46 +00:00
// Parse argument index, flags and width.
unsigned arg_index = parse_header(s, spec);
2014-06-23 14:16:46 +00:00
// Parse precision.
if (*s == '.') {
++s;
if ('0' <= *s && *s <= '9') {
spec.precision_ = parse_nonnegative_int(s);
2014-06-23 14:16:46 +00:00
} else if (*s == '*') {
++s;
2014-08-28 14:48:55 +00:00
spec.precision_ = PrecisionHandler().visit(get_arg(s));
2014-06-23 14:16:46 +00:00
}
}
2014-08-28 14:48:55 +00:00
Arg arg = get_arg(s, arg_index);
if (spec.flag(HASH_FLAG) && IsZeroInt().visit(arg))
2014-06-19 14:40:35 +00:00
spec.flags_ &= ~HASH_FLAG;
if (spec.fill_ == '0') {
if (arg.type <= Arg::LAST_NUMERIC_TYPE)
2014-06-19 14:40:35 +00:00
spec.align_ = ALIGN_NUMERIC;
else
spec.fill_ = ' '; // Ignore '0' flag for non-numeric types.
}
// Parse length and convert the argument to the required type.
switch (*s++) {
case 'h':
if (*s == 'h')
ArgConverter<signed char>(arg, *++s).visit(arg);
else
ArgConverter<short>(arg, *s).visit(arg);
2014-07-30 14:37:16 +00:00
break;
case 'l':
2014-08-12 15:36:19 +00:00
if (*s == 'l')
2014-08-13 13:53:43 +00:00
ArgConverter<fmt::LongLong>(arg, *++s).visit(arg);
2014-08-12 15:36:19 +00:00
else
ArgConverter<long>(arg, *s).visit(arg);
2014-07-31 14:43:14 +00:00
break;
case 'j':
ArgConverter<intmax_t>(arg, *s).visit(arg);
break;
case 'z':
ArgConverter<size_t>(arg, *s).visit(arg);
break;
case 't':
ArgConverter<ptrdiff_t>(arg, *s).visit(arg);
break;
case 'L':
2014-08-19 15:14:21 +00:00
// printf produces garbage when 'L' is omitted for long double, no
// need to do the same.
break;
default:
--s;
ArgConverter<int>(arg, *s).visit(arg);
}
// Parse type.
if (!*s)
throw FormatError("invalid format string");
spec.type_ = static_cast<char>(*s++);
2014-08-19 15:14:21 +00:00
if (arg.type <= Arg::LAST_INTEGER_TYPE) {
// Normalize type.
switch (spec.type_) {
case 'i': case 'u':
spec.type_ = 'd';
2014-08-19 15:14:21 +00:00
break;
case 'c':
// TODO: handle wchar_t
CharConverter(arg).visit(arg);
break;
}
}
start = s;
// Format argument.
2014-06-19 14:40:35 +00:00
switch (arg.type) {
case Arg::INT:
writer.write_int(arg.int_value, spec);
break;
case Arg::UINT:
writer.write_int(arg.uint_value, spec);
break;
case Arg::LONG_LONG:
writer.write_int(arg.long_long_value, spec);
break;
case Arg::ULONG_LONG:
writer.write_int(arg.ulong_long_value, spec);
break;
case Arg::CHAR: {
if (spec.type_ && spec.type_ != 'c')
writer.write_int(arg.int_value, spec);
typedef typename BasicWriter<Char>::CharPtr CharPtr;
CharPtr out = CharPtr();
if (spec.width_ > 1) {
Char fill = ' ';
out = writer.grow_buffer(spec.width_);
if (spec.align_ != ALIGN_LEFT) {
std::fill_n(out, spec.width_ - 1, fill);
out += spec.width_ - 1;
} else {
std::fill_n(out + 1, spec.width_ - 1, fill);
}
} else {
out = writer.grow_buffer(1);
}
2014-06-19 14:40:35 +00:00
*out = static_cast<Char>(arg.int_value);
break;
}
case Arg::DOUBLE:
writer.write_double(arg.double_value, spec);
break;
case Arg::LONG_DOUBLE:
writer.write_double(arg.long_double_value, spec);
break;
case Arg::STRING:
2014-06-30 21:26:29 +00:00
writer.write_str(arg.string, spec);
break;
case Arg::WSTRING:
writer.write_str(ignore_incompatible_str<Char>(arg.wstring), spec);
break;
case Arg::POINTER:
if (spec.type_ && spec.type_ != 'p')
internal::report_unknown_type(spec.type_, "pointer");
spec.flags_= HASH_FLAG;
spec.type_ = 'x';
writer.write_int(reinterpret_cast<uintptr_t>(arg.pointer_value), spec);
break;
case Arg::CUSTOM:
if (spec.type_)
internal::report_unknown_type(spec.type_, "object");
arg.custom.format(&writer, arg.custom.value, "s");
break;
default:
assert(false);
break;
}
}
write(writer, start, s);
}
2013-09-07 03:23:42 +00:00
template <typename Char>
const Char *fmt::BasicFormatter<Char>::format(
const Char *format_str, const Arg &arg) {
const Char *s = format_str;
const char *error = 0;
FormatSpec spec;
if (*s == ':') {
if (arg.type == Arg::CUSTOM) {
arg.custom.format(this, arg.custom.value, s);
return find_closing_brace(s) + 1;
2013-09-07 03:23:42 +00:00
}
++s;
// Parse fill and alignment.
if (Char c = *s) {
const Char *p = s + 1;
spec.align_ = ALIGN_DEFAULT;
do {
switch (*p) {
2013-09-07 03:23:42 +00:00
case '<':
spec.align_ = ALIGN_LEFT;
break;
case '>':
spec.align_ = ALIGN_RIGHT;
break;
case '=':
spec.align_ = ALIGN_NUMERIC;
break;
case '^':
spec.align_ = ALIGN_CENTER;
break;
}
if (spec.align_ != ALIGN_DEFAULT) {
if (p != s) {
if (c == '}') break;
if (c == '{')
2014-08-27 15:24:31 +00:00
throw FormatError("invalid fill character '{'");
s += 2;
spec.fill_ = c;
} else ++s;
if (spec.align_ == ALIGN_NUMERIC && arg.type > Arg::LAST_NUMERIC_TYPE)
2014-08-27 15:24:31 +00:00
throw FormatError("format specifier '=' requires numeric argument");
break;
}
} while (--p >= s);
}
2013-09-07 03:23:42 +00:00
// Parse sign.
switch (*s) {
2013-09-07 03:23:42 +00:00
case '+':
check_sign(s, arg);
2013-09-07 03:23:42 +00:00
spec.flags_ |= SIGN_FLAG | PLUS_FLAG;
break;
case '-':
check_sign(s, arg);
spec.flags_ |= MINUS_FLAG;
2013-09-07 03:23:42 +00:00
break;
case ' ':
check_sign(s, arg);
2013-09-07 03:23:42 +00:00
spec.flags_ |= SIGN_FLAG;
break;
}
if (*s == '#') {
if (arg.type > Arg::LAST_NUMERIC_TYPE)
2014-08-27 15:24:31 +00:00
// TODO: make FormatError accept arguments
throw FormatError("format specifier '#' requires numeric argument");
spec.flags_ |= HASH_FLAG;
++s;
}
2013-09-07 03:23:42 +00:00
// Parse width and zero flag.
if ('0' <= *s && *s <= '9') {
if (*s == '0') {
if (arg.type > Arg::LAST_NUMERIC_TYPE)
2014-08-27 15:24:31 +00:00
throw FormatError("format specifier '0' requires numeric argument");
spec.align_ = ALIGN_NUMERIC;
spec.fill_ = '0';
2013-09-07 03:23:42 +00:00
}
// Zero may be parsed again as a part of the width, but it is simpler
// and more efficient than checking if the next char is a digit.
spec.width_ = parse_nonnegative_int(s);
if (error)
2014-08-27 15:24:31 +00:00
throw FormatError(error);
}
2013-09-07 03:23:42 +00:00
// Parse precision.
if (*s == '.') {
++s;
spec.precision_ = 0;
2013-09-07 03:23:42 +00:00
if ('0' <= *s && *s <= '9') {
spec.precision_ = parse_nonnegative_int(s);
if (error)
2014-08-27 15:24:31 +00:00
throw FormatError(error);
} else if (*s == '{') {
2013-09-07 03:23:42 +00:00
++s;
const Arg &precision_arg = parse_arg_index(s);
2014-08-27 15:24:31 +00:00
if (*s++ != '}')
throw FormatError("unmatched '{' in format");
ULongLong value = 0;
switch (precision_arg.type) {
case Arg::INT:
2013-09-07 03:23:42 +00:00
if (precision_arg.int_value < 0)
2014-08-27 15:24:31 +00:00
throw FormatError("negative precision in format");
2013-09-07 03:23:42 +00:00
value = precision_arg.int_value;
break;
case Arg::UINT:
2013-09-07 03:23:42 +00:00
value = precision_arg.uint_value;
break;
case Arg::LONG_LONG:
if (precision_arg.long_long_value < 0)
2014-08-27 15:24:31 +00:00
throw FormatError("negative precision in format");
value = precision_arg.long_long_value;
break;
case Arg::ULONG_LONG:
value = precision_arg.ulong_long_value;
break;
2013-09-07 03:23:42 +00:00
default:
2014-08-27 15:24:31 +00:00
throw FormatError("precision is not integer");
2013-09-07 03:23:42 +00:00
}
if (value > INT_MAX)
throw FormatError("number is too big");
spec.precision_ = static_cast<int>(value);
} else {
2014-08-27 15:24:31 +00:00
throw FormatError("missing precision in format");
}
if (arg.type != Arg::DOUBLE && arg.type != Arg::LONG_DOUBLE) {
2014-08-27 15:24:31 +00:00
throw FormatError(
"precision specifier requires floating-point argument");
2013-09-07 03:23:42 +00:00
}
}
// Parse type.
if (*s != '}' && *s)
spec.type_ = static_cast<char>(*s++);
}
2013-09-07 03:23:42 +00:00
if (*s++ != '}')
throw FormatError("unmatched '{' in format");
start_ = s;
// Format argument.
2014-07-16 15:38:15 +00:00
internal::ArgFormatter<Char>(*this, spec, s - 1).visit(arg);
return s;
}
template <typename Char>
void fmt::BasicFormatter<Char>::format(
BasicStringRef<Char> format_str, const ArgList &args) {
const Char *s = start_ = format_str.c_str();
args_ = args;
next_arg_index_ = 0;
while (*s) {
Char c = *s++;
if (c != '{' && c != '}') continue;
if (*s == c) {
write(writer_, start_, s);
start_ = ++s;
continue;
2013-09-07 03:23:42 +00:00
}
if (c == '}')
throw FormatError("unmatched '}' in format");
write(writer_, start_, s - 1);
Arg arg = parse_arg_index(s);
s = format(s, arg);
2013-09-07 03:23:42 +00:00
}
write(writer_, start_, s);
2013-09-07 03:23:42 +00:00
}
void fmt::report_system_error(
int error_code, fmt::StringRef message) FMT_NOEXCEPT(true) {
// FIXME: format_system_error may throw
report_error(internal::format_system_error, error_code, message);
}
2014-04-30 19:38:17 +00:00
#ifdef _WIN32
void fmt::report_windows_error(
int error_code, fmt::StringRef message) FMT_NOEXCEPT(true) {
// FIXME: format_windows_error may throw
report_error(internal::format_windows_error, error_code, message);
}
2014-04-30 19:38:17 +00:00
#endif
2014-08-21 15:49:13 +00:00
void fmt::print(std::FILE *f, StringRef format_str, const ArgList &args) {
Writer w;
2014-08-21 15:49:13 +00:00
w.write(format_str, args);
2014-06-29 04:56:40 +00:00
std::fwrite(w.data(), 1, w.size(), f);
}
2014-08-21 15:49:13 +00:00
void fmt::print(std::ostream &os, StringRef format_str, const ArgList &args) {
Writer w;
2014-08-21 15:49:13 +00:00
w.write(format_str, args);
os.write(w.data(), w.size());
}
void fmt::print_colored(Color c, StringRef format, const ArgList &args) {
char escape[] = "\x1b[30m";
escape[3] = '0' + static_cast<char>(c);
std::fputs(escape, stdout);
print(format, args);
std::fputs(RESET_COLOR, stdout);
}
2014-08-21 15:49:13 +00:00
int fmt::fprintf(std::FILE *f, StringRef format, const ArgList &args) {
2014-06-29 04:56:40 +00:00
Writer w;
printf(w, format, args);
2014-08-21 15:49:13 +00:00
return std::fwrite(w.data(), 1, w.size(), f);
2014-06-29 04:56:40 +00:00
}
2013-09-07 17:15:08 +00:00
// Explicit instantiations for char.
template fmt::BasicWriter<char>::CharPtr
fmt::BasicWriter<char>::fill_padding(CharPtr buffer,
unsigned total_size, std::size_t content_size, wchar_t fill);
2013-09-07 17:15:08 +00:00
template void fmt::BasicFormatter<char>::format(
BasicStringRef<char> format, const ArgList &args);
2013-09-07 03:23:42 +00:00
template void fmt::internal::PrintfFormatter<char>::format(
BasicWriter<char> &writer, BasicStringRef<char> format, const ArgList &args);
2013-09-07 17:15:08 +00:00
// Explicit instantiations for wchar_t.
2013-09-07 17:15:08 +00:00
2013-09-07 03:23:42 +00:00
template fmt::BasicWriter<wchar_t>::CharPtr
fmt::BasicWriter<wchar_t>::fill_padding(CharPtr buffer,
unsigned total_size, std::size_t content_size, wchar_t fill);
2013-09-07 17:15:08 +00:00
template void fmt::BasicFormatter<wchar_t>::format(
BasicStringRef<wchar_t> format, const ArgList &args);
template void fmt::internal::PrintfFormatter<wchar_t>::format(
BasicWriter<wchar_t> &writer, BasicStringRef<wchar_t> format,
const ArgList &args);
2014-03-11 18:56:24 +00:00
#if _MSC_VER
# pragma warning(pop)
#endif