Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
F
fmt
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Libraries
fmt
Commits
f66ba650
Commit
f66ba650
authored
Jul 07, 2018
by
Victor Zverovich
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Optimize format string parsing
parent
f21268aa
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
52 additions
and
41 deletions
+52
-41
include/fmt/core.h
include/fmt/core.h
+17
-14
include/fmt/format.h
include/fmt/format.h
+30
-24
test/format-test.cc
test/format-test.cc
+5
-3
No files found.
include/fmt/core.h
View file @
f66ba650
...
@@ -71,16 +71,14 @@
...
@@ -71,16 +71,14 @@
#ifndef FMT_OVERRIDE
#ifndef FMT_OVERRIDE
# if FMT_HAS_FEATURE(cxx_override) || \
# if FMT_HAS_FEATURE(cxx_override) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1900
FMT_MSC_VER >= 1900
# define FMT_OVERRIDE override
# define FMT_OVERRIDE override
# else
# else
# define FMT_OVERRIDE
# define FMT_OVERRIDE
# endif
# endif
#endif
#endif
#if FMT_HAS_FEATURE(cxx_explicit_conversions) || \
#if FMT_HAS_FEATURE(cxx_explicit_conversions) || FMT_MSC_VER >= 1800
FMT_MSC_VER >= 1800
# define FMT_EXPLICIT explicit
# define FMT_EXPLICIT explicit
#else
#else
# define FMT_EXPLICIT
# define FMT_EXPLICIT
...
@@ -88,8 +86,7 @@
...
@@ -88,8 +86,7 @@
#ifndef FMT_NULL
#ifndef FMT_NULL
# if FMT_HAS_FEATURE(cxx_nullptr) || \
# if FMT_HAS_FEATURE(cxx_nullptr) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1600
FMT_MSC_VER >= 1600
# define FMT_NULL nullptr
# define FMT_NULL nullptr
# define FMT_USE_NULLPTR 1
# define FMT_USE_NULLPTR 1
# else
# else
...
@@ -123,8 +120,7 @@
...
@@ -123,8 +120,7 @@
#endif
#endif
#if FMT_USE_NOEXCEPT || FMT_HAS_FEATURE(cxx_noexcept) || \
#if FMT_USE_NOEXCEPT || FMT_HAS_FEATURE(cxx_noexcept) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1900
FMT_MSC_VER >= 1900
# define FMT_DETECTED_NOEXCEPT noexcept
# define FMT_DETECTED_NOEXCEPT noexcept
#else
#else
# define FMT_DETECTED_NOEXCEPT throw()
# define FMT_DETECTED_NOEXCEPT throw()
...
@@ -198,7 +194,6 @@
...
@@ -198,7 +194,6 @@
#endif
#endif
FMT_BEGIN_NAMESPACE
FMT_BEGIN_NAMESPACE
namespace
internal
{
namespace
internal
{
// An implementation of declval for pre-C++11 compilers such as gcc 4.
// An implementation of declval for pre-C++11 compilers such as gcc 4.
...
@@ -212,7 +207,14 @@ FMT_CONSTEXPR typename std::make_unsigned<Int>::type to_unsigned(Int value) {
...
@@ -212,7 +207,14 @@ FMT_CONSTEXPR typename std::make_unsigned<Int>::type to_unsigned(Int value) {
return
static_cast
<
typename
std
::
make_unsigned
<
Int
>::
type
>
(
value
);
return
static_cast
<
typename
std
::
make_unsigned
<
Int
>::
type
>
(
value
);
}
}
// A constexpr std::char_traits::length replacement for pre-C++17.
template
<
typename
Char
>
FMT_CONSTEXPR
size_t
length
(
const
Char
*
s
)
{
const
Char
*
start
=
s
;
while
(
*
s
)
++
s
;
return
s
-
start
;
}
}
}
// namespace internal
/**
/**
An implementation of ``std::basic_string_view`` for pre-C++17. It provides a
An implementation of ``std::basic_string_view`` for pre-C++17. It provides a
...
@@ -255,8 +257,8 @@ class basic_string_view {
...
@@ -255,8 +257,8 @@ class basic_string_view {
the size with ``std::char_traits<Char>::length``.
the size with ``std::char_traits<Char>::length``.
\endrst
\endrst
*/
*/
basic_string_view
(
const
Char
*
s
)
FMT_CONSTEXPR
basic_string_view
(
const
Char
*
s
)
:
data_
(
s
),
size_
(
std
::
char_traits
<
Char
>
::
length
(
s
))
{}
:
data_
(
s
),
size_
(
internal
::
length
(
s
))
{}
/** Constructs a string reference from a ``std::basic_string`` object. */
/** Constructs a string reference from a ``std::basic_string`` object. */
template
<
typename
Alloc
>
template
<
typename
Alloc
>
...
@@ -268,7 +270,7 @@ class basic_string_view {
...
@@ -268,7 +270,7 @@ class basic_string_view {
:
data_
(
s
.
data
()),
size_
(
s
.
size
())
{}
:
data_
(
s
.
data
()),
size_
(
s
.
size
())
{}
/** Returns a pointer to the string data. */
/** Returns a pointer to the string data. */
const
Char
*
data
()
const
{
return
data_
;
}
FMT_CONSTEXPR
const
Char
*
data
()
const
{
return
data_
;
}
/** Returns the string size. */
/** Returns the string size. */
FMT_CONSTEXPR
size_t
size
()
const
{
return
size_
;
}
FMT_CONSTEXPR
size_t
size
()
const
{
return
size_
;
}
...
@@ -974,7 +976,7 @@ inline typename std::enable_if<!IS_PACKED, basic_format_arg<Context>>::type
...
@@ -974,7 +976,7 @@ inline typename std::enable_if<!IS_PACKED, basic_format_arg<Context>>::type
make_arg
(
const
T
&
value
)
{
make_arg
(
const
T
&
value
)
{
return
make_arg
<
Context
>
(
value
);
return
make_arg
<
Context
>
(
value
);
}
}
}
}
// namespace internal
/**
/**
\rst
\rst
...
@@ -1088,7 +1090,8 @@ class basic_format_args {
...
@@ -1088,7 +1090,8 @@ class basic_format_args {
format_arg
do_get
(
size_type
index
)
const
{
format_arg
do_get
(
size_type
index
)
const
{
long
long
signed_types
=
static_cast
<
long
long
>
(
types_
);
long
long
signed_types
=
static_cast
<
long
long
>
(
types_
);
if
(
signed_types
<
0
)
{
if
(
signed_types
<
0
)
{
unsigned
long
long
num_args
=
static_cast
<
unsigned
long
long
>
(
-
signed_types
);
unsigned
long
long
num_args
=
static_cast
<
unsigned
long
long
>
(
-
signed_types
);
return
index
<
num_args
?
args_
[
index
]
:
format_arg
();
return
index
<
num_args
?
args_
[
index
]
:
format_arg
();
}
}
format_arg
arg
;
format_arg
arg
;
...
...
include/fmt/format.h
View file @
f66ba650
...
@@ -737,7 +737,7 @@ class null_terminating_iterator {
...
@@ -737,7 +737,7 @@ class null_terminating_iterator {
FMT_CONSTEXPR
explicit
null_terminating_iterator
(
const
Range
&
r
)
FMT_CONSTEXPR
explicit
null_terminating_iterator
(
const
Range
&
r
)
:
ptr_
(
r
.
begin
()),
end_
(
r
.
end
())
{}
:
ptr_
(
r
.
begin
()),
end_
(
r
.
end
())
{}
null_terminating_iterator
&
operator
=
(
const
Char
*
ptr
)
{
FMT_CONSTEXPR
null_terminating_iterator
&
operator
=
(
const
Char
*
ptr
)
{
assert
(
ptr
<=
end_
);
assert
(
ptr
<=
end_
);
ptr_
=
ptr
;
ptr_
=
ptr
;
return
*
this
;
return
*
this
;
...
@@ -2138,25 +2138,28 @@ struct id_adapter {
...
@@ -2138,25 +2138,28 @@ struct id_adapter {
Handler
&
handler
;
Handler
&
handler
;
};
};
template
<
typename
Iterator
,
typename
Handler
>
template
<
typename
Char
,
typename
Handler
>
FMT_CONSTEXPR
void
parse_format_string
(
Iterator
it
,
Handler
&&
handler
)
{
FMT_CONSTEXPR
void
parse_format_string
(
typedef
typename
std
::
iterator_traits
<
Iterator
>::
value_type
char_type
;
basic_string_view
<
Char
>
format_str
,
Handler
&&
handler
)
{
auto
start
=
it
;
auto
begin
=
format_str
.
data
();
while
(
*
it
)
{
auto
end
=
begin
+
format_str
.
size
();
char_type
ch
=
*
it
++
;
auto
p
=
begin
;
while
(
p
!=
end
)
{
Char
ch
=
*
p
++
;
if
(
ch
!=
'{'
&&
ch
!=
'}'
)
continue
;
if
(
ch
!=
'{'
&&
ch
!=
'}'
)
continue
;
if
(
*
it
==
ch
)
{
if
(
p
!=
end
&&
*
p
==
ch
)
{
handler
.
on_text
(
start
,
it
);
handler
.
on_text
(
begin
,
p
);
start
=
++
it
;
begin
=
++
p
;
continue
;
continue
;
}
}
if
(
ch
==
'}'
)
{
if
(
ch
==
'}'
)
{
handler
.
on_error
(
"unmatched '}' in format string"
);
handler
.
on_error
(
"unmatched '}' in format string"
);
return
;
return
;
}
}
handler
.
on_text
(
start
,
it
-
1
);
handler
.
on_text
(
begin
,
p
-
1
);
it
=
parse_arg_id
(
it
,
id_adapter
<
Handler
,
char_type
>
(
handler
));
internal
::
null_terminating_iterator
<
Char
>
it
(
p
,
end
);
it
=
parse_arg_id
(
it
,
id_adapter
<
Handler
,
Char
>
(
handler
));
if
(
*
it
==
'}'
)
{
if
(
*
it
==
'}'
)
{
handler
.
on_replacement_field
(
it
);
handler
.
on_replacement_field
(
it
);
}
else
if
(
*
it
==
':'
)
{
}
else
if
(
*
it
==
':'
)
{
...
@@ -2170,10 +2173,11 @@ FMT_CONSTEXPR void parse_format_string(Iterator it, Handler &&handler) {
...
@@ -2170,10 +2173,11 @@ FMT_CONSTEXPR void parse_format_string(Iterator it, Handler &&handler) {
handler
.
on_error
(
"missing '}' in format string"
);
handler
.
on_error
(
"missing '}' in format string"
);
return
;
return
;
}
}
p
=
pointer_from
(
it
);
start
=
++
it
;
begin
=
++
p
;
}
}
handler
.
on_text
(
start
,
it
);
handler
.
on_text
(
begin
,
p
);
}
}
template
<
typename
T
,
typename
ParseContext
>
template
<
typename
T
,
typename
ParseContext
>
...
@@ -2192,6 +2196,8 @@ class format_string_checker {
...
@@ -2192,6 +2196,8 @@ class format_string_checker {
:
arg_id_
(
-
1
),
context_
(
format_str
,
eh
),
:
arg_id_
(
-
1
),
context_
(
format_str
,
eh
),
parse_funcs_
{
&
parse_format_specs
<
Args
,
parse_context_type
>
...}
{}
parse_funcs_
{
&
parse_format_specs
<
Args
,
parse_context_type
>
...}
{}
typedef
internal
::
null_terminating_iterator
<
Char
>
iterator
;
FMT_CONSTEXPR
void
on_text
(
const
Char
*
,
const
Char
*
)
{}
FMT_CONSTEXPR
void
on_text
(
const
Char
*
,
const
Char
*
)
{}
FMT_CONSTEXPR
void
on_arg_id
()
{
FMT_CONSTEXPR
void
on_arg_id
()
{
...
@@ -2205,12 +2211,12 @@ class format_string_checker {
...
@@ -2205,12 +2211,12 @@ class format_string_checker {
}
}
FMT_CONSTEXPR
void
on_arg_id
(
basic_string_view
<
Char
>
)
{}
FMT_CONSTEXPR
void
on_arg_id
(
basic_string_view
<
Char
>
)
{}
FMT_CONSTEXPR
void
on_replacement_field
(
const
Char
*
)
{}
FMT_CONSTEXPR
void
on_replacement_field
(
iterator
)
{}
FMT_CONSTEXPR
const
Char
*
on_format_specs
(
const
Char
*
s
)
{
FMT_CONSTEXPR
const
Char
*
on_format_specs
(
iterator
it
)
{
context_
.
advance_to
(
s
);
auto
p
=
pointer_from
(
it
);
return
to_unsigned
(
arg_id_
)
<
NUM_ARGS
?
context_
.
advance_to
(
p
);
parse_funcs_
[
arg_id_
](
context_
)
:
s
;
return
to_unsigned
(
arg_id_
)
<
NUM_ARGS
?
parse_funcs_
[
arg_id_
](
context_
)
:
p
;
}
}
FMT_CONSTEXPR
void
on_error
(
const
char
*
message
)
{
FMT_CONSTEXPR
void
on_error
(
const
char
*
message
)
{
...
@@ -2238,7 +2244,7 @@ template <typename Char, typename ErrorHandler, typename... Args>
...
@@ -2238,7 +2244,7 @@ template <typename Char, typename ErrorHandler, typename... Args>
FMT_CONSTEXPR
bool
check_format_string
(
FMT_CONSTEXPR
bool
check_format_string
(
basic_string_view
<
Char
>
s
,
ErrorHandler
eh
=
ErrorHandler
())
{
basic_string_view
<
Char
>
s
,
ErrorHandler
eh
=
ErrorHandler
())
{
format_string_checker
<
Char
,
ErrorHandler
,
Args
...
>
checker
(
s
,
eh
);
format_string_checker
<
Char
,
ErrorHandler
,
Args
...
>
checker
(
s
,
eh
);
parse_format_string
(
s
.
begin
()
,
checker
);
parse_format_string
(
s
,
checker
);
return
true
;
return
true
;
}
}
...
@@ -3297,7 +3303,7 @@ struct format_handler : internal::error_handler {
...
@@ -3297,7 +3303,7 @@ struct format_handler : internal::error_handler {
basic_format_args
<
Context
>
format_args
)
basic_format_args
<
Context
>
format_args
)
:
context
(
r
.
begin
(),
str
,
format_args
)
{}
:
context
(
r
.
begin
(),
str
,
format_args
)
{}
void
on_text
(
iterator
begin
,
iterator
end
)
{
void
on_text
(
const
Char
*
begin
,
const
Char
*
end
)
{
auto
size
=
internal
::
to_unsigned
(
end
-
begin
);
auto
size
=
internal
::
to_unsigned
(
end
-
begin
);
auto
out
=
context
.
out
();
auto
out
=
context
.
out
();
auto
&&
it
=
internal
::
reserve
(
out
,
size
);
auto
&&
it
=
internal
::
reserve
(
out
,
size
);
...
@@ -3348,9 +3354,8 @@ template <typename ArgFormatter, typename Char, typename Context>
...
@@ -3348,9 +3354,8 @@ template <typename ArgFormatter, typename Char, typename Context>
typename
Context
::
iterator
vformat_to
(
typename
ArgFormatter
::
range
out
,
typename
Context
::
iterator
vformat_to
(
typename
ArgFormatter
::
range
out
,
basic_string_view
<
Char
>
format_str
,
basic_string_view
<
Char
>
format_str
,
basic_format_args
<
Context
>
args
)
{
basic_format_args
<
Context
>
args
)
{
typedef
internal
::
null_terminating_iterator
<
Char
>
iterator
;
format_handler
<
ArgFormatter
,
Char
,
Context
>
h
(
out
,
format_str
,
args
);
format_handler
<
ArgFormatter
,
Char
,
Context
>
h
(
out
,
format_str
,
args
);
parse_format_string
(
iterator
(
format_str
.
begin
(),
format_str
.
end
())
,
h
);
parse_format_string
(
format_str
,
h
);
return
h
.
context
.
out
();
return
h
.
context
.
out
();
}
}
...
@@ -3900,8 +3905,9 @@ operator"" _a(const wchar_t *s, std::size_t) { return {s}; }
...
@@ -3900,8 +3905,9 @@ operator"" _a(const wchar_t *s, std::size_t) { return {s}; }
FMT_END_NAMESPACE
FMT_END_NAMESPACE
#define FMT_STRING(s) [] { \
#define FMT_STRING(s) [] { \
typedef typename std::decay<decltype(s)>::type pointer; \
struct S : fmt::format_string { \
struct S : fmt::format_string { \
static FMT_CONSTEXPR
decltype(s)
data() { return s; } \
static FMT_CONSTEXPR
pointer
data() { return s; } \
static FMT_CONSTEXPR size_t size() { return sizeof(s); } \
static FMT_CONSTEXPR size_t size() { return sizeof(s); } \
}; \
}; \
return S{}; \
return S{}; \
...
...
test/format-test.cc
View file @
f66ba650
...
@@ -1760,16 +1760,18 @@ struct test_format_string_handler {
...
@@ -1760,16 +1760,18 @@ struct test_format_string_handler {
template
<
typename
T
>
template
<
typename
T
>
FMT_CONSTEXPR
void
on_arg_id
(
T
)
{}
FMT_CONSTEXPR
void
on_arg_id
(
T
)
{}
FMT_CONSTEXPR
void
on_replacement_field
(
const
char
*
)
{}
template
<
typename
Iterator
>
FMT_CONSTEXPR
void
on_replacement_field
(
Iterator
)
{}
FMT_CONSTEXPR
const
char
*
on_format_specs
(
const
char
*
s
)
{
return
s
;
}
template
<
typename
Iterator
>
FMT_CONSTEXPR
Iterator
on_format_specs
(
Iterator
it
)
{
return
it
;
}
FMT_CONSTEXPR
void
on_error
(
const
char
*
)
{
error
=
true
;
}
FMT_CONSTEXPR
void
on_error
(
const
char
*
)
{
error
=
true
;
}
bool
error
=
false
;
bool
error
=
false
;
};
};
FMT_CONSTEXPR
bool
parse_string
(
const
char
*
s
)
{
FMT_CONSTEXPR
bool
parse_string
(
fmt
::
string_view
s
)
{
test_format_string_handler
h
;
test_format_string_handler
h
;
fmt
::
internal
::
parse_format_string
(
s
,
h
);
fmt
::
internal
::
parse_format_string
(
s
,
h
);
return
!
h
.
error
;
return
!
h
.
error
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment