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 @@
#ifndef FMT_OVERRIDE
# if FMT_HAS_FEATURE(cxx_override) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
FMT_MSC_VER >= 1900
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1900
# define FMT_OVERRIDE override
# else
# define FMT_OVERRIDE
# endif
#endif
#if FMT_HAS_FEATURE(cxx_explicit_conversions) || \
FMT_MSC_VER >= 1800
#if FMT_HAS_FEATURE(cxx_explicit_conversions) || FMT_MSC_VER >= 1800
# define FMT_EXPLICIT explicit
#else
# define FMT_EXPLICIT
...
...
@@ -88,8 +86,7 @@
#ifndef FMT_NULL
# if FMT_HAS_FEATURE(cxx_nullptr) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
FMT_MSC_VER >= 1600
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1600
# define FMT_NULL nullptr
# define FMT_USE_NULLPTR 1
# else
...
...
@@ -123,8 +120,7 @@
#endif
#if FMT_USE_NOEXCEPT || FMT_HAS_FEATURE(cxx_noexcept) || \
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || \
FMT_MSC_VER >= 1900
(FMT_GCC_VERSION >= 408 && FMT_HAS_GXX_CXX11) || FMT_MSC_VER >= 1900
# define FMT_DETECTED_NOEXCEPT noexcept
#else
# define FMT_DETECTED_NOEXCEPT throw()
...
...
@@ -198,7 +194,6 @@
#endif
FMT_BEGIN_NAMESPACE
namespace
internal
{
// 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) {
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
...
...
@@ -255,8 +257,8 @@ class basic_string_view {
the size with ``std::char_traits<Char>::length``.
\endrst
*/
basic_string_view
(
const
Char
*
s
)
:
data_
(
s
),
size_
(
std
::
char_traits
<
Char
>
::
length
(
s
))
{}
FMT_CONSTEXPR
basic_string_view
(
const
Char
*
s
)
:
data_
(
s
),
size_
(
internal
::
length
(
s
))
{}
/** Constructs a string reference from a ``std::basic_string`` object. */
template
<
typename
Alloc
>
...
...
@@ -268,7 +270,7 @@ class basic_string_view {
:
data_
(
s
.
data
()),
size_
(
s
.
size
())
{}
/** 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. */
FMT_CONSTEXPR
size_t
size
()
const
{
return
size_
;
}
...
...
@@ -974,7 +976,7 @@ inline typename std::enable_if<!IS_PACKED, basic_format_arg<Context>>::type
make_arg
(
const
T
&
value
)
{
return
make_arg
<
Context
>
(
value
);
}
}
}
// namespace internal
/**
\rst
...
...
@@ -1088,7 +1090,8 @@ class basic_format_args {
format_arg
do_get
(
size_type
index
)
const
{
long
long
signed_types
=
static_cast
<
long
long
>
(
types_
);
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
();
}
format_arg
arg
;
...
...
include/fmt/format.h
View file @
f66ba650
...
...
@@ -737,7 +737,7 @@ class null_terminating_iterator {
FMT_CONSTEXPR
explicit
null_terminating_iterator
(
const
Range
&
r
)
:
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_
);
ptr_
=
ptr
;
return
*
this
;
...
...
@@ -2138,25 +2138,28 @@ struct id_adapter {
Handler
&
handler
;
};
template
<
typename
Iterator
,
typename
Handler
>
FMT_CONSTEXPR
void
parse_format_string
(
Iterator
it
,
Handler
&&
handler
)
{
typedef
typename
std
::
iterator_traits
<
Iterator
>::
value_type
char_type
;
auto
start
=
it
;
while
(
*
it
)
{
char_type
ch
=
*
it
++
;
template
<
typename
Char
,
typename
Handler
>
FMT_CONSTEXPR
void
parse_format_string
(
basic_string_view
<
Char
>
format_str
,
Handler
&&
handler
)
{
auto
begin
=
format_str
.
data
();
auto
end
=
begin
+
format_str
.
size
();
auto
p
=
begin
;
while
(
p
!=
end
)
{
Char
ch
=
*
p
++
;
if
(
ch
!=
'{'
&&
ch
!=
'}'
)
continue
;
if
(
*
it
==
ch
)
{
handler
.
on_text
(
start
,
it
);
start
=
++
it
;
if
(
p
!=
end
&&
*
p
==
ch
)
{
handler
.
on_text
(
begin
,
p
);
begin
=
++
p
;
continue
;
}
if
(
ch
==
'}'
)
{
handler
.
on_error
(
"unmatched '}' in format string"
);
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
==
'}'
)
{
handler
.
on_replacement_field
(
it
);
}
else
if
(
*
it
==
':'
)
{
...
...
@@ -2170,10 +2173,11 @@ FMT_CONSTEXPR void parse_format_string(Iterator it, Handler &&handler) {
handler
.
on_error
(
"missing '}' in format string"
);
return
;
}
p
=
pointer_from
(
it
);
start
=
++
it
;
begin
=
++
p
;
}
handler
.
on_text
(
start
,
it
);
handler
.
on_text
(
begin
,
p
);
}
template
<
typename
T
,
typename
ParseContext
>
...
...
@@ -2192,6 +2196,8 @@ class format_string_checker {
:
arg_id_
(
-
1
),
context_
(
format_str
,
eh
),
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_arg_id
()
{
...
...
@@ -2205,12 +2211,12 @@ class format_string_checker {
}
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
)
{
context_
.
advance_to
(
s
);
return
to_unsigned
(
arg_id_
)
<
NUM_ARGS
?
parse_funcs_
[
arg_id_
](
context_
)
:
s
;
FMT_CONSTEXPR
const
Char
*
on_format_specs
(
iterator
it
)
{
auto
p
=
pointer_from
(
it
);
context_
.
advance_to
(
p
);
return
to_unsigned
(
arg_id_
)
<
NUM_ARGS
?
parse_funcs_
[
arg_id_
](
context_
)
:
p
;
}
FMT_CONSTEXPR
void
on_error
(
const
char
*
message
)
{
...
...
@@ -2238,7 +2244,7 @@ template <typename Char, typename ErrorHandler, typename... Args>
FMT_CONSTEXPR
bool
check_format_string
(
basic_string_view
<
Char
>
s
,
ErrorHandler
eh
=
ErrorHandler
())
{
format_string_checker
<
Char
,
ErrorHandler
,
Args
...
>
checker
(
s
,
eh
);
parse_format_string
(
s
.
begin
()
,
checker
);
parse_format_string
(
s
,
checker
);
return
true
;
}
...
...
@@ -3297,7 +3303,7 @@ struct format_handler : internal::error_handler {
basic_format_args
<
Context
>
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
out
=
context
.
out
();
auto
&&
it
=
internal
::
reserve
(
out
,
size
);
...
...
@@ -3348,9 +3354,8 @@ template <typename ArgFormatter, typename Char, typename Context>
typename
Context
::
iterator
vformat_to
(
typename
ArgFormatter
::
range
out
,
basic_string_view
<
Char
>
format_str
,
basic_format_args
<
Context
>
args
)
{
typedef
internal
::
null_terminating_iterator
<
Char
>
iterator
;
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
();
}
...
...
@@ -3900,8 +3905,9 @@ operator"" _a(const wchar_t *s, std::size_t) { return {s}; }
FMT_END_NAMESPACE
#define FMT_STRING(s) [] { \
typedef typename std::decay<decltype(s)>::type pointer; \
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); } \
}; \
return S{}; \
...
...
test/format-test.cc
View file @
f66ba650
...
...
@@ -1760,16 +1760,18 @@ struct test_format_string_handler {
template
<
typename
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
;
}
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
;
fmt
::
internal
::
parse_format_string
(
s
,
h
);
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