Refactor parse context and fix warnings

This commit is contained in:
Victor Zverovich 2017-11-19 08:49:58 -08:00
parent e7e270f511
commit 44e186512b
4 changed files with 87 additions and 86 deletions

View File

@ -144,14 +144,6 @@
# define FMT_EXCEPTIONS 1 # define FMT_EXCEPTIONS 1
#endif #endif
#ifndef FMT_THROW
# if FMT_EXCEPTIONS
# define FMT_THROW(x) throw x
# else
# define FMT_THROW(x) assert(false)
# endif
#endif
// Define FMT_USE_NOEXCEPT to make fmt use noexcept (C++11 feature). // Define FMT_USE_NOEXCEPT to make fmt use noexcept (C++11 feature).
#ifndef FMT_USE_NOEXCEPT #ifndef FMT_USE_NOEXCEPT
# define FMT_USE_NOEXCEPT 0 # define FMT_USE_NOEXCEPT 0
@ -479,6 +471,24 @@ inline stdext::checked_array_iterator<T*> make_ptr(T *ptr, std::size_t size) {
template <typename T> template <typename T>
inline T *make_ptr(T *ptr, std::size_t) { return ptr; } inline T *make_ptr(T *ptr, std::size_t) { return ptr; }
#endif #endif
#ifndef FMT_THROW
# if FMT_EXCEPTIONS
# define FMT_THROW(x) throw x
# else
# define FMT_THROW(x) assert(false)
# endif
#endif
struct error_handler {
constexpr error_handler() {}
constexpr error_handler(const error_handler &) {}
// This function is intentionally not constexpr to give a compile-time error.
void on_error(const char *message) {
FMT_THROW(format_error(message));
}
};
} // namespace internal } // namespace internal
/** /**
@ -1345,7 +1355,7 @@ class value {
// have different extension points, e.g. `formatter<T>` for `format` and // have different extension points, e.g. `formatter<T>` for `format` and
// `printf_formatter<T>` for `printf`. // `printf_formatter<T>` for `printf`.
typename Context::template formatter_type<T> f; typename Context::template formatter_type<T> f;
auto &&parse_ctx = ctx.get_parse_context(); auto &&parse_ctx = ctx.parse_context();
parse_ctx.advance_to(f.parse(parse_ctx)); parse_ctx.advance_to(f.parse(parse_ctx));
f.format(buffer, *static_cast<const T*>(arg), ctx); f.format(buffer, *static_cast<const T*>(arg), ctx);
} }
@ -1705,18 +1715,67 @@ class basic_format_specs : public align_spec {
typedef basic_format_specs<char> format_specs; typedef basic_format_specs<char> format_specs;
namespace internal { // Parsing context consisting of a format string range being parsed and an
// argument counter for automatic indexing.
template <typename Char, typename ErrorHandler = internal::error_handler>
class basic_parse_context : private ErrorHandler {
private:
basic_string_view<Char> format_str_;
int next_arg_index_;
struct error_handler { protected:
constexpr error_handler() {} constexpr bool check_no_auto_index() {
constexpr error_handler(const error_handler &) {} if (next_arg_index_ > 0) {
on_error("cannot switch from automatic to manual argument indexing");
// This function is intentionally not constexpr to give a compile-time error. return false;
void on_error(const char *message) { }
FMT_THROW(format_error(message)); next_arg_index_ = -1;
return true;
} }
public:
using char_type = Char;
using iterator = typename basic_string_view<Char>::iterator;
explicit constexpr basic_parse_context(
basic_string_view<Char> format_str, ErrorHandler eh = ErrorHandler())
: ErrorHandler(eh), format_str_(format_str), next_arg_index_(0) {}
// Returns an iterator to the beginning of the format string range being
// parsed.
constexpr iterator begin() const { return format_str_.begin(); }
// Returns an iterator past the end of the format string range being parsed.
constexpr iterator end() const { return format_str_.end(); }
// Advances the begin iterator to ``it``.
constexpr void advance_to(iterator it) {
format_str_.remove_prefix(it - begin());
}
// Returns the next argument index.
constexpr unsigned next_arg_index() {
if (next_arg_index_ >= 0)
return internal::to_unsigned(next_arg_index_++);
on_error("cannot switch from manual to automatic argument indexing");
return 0;
}
constexpr void check_arg_id(unsigned) { check_no_auto_index(); }
void check_arg_id(basic_string_view<Char>) {}
constexpr void on_error(const char *message) {
ErrorHandler::on_error(message);
}
constexpr ErrorHandler error_handler() const { return *this; }
}; };
using parse_context = basic_parse_context<char>;
using wparse_context = basic_parse_context<wchar_t>;
namespace internal {
template <typename Handler> template <typename Handler>
constexpr void handle_integral_type_spec(char c, Handler &&handler) { constexpr void handle_integral_type_spec(char c, Handler &&handler) {
switch (c) { switch (c) {
@ -1947,64 +2006,8 @@ class arg_formatter_base {
} }
}; };
// Parsing context representing a format string range being parsed and an
// argument counter for automatic indexing.
template <typename Char, typename ErrorHandler = error_handler>
class parse_context : public ErrorHandler {
private:
basic_string_view<Char> format_str_;
int next_arg_index_;
protected:
constexpr bool check_no_auto_index() {
if (next_arg_index_ > 0) {
on_error("cannot switch from automatic to manual argument indexing");
return false;
}
next_arg_index_ = -1;
return true;
}
public:
using char_type = Char;
using iterator = const Char*;
explicit constexpr parse_context(
basic_string_view<Char> format_str, ErrorHandler eh = ErrorHandler())
: ErrorHandler(eh), format_str_(format_str), next_arg_index_(0) {}
// Returns an iterator to the beginning of the format string range being
// parsed.
constexpr iterator begin() const { return format_str_.begin(); }
// Returns an iterator past the end of the format string range being parsed.
constexpr iterator end() const { return format_str_.end(); }
// Advances the begin iterator to ``it``.
constexpr void advance_to(iterator it) {
format_str_.remove_prefix(it - begin());
}
// Returns the next argument index.
constexpr unsigned next_arg_index() {
if (next_arg_index_ >= 0)
return internal::to_unsigned(next_arg_index_++);
on_error("cannot switch from manual to automatic argument indexing");
return 0;
}
constexpr void check_arg_id(unsigned) { check_no_auto_index(); }
void check_arg_id(basic_string_view<Char>) {}
constexpr void on_error(const char *message) {
ErrorHandler::on_error(message);
}
constexpr ErrorHandler error_handler() const { return *this; }
};
template <typename Char, typename Context> template <typename Char, typename Context>
class context_base : public parse_context<Char>{ class context_base : public basic_parse_context<Char>{
private: private:
basic_args<Context> args_; basic_args<Context> args_;
@ -2012,7 +2015,7 @@ class context_base : public parse_context<Char>{
typedef basic_arg<Context> format_arg; typedef basic_arg<Context> format_arg;
context_base(basic_string_view<Char> format_str, basic_args<Context> args) context_base(basic_string_view<Char> format_str, basic_args<Context> args)
: parse_context<Char>(format_str), args_(args) {} : basic_parse_context<Char>(format_str), args_(args) {}
~context_base() {} ~context_base() {}
basic_args<Context> args() const { return args_; } basic_args<Context> args() const { return args_; }
@ -2033,7 +2036,7 @@ class context_base : public parse_context<Char>{
} }
public: public:
parse_context<Char> &get_parse_context() { return *this; } basic_parse_context<Char> &parse_context() { return *this; }
}; };
struct format_string {}; struct format_string {};
@ -2634,7 +2637,7 @@ class format_string_checker {
} }
private: private:
using parse_context_type = parse_context<Char, ErrorHandler>; using parse_context_type = basic_parse_context<Char, ErrorHandler>;
constexpr static size_t NUM_ARGS = sizeof...(Args); constexpr static size_t NUM_ARGS = sizeof...(Args);
constexpr void check_arg_index() { constexpr void check_arg_index() {

View File

@ -331,7 +331,7 @@ class printf_context :
basic_string_view<Char> format_str, basic_args<printf_context> args) basic_string_view<Char> format_str, basic_args<printf_context> args)
: Base(format_str, args) {} : Base(format_str, args) {}
using Base::get_parse_context; using Base::parse_context;
/** Formats stored arguments and writes the output to the buffer. */ /** Formats stored arguments and writes the output to the buffer. */
FMT_API void format(basic_buffer<Char> &buffer); FMT_API void format(basic_buffer<Char> &buffer);

View File

@ -14,8 +14,8 @@ using fmt::string_buffer;
TEST(StringBufferTest, Empty) { TEST(StringBufferTest, Empty) {
string_buffer buffer; string_buffer buffer;
EXPECT_EQ(0, buffer.size()); EXPECT_EQ(0u, buffer.size());
EXPECT_EQ(0, buffer.capacity()); EXPECT_EQ(0u, buffer.capacity());
std::string data; std::string data;
// std::string may have initial capacity. // std::string may have initial capacity.
std::size_t capacity = data.capacity(); std::size_t capacity = data.capacity();
@ -28,7 +28,7 @@ TEST(StringBufferTest, Reserve) {
string_buffer buffer; string_buffer buffer;
std::size_t capacity = std::string().capacity() + 10; std::size_t capacity = std::string().capacity() + 10;
buffer.reserve(capacity); buffer.reserve(capacity);
EXPECT_EQ(0, buffer.size()); EXPECT_EQ(0u, buffer.size());
EXPECT_EQ(capacity, buffer.capacity()); EXPECT_EQ(capacity, buffer.capacity());
std::string data; std::string data;
buffer.move_to(data); buffer.move_to(data);
@ -54,8 +54,8 @@ TEST(StringBufferTest, MoveTo) {
std::string data; std::string data;
buffer.move_to(data); buffer.move_to(data);
EXPECT_EQ(p, &data[0]); EXPECT_EQ(p, &data[0]);
EXPECT_EQ(0, buffer.size()); EXPECT_EQ(0u, buffer.size());
EXPECT_EQ(0, buffer.capacity()); EXPECT_EQ(0u, buffer.capacity());
} }
TEST(StringBufferTest, WString) { TEST(StringBufferTest, WString) {

View File

@ -448,9 +448,7 @@ struct CustomContext {
bool called; bool called;
fmt::internal::parse_context<char> get_parse_context() { fmt::parse_context parse_context() { return fmt::parse_context(""); }
return fmt::internal::parse_context<char>("");
}
}; };
TEST(UtilTest, MakeValueWithCustomFormatter) { TEST(UtilTest, MakeValueWithCustomFormatter) {