Parameterize formatting argument on char type.
diff --git a/fmt/format.cc b/fmt/format.cc
index 499e838..3e2a6ee 100644
--- a/fmt/format.cc
+++ b/fmt/format.cc
@@ -443,8 +443,7 @@
void printf(BasicWriter<Char> &w, BasicCStringRef<Char> format,
format_args args);
-FMT_FUNC int vfprintf(std::FILE *f, CStringRef format,
- basic_format_args<printf_context<char>> args) {
+FMT_FUNC int vfprintf(std::FILE *f, CStringRef format, printf_args args) {
MemoryWriter w;
printf(w, format, args);
std::size_t size = w.size();
@@ -475,7 +474,7 @@
template void internal::FixedBuffer<wchar_t>::grow(std::size_t);
-template void internal::ArgMap<wchar_t>::init(const format_args &args);
+template void internal::ArgMap<wchar_t>::init(const wformat_args &args);
template void printf_context<wchar_t>::format(WWriter &writer);
diff --git a/fmt/format.h b/fmt/format.h
index e03c66a..ed70646 100644
--- a/fmt/format.h
+++ b/fmt/format.h
@@ -414,6 +414,8 @@
std::size_t size_;
public:
+ BasicStringRef() : data_(0), size_(0) {}
+
/** Constructs a string reference object from a C string and a size. */
BasicStringRef(const Char *s, std::size_t size) : data_(s), size_(size) {}
@@ -1027,16 +1029,102 @@
CSTRING, STRING, WSTRING, POINTER, CUSTOM
};
};
+
+template <typename Char>
+class ArgMap;
} // namespace internal
+template <typename Context, typename Char>
+class basic_format_args;
+
// A formatting argument. It is a trivially copyable/constructible type to
// allow storage in internal::MemoryBuffer.
-struct format_arg : internal::Value {
- Type type;
+template <typename Char>
+class basic_format_arg : public internal::Value {
+ protected:
+ Type type_;
- explicit operator bool() const noexcept { return type != NONE; }
+ template <typename Visitor, typename CharType>
+ friend typename std::result_of<Visitor(int)>::type
+ visit(Visitor &&vis, basic_format_arg<CharType> arg);
+
+ template <typename Context, typename CharType>
+ friend class basic_format_args;
+
+ template <typename CharType>
+ friend class internal::ArgMap;
+
+ void check_type() const {
+ FMT_ASSERT(type_ > NAMED_ARG, "invalid argument type");
+ }
+
+ public:
+ explicit operator bool() const noexcept { return type_ != NONE; }
+
+ bool is_integral() const {
+ check_type();
+ return type_ <= LAST_INTEGER_TYPE;
+ }
+
+ bool is_numeric() const {
+ check_type();
+ return type_ <= LAST_NUMERIC_TYPE;
+ }
+
+ bool is_pointer() const {
+ check_type();
+ return type_ == POINTER;
+ }
};
+typedef basic_format_arg<char> format_arg;
+typedef basic_format_arg<wchar_t> wformat_arg;
+
+/**
+ \rst
+ Visits an argument dispatching to the appropriate visit method based on
+ the argument type. For example, if the argument type is ``double`` then
+ ``vis(value)`` will be called with the value of type ``double``.
+ \endrst
+ */
+template <typename Visitor, typename Char>
+typename std::result_of<Visitor(int)>::type
+ visit(Visitor &&vis, basic_format_arg<Char> arg) {
+ switch (arg.type_) {
+ case format_arg::NONE:
+ case format_arg::NAMED_ARG:
+ FMT_ASSERT(false, "invalid argument type");
+ break;
+ case format_arg::INT:
+ return vis(arg.int_value);
+ case format_arg::UINT:
+ return vis(arg.uint_value);
+ case format_arg::LONG_LONG:
+ return vis(arg.long_long_value);
+ case format_arg::ULONG_LONG:
+ return vis(arg.ulong_long_value);
+ case format_arg::BOOL:
+ return vis(arg.int_value != 0);
+ case format_arg::CHAR:
+ return vis(static_cast<Char>(arg.int_value));
+ case format_arg::DOUBLE:
+ return vis(arg.double_value);
+ case format_arg::LONG_DOUBLE:
+ return vis(arg.long_double_value);
+ case format_arg::CSTRING:
+ return vis(arg.string.value);
+ case format_arg::STRING:
+ return vis(arg.string);
+ case format_arg::WSTRING:
+ return vis(arg.wstring);
+ case format_arg::POINTER:
+ return vis(arg.pointer);
+ case format_arg::CUSTOM:
+ return vis(arg.custom);
+ }
+ return typename std::result_of<Visitor(int)>::type();
+}
+
namespace internal {
template <typename Char>
@@ -1251,7 +1339,7 @@
// Makes a format_arg object from any type.
template <typename Context>
-class MakeValue : public format_arg {
+class MakeValue : public basic_format_arg<typename Context::char_type> {
public:
typedef typename Context::char_type Char;
@@ -1279,13 +1367,13 @@
MakeValue(typename WCharHelper<WStringRef, Char>::Unsupported);
void set_string(StringRef str) {
- string.value = str.data();
- string.size = str.size();
+ this->string.value = str.data();
+ this->string.size = str.size();
}
void set_string(WStringRef str) {
- wstring.value = str.data();
- wstring.size = str.size();
+ this->wstring.value = str.data();
+ this->wstring.size = str.size();
}
// Formats an argument of a custom type, such as a user-defined class.
@@ -1302,8 +1390,8 @@
#define FMT_MAKE_VALUE_(Type, field, TYPE, rhs) \
MakeValue(Type value) { \
- static_assert(internal::type<Type>() == TYPE, "invalid type"); \
- field = rhs; \
+ static_assert(internal::type<Type>() == MakeValue::TYPE, "invalid type"); \
+ this->field = rhs; \
}
#define FMT_MAKE_VALUE(Type, field, TYPE) \
@@ -1319,16 +1407,16 @@
// To minimize the number of types we need to deal with, long is
// translated either to int or to long long depending on its size.
if (const_check(sizeof(long) == sizeof(int)))
- int_value = static_cast<int>(value);
+ this->int_value = static_cast<int>(value);
else
- long_long_value = value;
+ this->long_long_value = value;
}
MakeValue(unsigned long value) {
if (const_check(sizeof(unsigned long) == sizeof(unsigned)))
- uint_value = static_cast<unsigned>(value);
+ this->uint_value = static_cast<unsigned>(value);
else
- ulong_long_value = value;
+ this->ulong_long_value = value;
}
FMT_MAKE_VALUE(LongLong, long_long_value, LONG_LONG)
@@ -1343,14 +1431,14 @@
#if !defined(_MSC_VER) || defined(_NATIVE_WCHAR_T_DEFINED)
typedef typename WCharHelper<wchar_t, Char>::Supported WChar;
MakeValue(WChar value) {
- static_assert(internal::type<WChar>() == CHAR, "invalid type");
- int_value = value;
+ static_assert(internal::type<WChar>() == MakeValue::CHAR, "invalid type");
+ this->int_value = value;
}
#endif
#define FMT_MAKE_STR_VALUE(Type, TYPE) \
MakeValue(Type value) { \
- static_assert(internal::type<Type>() == TYPE, "invalid type"); \
+ static_assert(internal::type<Type>() == MakeValue::TYPE, "invalid type"); \
set_string(value); \
}
@@ -1366,7 +1454,7 @@
#define FMT_MAKE_WSTR_VALUE(Type, TYPE) \
MakeValue(typename WCharHelper<Type, Char>::Supported value) { \
- static_assert(internal::type<Type>() == TYPE, "invalid type"); \
+ static_assert(internal::type<Type>() == MakeValue::TYPE, "invalid type"); \
set_string(value); \
}
@@ -1382,49 +1470,51 @@
MakeValue(const T &value,
typename EnableIf<Not<
ConvertToInt<T>::value>::value, int>::type = 0) {
- static_assert(internal::type<T>() == CUSTOM, "invalid type");
- custom.value = &value;
- custom.format = &format_custom_arg<T>;
+ static_assert(internal::type<T>() == MakeValue::CUSTOM, "invalid type");
+ this->custom.value = &value;
+ this->custom.format = &format_custom_arg<T>;
}
template <typename T>
MakeValue(const T &value,
typename EnableIf<ConvertToInt<T>::value, int>::type = 0) {
- static_assert(internal::type<T>() == INT, "invalid type");
- int_value = value;
+ static_assert(internal::type<T>() == MakeValue::INT, "invalid type");
+ this->int_value = value;
}
// Additional template param `Char_` is needed here because make_type always
// uses char.
template <typename Char_>
MakeValue(const NamedArg<Char_> &value) {
- static_assert(internal::type<const NamedArg<Char_> &>() == NAMED_ARG,
- "invalid type");
- pointer = &value;
+ static_assert(
+ internal::type<const NamedArg<Char_> &>() == MakeValue::NAMED_ARG,
+ "invalid type");
+ this->pointer = &value;
}
};
template <typename Context>
-class MakeArg : public format_arg {
+ class MakeArg : public basic_format_arg<typename Context::char_type> {
public:
MakeArg() {
- type = format_arg::NONE;
+ this->type_ = format_arg::NONE;
}
template <typename T>
MakeArg(const T &value)
- : format_arg(MakeValue<Context>(value)) {
- type = internal::type<T>();
+ : basic_format_arg<typename Context::char_type>(MakeValue<Context>(value)) {
+ this->type_ = internal::type<T>();
}
};
template <typename Char>
-struct NamedArg : format_arg {
+struct NamedArg : basic_format_arg<Char> {
BasicStringRef<Char> name;
template <typename T>
NamedArg(BasicStringRef<Char> argname, const T &value)
- : format_arg(MakeArg< basic_format_context<Char> >(value)), name(argname) {}
+ : basic_format_arg<Char>(MakeArg< basic_format_context<Char> >(value)),
+ name(argname) {}
};
class RuntimeError : public std::runtime_error {
@@ -1433,9 +1523,6 @@
~RuntimeError() throw();
};
-template <typename Char>
-class ArgMap;
-
template <typename Arg, typename... Args>
constexpr uint64_t make_type() {
return type<Arg>() | (make_type<Args...>() << 4);
@@ -1482,8 +1569,12 @@
}
/** Formatting arguments. */
-template <typename Context>
+template <typename Context, typename Char>
class basic_format_args {
+ public:
+ typedef unsigned size_type;
+ typedef basic_format_arg<Char> format_arg;
+
private:
// To reduce compiled code size per formatting function call, types of first
// MAX_PACKED_ARGS arguments are passed in the types_ field.
@@ -1498,21 +1589,43 @@
const format_arg *args_;
};
- format_arg::Type type(unsigned index) const {
+ typename format_arg::Type type(unsigned index) const {
unsigned shift = index * 4;
uint64_t mask = 0xf;
- return static_cast<format_arg::Type>((types_ & (mask << shift)) >> shift);
+ return static_cast<typename format_arg::Type>(
+ (types_ & (mask << shift)) >> shift);
}
- template <typename Char>
- friend class internal::ArgMap;
+ friend class internal::ArgMap<Char>;
void set_data(const internal::Value *values) { values_ = values; }
void set_data(const format_arg *args) { args_ = args; }
- public:
- typedef unsigned size_type;
+ format_arg get(size_type index) const {
+ format_arg arg;
+ bool use_values = type(internal::MAX_PACKED_ARGS - 1) == format_arg::NONE;
+ if (index < internal::MAX_PACKED_ARGS) {
+ typename format_arg::Type arg_type = type(index);
+ internal::Value &val = arg;
+ if (arg_type != format_arg::NONE)
+ val = use_values ? values_[index] : args_[index];
+ arg.type_ = arg_type;
+ return arg;
+ }
+ if (use_values) {
+ // The index is greater than the number of arguments that can be stored
+ // in values, so return a "none" argument.
+ arg.type_ = format_arg::NONE;
+ return arg;
+ }
+ for (unsigned i = internal::MAX_PACKED_ARGS; i <= index; ++i) {
+ if (args_[i].type_ == format_arg::NONE)
+ return args_[i];
+ }
+ return args_[index];
+ }
+ public:
basic_format_args() : types_(0) {}
template <typename... Args>
@@ -1523,77 +1636,14 @@
/** Returns the argument at specified index. */
format_arg operator[](size_type index) const {
- format_arg arg;
- bool use_values = type(internal::MAX_PACKED_ARGS - 1) == format_arg::NONE;
- if (index < internal::MAX_PACKED_ARGS) {
- format_arg::Type arg_type = type(index);
- internal::Value &val = arg;
- if (arg_type != format_arg::NONE)
- val = use_values ? values_[index] : args_[index];
- arg.type = arg_type;
- return arg;
- }
- if (use_values) {
- // The index is greater than the number of arguments that can be stored
- // in values, so return a "none" argument.
- arg.type = format_arg::NONE;
- return arg;
- }
- for (unsigned i = internal::MAX_PACKED_ARGS; i <= index; ++i) {
- if (args_[i].type == format_arg::NONE)
- return args_[i];
- }
- return args_[index];
+ format_arg arg = get(index);
+ return arg.type_ == format_arg::NAMED_ARG ?
+ *static_cast<const format_arg*>(arg.pointer) : arg;
}
};
-typedef basic_format_args<basic_format_context<char>> format_args;
-typedef basic_format_args<basic_format_context<wchar_t>> wformat_args;
-
-/**
- \rst
- Visits an argument dispatching to the appropriate visit method based on
- the argument type. For example, if the argument type is ``double`` then
- ``vis(value)`` will be called with the value of type ``double``.
- \endrst
- */
-template <typename Visitor>
-typename std::result_of<Visitor(int)>::type visit(Visitor &&vis,
- format_arg arg) {
- switch (arg.type) {
- case format_arg::NONE:
- case format_arg::NAMED_ARG:
- FMT_ASSERT(false, "invalid argument type");
- break;
- case format_arg::INT:
- return vis(arg.int_value);
- case format_arg::UINT:
- return vis(arg.uint_value);
- case format_arg::LONG_LONG:
- return vis(arg.long_long_value);
- case format_arg::ULONG_LONG:
- return vis(arg.ulong_long_value);
- case format_arg::BOOL:
- return vis(arg.int_value != 0);
- case format_arg::CHAR:
- return vis(static_cast<wchar_t>(arg.int_value));
- case format_arg::DOUBLE:
- return vis(arg.double_value);
- case format_arg::LONG_DOUBLE:
- return vis(arg.long_double_value);
- case format_arg::CSTRING:
- return vis(arg.string.value);
- case format_arg::STRING:
- return vis(arg.string);
- case format_arg::WSTRING:
- return vis(arg.wstring);
- case format_arg::POINTER:
- return vis(arg.pointer);
- case format_arg::CUSTOM:
- return vis(arg.custom);
- }
- return typename std::result_of<Visitor(int)>::type();
-}
+typedef basic_format_args<basic_format_context<char>, char> format_args;
+typedef basic_format_args<basic_format_context<wchar_t>, wchar_t> wformat_args;
enum Alignment {
ALIGN_DEFAULT, ALIGN_LEFT, ALIGN_RIGHT, ALIGN_CENTER, ALIGN_NUMERIC
@@ -1822,16 +1872,17 @@
class ArgMap {
private:
typedef std::vector<
- std::pair<fmt::BasicStringRef<Char>, format_arg> > MapType;
+ std::pair<fmt::BasicStringRef<Char>, basic_format_arg<Char> > > MapType;
typedef typename MapType::value_type Pair;
MapType map_;
public:
- template <typename Formatter>
- void init(const basic_format_args<Formatter> &args);
+ template <typename Context>
+ void init(const basic_format_args<Context, Char> &args);
- const format_arg *find(const fmt::BasicStringRef<Char> &name) const {
+ const basic_format_arg<Char>
+ *find(const fmt::BasicStringRef<Char> &name) const {
// The list is unsorted, so just return the first matching name.
for (typename MapType::const_iterator it = map_.begin(), end = map_.end();
it != end; ++it) {
@@ -1843,8 +1894,8 @@
};
template <typename Char>
-template <typename Formatter>
-void ArgMap<Char>::init(const basic_format_args<Formatter> &args) {
+template <typename Context>
+void ArgMap<Char>::init(const basic_format_args<Context, Char> &args) {
if (!map_.empty())
return;
typedef internal::NamedArg<Char> NamedArg;
@@ -1874,8 +1925,8 @@
map_.push_back(Pair(named_arg->name, *named_arg));
}
}
- for (unsigned i = MAX_PACKED_ARGS;/*nothing*/; ++i) {
- switch (args.args_[i].type) {
+ for (unsigned i = MAX_PACKED_ARGS; ; ++i) {
+ switch (args.args_[i].type_) {
case format_arg::NONE:
return;
case format_arg::NAMED_ARG:
@@ -1955,7 +2006,7 @@
write(value);
}
- void operator()(wchar_t value) {
+ void operator()(Char value) {
if (spec_.type_ && spec_.type_ != 'c') {
spec_.flags_ |= CHAR_FLAG;
writer_.write_int(value, spec_);
@@ -2016,29 +2067,24 @@
class format_context_base {
private:
const Char *ptr_;
- basic_format_args<Context> args_;
+ basic_format_args<Context, Char> args_;
int next_arg_index_;
protected:
- format_context_base(const Char *format_str, basic_format_args<Context> args)
+ typedef basic_format_arg<Char> format_arg;
+
+ format_context_base(const Char *format_str,
+ basic_format_args<Context, Char> args)
: ptr_(format_str), args_(args), next_arg_index_(0) {}
~format_context_base() {}
- basic_format_args<Context> args() const { return args_; }
+ basic_format_args<Context, Char> args() const { return args_; }
// Returns the argument with specified index.
format_arg do_get_arg(unsigned arg_index, const char *&error) {
format_arg arg = args_[arg_index];
- switch (arg.type) {
- case format_arg::NONE:
- error = "argument index out of range";
- break;
- case format_arg::NAMED_ARG:
- arg = *static_cast<const format_arg*>(arg.pointer);
- break;
- default:
- /*nothing*/;
- }
+ if (!arg)
+ error = "argument index out of range";
return arg;
}
@@ -2107,13 +2153,14 @@
FMT_DISALLOW_COPY_AND_ASSIGN(basic_format_context);
- typedef internal::format_context_base<Char, basic_format_context> Base;
+ typedef internal::format_context_base<Char, basic_format_context<Char>> Base;
+ using typename Base::format_arg;
using Base::get_arg;
// Checks if manual indexing is used and returns the argument with
// specified name.
- format_arg get_arg(BasicStringRef<Char> name, const char *&error);
+ basic_format_arg<Char> get_arg(BasicStringRef<Char> name, const char *&error);
public:
/** The character type for the output. */
@@ -2126,11 +2173,11 @@
\endrst
*/
basic_format_context(const Char *format_str,
- basic_format_args<basic_format_context> args)
+ basic_format_args<basic_format_context, Char> args)
: Base(format_str, args) {}
// Parses argument id and returns corresponding argument.
- format_arg parse_arg_id();
+ basic_format_arg<Char> parse_arg_id();
using Base::ptr;
};
@@ -2364,7 +2411,7 @@
}
void vwrite(BasicCStringRef<Char> format,
- basic_format_args<basic_format_context<Char>> args);
+ basic_format_args<basic_format_context<Char>, Char> args);
/**
\rst
Writes formatted data.
@@ -3281,45 +3328,121 @@
return value;
}
-inline void require_numeric_argument(const format_arg &arg, char spec) {
- if (arg.type > format_arg::LAST_NUMERIC_TYPE) {
- std::string message =
- fmt::format("format specifier '{}' requires numeric argument", spec);
- FMT_THROW(fmt::format_error(message));
+template <typename Char>
+inline void require_numeric_argument(
+ const basic_format_arg<Char> &arg, char spec) {
+ if (!arg.is_numeric()) {
+ FMT_THROW(fmt::format_error(
+ fmt::format("format specifier '{}' requires numeric argument", spec)));
}
}
+struct IsUnsigned {
+ template <typename T>
+ typename std::enable_if<std::is_unsigned<T>::value, bool>::type
+ operator()(T value) {
+ return true;
+ }
+
+ template <typename T>
+ typename std::enable_if<!std::is_unsigned<T>::value, bool>::type
+ operator()(T value) {
+ return false;
+ }
+};
+
template <typename Char>
-void check_sign(const Char *&s, const format_arg &arg) {
+void check_sign(const Char *&s, const basic_format_arg<Char> &arg) {
char sign = static_cast<char>(*s);
require_numeric_argument(arg, sign);
- if (arg.type == format_arg::UINT || arg.type == format_arg::ULONG_LONG) {
+ if (visit(IsUnsigned(), arg)) {
FMT_THROW(format_error(fmt::format(
"format specifier '{}' requires signed argument", sign)));
}
++s;
}
+
+template <typename Char, typename Context>
+class CustomFormatter {
+ private:
+ BasicWriter<Char> &writer_;
+ Context &ctx_;
+
+ public:
+ CustomFormatter(BasicWriter<Char> &writer, Context &ctx)
+ : writer_(writer), ctx_(ctx) {}
+
+ bool operator()(format_arg::CustomValue custom) {
+ custom.format(&writer_, custom.value, &ctx_);
+ return true;
+ }
+
+ template <typename T>
+ bool operator()(T) { return false; }
+};
+
+template <typename T>
+struct IsInteger {
+ enum {
+ value = std::is_integral<T>::value && !std::is_same<T, bool>::value &&
+ !std::is_same<T, char>::value && !std::is_same<T, wchar_t>::value
+ };
+};
+
+struct WidthHandler {
+ template <typename T>
+ typename std::enable_if<IsInteger<T>::value, ULongLong>::type
+ operator()(T value) {
+ if (is_negative(value))
+ FMT_THROW(format_error("negative width"));
+ return value;
+ }
+
+ template <typename T>
+ typename std::enable_if<!IsInteger<T>::value, ULongLong>::type
+ operator()(T value) {
+ FMT_THROW(format_error("width is not integer"));
+ return 0;
+ }
+};
+
+struct PrecisionHandler {
+ template <typename T>
+ typename std::enable_if<IsInteger<T>::value, ULongLong>::type
+ operator()(T value) {
+ if (is_negative(value))
+ FMT_THROW(format_error("negative precision"));
+ return value;
+ }
+
+ template <typename T>
+ typename std::enable_if<!IsInteger<T>::value, ULongLong>::type
+ operator()(T value) {
+ FMT_THROW(format_error("precision is not integer"));
+ return 0;
+ }
+};
} // namespace internal
template <typename Char>
-inline format_arg basic_format_context<Char>::get_arg(
+inline basic_format_arg<Char> basic_format_context<Char>::get_arg(
BasicStringRef<Char> name, const char *&error) {
if (this->check_no_auto_index(error)) {
map_.init(this->args());
- const format_arg *arg = map_.find(name);
+ const basic_format_arg<Char> *arg = map_.find(name);
if (arg)
return *arg;
error = "argument not found";
}
- return format_arg();
+ return basic_format_arg<Char>();
}
template <typename Char>
-inline format_arg basic_format_context<Char>::parse_arg_id() {
+inline basic_format_arg<Char> basic_format_context<Char>::parse_arg_id() {
const Char *&s = this->ptr();
if (!internal::is_name_start(*s)) {
const char *error = 0;
- format_arg arg = *s < '0' || *s > '9' ?
+ basic_format_arg<Char> arg = *s < '0' || *s > '9' ?
this->next_arg(error) : get_arg(internal::parse_nonnegative_int(s), error);
if (error) {
FMT_THROW(format_error(
@@ -3333,7 +3456,8 @@
c = *++s;
} while (internal::is_name_start(c) || ('0' <= c && c <= '9'));
const char *error = 0;
- format_arg arg = get_arg(BasicStringRef<Char>(start, s - start), error);
+ basic_format_arg<Char> arg =
+ get_arg(BasicStringRef<Char>(start, s - start), error);
if (error)
FMT_THROW(format_error(error));
return arg;
@@ -3341,15 +3465,13 @@
// Formats a single argument.
template <typename ArgFormatter, typename Char, typename Context>
-void do_format_arg(BasicWriter<Char> &writer, const format_arg &arg,
+void do_format_arg(BasicWriter<Char> &writer, const basic_format_arg<Char> &arg,
Context &ctx) {
const Char *&s = ctx.ptr();
FormatSpec spec;
if (*s == ':') {
- if (arg.type == format_arg::CUSTOM) {
- arg.custom.format(&writer, arg.custom.value, &ctx);
+ if (visit(internal::CustomFormatter<Char, Context>(writer, ctx), arg))
return;
- }
++s;
// Parse fill and alignment.
if (Char c = *s) {
@@ -3420,33 +3542,13 @@
spec.width_ = internal::parse_nonnegative_int(s);
} else if (*s == '{') {
++s;
- format_arg width_arg = ctx.parse_arg_id();
+ auto width_arg = ctx.parse_arg_id();
if (*s++ != '}')
FMT_THROW(format_error("invalid format string"));
- ULongLong value = 0;
- switch (width_arg.type) {
- case format_arg::INT:
- if (width_arg.int_value < 0)
- FMT_THROW(format_error("negative width"));
- value = width_arg.int_value;
- break;
- case format_arg::UINT:
- value = width_arg.uint_value;
- break;
- case format_arg::LONG_LONG:
- if (width_arg.long_long_value < 0)
- FMT_THROW(format_error("negative width"));
- value = width_arg.long_long_value;
- break;
- case format_arg::ULONG_LONG:
- value = width_arg.ulong_long_value;
- break;
- default:
- FMT_THROW(format_error("width is not integer"));
- }
- if (value > (std::numeric_limits<int>::max)())
+ ULongLong width = visit(internal::WidthHandler(), width_arg);
+ if (width > (std::numeric_limits<int>::max)())
FMT_THROW(format_error("number is too big"));
- spec.width_ = static_cast<int>(value);
+ spec.width_ = static_cast<int>(width);
}
// Parse precision.
@@ -3457,41 +3559,21 @@
spec.precision_ = internal::parse_nonnegative_int(s);
} else if (*s == '{') {
++s;
- format_arg precision_arg = ctx.parse_arg_id();
+ auto precision_arg = ctx.parse_arg_id();
if (*s++ != '}')
FMT_THROW(format_error("invalid format string"));
- ULongLong value = 0;
- switch (precision_arg.type) {
- case format_arg::INT:
- if (precision_arg.int_value < 0)
- FMT_THROW(format_error("negative precision"));
- value = precision_arg.int_value;
- break;
- case format_arg::UINT:
- value = precision_arg.uint_value;
- break;
- case format_arg::LONG_LONG:
- if (precision_arg.long_long_value < 0)
- FMT_THROW(format_error("negative precision"));
- value = precision_arg.long_long_value;
- break;
- case format_arg::ULONG_LONG:
- value = precision_arg.ulong_long_value;
- break;
- default:
- FMT_THROW(format_error("precision is not integer"));
- }
- if (value > (std::numeric_limits<int>::max)())
+ ULongLong precision =
+ visit(internal::PrecisionHandler(), precision_arg);
+ if (precision > (std::numeric_limits<int>::max)())
FMT_THROW(format_error("number is too big"));
- spec.precision_ = static_cast<int>(value);
+ spec.precision_ = static_cast<int>(precision);
} else {
FMT_THROW(format_error("missing precision specifier"));
}
- if (arg.type <= format_arg::LAST_INTEGER_TYPE ||
- arg.type == format_arg::POINTER) {
+ if (arg.is_integral() || arg.is_pointer()) {
FMT_THROW(format_error(
fmt::format("precision not allowed in {} format specifier",
- arg.type == format_arg::POINTER ? "pointer" : "integer")));
+ arg.is_pointer() ? "pointer" : "integer")));
}
}
@@ -3510,7 +3592,7 @@
/** Formats arguments and writes the output to the writer. */
template <typename ArgFormatter, typename Char, typename Context>
void vformat(BasicWriter<Char> &writer, BasicCStringRef<Char> format_str,
- basic_format_args<Context> args) {
+ basic_format_args<Context, Char> args) {
basic_format_context<Char> ctx(format_str.c_str(), args);
const Char *&s = ctx.ptr();
const Char *start = s;
@@ -3536,7 +3618,7 @@
template <typename Char>
inline void BasicWriter<Char>::vwrite(
BasicCStringRef<Char> format,
- basic_format_args<basic_format_context<Char>> args) {
+ basic_format_args<basic_format_context<Char>, Char> args) {
vformat<ArgFormatter<Char>>(*this, format, args);
}
} // namespace fmt
diff --git a/fmt/printf.h b/fmt/printf.h
index 386acf9..c670018 100644
--- a/fmt/printf.h
+++ b/fmt/printf.h
@@ -40,7 +40,7 @@
static bool fits_in_int(int) { return true; }
};
-class PrecisionHandler {
+class PrintfPrecisionHandler {
public:
template <typename T>
typename std::enable_if<std::is_integral<T>::value, int>::type
@@ -80,14 +80,14 @@
enum { value = 1 };
};
-template <typename T>
+template <typename T, typename Char>
class ArgConverter {
private:
- format_arg &arg_;
- wchar_t type_;
+ basic_format_arg<Char> &arg_;
+ Char type_;
public:
- ArgConverter(format_arg &arg, wchar_t type)
+ ArgConverter(basic_format_arg<Char> &arg, Char type)
: arg_(arg), type_(type) {}
void operator()(bool value) {
@@ -101,27 +101,27 @@
bool is_signed = type_ == 'd' || type_ == 'i';
typedef typename internal::Conditional<
is_same<T, void>::value, U, T>::type TargetType;
+ typedef basic_format_context<Char> format_context;
if (sizeof(TargetType) <= sizeof(int)) {
// Extra casts are used to silence warnings.
if (is_signed) {
- arg_.type = format_arg::INT;
- arg_.int_value = static_cast<int>(static_cast<TargetType>(value));
+ arg_ = internal::MakeArg<format_context>(
+ static_cast<int>(static_cast<TargetType>(value)));
} else {
- arg_.type = format_arg::UINT;
typedef typename internal::MakeUnsigned<TargetType>::Type Unsigned;
- arg_.uint_value = static_cast<unsigned>(static_cast<Unsigned>(value));
+ arg_ = internal::MakeArg<format_context>(
+ static_cast<unsigned>(static_cast<Unsigned>(value)));
}
} else {
if (is_signed) {
- arg_.type = format_arg::LONG_LONG;
// glibc's printf doesn't sign extend arguments of smaller types:
// std::printf("%lld", -42); // prints "4294967254"
// but we don't have to do the same because it's a UB.
- arg_.long_long_value = static_cast<LongLong>(value);
+ arg_ = internal::MakeArg<format_context>(
+ static_cast<LongLong>(value));
} else {
- arg_.type = format_arg::ULONG_LONG;
- arg_.ulong_long_value =
- static_cast<typename internal::MakeUnsigned<U>::Type>(value);
+ arg_ = internal::MakeArg<format_context>(
+ static_cast<typename internal::MakeUnsigned<U>::Type>(value));
}
}
}
@@ -137,26 +137,27 @@
// If T is void, the argument is converted to corresponding signed or unsigned
// type depending on the type specifier: 'd' and 'i' - signed, other -
// unsigned).
-template <typename T>
-void convert_arg(format_arg &arg, wchar_t type) {
- visit(ArgConverter<T>(arg, type), arg);
+template <typename T, typename Char>
+void convert_arg(basic_format_arg<Char> &arg, Char type) {
+ visit(ArgConverter<T, Char>(arg, type), arg);
}
// Converts an integer argument to char for printf.
+template <typename Char>
class CharConverter {
private:
- format_arg &arg_;
+ basic_format_arg<Char> &arg_;
FMT_DISALLOW_COPY_AND_ASSIGN(CharConverter);
public:
- explicit CharConverter(format_arg &arg) : arg_(arg) {}
+ explicit CharConverter(basic_format_arg<Char> &arg) : arg_(arg) {}
template <typename T>
typename std::enable_if<std::is_integral<T>::value>::type
operator()(T value) {
- arg_.type = format_arg::CHAR;
- arg_.int_value = static_cast<char>(value);
+ arg_ =
+ internal::MakeArg<basic_format_context<Char>>(static_cast<char>(value));
}
template <typename T>
@@ -168,14 +169,14 @@
// Checks if an argument is a valid printf width specifier and sets
// left alignment if it is negative.
-class WidthHandler {
+class PrintfWidthHandler {
private:
FormatSpec &spec_;
- FMT_DISALLOW_COPY_AND_ASSIGN(WidthHandler);
+ FMT_DISALLOW_COPY_AND_ASSIGN(PrintfWidthHandler);
public:
- explicit WidthHandler(FormatSpec &spec) : spec_(spec) {}
+ explicit PrintfWidthHandler(FormatSpec &spec) : spec_(spec) {}
template <typename T>
typename std::enable_if<std::is_integral<T>::value, unsigned>::type
@@ -239,7 +240,7 @@
}
/** Formats a character. */
- void operator()(wchar_t value) {
+ void operator()(Char value) {
const FormatSpec &fmt_spec = this->spec();
BasicWriter<Char> &w = this->writer();
if (fmt_spec.type_ && fmt_spec.type_ != 'c')
@@ -282,7 +283,7 @@
/** Formats an argument of a custom (user-defined) type. */
void operator()(format_arg::CustomValue c) {
const Char format_str[] = {'}', '\0'};
- auto args = basic_format_args<basic_format_context<Char>>();
+ auto args = basic_format_args<basic_format_context<Char>, Char>();
basic_format_context<Char> ctx(format_str, args);
c.format(&this->writer(), c.value, &ctx);
}
@@ -305,7 +306,7 @@
// Returns the argument with specified index or, if arg_index is equal
// to the maximum unsigned value, the next argument.
- format_arg get_arg(
+ basic_format_arg<Char> get_arg(
const Char *s,
unsigned arg_index = (std::numeric_limits<unsigned>::max)());
@@ -321,7 +322,7 @@
\endrst
*/
explicit printf_context(BasicCStringRef<Char> format_str,
- basic_format_args<printf_context> args)
+ basic_format_args<printf_context, Char> args)
: Base(format_str.c_str(), args) {}
/** Formats stored arguments and writes the output to the writer. */
@@ -355,11 +356,12 @@
}
template <typename Char, typename AF>
-format_arg printf_context<Char, AF>::get_arg(const Char *s,
- unsigned arg_index) {
+basic_format_arg<Char> printf_context<Char, AF>::get_arg(
+ const Char *s, unsigned arg_index) {
(void)s;
const char *error = 0;
- format_arg arg = arg_index == std::numeric_limits<unsigned>::max() ?
+ basic_format_arg<Char> arg =
+ arg_index == std::numeric_limits<unsigned>::max() ?
this->next_arg(error) : Base::get_arg(arg_index - 1, error);
if (error)
FMT_THROW(format_error(!*s ? "invalid format string" : error));
@@ -395,7 +397,7 @@
spec.width_ = internal::parse_nonnegative_int(s);
} else if (*s == '*') {
++s;
- spec.width_ = visit(internal::WidthHandler(spec), get_arg(s));
+ spec.width_ = visit(internal::PrintfWidthHandler(spec), get_arg(s));
}
return arg_index;
}
@@ -427,15 +429,15 @@
spec.precision_ = static_cast<int>(internal::parse_nonnegative_int(s));
} else if (*s == '*') {
++s;
- spec.precision_ = visit(internal::PrecisionHandler(), get_arg(s));
+ spec.precision_ = visit(internal::PrintfPrecisionHandler(), get_arg(s));
}
}
- format_arg arg = get_arg(s, arg_index);
+ basic_format_arg<Char> arg = get_arg(s, arg_index);
if (spec.flag(HASH_FLAG) && visit(internal::IsZeroInt(), arg))
spec.flags_ &= ~internal::to_unsigned<int>(HASH_FLAG);
if (spec.fill_ == '0') {
- if (arg.type <= format_arg::LAST_NUMERIC_TYPE)
+ if (arg.is_numeric())
spec.align_ = ALIGN_NUMERIC;
else
spec.fill_ = ' '; // Ignore '0' flag for non-numeric types.
@@ -478,7 +480,7 @@
if (!*s)
FMT_THROW(format_error("invalid format string"));
spec.type_ = static_cast<char>(*s++);
- if (arg.type <= format_arg::LAST_INTEGER_TYPE) {
+ if (arg.is_integral()) {
// Normalize type.
switch (spec.type_) {
case 'i': case 'u':
@@ -486,7 +488,7 @@
break;
case 'c':
// TODO: handle wchar_t
- visit(internal::CharConverter(arg), arg);
+ visit(internal::CharConverter<Char>(arg), arg);
break;
}
}
@@ -509,12 +511,13 @@
template <typename Char>
void printf(BasicWriter<Char> &w, BasicCStringRef<Char> format,
- basic_format_args<printf_context<Char>> args) {
+ basic_format_args<printf_context<Char>, Char> args) {
printf_context<Char>(format, args).format(w);
}
-inline std::string vsprintf(CStringRef format,
- basic_format_args<printf_context<char>> args) {
+typedef basic_format_args<printf_context<char>, char> printf_args;
+
+inline std::string vsprintf(CStringRef format, printf_args args) {
MemoryWriter w;
printf(w, format, args);
return w.str();
@@ -534,8 +537,9 @@
return vsprintf(format_str, make_xformat_args<printf_context<char>>(args...));
}
-inline std::wstring vsprintf(WCStringRef format,
- basic_format_args<printf_context<wchar_t>> args) {
+inline std::wstring vsprintf(
+ WCStringRef format,
+ basic_format_args<printf_context<wchar_t>, wchar_t> args) {
WMemoryWriter w;
printf(w, format, args);
return w.str();
@@ -547,8 +551,7 @@
return vsprintf(format_str, vargs);
}
-FMT_API int vfprintf(std::FILE *f, CStringRef format,
- basic_format_args<printf_context<char>> args);
+FMT_API int vfprintf(std::FILE *f, CStringRef format, printf_args args);
/**
\rst
@@ -565,8 +568,7 @@
return vfprintf(f, format_str, vargs);
}
-inline int vprintf(CStringRef format,
- basic_format_args<printf_context<char>> args) {
+inline int vprintf(CStringRef format, printf_args args) {
return vfprintf(stdout, format, args);
}
@@ -584,8 +586,7 @@
return vprintf(format_str, make_xformat_args<printf_context<char>>(args...));
}
-inline int vfprintf(std::ostream &os, CStringRef format_str,
- basic_format_args<printf_context<char>> args) {
+inline int vfprintf(std::ostream &os, CStringRef format_str, printf_args args) {
MemoryWriter w;
printf(w, format_str, args);
internal::write(os, w);
diff --git a/test/custom-formatter-test.cc b/test/custom-formatter-test.cc
index 7601aeb..51bed76 100644
--- a/test/custom-formatter-test.cc
+++ b/test/custom-formatter-test.cc
@@ -63,7 +63,7 @@
std::string custom_vsprintf(
const char* format_str,
- fmt::basic_format_args<CustomPrintfFormatter> args) {
+ fmt::basic_format_args<CustomPrintfFormatter, char> args) {
fmt::MemoryWriter writer;
CustomPrintfFormatter formatter(format_str, args);
formatter.format(writer);
diff --git a/test/format-impl-test.cc b/test/format-impl-test.cc
index c63b6b9..ff702a5 100644
--- a/test/format-impl-test.cc
+++ b/test/format-impl-test.cc
@@ -41,13 +41,25 @@
#undef min
#undef max
+template <typename T>
+struct ValueExtractor {
+ T operator()(T value) {
+ return value;
+ }
+
+ template <typename U>
+ T operator()(U) {
+ throw std::runtime_error(fmt::format("invalid type {}", typeid(U).name()));
+ return T();
+ }
+};
+
TEST(FormatTest, ArgConverter) {
using fmt::format_arg;
- format_arg arg = format_arg();
- arg.type = format_arg::LONG_LONG;
- arg.long_long_value = std::numeric_limits<fmt::LongLong>::max();
- visit(fmt::internal::ArgConverter<fmt::LongLong>(arg, 'd'), arg);
- EXPECT_EQ(format_arg::LONG_LONG, arg.type);
+ fmt::LongLong value = std::numeric_limits<fmt::LongLong>::max();
+ format_arg arg = fmt::internal::MakeArg<fmt::format_context>(value);
+ visit(fmt::internal::ArgConverter<fmt::LongLong, char>(arg, 'd'), arg);
+ EXPECT_EQ(value, visit(ValueExtractor<fmt::LongLong>(), arg));
}
TEST(FormatTest, FormatNegativeNaN) {
diff --git a/test/util-test.cc b/test/util-test.cc
index 3371ef2..c39b0ab 100644
--- a/test/util-test.cc
+++ b/test/util-test.cc
@@ -49,13 +49,17 @@
#include "fmt/format.h"
+#undef min
#undef max
+using fmt::basic_format_arg;
using fmt::format_arg;
using fmt::Buffer;
using fmt::StringRef;
using fmt::internal::MemoryBuffer;
+using fmt::internal::Value;
+using testing::_;
using testing::Return;
using testing::StrictMock;
@@ -70,11 +74,9 @@
}
template <typename Char, typename T>
-format_arg make_arg(const T &value) {
- typedef fmt::internal::MakeValue< fmt::basic_format_context<Char> > MakeValue;
- format_arg arg = MakeValue(value);
- arg.type = fmt::internal::type<T>();
- return arg;
+basic_format_arg<Char> make_arg(const T &value) {
+ typedef fmt::internal::MakeArg< fmt::basic_format_context<Char> > MakeArg;
+ return MakeArg(value);
}
} // namespace
@@ -406,175 +408,9 @@
EXPECT_STREQ("200", s);
}
-template <format_arg::Type>
-struct ArgInfo;
-
-#define ARG_INFO(type_code, Type, field) \
- template <> \
- struct ArgInfo<format_arg::type_code> { \
- static Type get(const format_arg &arg) { return arg.field; } \
- }
-
-ARG_INFO(INT, int, int_value);
-ARG_INFO(UINT, unsigned, uint_value);
-ARG_INFO(LONG_LONG, fmt::LongLong, long_long_value);
-ARG_INFO(ULONG_LONG, fmt::ULongLong, ulong_long_value);
-ARG_INFO(BOOL, int, int_value);
-ARG_INFO(CHAR, int, int_value);
-ARG_INFO(DOUBLE, double, double_value);
-ARG_INFO(LONG_DOUBLE, long double, long_double_value);
-ARG_INFO(CSTRING, const char *, string.value);
-ARG_INFO(STRING, const char *, string.value);
-ARG_INFO(WSTRING, const wchar_t *, wstring.value);
-ARG_INFO(POINTER, const void *, pointer);
-ARG_INFO(CUSTOM, format_arg::CustomValue, custom);
-
-#define CHECK_ARG_INFO(Type, field, value) { \
- format_arg arg = format_arg(); \
- arg.field = value; \
- EXPECT_EQ(value, ArgInfo<format_arg::Type>::get(arg)); \
-}
-
-TEST(ArgTest, ArgInfo) {
- CHECK_ARG_INFO(INT, int_value, 42);
- CHECK_ARG_INFO(UINT, uint_value, 42u);
- CHECK_ARG_INFO(LONG_LONG, long_long_value, 42);
- CHECK_ARG_INFO(ULONG_LONG, ulong_long_value, 42u);
- CHECK_ARG_INFO(DOUBLE, double_value, 4.2);
- CHECK_ARG_INFO(LONG_DOUBLE, long_double_value, 4.2);
- CHECK_ARG_INFO(CHAR, int_value, 'x');
- const char STR[] = "abc";
- CHECK_ARG_INFO(CSTRING, string.value, STR);
- const wchar_t WSTR[] = L"abc";
- CHECK_ARG_INFO(WSTRING, wstring.value, WSTR);
- int p = 0;
- CHECK_ARG_INFO(POINTER, pointer, &p);
- format_arg arg = format_arg();
- arg.custom.value = &p;
- EXPECT_EQ(&p, ArgInfo<format_arg::CUSTOM>::get(arg).value);
-}
-
-#define EXPECT_ARG_(Char, type_code, MakeArgType, ExpectedType, value) { \
- MakeArgType input = static_cast<MakeArgType>(value); \
- format_arg arg = make_arg<Char>(input); \
- EXPECT_EQ(format_arg::type_code, arg.type); \
- ExpectedType expected_value = static_cast<ExpectedType>(value); \
- EXPECT_EQ(expected_value, ArgInfo<format_arg::type_code>::get(arg)); \
-}
-
-#define EXPECT_ARG(type_code, Type, value) \
- EXPECT_ARG_(char, type_code, Type, Type, value)
-
-#define EXPECT_ARGW(type_code, Type, value) \
- EXPECT_ARG_(wchar_t, type_code, Type, Type, value)
-
-TEST(ArgTest, MakeArg) {
- // Test bool.
- EXPECT_ARG_(char, BOOL, bool, int, true);
- EXPECT_ARG_(wchar_t, BOOL, bool, int, true);
-
- // Test char.
- EXPECT_ARG(CHAR, char, 'a');
- EXPECT_ARG(CHAR, char, CHAR_MIN);
- EXPECT_ARG(CHAR, char, CHAR_MAX);
-
- // Test wchar_t.
- EXPECT_ARGW(CHAR, wchar_t, L'a');
- EXPECT_ARGW(CHAR, wchar_t, WCHAR_MIN);
- EXPECT_ARGW(CHAR, wchar_t, WCHAR_MAX);
-
- // Test signed/unsigned char.
- EXPECT_ARG(INT, signed char, 42);
- EXPECT_ARG(INT, signed char, SCHAR_MIN);
- EXPECT_ARG(INT, signed char, SCHAR_MAX);
- EXPECT_ARG(UINT, unsigned char, 42);
- EXPECT_ARG(UINT, unsigned char, UCHAR_MAX );
-
- // Test short.
- EXPECT_ARG(INT, short, 42);
- EXPECT_ARG(INT, short, SHRT_MIN);
- EXPECT_ARG(INT, short, SHRT_MAX);
- EXPECT_ARG(UINT, unsigned short, 42);
- EXPECT_ARG(UINT, unsigned short, USHRT_MAX);
-
- // Test int.
- EXPECT_ARG(INT, int, 42);
- EXPECT_ARG(INT, int, INT_MIN);
- EXPECT_ARG(INT, int, INT_MAX);
- EXPECT_ARG(UINT, unsigned, 42);
- EXPECT_ARG(UINT, unsigned, UINT_MAX);
-
- // Test long.
-#if LONG_MAX == INT_MAX
-# define LONG INT
-# define ULONG UINT
-# define long_value int_value
-# define ulong_value uint_value
-#else
-# define LONG LONG_LONG
-# define ULONG ULONG_LONG
-# define long_value long_long_value
-# define ulong_value ulong_long_value
-#endif
- EXPECT_ARG(LONG, long, 42);
- EXPECT_ARG(LONG, long, LONG_MIN);
- EXPECT_ARG(LONG, long, LONG_MAX);
- EXPECT_ARG(ULONG, unsigned long, 42);
- EXPECT_ARG(ULONG, unsigned long, ULONG_MAX);
-
- // Test long long.
- EXPECT_ARG(LONG_LONG, fmt::LongLong, 42);
- EXPECT_ARG(LONG_LONG, fmt::LongLong, LLONG_MIN);
- EXPECT_ARG(LONG_LONG, fmt::LongLong, LLONG_MAX);
- EXPECT_ARG(ULONG_LONG, fmt::ULongLong, 42);
- EXPECT_ARG(ULONG_LONG, fmt::ULongLong, ULLONG_MAX);
-
- // Test float.
- EXPECT_ARG(DOUBLE, float, 4.2);
- EXPECT_ARG(DOUBLE, float, FLT_MIN);
- EXPECT_ARG(DOUBLE, float, FLT_MAX);
-
- // Test double.
- EXPECT_ARG(DOUBLE, double, 4.2);
- EXPECT_ARG(DOUBLE, double, DBL_MIN);
- EXPECT_ARG(DOUBLE, double, DBL_MAX);
-
- // Test long double.
- EXPECT_ARG(LONG_DOUBLE, long double, 4.2);
- EXPECT_ARG(LONG_DOUBLE, long double, LDBL_MIN);
- EXPECT_ARG(LONG_DOUBLE, long double, LDBL_MAX);
-
- // Test string.
- char STR[] = "test";
- EXPECT_ARG(CSTRING, char*, STR);
- EXPECT_ARG(CSTRING, const char*, STR);
- EXPECT_ARG(STRING, std::string, STR);
- EXPECT_ARG(STRING, fmt::StringRef, STR);
-
- // Test wide string.
- wchar_t WSTR[] = L"test";
- EXPECT_ARGW(WSTRING, wchar_t*, WSTR);
- EXPECT_ARGW(WSTRING, const wchar_t*, WSTR);
- EXPECT_ARGW(WSTRING, std::wstring, WSTR);
- EXPECT_ARGW(WSTRING, fmt::WStringRef, WSTR);
-
- int n = 42;
- EXPECT_ARG(POINTER, void*, &n);
- EXPECT_ARG(POINTER, const void*, &n);
-
- ::Test t;
- format_arg arg = make_arg<char>(t);
- EXPECT_EQ(format_arg::CUSTOM, arg.type);
- EXPECT_EQ(&t, arg.custom.value);
- fmt::MemoryWriter w;
- fmt::format_context ctx("}", fmt::format_args());
- arg.custom.format(&w, &t, &ctx);
- EXPECT_EQ("test", w.str());
-}
-
TEST(UtilTest, FormatArgs) {
fmt::format_args args;
- EXPECT_EQ(format_arg::NONE, args[1].type);
+ EXPECT_FALSE(args[1]);
}
struct CustomFormatter {
@@ -595,73 +431,163 @@
EXPECT_TRUE(ctx.called);
}
-struct Result {
- format_arg arg;
+namespace fmt {
+namespace internal {
- Result() : arg(make_arg<char>(0xdeadbeef)) {}
-
- template <typename T>
- Result(const T& value) : arg(make_arg<char>(value)) {}
- Result(const wchar_t *s) : arg(make_arg<wchar_t>(s)) {}
-};
-
-struct TestVisitor {
- Result operator()(int value) { return value; }
- Result operator()(unsigned value) { return value; }
- Result operator()(fmt::LongLong value) { return value; }
- Result operator()(fmt::ULongLong value) { return value; }
- Result operator()(double value) { return value; }
- Result operator()(long double value) { return value; }
- Result operator()(wchar_t value) { return static_cast<char>(value); }
- Result operator()(const char *s) { return s; }
- Result operator()(fmt::format_arg::StringValue<char> s) {
- return s.value;
- }
- Result operator()(fmt::format_arg::StringValue<wchar_t> s) {
- return s.value;
- }
- Result operator()(const void *p) { return p; }
- Result operator()(fmt::format_arg::CustomValue c) {
- return *static_cast<const ::Test*>(c.value);
- }
-};
-
-#define EXPECT_RESULT_(Char, type_code, value) { \
- format_arg arg = make_arg<Char>(value); \
- Result result = fmt::visit(TestVisitor(), arg); \
- EXPECT_EQ(format_arg::type_code, result.arg.type); \
- EXPECT_EQ(value, ArgInfo<format_arg::type_code>::get(result.arg)); \
+bool operator==(Value::CustomValue lhs, Value::CustomValue rhs) {
+ return lhs.value == rhs.value;
}
-#define EXPECT_RESULT(type_code, value) \
- EXPECT_RESULT_(char, type_code, value)
-#define EXPECT_RESULTW(type_code, value) \
- EXPECT_RESULT_(wchar_t, type_code, value)
+template <typename T>
+bool operator==(Value::StringValue<T> lhs, Value::StringValue<T> rhs) {
+ return std::basic_string<T>(lhs.value, lhs.size) ==
+ std::basic_string<T>(rhs.value, rhs.size);
+}
+}
+}
-TEST(ArgVisitorTest, VisitAll) {
- EXPECT_RESULT(INT, 42);
- EXPECT_RESULT(UINT, 42u);
- EXPECT_RESULT(LONG_LONG, 42ll);
- EXPECT_RESULT(ULONG_LONG, 42ull);
- EXPECT_RESULT(DOUBLE, 4.2);
- EXPECT_RESULT(LONG_DOUBLE, 4.2l);
- EXPECT_RESULT(CHAR, 'x');
- const char STR[] = "abc";
- EXPECT_RESULT(CSTRING, STR);
- const wchar_t WSTR[] = L"abc";
- EXPECT_RESULTW(WSTRING, WSTR);
- const void *p = STR;
- EXPECT_RESULT(POINTER, p);
- ::Test t;
- Result result = visit(TestVisitor(), make_arg<char>(t));
- EXPECT_EQ(format_arg::CUSTOM, result.arg.type);
- EXPECT_EQ(&t, result.arg.custom.value);
+template <typename T>
+struct MockVisitor {
+ // Use a unique result type to make sure that there are no undesirable
+ // conversions.
+ struct Result {};
+
+ MockVisitor() {
+ ON_CALL(*this, visit(_)).WillByDefault(Return(Result()));
+ }
+
+ MOCK_METHOD1_T(visit, Result (T value));
+ MOCK_METHOD0_T(unexpected, void ());
+
+ Result operator()(T value) { return visit(value); }
+
+ template <typename U>
+ Result operator()(U value) {
+ unexpected();
+ return Result();
+ }
+};
+
+template <typename T>
+struct VisitType { typedef T Type; };
+
+#define VISIT_TYPE(Type_, VisitType_) \
+ template <> \
+ struct VisitType<Type_> { typedef VisitType_ Type; }
+
+VISIT_TYPE(signed char, int);
+VISIT_TYPE(unsigned char, unsigned);
+VISIT_TYPE(short, int);
+VISIT_TYPE(unsigned short, unsigned);
+
+#if LONG_MAX == INT_MAX
+VISIT_TYPE(long, int);
+VISIT_TYPE(unsigned long, unsigned);
+#else
+VISIT_TYPE(long, fmt::LongLong);
+VISIT_TYPE(unsigned long, fmt::ULongLong);
+#endif
+
+VISIT_TYPE(float, double);
+
+#define CHECK_ARG_(Char, expected, value) { \
+ testing::StrictMock<MockVisitor<decltype(expected)>> visitor; \
+ EXPECT_CALL(visitor, visit(expected)); \
+ fmt::visit(visitor, make_arg<Char>(value)); \
+}
+
+#define CHECK_ARG(value) { \
+ typename VisitType<decltype(value)>::Type expected = value; \
+ CHECK_ARG_(char, expected, value) \
+ CHECK_ARG_(wchar_t, expected, value) \
+}
+
+template <typename T>
+class NumericArgTest : public testing::Test {};
+
+typedef ::testing::Types<
+ bool, signed char, unsigned char, signed, unsigned short,
+ int, unsigned, long, unsigned long, fmt::LongLong, fmt::ULongLong,
+ float, double, long double> Types;
+TYPED_TEST_CASE(NumericArgTest, Types);
+
+template <typename T>
+typename std::enable_if<std::is_integral<T>::value, T>::type test_value() {
+ return static_cast<T>(42);
+}
+
+template <typename T>
+typename std::enable_if<std::is_floating_point<T>::value, T>::type
+ test_value() {
+ return static_cast<T>(4.2);
+}
+
+TYPED_TEST(NumericArgTest, MakeAndVisit) {
+ CHECK_ARG(test_value<TypeParam>());
+ CHECK_ARG(std::numeric_limits<TypeParam>::min());
+ CHECK_ARG(std::numeric_limits<TypeParam>::max());
+}
+
+TEST(UtilTest, CharArg) {
+ CHECK_ARG_(char, 'a', 'a');
+ CHECK_ARG_(wchar_t, L'a', 'a');
+ CHECK_ARG_(wchar_t, L'a', L'a');
+}
+
+TEST(UtilTest, StringArg) {
+ char str_data[] = "test";
+ char *str = str_data;
+ const char *cstr = str;
+ CHECK_ARG_(char, cstr, str);
+ CHECK_ARG_(wchar_t, cstr, str);
+ CHECK_ARG(cstr);
+
+ Value::StringValue<char> strval = {str, 4};
+ CHECK_ARG_(char, strval, std::string(str));
+ CHECK_ARG_(wchar_t, strval, std::string(str));
+ CHECK_ARG_(char, strval, fmt::StringRef(str));
+ CHECK_ARG_(wchar_t, strval, fmt::StringRef(str));
+}
+
+TEST(UtilTest, WStringArg) {
+ wchar_t str_data[] = L"test";
+ wchar_t *str = str_data;
+ const wchar_t *cstr = str;
+
+ Value::StringValue<wchar_t> strval = {str, 4};
+ CHECK_ARG_(wchar_t, strval, str);
+ CHECK_ARG_(wchar_t, strval, cstr);
+ CHECK_ARG_(wchar_t, strval, std::wstring(str));
+ CHECK_ARG_(wchar_t, strval, fmt::WStringRef(str));
+}
+
+TEST(UtilTest, PointerArg) {
+ void *p = 0;
+ const void *cp = 0;
+ CHECK_ARG_(char, cp, p);
+ CHECK_ARG_(wchar_t, cp, p);
+ CHECK_ARG(cp);
+}
+
+TEST(UtilTest, CustomArg) {
+ ::Test test;
+ typedef MockVisitor<Value::CustomValue> Visitor;
+ testing::StrictMock<Visitor> visitor;
+ EXPECT_CALL(visitor, visit(_)).WillOnce(
+ testing::Invoke([&](Value::CustomValue custom) {
+ EXPECT_EQ(&test, custom.value);
+ fmt::MemoryWriter w;
+ fmt::format_context ctx("}", fmt::format_args());
+ custom.format(&w, &test, &ctx);
+ EXPECT_EQ("test", w.str());
+ return Visitor::Result();
+ }));
+ fmt::visit(visitor, make_arg<char>(test));
}
TEST(ArgVisitorTest, VisitInvalidArg) {
format_arg arg = format_arg();
- arg.type = static_cast<format_arg::Type>(format_arg::NONE);
- EXPECT_ASSERT(visit(TestVisitor(), arg), "invalid argument type");
+ EXPECT_ASSERT(visit(MockVisitor<int>(), arg), "invalid argument type");
}
// Tests fmt::internal::count_digits for integer type Int.