chore(deps/fmt): Bump fmt to 12.1.0 (#25444)
Co-authored-by: Shauren <shauren.trinity@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
This commit is contained in:
parent
472745a944
commit
2a689d7084
32 changed files with 12209 additions and 14171 deletions
2
deps/PackageList.txt
vendored
2
deps/PackageList.txt
vendored
|
|
@ -80,7 +80,7 @@ recastnavigation (Recast is state of the art navigation mesh construction toolse
|
||||||
|
|
||||||
{fmt} is an open-source formatting library providing a fast and safe alternative to C stdio and C++ iostreams.
|
{fmt} is an open-source formatting library providing a fast and safe alternative to C stdio and C++ iostreams.
|
||||||
https://github.com/fmtlib/fmt
|
https://github.com/fmtlib/fmt
|
||||||
Version: 7.1.3
|
Version: 12.1.0 https://github.com/fmtlib/fmt/releases/tag/12.1.0
|
||||||
|
|
||||||
fkYAML (A C++ header-only YAML library)
|
fkYAML (A C++ header-only YAML library)
|
||||||
https://github.com/fktn-k/fkYAML
|
https://github.com/fktn-k/fkYAML
|
||||||
|
|
|
||||||
74
deps/fmt/CMakeLists.txt
vendored
74
deps/fmt/CMakeLists.txt
vendored
|
|
@ -8,67 +8,31 @@
|
||||||
# This program is distributed in the hope that it will be useful, but
|
# This program is distributed in the hope that it will be useful, but
|
||||||
# WITHOUT ANY WARRANTY, to the extent permitted by law; without even the
|
# WITHOUT ANY WARRANTY, to the extent permitted by law; without even the
|
||||||
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
#
|
|
||||||
|
|
||||||
include(CheckSymbolExists)
|
add_library(fmt STATIC)
|
||||||
|
|
||||||
set(strtod_l_headers stdlib.h)
|
target_sources(fmt
|
||||||
|
PRIVATE
|
||||||
if (APPLE)
|
include/fmt/args.h
|
||||||
set(strtod_l_headers ${strtod_l_headers} xlocale.h)
|
include/fmt/base.h
|
||||||
endif ()
|
include/fmt/chrono.h
|
||||||
|
include/fmt/color.h
|
||||||
if(WIN32)
|
include/fmt/compile.h
|
||||||
check_symbol_exists(_strtod_l "${strtod_l_headers}" HAVE_STRTOD_L)
|
include/fmt/core.h
|
||||||
else()
|
include/fmt/format.h
|
||||||
check_symbol_exists(strtod_l "${strtod_l_headers}" HAVE_STRTOD_L)
|
include/fmt/format-inl.h
|
||||||
endif()
|
include/fmt/os.h
|
||||||
|
include/fmt/ostream.h
|
||||||
function(add_headers VAR)
|
include/fmt/printf.h
|
||||||
set(headers ${${VAR}})
|
include/fmt/ranges.h
|
||||||
foreach (header ${ARGN})
|
include/fmt/std.h
|
||||||
set(headers ${headers} include/fmt/${header})
|
include/fmt/xchar.h
|
||||||
endforeach()
|
|
||||||
set(${VAR} ${headers} PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
# Define the fmt library, its includes and the needed defines.
|
|
||||||
add_headers(FMT_HEADERS
|
|
||||||
args.h
|
|
||||||
chrono.h
|
|
||||||
color.h
|
|
||||||
compile.h
|
|
||||||
core.h
|
|
||||||
format.h
|
|
||||||
format-inl.h
|
|
||||||
os.h
|
|
||||||
ostream.h
|
|
||||||
printf.h
|
|
||||||
ranges.h
|
|
||||||
std.h
|
|
||||||
xchar.h)
|
|
||||||
|
|
||||||
set(FMT_SOURCES
|
|
||||||
src/format.cc
|
src/format.cc
|
||||||
src/os.cc)
|
src/os.cc)
|
||||||
|
|
||||||
add_library(fmt STATIC ${FMT_SOURCES} ${FMT_HEADERS})
|
|
||||||
|
|
||||||
if (HAVE_STRTOD_L)
|
|
||||||
target_compile_definitions(fmt
|
|
||||||
PUBLIC
|
|
||||||
FMT_LOCALE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Define FMT_CONSTEVAL for current using in c++20
|
|
||||||
target_compile_definitions(fmt
|
|
||||||
PUBLIC
|
|
||||||
FMT_CONSTEVAL=)
|
|
||||||
|
|
||||||
target_include_directories(fmt
|
target_include_directories(fmt
|
||||||
PUBLIC
|
PUBLIC
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/include)
|
include)
|
||||||
|
|
||||||
target_link_libraries(fmt
|
target_link_libraries(fmt
|
||||||
PRIVATE
|
PRIVATE
|
||||||
acore-dependency-interface)
|
acore-dependency-interface)
|
||||||
|
|
@ -76,4 +40,4 @@ target_link_libraries(fmt
|
||||||
set_target_properties(fmt
|
set_target_properties(fmt
|
||||||
PROPERTIES
|
PROPERTIES
|
||||||
FOLDER
|
FOLDER
|
||||||
"deps")
|
"deps")
|
||||||
|
|
|
||||||
3299
deps/fmt/ChangeLog.md
vendored
Normal file
3299
deps/fmt/ChangeLog.md
vendored
Normal file
File diff suppressed because it is too large
Load diff
5922
deps/fmt/ChangeLog.rst
vendored
5922
deps/fmt/ChangeLog.rst
vendored
File diff suppressed because it is too large
Load diff
0
deps/fmt/LICENSE.rst → deps/fmt/LICENSE
vendored
0
deps/fmt/LICENSE.rst → deps/fmt/LICENSE
vendored
486
deps/fmt/README.md
vendored
Normal file
486
deps/fmt/README.md
vendored
Normal file
|
|
@ -0,0 +1,486 @@
|
||||||
|
<img src="https://user-images.githubusercontent.com/576385/156254208-f5b743a9-88cf-439d-b0c0-923d53e8d551.png" alt="{fmt}" width="25%"/>
|
||||||
|
|
||||||
|
[](https://github.com/fmtlib/fmt/actions?query=workflow%3Alinux)
|
||||||
|
[](https://github.com/fmtlib/fmt/actions?query=workflow%3Amacos)
|
||||||
|
[](https://github.com/fmtlib/fmt/actions?query=workflow%3Awindows)
|
||||||
|
[](https://bugs.chromium.org/p/oss-fuzz/issues/list?\%0Acolspec=ID%20Type%20Component%20Status%20Proj%20Reported%20Owner%20\%0ASummary&q=proj%3Dfmt&can=1)
|
||||||
|
[](https://www.bestpractices.dev/projects/8880)
|
||||||
|
[](https://securityscorecards.dev/viewer/?uri=github.com/fmtlib/fmt)
|
||||||
|
[](https://stackoverflow.com/questions/tagged/fmt)
|
||||||
|
|
||||||
|
**{fmt}** is an open-source formatting library providing a fast and safe
|
||||||
|
alternative to C stdio and C++ iostreams.
|
||||||
|
|
||||||
|
If you like this project, please consider donating to one of the funds
|
||||||
|
that help victims of the war in Ukraine: <https://www.stopputin.net/>.
|
||||||
|
|
||||||
|
[Documentation](https://fmt.dev)
|
||||||
|
|
||||||
|
[Cheat Sheets](https://hackingcpp.com/cpp/libs/fmt.html)
|
||||||
|
|
||||||
|
Q&A: ask questions on [StackOverflow with the tag
|
||||||
|
fmt](https://stackoverflow.com/questions/tagged/fmt).
|
||||||
|
|
||||||
|
Try {fmt} in [Compiler Explorer](https://godbolt.org/z/8Mx1EW73v).
|
||||||
|
|
||||||
|
# Features
|
||||||
|
|
||||||
|
- Simple [format API](https://fmt.dev/latest/api/) with positional
|
||||||
|
arguments for localization
|
||||||
|
- Implementation of [C++20
|
||||||
|
std::format](https://en.cppreference.com/w/cpp/utility/format) and
|
||||||
|
[C++23 std::print](https://en.cppreference.com/w/cpp/io/print)
|
||||||
|
- [Format string syntax](https://fmt.dev/latest/syntax/) similar
|
||||||
|
to Python\'s
|
||||||
|
[format](https://docs.python.org/3/library/stdtypes.html#str.format)
|
||||||
|
- Fast IEEE 754 floating-point formatter with correct rounding,
|
||||||
|
shortness and round-trip guarantees using the
|
||||||
|
[Dragonbox](https://github.com/jk-jeon/dragonbox) algorithm
|
||||||
|
- Portable Unicode support
|
||||||
|
- Safe [printf
|
||||||
|
implementation](https://fmt.dev/latest/api/#printf-formatting)
|
||||||
|
including the POSIX extension for positional arguments
|
||||||
|
- Extensibility: [support for user-defined
|
||||||
|
types](https://fmt.dev/latest/api/#formatting-user-defined-types)
|
||||||
|
- High performance: faster than common standard library
|
||||||
|
implementations of `(s)printf`, iostreams, `to_string` and
|
||||||
|
`to_chars`, see [Speed tests](#speed-tests) and [Converting a
|
||||||
|
hundred million integers to strings per
|
||||||
|
second](http://www.zverovich.net/2020/06/13/fast-int-to-string-revisited.html)
|
||||||
|
- Small code size both in terms of source code with the minimum
|
||||||
|
configuration consisting of just three files, `base.h`, `format.h`
|
||||||
|
and `format-inl.h`, and compiled code; see [Compile time and code
|
||||||
|
bloat](#compile-time-and-code-bloat)
|
||||||
|
- Reliability: the library has an extensive set of
|
||||||
|
[tests](https://github.com/fmtlib/fmt/tree/master/test) and is
|
||||||
|
[continuously fuzzed](https://bugs.chromium.org/p/oss-fuzz/issues/list?colspec=ID%20Type%20Component%20Status%20Proj%20Reported%20Owner%20Summary&q=proj%3Dfmt&can=1)
|
||||||
|
- Safety: the library is fully type-safe, errors in format strings can
|
||||||
|
be reported at compile time, automatic memory management prevents
|
||||||
|
buffer overflow errors
|
||||||
|
- Ease of use: small self-contained code base, no external
|
||||||
|
dependencies, permissive MIT
|
||||||
|
[license](https://github.com/fmtlib/fmt/blob/master/LICENSE)
|
||||||
|
- [Portability](https://fmt.dev/latest/#portability) with
|
||||||
|
consistent output across platforms and support for older compilers
|
||||||
|
- Clean warning-free codebase even on high warning levels such as
|
||||||
|
`-Wall -Wextra -pedantic`
|
||||||
|
- Locale independence by default
|
||||||
|
- Optional header-only configuration enabled with the
|
||||||
|
`FMT_HEADER_ONLY` macro
|
||||||
|
|
||||||
|
See the [documentation](https://fmt.dev) for more details.
|
||||||
|
|
||||||
|
# Examples
|
||||||
|
|
||||||
|
**Print to stdout** ([run](https://godbolt.org/z/Tevcjh))
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
#include <fmt/base.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
fmt::print("Hello, world!\n");
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Format a string** ([run](https://godbolt.org/z/oK8h33))
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
std::string s = fmt::format("The answer is {}.", 42);
|
||||||
|
// s == "The answer is 42."
|
||||||
|
```
|
||||||
|
|
||||||
|
**Format a string using positional arguments**
|
||||||
|
([run](https://godbolt.org/z/Yn7Txe))
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
std::string s = fmt::format("I'd rather be {1} than {0}.", "right", "happy");
|
||||||
|
// s == "I'd rather be happy than right."
|
||||||
|
```
|
||||||
|
|
||||||
|
**Print dates and times** ([run](https://godbolt.org/z/c31ExdY3W))
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
#include <fmt/chrono.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
auto now = std::chrono::system_clock::now();
|
||||||
|
fmt::print("Date and time: {}\n", now);
|
||||||
|
fmt::print("Time: {:%H:%M}\n", now);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Output:
|
||||||
|
|
||||||
|
Date and time: 2023-12-26 19:10:31.557195597
|
||||||
|
Time: 19:10
|
||||||
|
|
||||||
|
**Print a container** ([run](https://godbolt.org/z/MxM1YqjE7))
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
#include <vector>
|
||||||
|
#include <fmt/ranges.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
std::vector<int> v = {1, 2, 3};
|
||||||
|
fmt::print("{}\n", v);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Output:
|
||||||
|
|
||||||
|
[1, 2, 3]
|
||||||
|
|
||||||
|
**Check a format string at compile time**
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
std::string s = fmt::format("{:d}", "I am not a number");
|
||||||
|
```
|
||||||
|
|
||||||
|
This gives a compile-time error in C++20 because `d` is an invalid
|
||||||
|
format specifier for a string.
|
||||||
|
|
||||||
|
**Write a file from a single thread**
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
#include <fmt/os.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
auto out = fmt::output_file("guide.txt");
|
||||||
|
out.print("Don't {}", "Panic");
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This can be [5 to 9 times faster than
|
||||||
|
fprintf](http://www.zverovich.net/2020/08/04/optimal-file-buffer-size.html).
|
||||||
|
|
||||||
|
**Print with colors and text styles**
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
#include <fmt/color.h>
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
fmt::print(fg(fmt::color::crimson) | fmt::emphasis::bold,
|
||||||
|
"Hello, {}!\n", "world");
|
||||||
|
fmt::print(fg(fmt::color::floral_white) | bg(fmt::color::slate_gray) |
|
||||||
|
fmt::emphasis::underline, "Olá, {}!\n", "Mundo");
|
||||||
|
fmt::print(fg(fmt::color::steel_blue) | fmt::emphasis::italic,
|
||||||
|
"你好{}!\n", "世界");
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Output on a modern terminal with Unicode support:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
# Benchmarks
|
||||||
|
|
||||||
|
## Speed tests
|
||||||
|
|
||||||
|
| Library | Method | Run Time, s |
|
||||||
|
|-------------------|---------------|-------------|
|
||||||
|
| libc | printf | 0.91 |
|
||||||
|
| libc++ | std::ostream | 2.49 |
|
||||||
|
| {fmt} 9.1 | fmt::print | 0.74 |
|
||||||
|
| Boost Format 1.80 | boost::format | 6.26 |
|
||||||
|
| Folly Format | folly::format | 1.87 |
|
||||||
|
|
||||||
|
{fmt} is the fastest of the benchmarked methods, \~20% faster than
|
||||||
|
`printf`.
|
||||||
|
|
||||||
|
The above results were generated by building `tinyformat_test.cpp` on
|
||||||
|
macOS 12.6.1 with `clang++ -O3 -DNDEBUG -DSPEED_TEST -DHAVE_FORMAT`, and
|
||||||
|
taking the best of three runs. In the test, the format string
|
||||||
|
`"%0.10f:%04d:%+g:%s:%p:%c:%%\n"` or equivalent is filled 2,000,000
|
||||||
|
times with output sent to `/dev/null`; for further details refer to the
|
||||||
|
[source](https://github.com/fmtlib/format-benchmark/blob/master/src/tinyformat-test.cc).
|
||||||
|
|
||||||
|
{fmt} is up to 20-30x faster than `std::ostringstream` and `sprintf` on
|
||||||
|
IEEE754 `float` and `double` formatting
|
||||||
|
([dtoa-benchmark](https://github.com/fmtlib/dtoa-benchmark)) and faster
|
||||||
|
than [double-conversion](https://github.com/google/double-conversion)
|
||||||
|
and [ryu](https://github.com/ulfjack/ryu):
|
||||||
|
|
||||||
|
[](https://fmt.dev/unknown_mac64_clang12.0.html)
|
||||||
|
|
||||||
|
## Compile time and code bloat
|
||||||
|
|
||||||
|
The script [bloat-test.py][test] from [format-benchmark][bench] tests compile
|
||||||
|
time and code bloat for nontrivial projects. It generates 100 translation units
|
||||||
|
and uses `printf()` or its alternative five times in each to simulate a
|
||||||
|
medium-sized project. The resulting executable size and compile time (Apple
|
||||||
|
clang version 15.0.0 (clang-1500.1.0.2.5), macOS Sonoma, best of three) is shown
|
||||||
|
in the following tables.
|
||||||
|
|
||||||
|
[test]: https://github.com/fmtlib/format-benchmark/blob/master/bloat-test.py
|
||||||
|
[bench]: https://github.com/fmtlib/format-benchmark
|
||||||
|
|
||||||
|
**Optimized build (-O3)**
|
||||||
|
|
||||||
|
| Method | Compile Time, s | Executable size, KiB | Stripped size, KiB |
|
||||||
|
|---------------|-----------------|----------------------|--------------------|
|
||||||
|
| printf | 1.6 | 54 | 50 |
|
||||||
|
| IOStreams | 25.9 | 98 | 84 |
|
||||||
|
| fmt 83652df | 4.8 | 54 | 50 |
|
||||||
|
| tinyformat | 29.1 | 161 | 136 |
|
||||||
|
| Boost Format | 55.0 | 530 | 317 |
|
||||||
|
|
||||||
|
{fmt} is fast to compile and is comparable to `printf` in terms of per-call
|
||||||
|
binary size (within a rounding error on this system).
|
||||||
|
|
||||||
|
**Non-optimized build**
|
||||||
|
|
||||||
|
| Method | Compile Time, s | Executable size, KiB | Stripped size, KiB |
|
||||||
|
|---------------|-----------------|----------------------|--------------------|
|
||||||
|
| printf | 1.4 | 54 | 50 |
|
||||||
|
| IOStreams | 23.4 | 92 | 68 |
|
||||||
|
| {fmt} 83652df | 4.4 | 89 | 85 |
|
||||||
|
| tinyformat | 24.5 | 204 | 161 |
|
||||||
|
| Boost Format | 36.4 | 831 | 462 |
|
||||||
|
|
||||||
|
`libc`, `lib(std)c++`, and `libfmt` are all linked as shared libraries
|
||||||
|
to compare formatting function overhead only. Boost Format is a
|
||||||
|
header-only library so it doesn\'t provide any linkage options.
|
||||||
|
|
||||||
|
## Running the tests
|
||||||
|
|
||||||
|
Please refer to [Building the
|
||||||
|
library](https://fmt.dev/latest/get-started/#building-from-source) for
|
||||||
|
instructions on how to build the library and run the unit tests.
|
||||||
|
|
||||||
|
Benchmarks reside in a separate repository,
|
||||||
|
[format-benchmarks](https://github.com/fmtlib/format-benchmark), so to
|
||||||
|
run the benchmarks you first need to clone this repository and generate
|
||||||
|
Makefiles with CMake:
|
||||||
|
|
||||||
|
$ git clone --recursive https://github.com/fmtlib/format-benchmark.git
|
||||||
|
$ cd format-benchmark
|
||||||
|
$ cmake .
|
||||||
|
|
||||||
|
Then you can run the speed test:
|
||||||
|
|
||||||
|
$ make speed-test
|
||||||
|
|
||||||
|
or the bloat test:
|
||||||
|
|
||||||
|
$ make bloat-test
|
||||||
|
|
||||||
|
# Migrating code
|
||||||
|
|
||||||
|
[clang-tidy](https://clang.llvm.org/extra/clang-tidy/) v18 provides the
|
||||||
|
[modernize-use-std-print](https://clang.llvm.org/extra/clang-tidy/checks/modernize/use-std-print.html)
|
||||||
|
check that is capable of converting occurrences of `printf` and
|
||||||
|
`fprintf` to `fmt::print` if configured to do so. (By default it
|
||||||
|
converts to `std::print`.)
|
||||||
|
|
||||||
|
# Notable projects using this library
|
||||||
|
|
||||||
|
- [0 A.D.](https://play0ad.com/): a free, open-source, cross-platform
|
||||||
|
real-time strategy game
|
||||||
|
- [AMPL/MP](https://github.com/ampl/mp): an open-source library for
|
||||||
|
mathematical programming
|
||||||
|
- [Apple's FoundationDB](https://github.com/apple/foundationdb): an open-source,
|
||||||
|
distributed, transactional key-value store
|
||||||
|
- [Aseprite](https://github.com/aseprite/aseprite): animated sprite
|
||||||
|
editor & pixel art tool
|
||||||
|
- [AvioBook](https://www.aviobook.aero/en): a comprehensive aircraft
|
||||||
|
operations suite
|
||||||
|
- [Blizzard Battle.net](https://battle.net/): an online gaming
|
||||||
|
platform
|
||||||
|
- [Celestia](https://celestia.space/): real-time 3D visualization of
|
||||||
|
space
|
||||||
|
- [Ceph](https://ceph.com/): a scalable distributed storage system
|
||||||
|
- [ccache](https://ccache.dev/): a compiler cache
|
||||||
|
- [ClickHouse](https://github.com/ClickHouse/ClickHouse): an
|
||||||
|
analytical database management system
|
||||||
|
- [ContextVision](https://www.contextvision.com/): medical imaging software
|
||||||
|
- [Contour](https://github.com/contour-terminal/contour/): a modern
|
||||||
|
terminal emulator
|
||||||
|
- [CUAUV](https://cuauv.org/): Cornell University\'s autonomous
|
||||||
|
underwater vehicle
|
||||||
|
- [Drake](https://drake.mit.edu/): a planning, control, and analysis
|
||||||
|
toolbox for nonlinear dynamical systems (MIT)
|
||||||
|
- [Envoy](https://github.com/envoyproxy/envoy): C++ L7 proxy and
|
||||||
|
communication bus (Lyft)
|
||||||
|
- [FiveM](https://fivem.net/): a modification framework for GTA V
|
||||||
|
- [fmtlog](https://github.com/MengRao/fmtlog): a performant
|
||||||
|
fmtlib-style logging library with latency in nanoseconds
|
||||||
|
- [Folly](https://github.com/facebook/folly): Facebook open-source
|
||||||
|
library
|
||||||
|
- [GemRB](https://gemrb.org/): a portable open-source implementation
|
||||||
|
of Bioware's Infinity Engine
|
||||||
|
- [Grand Mountain
|
||||||
|
Adventure](https://store.steampowered.com/app/1247360/Grand_Mountain_Adventure/):
|
||||||
|
a beautiful open-world ski & snowboarding game
|
||||||
|
- [HarpyWar/pvpgn](https://github.com/pvpgn/pvpgn-server): Player vs
|
||||||
|
Player Gaming Network with tweaks
|
||||||
|
- [KBEngine](https://github.com/kbengine/kbengine): an open-source
|
||||||
|
MMOG server engine
|
||||||
|
- [Keypirinha](https://keypirinha.com/): a semantic launcher for
|
||||||
|
Windows
|
||||||
|
- [Kodi](https://kodi.tv/) (formerly xbmc): home theater software
|
||||||
|
- [Knuth](https://kth.cash/): high-performance Bitcoin full-node
|
||||||
|
- [libunicode](https://github.com/contour-terminal/libunicode/): a
|
||||||
|
modern C++17 Unicode library
|
||||||
|
- [MariaDB](https://mariadb.org/): relational database management
|
||||||
|
system
|
||||||
|
- [Microsoft Verona](https://github.com/microsoft/verona): research
|
||||||
|
programming language for concurrent ownership
|
||||||
|
- [MongoDB](https://mongodb.com/): distributed document database
|
||||||
|
- [MongoDB Smasher](https://github.com/duckie/mongo_smasher): a small
|
||||||
|
tool to generate randomized datasets
|
||||||
|
- [OpenSpace](https://openspaceproject.com/): an open-source
|
||||||
|
astrovisualization framework
|
||||||
|
- [PenUltima Online (POL)](https://www.polserver.com/): an MMO server,
|
||||||
|
compatible with most Ultima Online clients
|
||||||
|
- [PyTorch](https://github.com/pytorch/pytorch): an open-source
|
||||||
|
machine learning library
|
||||||
|
- [quasardb](https://www.quasardb.net/): a distributed,
|
||||||
|
high-performance, associative database
|
||||||
|
- [Quill](https://github.com/odygrd/quill): asynchronous low-latency
|
||||||
|
logging library
|
||||||
|
- [QKW](https://github.com/ravijanjam/qkw): generalizing aliasing to
|
||||||
|
simplify navigation, and execute complex multi-line terminal
|
||||||
|
command sequences
|
||||||
|
- [redis-cerberus](https://github.com/HunanTV/redis-cerberus): a Redis
|
||||||
|
cluster proxy
|
||||||
|
- [redpanda](https://vectorized.io/redpanda): a 10x faster Kafka®
|
||||||
|
replacement for mission-critical systems written in C++
|
||||||
|
- [rpclib](http://rpclib.net/): a modern C++ msgpack-RPC server and
|
||||||
|
client library
|
||||||
|
- [Salesforce Analytics
|
||||||
|
Cloud](https://www.salesforce.com/analytics-cloud/overview/):
|
||||||
|
business intelligence software
|
||||||
|
- [Scylla](https://www.scylladb.com/): a Cassandra-compatible NoSQL
|
||||||
|
data store that can handle 1 million transactions per second on a
|
||||||
|
single server
|
||||||
|
- [Seastar](http://www.seastar-project.org/): an advanced, open-source
|
||||||
|
C++ framework for high-performance server applications on modern
|
||||||
|
hardware
|
||||||
|
- [spdlog](https://github.com/gabime/spdlog): super fast C++ logging
|
||||||
|
library
|
||||||
|
- [Stellar](https://www.stellar.org/): financial platform
|
||||||
|
- [Touch Surgery](https://www.touchsurgery.com/): surgery simulator
|
||||||
|
- [TrinityCore](https://github.com/TrinityCore/TrinityCore):
|
||||||
|
open-source MMORPG framework
|
||||||
|
- [🐙 userver framework](https://userver.tech/): open-source
|
||||||
|
asynchronous framework with a rich set of abstractions and database
|
||||||
|
drivers
|
||||||
|
- [Windows Terminal](https://github.com/microsoft/terminal): the new
|
||||||
|
Windows terminal
|
||||||
|
|
||||||
|
[More\...](https://github.com/search?q=fmtlib&type=Code)
|
||||||
|
|
||||||
|
If you are aware of other projects using this library, please let me
|
||||||
|
know by [email](mailto:victor.zverovich@gmail.com) or by submitting an
|
||||||
|
[issue](https://github.com/fmtlib/fmt/issues).
|
||||||
|
|
||||||
|
# Motivation
|
||||||
|
|
||||||
|
So why yet another formatting library?
|
||||||
|
|
||||||
|
There are plenty of methods for doing this task, from standard ones like
|
||||||
|
the printf family of function and iostreams to Boost Format and
|
||||||
|
FastFormat libraries. The reason for creating a new library is that
|
||||||
|
every existing solution that I found either had serious issues or
|
||||||
|
didn\'t provide all the features I needed.
|
||||||
|
|
||||||
|
## printf
|
||||||
|
|
||||||
|
The good thing about `printf` is that it is pretty fast and readily
|
||||||
|
available being a part of the C standard library. The main drawback is
|
||||||
|
that it doesn\'t support user-defined types. `printf` also has safety
|
||||||
|
issues although they are somewhat mitigated with [\_\_attribute\_\_
|
||||||
|
((format (printf,
|
||||||
|
\...))](https://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html) in
|
||||||
|
GCC. There is a POSIX extension that adds positional arguments required
|
||||||
|
for
|
||||||
|
[i18n](https://en.wikipedia.org/wiki/Internationalization_and_localization)
|
||||||
|
to `printf` but it is not a part of C99 and may not be available on some
|
||||||
|
platforms.
|
||||||
|
|
||||||
|
## iostreams
|
||||||
|
|
||||||
|
The main issue with iostreams is best illustrated with an example:
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
std::cout << std::setprecision(2) << std::fixed << 1.23456 << "\n";
|
||||||
|
```
|
||||||
|
|
||||||
|
which is a lot of typing compared to printf:
|
||||||
|
|
||||||
|
``` c++
|
||||||
|
printf("%.2f\n", 1.23456);
|
||||||
|
```
|
||||||
|
|
||||||
|
Matthew Wilson, the author of FastFormat, called this \"chevron hell\".
|
||||||
|
iostreams don\'t support positional arguments by design.
|
||||||
|
|
||||||
|
The good part is that iostreams support user-defined types and are safe
|
||||||
|
although error handling is awkward.
|
||||||
|
|
||||||
|
## Boost Format
|
||||||
|
|
||||||
|
This is a very powerful library that supports both `printf`-like format
|
||||||
|
strings and positional arguments. Its main drawback is performance.
|
||||||
|
According to various benchmarks, it is much slower than other methods
|
||||||
|
considered here. Boost Format also has excessive build times and severe
|
||||||
|
code bloat issues (see [Benchmarks](#benchmarks)).
|
||||||
|
|
||||||
|
## FastFormat
|
||||||
|
|
||||||
|
This is an interesting library that is fast, safe and has positional
|
||||||
|
arguments. However, it has significant limitations, citing its author:
|
||||||
|
|
||||||
|
> Three features that have no hope of being accommodated within the
|
||||||
|
> current design are:
|
||||||
|
>
|
||||||
|
> - Leading zeros (or any other non-space padding)
|
||||||
|
> - Octal/hexadecimal encoding
|
||||||
|
> - Runtime width/alignment specification
|
||||||
|
|
||||||
|
It is also quite big and has a heavy dependency, on STLSoft, which might be
|
||||||
|
too restrictive for use in some projects.
|
||||||
|
|
||||||
|
## Boost Spirit.Karma
|
||||||
|
|
||||||
|
This is not a formatting library but I decided to include it here for
|
||||||
|
completeness. As iostreams, it suffers from the problem of mixing
|
||||||
|
verbatim text with arguments. The library is pretty fast, but slower on
|
||||||
|
integer formatting than `fmt::format_to` with format string compilation
|
||||||
|
on Karma\'s own benchmark, see [Converting a hundred million integers to
|
||||||
|
strings per
|
||||||
|
second](http://www.zverovich.net/2020/06/13/fast-int-to-string-revisited.html).
|
||||||
|
|
||||||
|
# License
|
||||||
|
|
||||||
|
{fmt} is distributed under the MIT
|
||||||
|
[license](https://github.com/fmtlib/fmt/blob/master/LICENSE).
|
||||||
|
|
||||||
|
# Documentation License
|
||||||
|
|
||||||
|
The [Format String Syntax](https://fmt.dev/latest/syntax/) section
|
||||||
|
in the documentation is based on the one from Python [string module
|
||||||
|
documentation](https://docs.python.org/3/library/string.html#module-string).
|
||||||
|
For this reason, the documentation is distributed under the Python
|
||||||
|
Software Foundation license available in
|
||||||
|
[doc/python-license.txt](https://raw.github.com/fmtlib/fmt/master/doc/python-license.txt).
|
||||||
|
It only applies if you distribute the documentation of {fmt}.
|
||||||
|
|
||||||
|
# Maintainers
|
||||||
|
|
||||||
|
The {fmt} library is maintained by Victor Zverovich
|
||||||
|
([vitaut](https://github.com/vitaut)) with contributions from many other
|
||||||
|
people. See
|
||||||
|
[Contributors](https://github.com/fmtlib/fmt/graphs/contributors) and
|
||||||
|
[Releases](https://github.com/fmtlib/fmt/releases) for some of the
|
||||||
|
names. Let us know if your contribution is not listed or mentioned
|
||||||
|
incorrectly and we\'ll make it right.
|
||||||
|
|
||||||
|
# Security Policy
|
||||||
|
|
||||||
|
To report a security issue, please disclose it at [security
|
||||||
|
advisory](https://github.com/fmtlib/fmt/security/advisories/new).
|
||||||
|
|
||||||
|
This project is maintained by a team of volunteers on a
|
||||||
|
reasonable-effort basis. As such, please give us at least *90* days to
|
||||||
|
work on a fix before public exposure.
|
||||||
545
deps/fmt/README.rst
vendored
545
deps/fmt/README.rst
vendored
|
|
@ -1,545 +0,0 @@
|
||||||
.. image:: https://user-images.githubusercontent.com/
|
|
||||||
576385/156254208-f5b743a9-88cf-439d-b0c0-923d53e8d551.png
|
|
||||||
:width: 25%
|
|
||||||
:alt: {fmt}
|
|
||||||
|
|
||||||
.. image:: https://github.com/fmtlib/fmt/workflows/linux/badge.svg
|
|
||||||
:target: https://github.com/fmtlib/fmt/actions?query=workflow%3Alinux
|
|
||||||
|
|
||||||
.. image:: https://github.com/fmtlib/fmt/workflows/macos/badge.svg
|
|
||||||
:target: https://github.com/fmtlib/fmt/actions?query=workflow%3Amacos
|
|
||||||
|
|
||||||
.. image:: https://github.com/fmtlib/fmt/workflows/windows/badge.svg
|
|
||||||
:target: https://github.com/fmtlib/fmt/actions?query=workflow%3Awindows
|
|
||||||
|
|
||||||
.. image:: https://oss-fuzz-build-logs.storage.googleapis.com/badges/fmt.svg
|
|
||||||
:alt: fmt is continuously fuzzed at oss-fuzz
|
|
||||||
:target: https://bugs.chromium.org/p/oss-fuzz/issues/list?\
|
|
||||||
colspec=ID%20Type%20Component%20Status%20Proj%20Reported%20Owner%20\
|
|
||||||
Summary&q=proj%3Dfmt&can=1
|
|
||||||
|
|
||||||
.. image:: https://img.shields.io/badge/stackoverflow-fmt-blue.svg
|
|
||||||
:alt: Ask questions at StackOverflow with the tag fmt
|
|
||||||
:target: https://stackoverflow.com/questions/tagged/fmt
|
|
||||||
|
|
||||||
.. image:: https://api.securityscorecards.dev/projects/github.com/fmtlib/fmt/badge
|
|
||||||
:target: https://securityscorecards.dev/viewer/?uri=github.com/fmtlib/fmt
|
|
||||||
|
|
||||||
**{fmt}** is an open-source formatting library providing a fast and safe
|
|
||||||
alternative to C stdio and C++ iostreams.
|
|
||||||
|
|
||||||
If you like this project, please consider donating to one of the funds that
|
|
||||||
help victims of the war in Ukraine: https://www.stopputin.net/.
|
|
||||||
|
|
||||||
`Documentation <https://fmt.dev>`__
|
|
||||||
|
|
||||||
`Cheat Sheets <https://hackingcpp.com/cpp/libs/fmt.html>`__
|
|
||||||
|
|
||||||
Q&A: ask questions on `StackOverflow with the tag fmt
|
|
||||||
<https://stackoverflow.com/questions/tagged/fmt>`_.
|
|
||||||
|
|
||||||
Try {fmt} in `Compiler Explorer <https://godbolt.org/z/Eq5763>`_.
|
|
||||||
|
|
||||||
Features
|
|
||||||
--------
|
|
||||||
|
|
||||||
* Simple `format API <https://fmt.dev/latest/api.html>`_ with positional arguments
|
|
||||||
for localization
|
|
||||||
* Implementation of `C++20 std::format
|
|
||||||
<https://en.cppreference.com/w/cpp/utility/format>`__
|
|
||||||
* `Format string syntax <https://fmt.dev/latest/syntax.html>`_ similar to Python's
|
|
||||||
`format <https://docs.python.org/3/library/stdtypes.html#str.format>`_
|
|
||||||
* Fast IEEE 754 floating-point formatter with correct rounding, shortness and
|
|
||||||
round-trip guarantees using the `Dragonbox <https://github.com/jk-jeon/dragonbox>`_
|
|
||||||
algorithm
|
|
||||||
* Portable Unicode support
|
|
||||||
* Safe `printf implementation
|
|
||||||
<https://fmt.dev/latest/api.html#printf-formatting>`_ including the POSIX
|
|
||||||
extension for positional arguments
|
|
||||||
* Extensibility: `support for user-defined types
|
|
||||||
<https://fmt.dev/latest/api.html#formatting-user-defined-types>`_
|
|
||||||
* High performance: faster than common standard library implementations of
|
|
||||||
``(s)printf``, iostreams, ``to_string`` and ``to_chars``, see `Speed tests`_
|
|
||||||
and `Converting a hundred million integers to strings per second
|
|
||||||
<http://www.zverovich.net/2020/06/13/fast-int-to-string-revisited.html>`_
|
|
||||||
* Small code size both in terms of source code with the minimum configuration
|
|
||||||
consisting of just three files, ``core.h``, ``format.h`` and ``format-inl.h``,
|
|
||||||
and compiled code; see `Compile time and code bloat`_
|
|
||||||
* Reliability: the library has an extensive set of `tests
|
|
||||||
<https://github.com/fmtlib/fmt/tree/master/test>`_ and is `continuously fuzzed
|
|
||||||
<https://bugs.chromium.org/p/oss-fuzz/issues/list?colspec=ID%20Type%20
|
|
||||||
Component%20Status%20Proj%20Reported%20Owner%20Summary&q=proj%3Dfmt&can=1>`_
|
|
||||||
* Safety: the library is fully type-safe, errors in format strings can be
|
|
||||||
reported at compile time, automatic memory management prevents buffer overflow
|
|
||||||
errors
|
|
||||||
* Ease of use: small self-contained code base, no external dependencies,
|
|
||||||
permissive MIT `license
|
|
||||||
<https://github.com/fmtlib/fmt/blob/master/LICENSE.rst>`_
|
|
||||||
* `Portability <https://fmt.dev/latest/index.html#portability>`_ with
|
|
||||||
consistent output across platforms and support for older compilers
|
|
||||||
* Clean warning-free codebase even on high warning levels such as
|
|
||||||
``-Wall -Wextra -pedantic``
|
|
||||||
* Locale independence by default
|
|
||||||
* Optional header-only configuration enabled with the ``FMT_HEADER_ONLY`` macro
|
|
||||||
|
|
||||||
See the `documentation <https://fmt.dev>`_ for more details.
|
|
||||||
|
|
||||||
Examples
|
|
||||||
--------
|
|
||||||
|
|
||||||
**Print to stdout** (`run <https://godbolt.org/z/Tevcjh>`_)
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
#include <fmt/core.h>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
fmt::print("Hello, world!\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
**Format a string** (`run <https://godbolt.org/z/oK8h33>`_)
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
std::string s = fmt::format("The answer is {}.", 42);
|
|
||||||
// s == "The answer is 42."
|
|
||||||
|
|
||||||
**Format a string using positional arguments** (`run <https://godbolt.org/z/Yn7Txe>`_)
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
std::string s = fmt::format("I'd rather be {1} than {0}.", "right", "happy");
|
|
||||||
// s == "I'd rather be happy than right."
|
|
||||||
|
|
||||||
**Print chrono durations** (`run <https://godbolt.org/z/K8s4Mc>`_)
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
#include <fmt/chrono.h>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
using namespace std::literals::chrono_literals;
|
|
||||||
fmt::print("Default format: {} {}\n", 42s, 100ms);
|
|
||||||
fmt::print("strftime-like format: {:%H:%M:%S}\n", 3h + 15min + 30s);
|
|
||||||
}
|
|
||||||
|
|
||||||
Output::
|
|
||||||
|
|
||||||
Default format: 42s 100ms
|
|
||||||
strftime-like format: 03:15:30
|
|
||||||
|
|
||||||
**Print a container** (`run <https://godbolt.org/z/MxM1YqjE7>`_)
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
#include <vector>
|
|
||||||
#include <fmt/ranges.h>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
std::vector<int> v = {1, 2, 3};
|
|
||||||
fmt::print("{}\n", v);
|
|
||||||
}
|
|
||||||
|
|
||||||
Output::
|
|
||||||
|
|
||||||
[1, 2, 3]
|
|
||||||
|
|
||||||
**Check a format string at compile time**
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
std::string s = fmt::format("{:d}", "I am not a number");
|
|
||||||
|
|
||||||
This gives a compile-time error in C++20 because ``d`` is an invalid format
|
|
||||||
specifier for a string.
|
|
||||||
|
|
||||||
**Write a file from a single thread**
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
#include <fmt/os.h>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
auto out = fmt::output_file("guide.txt");
|
|
||||||
out.print("Don't {}", "Panic");
|
|
||||||
}
|
|
||||||
|
|
||||||
This can be `5 to 9 times faster than fprintf
|
|
||||||
<http://www.zverovich.net/2020/08/04/optimal-file-buffer-size.html>`_.
|
|
||||||
|
|
||||||
**Print with colors and text styles**
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
#include <fmt/color.h>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
fmt::print(fg(fmt::color::crimson) | fmt::emphasis::bold,
|
|
||||||
"Hello, {}!\n", "world");
|
|
||||||
fmt::print(fg(fmt::color::floral_white) | bg(fmt::color::slate_gray) |
|
|
||||||
fmt::emphasis::underline, "Hello, {}!\n", "мир");
|
|
||||||
fmt::print(fg(fmt::color::steel_blue) | fmt::emphasis::italic,
|
|
||||||
"Hello, {}!\n", "世界");
|
|
||||||
}
|
|
||||||
|
|
||||||
Output on a modern terminal:
|
|
||||||
|
|
||||||
.. image:: https://user-images.githubusercontent.com/
|
|
||||||
576385/88485597-d312f600-cf2b-11ea-9cbe-61f535a86e28.png
|
|
||||||
|
|
||||||
Benchmarks
|
|
||||||
----------
|
|
||||||
|
|
||||||
Speed tests
|
|
||||||
~~~~~~~~~~~
|
|
||||||
|
|
||||||
================= ============= ===========
|
|
||||||
Library Method Run Time, s
|
|
||||||
================= ============= ===========
|
|
||||||
libc printf 0.91
|
|
||||||
libc++ std::ostream 2.49
|
|
||||||
{fmt} 9.1 fmt::print 0.74
|
|
||||||
Boost Format 1.80 boost::format 6.26
|
|
||||||
Folly Format folly::format 1.87
|
|
||||||
================= ============= ===========
|
|
||||||
|
|
||||||
{fmt} is the fastest of the benchmarked methods, ~20% faster than ``printf``.
|
|
||||||
|
|
||||||
The above results were generated by building ``tinyformat_test.cpp`` on macOS
|
|
||||||
12.6.1 with ``clang++ -O3 -DNDEBUG -DSPEED_TEST -DHAVE_FORMAT``, and taking the
|
|
||||||
best of three runs. In the test, the format string ``"%0.10f:%04d:%+g:%s:%p:%c:%%\n"``
|
|
||||||
or equivalent is filled 2,000,000 times with output sent to ``/dev/null``; for
|
|
||||||
further details refer to the `source
|
|
||||||
<https://github.com/fmtlib/format-benchmark/blob/master/src/tinyformat-test.cc>`_.
|
|
||||||
|
|
||||||
{fmt} is up to 20-30x faster than ``std::ostringstream`` and ``sprintf`` on
|
|
||||||
IEEE754 ``float`` and ``double`` formatting (`dtoa-benchmark <https://github.com/fmtlib/dtoa-benchmark>`_)
|
|
||||||
and faster than `double-conversion <https://github.com/google/double-conversion>`_ and
|
|
||||||
`ryu <https://github.com/ulfjack/ryu>`_:
|
|
||||||
|
|
||||||
.. image:: https://user-images.githubusercontent.com/576385/
|
|
||||||
95684665-11719600-0ba8-11eb-8e5b-972ff4e49428.png
|
|
||||||
:target: https://fmt.dev/unknown_mac64_clang12.0.html
|
|
||||||
|
|
||||||
Compile time and code bloat
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
||||||
|
|
||||||
The script `bloat-test.py
|
|
||||||
<https://github.com/fmtlib/format-benchmark/blob/master/bloat-test.py>`_
|
|
||||||
from `format-benchmark <https://github.com/fmtlib/format-benchmark>`_
|
|
||||||
tests compile time and code bloat for nontrivial projects.
|
|
||||||
It generates 100 translation units and uses ``printf()`` or its alternative
|
|
||||||
five times in each to simulate a medium-sized project. The resulting
|
|
||||||
executable size and compile time (Apple LLVM version 8.1.0 (clang-802.0.42),
|
|
||||||
macOS Sierra, best of three) is shown in the following tables.
|
|
||||||
|
|
||||||
**Optimized build (-O3)**
|
|
||||||
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
Method Compile Time, s Executable size, KiB Stripped size, KiB
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
printf 2.6 29 26
|
|
||||||
printf+string 16.4 29 26
|
|
||||||
iostreams 31.1 59 55
|
|
||||||
{fmt} 19.0 37 34
|
|
||||||
Boost Format 91.9 226 203
|
|
||||||
Folly Format 115.7 101 88
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
|
|
||||||
As you can see, {fmt} has 60% less overhead in terms of resulting binary code
|
|
||||||
size compared to iostreams and comes pretty close to ``printf``. Boost Format
|
|
||||||
and Folly Format have the largest overheads.
|
|
||||||
|
|
||||||
``printf+string`` is the same as ``printf`` but with an extra ``<string>``
|
|
||||||
include to measure the overhead of the latter.
|
|
||||||
|
|
||||||
**Non-optimized build**
|
|
||||||
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
Method Compile Time, s Executable size, KiB Stripped size, KiB
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
printf 2.2 33 30
|
|
||||||
printf+string 16.0 33 30
|
|
||||||
iostreams 28.3 56 52
|
|
||||||
{fmt} 18.2 59 50
|
|
||||||
Boost Format 54.1 365 303
|
|
||||||
Folly Format 79.9 445 430
|
|
||||||
============= =============== ==================== ==================
|
|
||||||
|
|
||||||
``libc``, ``lib(std)c++``, and ``libfmt`` are all linked as shared libraries to
|
|
||||||
compare formatting function overhead only. Boost Format is a
|
|
||||||
header-only library so it doesn't provide any linkage options.
|
|
||||||
|
|
||||||
Running the tests
|
|
||||||
~~~~~~~~~~~~~~~~~
|
|
||||||
|
|
||||||
Please refer to `Building the library`__ for instructions on how to build
|
|
||||||
the library and run the unit tests.
|
|
||||||
|
|
||||||
__ https://fmt.dev/latest/usage.html#building-the-library
|
|
||||||
|
|
||||||
Benchmarks reside in a separate repository,
|
|
||||||
`format-benchmarks <https://github.com/fmtlib/format-benchmark>`_,
|
|
||||||
so to run the benchmarks you first need to clone this repository and
|
|
||||||
generate Makefiles with CMake::
|
|
||||||
|
|
||||||
$ git clone --recursive https://github.com/fmtlib/format-benchmark.git
|
|
||||||
$ cd format-benchmark
|
|
||||||
$ cmake .
|
|
||||||
|
|
||||||
Then you can run the speed test::
|
|
||||||
|
|
||||||
$ make speed-test
|
|
||||||
|
|
||||||
or the bloat test::
|
|
||||||
|
|
||||||
$ make bloat-test
|
|
||||||
|
|
||||||
Migrating code
|
|
||||||
--------------
|
|
||||||
|
|
||||||
`clang-tidy <https://clang.llvm.org/extra/clang-tidy/>`_ v17 (not yet
|
|
||||||
released) provides the `modernize-use-std-print
|
|
||||||
<https://clang.llvm.org/extra/clang-tidy/checks/modernize/use-std-print.html>`_
|
|
||||||
check that is capable of converting occurrences of ``printf`` and
|
|
||||||
``fprintf`` to ``fmt::print`` if configured to do so. (By default it
|
|
||||||
converts to ``std::print``.)
|
|
||||||
|
|
||||||
Projects using this library
|
|
||||||
---------------------------
|
|
||||||
|
|
||||||
* `0 A.D. <https://play0ad.com/>`_: a free, open-source, cross-platform
|
|
||||||
real-time strategy game
|
|
||||||
|
|
||||||
* `AMPL/MP <https://github.com/ampl/mp>`_:
|
|
||||||
an open-source library for mathematical programming
|
|
||||||
|
|
||||||
* `Aseprite <https://github.com/aseprite/aseprite>`_:
|
|
||||||
animated sprite editor & pixel art tool
|
|
||||||
|
|
||||||
* `AvioBook <https://www.aviobook.aero/en>`_: a comprehensive aircraft
|
|
||||||
operations suite
|
|
||||||
|
|
||||||
* `Blizzard Battle.net <https://battle.net/>`_: an online gaming platform
|
|
||||||
|
|
||||||
* `Celestia <https://celestia.space/>`_: real-time 3D visualization of space
|
|
||||||
|
|
||||||
* `Ceph <https://ceph.com/>`_: a scalable distributed storage system
|
|
||||||
|
|
||||||
* `ccache <https://ccache.dev/>`_: a compiler cache
|
|
||||||
|
|
||||||
* `ClickHouse <https://github.com/ClickHouse/ClickHouse>`_: an analytical database
|
|
||||||
management system
|
|
||||||
|
|
||||||
* `Contour <https://github.com/contour-terminal/contour/>`_: a modern terminal emulator
|
|
||||||
|
|
||||||
* `CUAUV <https://cuauv.org/>`_: Cornell University's autonomous underwater
|
|
||||||
vehicle
|
|
||||||
|
|
||||||
* `Drake <https://drake.mit.edu/>`_: a planning, control, and analysis toolbox
|
|
||||||
for nonlinear dynamical systems (MIT)
|
|
||||||
|
|
||||||
* `Envoy <https://lyft.github.io/envoy/>`_: C++ L7 proxy and communication bus
|
|
||||||
(Lyft)
|
|
||||||
|
|
||||||
* `FiveM <https://fivem.net/>`_: a modification framework for GTA V
|
|
||||||
|
|
||||||
* `fmtlog <https://github.com/MengRao/fmtlog>`_: a performant fmtlib-style
|
|
||||||
logging library with latency in nanoseconds
|
|
||||||
|
|
||||||
* `Folly <https://github.com/facebook/folly>`_: Facebook open-source library
|
|
||||||
|
|
||||||
* `GemRB <https://gemrb.org/>`_: a portable open-source implementation of
|
|
||||||
Bioware’s Infinity Engine
|
|
||||||
|
|
||||||
* `Grand Mountain Adventure
|
|
||||||
<https://store.steampowered.com/app/1247360/Grand_Mountain_Adventure/>`_:
|
|
||||||
a beautiful open-world ski & snowboarding game
|
|
||||||
|
|
||||||
* `HarpyWar/pvpgn <https://github.com/pvpgn/pvpgn-server>`_:
|
|
||||||
Player vs Player Gaming Network with tweaks
|
|
||||||
|
|
||||||
* `KBEngine <https://github.com/kbengine/kbengine>`_: an open-source MMOG server
|
|
||||||
engine
|
|
||||||
|
|
||||||
* `Keypirinha <https://keypirinha.com/>`_: a semantic launcher for Windows
|
|
||||||
|
|
||||||
* `Kodi <https://kodi.tv/>`_ (formerly xbmc): home theater software
|
|
||||||
|
|
||||||
* `Knuth <https://kth.cash/>`_: high-performance Bitcoin full-node
|
|
||||||
|
|
||||||
* `libunicode <https://github.com/contour-terminal/libunicode/>`_: a modern C++17 Unicode library
|
|
||||||
|
|
||||||
* `MariaDB <https://mariadb.org/>`_: relational database management system
|
|
||||||
|
|
||||||
* `Microsoft Verona <https://github.com/microsoft/verona>`_:
|
|
||||||
research programming language for concurrent ownership
|
|
||||||
|
|
||||||
* `MongoDB <https://mongodb.com/>`_: distributed document database
|
|
||||||
|
|
||||||
* `MongoDB Smasher <https://github.com/duckie/mongo_smasher>`_: a small tool to
|
|
||||||
generate randomized datasets
|
|
||||||
|
|
||||||
* `OpenSpace <https://openspaceproject.com/>`_: an open-source
|
|
||||||
astrovisualization framework
|
|
||||||
|
|
||||||
* `PenUltima Online (POL) <https://www.polserver.com/>`_:
|
|
||||||
an MMO server, compatible with most Ultima Online clients
|
|
||||||
|
|
||||||
* `PyTorch <https://github.com/pytorch/pytorch>`_: an open-source machine
|
|
||||||
learning library
|
|
||||||
|
|
||||||
* `quasardb <https://www.quasardb.net/>`_: a distributed, high-performance,
|
|
||||||
associative database
|
|
||||||
|
|
||||||
* `Quill <https://github.com/odygrd/quill>`_: asynchronous low-latency logging library
|
|
||||||
|
|
||||||
* `QKW <https://github.com/ravijanjam/qkw>`_: generalizing aliasing to simplify
|
|
||||||
navigation, and executing complex multi-line terminal command sequences
|
|
||||||
|
|
||||||
* `redis-cerberus <https://github.com/HunanTV/redis-cerberus>`_: a Redis cluster
|
|
||||||
proxy
|
|
||||||
|
|
||||||
* `redpanda <https://vectorized.io/redpanda>`_: a 10x faster Kafka® replacement
|
|
||||||
for mission-critical systems written in C++
|
|
||||||
|
|
||||||
* `rpclib <http://rpclib.net/>`_: a modern C++ msgpack-RPC server and client
|
|
||||||
library
|
|
||||||
|
|
||||||
* `Salesforce Analytics Cloud
|
|
||||||
<https://www.salesforce.com/analytics-cloud/overview/>`_:
|
|
||||||
business intelligence software
|
|
||||||
|
|
||||||
* `Scylla <https://www.scylladb.com/>`_: a Cassandra-compatible NoSQL data store
|
|
||||||
that can handle 1 million transactions per second on a single server
|
|
||||||
|
|
||||||
* `Seastar <http://www.seastar-project.org/>`_: an advanced, open-source C++
|
|
||||||
framework for high-performance server applications on modern hardware
|
|
||||||
|
|
||||||
* `spdlog <https://github.com/gabime/spdlog>`_: super fast C++ logging library
|
|
||||||
|
|
||||||
* `Stellar <https://www.stellar.org/>`_: financial platform
|
|
||||||
|
|
||||||
* `Touch Surgery <https://www.touchsurgery.com/>`_: surgery simulator
|
|
||||||
|
|
||||||
* `TrinityCore <https://github.com/TrinityCore/TrinityCore>`_: open-source
|
|
||||||
MMORPG framework
|
|
||||||
|
|
||||||
* `🐙 userver framework <https://userver.tech/>`_: open-source asynchronous
|
|
||||||
framework with a rich set of abstractions and database drivers
|
|
||||||
|
|
||||||
* `Windows Terminal <https://github.com/microsoft/terminal>`_: the new Windows
|
|
||||||
terminal
|
|
||||||
|
|
||||||
`More... <https://github.com/search?q=fmtlib&type=Code>`_
|
|
||||||
|
|
||||||
If you are aware of other projects using this library, please let me know
|
|
||||||
by `email <mailto:victor.zverovich@gmail.com>`_ or by submitting an
|
|
||||||
`issue <https://github.com/fmtlib/fmt/issues>`_.
|
|
||||||
|
|
||||||
Motivation
|
|
||||||
----------
|
|
||||||
|
|
||||||
So why yet another formatting library?
|
|
||||||
|
|
||||||
There are plenty of methods for doing this task, from standard ones like
|
|
||||||
the printf family of function and iostreams to Boost Format and FastFormat
|
|
||||||
libraries. The reason for creating a new library is that every existing
|
|
||||||
solution that I found either had serious issues or didn't provide
|
|
||||||
all the features I needed.
|
|
||||||
|
|
||||||
printf
|
|
||||||
~~~~~~
|
|
||||||
|
|
||||||
The good thing about ``printf`` is that it is pretty fast and readily available
|
|
||||||
being a part of the C standard library. The main drawback is that it
|
|
||||||
doesn't support user-defined types. ``printf`` also has safety issues although
|
|
||||||
they are somewhat mitigated with `__attribute__ ((format (printf, ...))
|
|
||||||
<https://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html>`_ in GCC.
|
|
||||||
There is a POSIX extension that adds positional arguments required for
|
|
||||||
`i18n <https://en.wikipedia.org/wiki/Internationalization_and_localization>`_
|
|
||||||
to ``printf`` but it is not a part of C99 and may not be available on some
|
|
||||||
platforms.
|
|
||||||
|
|
||||||
iostreams
|
|
||||||
~~~~~~~~~
|
|
||||||
|
|
||||||
The main issue with iostreams is best illustrated with an example:
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
std::cout << std::setprecision(2) << std::fixed << 1.23456 << "\n";
|
|
||||||
|
|
||||||
which is a lot of typing compared to printf:
|
|
||||||
|
|
||||||
.. code:: c++
|
|
||||||
|
|
||||||
printf("%.2f\n", 1.23456);
|
|
||||||
|
|
||||||
Matthew Wilson, the author of FastFormat, called this "chevron hell". iostreams
|
|
||||||
don't support positional arguments by design.
|
|
||||||
|
|
||||||
The good part is that iostreams support user-defined types and are safe although
|
|
||||||
error handling is awkward.
|
|
||||||
|
|
||||||
Boost Format
|
|
||||||
~~~~~~~~~~~~
|
|
||||||
|
|
||||||
This is a very powerful library that supports both ``printf``-like format
|
|
||||||
strings and positional arguments. Its main drawback is performance. According to
|
|
||||||
various benchmarks, it is much slower than other methods considered here. Boost
|
|
||||||
Format also has excessive build times and severe code bloat issues (see
|
|
||||||
`Benchmarks`_).
|
|
||||||
|
|
||||||
FastFormat
|
|
||||||
~~~~~~~~~~
|
|
||||||
|
|
||||||
This is an interesting library that is fast, safe, and has positional arguments.
|
|
||||||
However, it has significant limitations, citing its author:
|
|
||||||
|
|
||||||
Three features that have no hope of being accommodated within the
|
|
||||||
current design are:
|
|
||||||
|
|
||||||
* Leading zeros (or any other non-space padding)
|
|
||||||
* Octal/hexadecimal encoding
|
|
||||||
* Runtime width/alignment specification
|
|
||||||
|
|
||||||
It is also quite big and has a heavy dependency, STLSoft, which might be too
|
|
||||||
restrictive for using it in some projects.
|
|
||||||
|
|
||||||
Boost Spirit.Karma
|
|
||||||
~~~~~~~~~~~~~~~~~~
|
|
||||||
|
|
||||||
This is not a formatting library but I decided to include it here for
|
|
||||||
completeness. As iostreams, it suffers from the problem of mixing verbatim text
|
|
||||||
with arguments. The library is pretty fast, but slower on integer formatting
|
|
||||||
than ``fmt::format_to`` with format string compilation on Karma's own benchmark,
|
|
||||||
see `Converting a hundred million integers to strings per second
|
|
||||||
<http://www.zverovich.net/2020/06/13/fast-int-to-string-revisited.html>`_.
|
|
||||||
|
|
||||||
License
|
|
||||||
-------
|
|
||||||
|
|
||||||
{fmt} is distributed under the MIT `license
|
|
||||||
<https://github.com/fmtlib/fmt/blob/master/LICENSE.rst>`_.
|
|
||||||
|
|
||||||
Documentation License
|
|
||||||
---------------------
|
|
||||||
|
|
||||||
The `Format String Syntax <https://fmt.dev/latest/syntax.html>`_
|
|
||||||
section in the documentation is based on the one from Python `string module
|
|
||||||
documentation <https://docs.python.org/3/library/string.html#module-string>`_.
|
|
||||||
For this reason, the documentation is distributed under the Python Software
|
|
||||||
Foundation license available in `doc/python-license.txt
|
|
||||||
<https://raw.github.com/fmtlib/fmt/master/doc/python-license.txt>`_.
|
|
||||||
It only applies if you distribute the documentation of {fmt}.
|
|
||||||
|
|
||||||
Maintainers
|
|
||||||
-----------
|
|
||||||
|
|
||||||
The {fmt} library is maintained by Victor Zverovich (`vitaut
|
|
||||||
<https://github.com/vitaut>`_) with contributions from many other people.
|
|
||||||
See `Contributors <https://github.com/fmtlib/fmt/graphs/contributors>`_ and
|
|
||||||
`Releases <https://github.com/fmtlib/fmt/releases>`_ for some of the names.
|
|
||||||
Let us know if your contribution is not listed or mentioned incorrectly and
|
|
||||||
we'll make it right.
|
|
||||||
172
deps/fmt/include/fmt/args.h
vendored
172
deps/fmt/include/fmt/args.h
vendored
|
|
@ -8,34 +8,39 @@
|
||||||
#ifndef FMT_ARGS_H_
|
#ifndef FMT_ARGS_H_
|
||||||
#define FMT_ARGS_H_
|
#define FMT_ARGS_H_
|
||||||
|
|
||||||
#include <functional> // std::reference_wrapper
|
#ifndef FMT_MODULE
|
||||||
#include <memory> // std::unique_ptr
|
# include <functional> // std::reference_wrapper
|
||||||
#include <vector>
|
# include <memory> // std::unique_ptr
|
||||||
|
# include <vector>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "core.h"
|
#include "format.h" // std_string_view
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template <typename T> struct is_reference_wrapper : std::false_type {};
|
template <typename T> struct is_reference_wrapper : std::false_type {};
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct is_reference_wrapper<std::reference_wrapper<T>> : std::true_type {};
|
struct is_reference_wrapper<std::reference_wrapper<T>> : std::true_type {};
|
||||||
|
|
||||||
template <typename T> const T& unwrap(const T& v) { return v; }
|
template <typename T> auto unwrap(const T& v) -> const T& { return v; }
|
||||||
template <typename T> const T& unwrap(const std::reference_wrapper<T>& v) {
|
template <typename T>
|
||||||
|
auto unwrap(const std::reference_wrapper<T>& v) -> const T& {
|
||||||
return static_cast<const T&>(v);
|
return static_cast<const T&>(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
class dynamic_arg_list {
|
// node is defined outside dynamic_arg_list to workaround a C2504 bug in MSVC
|
||||||
// Workaround for clang's -Wweak-vtables. Unlike for regular classes, for
|
// 2022 (v17.10.0).
|
||||||
// templates it doesn't complain about inability to deduce single translation
|
//
|
||||||
// unit for placing vtable. So storage_node_base is made a fake template.
|
// Workaround for clang's -Wweak-vtables. Unlike for regular classes, for
|
||||||
template <typename = void> struct node {
|
// templates it doesn't complain about inability to deduce single translation
|
||||||
virtual ~node() = default;
|
// unit for placing vtable. So node is made a fake template.
|
||||||
std::unique_ptr<node<>> next;
|
template <typename = void> struct node {
|
||||||
};
|
virtual ~node() = default;
|
||||||
|
std::unique_ptr<node<>> next;
|
||||||
|
};
|
||||||
|
|
||||||
|
class dynamic_arg_list {
|
||||||
template <typename T> struct typed_node : node<> {
|
template <typename T> struct typed_node : node<> {
|
||||||
T value;
|
T value;
|
||||||
|
|
||||||
|
|
@ -50,7 +55,7 @@ class dynamic_arg_list {
|
||||||
std::unique_ptr<node<>> head_;
|
std::unique_ptr<node<>> head_;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
template <typename T, typename Arg> const T& push(const Arg& arg) {
|
template <typename T, typename Arg> auto push(const Arg& arg) -> const T& {
|
||||||
auto new_node = std::unique_ptr<typed_node<T>>(new typed_node<T>(arg));
|
auto new_node = std::unique_ptr<typed_node<T>>(new typed_node<T>(arg));
|
||||||
auto& value = new_node->value;
|
auto& value = new_node->value;
|
||||||
new_node->next = std::move(head_);
|
new_node->next = std::move(head_);
|
||||||
|
|
@ -61,28 +66,18 @@ class dynamic_arg_list {
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* A dynamic list of formatting arguments with storage.
|
||||||
A dynamic version of `fmt::format_arg_store`.
|
*
|
||||||
It's equipped with a storage to potentially temporary objects which lifetimes
|
* It can be implicitly converted into `fmt::basic_format_args` for passing
|
||||||
could be shorter than the format arguments object.
|
* into type-erased formatting functions such as `fmt::vformat`.
|
||||||
|
|
||||||
It can be implicitly converted into `~fmt::basic_format_args` for passing
|
|
||||||
into type-erased formatting functions such as `~fmt::vformat`.
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename Context>
|
FMT_EXPORT template <typename Context> class dynamic_format_arg_store {
|
||||||
class dynamic_format_arg_store
|
|
||||||
#if FMT_GCC_VERSION && FMT_GCC_VERSION < 409
|
|
||||||
// Workaround a GCC template argument substitution bug.
|
|
||||||
: public basic_format_args<Context>
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
private:
|
private:
|
||||||
using char_type = typename Context::char_type;
|
using char_type = typename Context::char_type;
|
||||||
|
|
||||||
template <typename T> struct need_copy {
|
template <typename T> struct need_copy {
|
||||||
static constexpr detail::type mapped_type =
|
static constexpr detail::type mapped_type =
|
||||||
detail::mapped_type_constant<T, Context>::value;
|
detail::mapped_type_constant<T, char_type>::value;
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
value = !(detail::is_reference_wrapper<T>::value ||
|
value = !(detail::is_reference_wrapper<T>::value ||
|
||||||
|
|
@ -95,7 +90,7 @@ class dynamic_format_arg_store
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
using stored_type = conditional_t<
|
using stored_t = conditional_t<
|
||||||
std::is_convertible<T, std::basic_string<char_type>>::value &&
|
std::is_convertible<T, std::basic_string<char_type>>::value &&
|
||||||
!detail::is_reference_wrapper<T>::value,
|
!detail::is_reference_wrapper<T>::value,
|
||||||
std::basic_string<char_type>, T>;
|
std::basic_string<char_type>, T>;
|
||||||
|
|
@ -110,80 +105,72 @@ class dynamic_format_arg_store
|
||||||
|
|
||||||
friend class basic_format_args<Context>;
|
friend class basic_format_args<Context>;
|
||||||
|
|
||||||
unsigned long long get_types() const {
|
auto data() const -> const basic_format_arg<Context>* {
|
||||||
return detail::is_unpacked_bit | data_.size() |
|
|
||||||
(named_info_.empty()
|
|
||||||
? 0ULL
|
|
||||||
: static_cast<unsigned long long>(detail::has_named_args_bit));
|
|
||||||
}
|
|
||||||
|
|
||||||
const basic_format_arg<Context>* data() const {
|
|
||||||
return named_info_.empty() ? data_.data() : data_.data() + 1;
|
return named_info_.empty() ? data_.data() : data_.data() + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T> void emplace_arg(const T& arg) {
|
template <typename T> void emplace_arg(const T& arg) {
|
||||||
data_.emplace_back(detail::make_arg<Context>(arg));
|
data_.emplace_back(arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void emplace_arg(const detail::named_arg<char_type, T>& arg) {
|
void emplace_arg(const detail::named_arg<char_type, T>& arg) {
|
||||||
if (named_info_.empty()) {
|
if (named_info_.empty())
|
||||||
constexpr const detail::named_arg_info<char_type>* zero_ptr{nullptr};
|
data_.insert(data_.begin(), basic_format_arg<Context>(nullptr, 0));
|
||||||
data_.insert(data_.begin(), {zero_ptr, 0});
|
data_.emplace_back(detail::unwrap(arg.value));
|
||||||
}
|
|
||||||
data_.emplace_back(detail::make_arg<Context>(detail::unwrap(arg.value)));
|
|
||||||
auto pop_one = [](std::vector<basic_format_arg<Context>>* data) {
|
auto pop_one = [](std::vector<basic_format_arg<Context>>* data) {
|
||||||
data->pop_back();
|
data->pop_back();
|
||||||
};
|
};
|
||||||
std::unique_ptr<std::vector<basic_format_arg<Context>>, decltype(pop_one)>
|
std::unique_ptr<std::vector<basic_format_arg<Context>>, decltype(pop_one)>
|
||||||
guard{&data_, pop_one};
|
guard{&data_, pop_one};
|
||||||
named_info_.push_back({arg.name, static_cast<int>(data_.size() - 2u)});
|
named_info_.push_back({arg.name, static_cast<int>(data_.size() - 2u)});
|
||||||
data_[0].value_.named_args = {named_info_.data(), named_info_.size()};
|
data_[0] = {named_info_.data(), named_info_.size()};
|
||||||
guard.release();
|
guard.release();
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
constexpr dynamic_format_arg_store() = default;
|
constexpr dynamic_format_arg_store() = default;
|
||||||
|
|
||||||
|
operator basic_format_args<Context>() const {
|
||||||
|
return basic_format_args<Context>(data(), static_cast<int>(data_.size()),
|
||||||
|
!named_info_.empty());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Adds an argument into the dynamic store for later passing to a formatting
|
||||||
Adds an argument into the dynamic store for later passing to a formatting
|
* function.
|
||||||
function.
|
*
|
||||||
|
* Note that custom types and string types (but not string views) are copied
|
||||||
Note that custom types and string types (but not string views) are copied
|
* into the store dynamically allocating memory if necessary.
|
||||||
into the store dynamically allocating memory if necessary.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
* store.push_back(42);
|
||||||
store.push_back(42);
|
* store.push_back("abc");
|
||||||
store.push_back("abc");
|
* store.push_back(1.5f);
|
||||||
store.push_back(1.5f);
|
* std::string result = fmt::vformat("{} and {} and {}", store);
|
||||||
std::string result = fmt::vformat("{} and {} and {}", store);
|
*/
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename T> void push_back(const T& arg) {
|
template <typename T> void push_back(const T& arg) {
|
||||||
if (detail::const_check(need_copy<T>::value))
|
if (detail::const_check(need_copy<T>::value))
|
||||||
emplace_arg(dynamic_args_.push<stored_type<T>>(arg));
|
emplace_arg(dynamic_args_.push<stored_t<T>>(arg));
|
||||||
else
|
else
|
||||||
emplace_arg(detail::unwrap(arg));
|
emplace_arg(detail::unwrap(arg));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Adds a reference to the argument into the dynamic store for later passing
|
||||||
Adds a reference to the argument into the dynamic store for later passing to
|
* to a formatting function.
|
||||||
a formatting function.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
* char band[] = "Rolling Stones";
|
||||||
char band[] = "Rolling Stones";
|
* store.push_back(std::cref(band));
|
||||||
store.push_back(std::cref(band));
|
* band[9] = 'c'; // Changing str affects the output.
|
||||||
band[9] = 'c'; // Changing str affects the output.
|
* std::string result = fmt::vformat("{}", store);
|
||||||
std::string result = fmt::vformat("{}", store);
|
* // result == "Rolling Scones"
|
||||||
// result == "Rolling Scones"
|
*/
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename T> void push_back(std::reference_wrapper<T> arg) {
|
template <typename T> void push_back(std::reference_wrapper<T> arg) {
|
||||||
static_assert(
|
static_assert(
|
||||||
need_copy<T>::value,
|
need_copy<T>::value,
|
||||||
|
|
@ -192,41 +179,40 @@ class dynamic_format_arg_store
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Adds named argument into the dynamic store for later passing to a formatting
|
* Adds named argument into the dynamic store for later passing to a
|
||||||
function. ``std::reference_wrapper`` is supported to avoid copying of the
|
* formatting function. `std::reference_wrapper` is supported to avoid
|
||||||
argument. The name is always copied into the store.
|
* copying of the argument. The name is always copied into the store.
|
||||||
*/
|
*/
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void push_back(const detail::named_arg<char_type, T>& arg) {
|
void push_back(const detail::named_arg<char_type, T>& arg) {
|
||||||
const char_type* arg_name =
|
const char_type* arg_name =
|
||||||
dynamic_args_.push<std::basic_string<char_type>>(arg.name).c_str();
|
dynamic_args_.push<std::basic_string<char_type>>(arg.name).c_str();
|
||||||
if (detail::const_check(need_copy<T>::value)) {
|
if (detail::const_check(need_copy<T>::value)) {
|
||||||
emplace_arg(
|
emplace_arg(
|
||||||
fmt::arg(arg_name, dynamic_args_.push<stored_type<T>>(arg.value)));
|
fmt::arg(arg_name, dynamic_args_.push<stored_t<T>>(arg.value)));
|
||||||
} else {
|
} else {
|
||||||
emplace_arg(fmt::arg(arg_name, arg.value));
|
emplace_arg(fmt::arg(arg_name, arg.value));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Erase all elements from the store */
|
/// Erase all elements from the store.
|
||||||
void clear() {
|
void clear() {
|
||||||
data_.clear();
|
data_.clear();
|
||||||
named_info_.clear();
|
named_info_.clear();
|
||||||
dynamic_args_ = detail::dynamic_arg_list();
|
dynamic_args_ = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/// Reserves space to store at least `new_cap` arguments including
|
||||||
\rst
|
/// `new_cap_named` named arguments.
|
||||||
Reserves space to store at least *new_cap* arguments including
|
|
||||||
*new_cap_named* named arguments.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
void reserve(size_t new_cap, size_t new_cap_named) {
|
void reserve(size_t new_cap, size_t new_cap_named) {
|
||||||
FMT_ASSERT(new_cap >= new_cap_named,
|
FMT_ASSERT(new_cap >= new_cap_named,
|
||||||
"Set of arguments includes set of named arguments");
|
"set of arguments includes set of named arguments");
|
||||||
data_.reserve(new_cap);
|
data_.reserve(new_cap);
|
||||||
named_info_.reserve(new_cap_named);
|
named_info_.reserve(new_cap_named);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the number of elements in the store.
|
||||||
|
auto size() const noexcept -> size_t { return data_.size(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
|
||||||
3010
deps/fmt/include/fmt/base.h
vendored
Normal file
3010
deps/fmt/include/fmt/base.h
vendored
Normal file
File diff suppressed because it is too large
Load diff
1670
deps/fmt/include/fmt/chrono.h
vendored
1670
deps/fmt/include/fmt/chrono.h
vendored
File diff suppressed because it is too large
Load diff
467
deps/fmt/include/fmt/color.h
vendored
467
deps/fmt/include/fmt/color.h
vendored
|
|
@ -190,11 +190,11 @@ enum class emphasis : uint8_t {
|
||||||
// rgb is a struct for red, green and blue colors.
|
// rgb is a struct for red, green and blue colors.
|
||||||
// Using the name "rgb" makes some editors show the color in a tooltip.
|
// Using the name "rgb" makes some editors show the color in a tooltip.
|
||||||
struct rgb {
|
struct rgb {
|
||||||
FMT_CONSTEXPR rgb() : r(0), g(0), b(0) {}
|
constexpr rgb() : r(0), g(0), b(0) {}
|
||||||
FMT_CONSTEXPR rgb(uint8_t r_, uint8_t g_, uint8_t b_) : r(r_), g(g_), b(b_) {}
|
constexpr rgb(uint8_t r_, uint8_t g_, uint8_t b_) : r(r_), g(g_), b(b_) {}
|
||||||
FMT_CONSTEXPR rgb(uint32_t hex)
|
constexpr rgb(uint32_t hex)
|
||||||
: r((hex >> 16) & 0xFF), g((hex >> 8) & 0xFF), b(hex & 0xFF) {}
|
: r((hex >> 16) & 0xFF), g((hex >> 8) & 0xFF), b(hex & 0xFF) {}
|
||||||
FMT_CONSTEXPR rgb(color hex)
|
constexpr rgb(color hex)
|
||||||
: r((uint32_t(hex) >> 16) & 0xFF),
|
: r((uint32_t(hex) >> 16) & 0xFF),
|
||||||
g((uint32_t(hex) >> 8) & 0xFF),
|
g((uint32_t(hex) >> 8) & 0xFF),
|
||||||
b(uint32_t(hex) & 0xFF) {}
|
b(uint32_t(hex) & 0xFF) {}
|
||||||
|
|
@ -205,161 +205,198 @@ struct rgb {
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
// color is a struct of either a rgb color or a terminal color.
|
// A bit-packed variant of an RGB color, a terminal color, or unset color.
|
||||||
|
// see text_style for the bit-packing scheme.
|
||||||
struct color_type {
|
struct color_type {
|
||||||
FMT_CONSTEXPR color_type() noexcept : is_rgb(), value{} {}
|
constexpr color_type() noexcept = default;
|
||||||
FMT_CONSTEXPR color_type(color rgb_color) noexcept : is_rgb(true), value{} {
|
constexpr color_type(color rgb_color) noexcept
|
||||||
value.rgb_color = static_cast<uint32_t>(rgb_color);
|
: value_(static_cast<uint32_t>(rgb_color) | (1 << 24)) {}
|
||||||
|
constexpr color_type(rgb rgb_color) noexcept
|
||||||
|
: color_type(static_cast<color>(
|
||||||
|
(static_cast<uint32_t>(rgb_color.r) << 16) |
|
||||||
|
(static_cast<uint32_t>(rgb_color.g) << 8) | rgb_color.b)) {}
|
||||||
|
constexpr color_type(terminal_color term_color) noexcept
|
||||||
|
: value_(static_cast<uint32_t>(term_color) | (3 << 24)) {}
|
||||||
|
|
||||||
|
constexpr auto is_terminal_color() const noexcept -> bool {
|
||||||
|
return (value_ & (1 << 25)) != 0;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR color_type(rgb rgb_color) noexcept : is_rgb(true), value{} {
|
|
||||||
value.rgb_color = (static_cast<uint32_t>(rgb_color.r) << 16) |
|
constexpr auto value() const noexcept -> uint32_t {
|
||||||
(static_cast<uint32_t>(rgb_color.g) << 8) | rgb_color.b;
|
return value_ & 0xFFFFFF;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR color_type(terminal_color term_color) noexcept
|
|
||||||
: is_rgb(), value{} {
|
constexpr color_type(uint32_t value) noexcept : value_(value) {}
|
||||||
value.term_color = static_cast<uint8_t>(term_color);
|
|
||||||
}
|
uint32_t value_ = 0;
|
||||||
bool is_rgb;
|
|
||||||
union color_union {
|
|
||||||
uint8_t term_color;
|
|
||||||
uint32_t rgb_color;
|
|
||||||
} value;
|
|
||||||
};
|
};
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
/** A text style consisting of foreground and background colors and emphasis. */
|
/// A text style consisting of foreground and background colors and emphasis.
|
||||||
class text_style {
|
class text_style {
|
||||||
|
// The information is packed as follows:
|
||||||
|
// ┌──┐
|
||||||
|
// │ 0│─┐
|
||||||
|
// │..│ ├── foreground color value
|
||||||
|
// │23│─┘
|
||||||
|
// ├──┤
|
||||||
|
// │24│─┬── discriminator for the above value. 00 if unset, 01 if it's
|
||||||
|
// │25│─┘ an RGB color, or 11 if it's a terminal color (10 is unused)
|
||||||
|
// ├──┤
|
||||||
|
// │26│──── overflow bit, always zero (see below)
|
||||||
|
// ├──┤
|
||||||
|
// │27│─┐
|
||||||
|
// │..│ │
|
||||||
|
// │50│ │
|
||||||
|
// ├──┤ │
|
||||||
|
// │51│ ├── background color (same format as the foreground color)
|
||||||
|
// │52│ │
|
||||||
|
// ├──┤ │
|
||||||
|
// │53│─┘
|
||||||
|
// ├──┤
|
||||||
|
// │54│─┐
|
||||||
|
// │..│ ├── emphases
|
||||||
|
// │61│─┘
|
||||||
|
// ├──┤
|
||||||
|
// │62│─┬── unused
|
||||||
|
// │63│─┘
|
||||||
|
// └──┘
|
||||||
|
// The overflow bits are there to make operator|= efficient.
|
||||||
|
// When ORing, we must throw if, for either the foreground or background,
|
||||||
|
// one style specifies a terminal color and the other specifies any color
|
||||||
|
// (terminal or RGB); in other words, if one discriminator is 11 and the
|
||||||
|
// other is 11 or 01.
|
||||||
|
//
|
||||||
|
// We do that check by adding the styles. Consider what adding does to each
|
||||||
|
// possible pair of discriminators:
|
||||||
|
// 00 + 00 = 000
|
||||||
|
// 01 + 00 = 001
|
||||||
|
// 11 + 00 = 011
|
||||||
|
// 01 + 01 = 010
|
||||||
|
// 11 + 01 = 100 (!!)
|
||||||
|
// 11 + 11 = 110 (!!)
|
||||||
|
// In the last two cases, the ones we want to catch, the third bit——the
|
||||||
|
// overflow bit——is set. Bingo.
|
||||||
|
//
|
||||||
|
// We must take into account the possible carry bit from the bits
|
||||||
|
// before the discriminator. The only potentially problematic case is
|
||||||
|
// 11 + 00 = 011 (a carry bit would make it 100, not good!), but a carry
|
||||||
|
// bit is impossible in that case, because 00 (unset color) means the
|
||||||
|
// 24 bits that precede the discriminator are all zero.
|
||||||
|
//
|
||||||
|
// This test can be applied to both colors simultaneously.
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR text_style(emphasis em = emphasis()) noexcept
|
FMT_CONSTEXPR text_style(emphasis em = emphasis()) noexcept
|
||||||
: set_foreground_color(), set_background_color(), ems(em) {}
|
: style_(static_cast<uint64_t>(em) << 54) {}
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style& operator|=(const text_style& rhs) {
|
FMT_CONSTEXPR auto operator|=(text_style rhs) -> text_style& {
|
||||||
if (!set_foreground_color) {
|
if (((style_ + rhs.style_) & ((1ULL << 26) | (1ULL << 53))) != 0)
|
||||||
set_foreground_color = rhs.set_foreground_color;
|
report_error("can't OR a terminal color");
|
||||||
foreground_color = rhs.foreground_color;
|
style_ |= rhs.style_;
|
||||||
} else if (rhs.set_foreground_color) {
|
|
||||||
if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb)
|
|
||||||
FMT_THROW(format_error("can't OR a terminal color"));
|
|
||||||
foreground_color.value.rgb_color |= rhs.foreground_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!set_background_color) {
|
|
||||||
set_background_color = rhs.set_background_color;
|
|
||||||
background_color = rhs.background_color;
|
|
||||||
} else if (rhs.set_background_color) {
|
|
||||||
if (!background_color.is_rgb || !rhs.background_color.is_rgb)
|
|
||||||
FMT_THROW(format_error("can't OR a terminal color"));
|
|
||||||
background_color.value.rgb_color |= rhs.background_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
ems = static_cast<emphasis>(static_cast<uint8_t>(ems) |
|
|
||||||
static_cast<uint8_t>(rhs.ems));
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
friend FMT_CONSTEXPR text_style operator|(text_style lhs,
|
friend FMT_CONSTEXPR auto operator|(text_style lhs, text_style rhs)
|
||||||
const text_style& rhs) {
|
-> text_style {
|
||||||
return lhs |= rhs;
|
return lhs |= rhs;
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_CONSTEXPR bool has_foreground() const noexcept {
|
FMT_CONSTEXPR auto operator==(text_style rhs) const noexcept -> bool {
|
||||||
return set_foreground_color;
|
return style_ == rhs.style_;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR bool has_background() const noexcept {
|
|
||||||
return set_background_color;
|
FMT_CONSTEXPR auto operator!=(text_style rhs) const noexcept -> bool {
|
||||||
|
return !(*this == rhs);
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR bool has_emphasis() const noexcept {
|
|
||||||
return static_cast<uint8_t>(ems) != 0;
|
FMT_CONSTEXPR auto has_foreground() const noexcept -> bool {
|
||||||
|
return (style_ & (1 << 24)) != 0;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR detail::color_type get_foreground() const noexcept {
|
FMT_CONSTEXPR auto has_background() const noexcept -> bool {
|
||||||
|
return (style_ & (1ULL << 51)) != 0;
|
||||||
|
}
|
||||||
|
FMT_CONSTEXPR auto has_emphasis() const noexcept -> bool {
|
||||||
|
return (style_ >> 54) != 0;
|
||||||
|
}
|
||||||
|
FMT_CONSTEXPR auto get_foreground() const noexcept -> detail::color_type {
|
||||||
FMT_ASSERT(has_foreground(), "no foreground specified for this style");
|
FMT_ASSERT(has_foreground(), "no foreground specified for this style");
|
||||||
return foreground_color;
|
return style_ & 0x3FFFFFF;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR detail::color_type get_background() const noexcept {
|
FMT_CONSTEXPR auto get_background() const noexcept -> detail::color_type {
|
||||||
FMT_ASSERT(has_background(), "no background specified for this style");
|
FMT_ASSERT(has_background(), "no background specified for this style");
|
||||||
return background_color;
|
return (style_ >> 27) & 0x3FFFFFF;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR emphasis get_emphasis() const noexcept {
|
FMT_CONSTEXPR auto get_emphasis() const noexcept -> emphasis {
|
||||||
FMT_ASSERT(has_emphasis(), "no emphasis specified for this style");
|
FMT_ASSERT(has_emphasis(), "no emphasis specified for this style");
|
||||||
return ems;
|
return static_cast<emphasis>(style_ >> 54);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
FMT_CONSTEXPR text_style(bool is_foreground,
|
FMT_CONSTEXPR text_style(uint64_t style) noexcept : style_(style) {}
|
||||||
detail::color_type text_color) noexcept
|
|
||||||
: set_foreground_color(), set_background_color(), ems() {
|
|
||||||
if (is_foreground) {
|
|
||||||
foreground_color = text_color;
|
|
||||||
set_foreground_color = true;
|
|
||||||
} else {
|
|
||||||
background_color = text_color;
|
|
||||||
set_background_color = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
friend FMT_CONSTEXPR text_style fg(detail::color_type foreground) noexcept;
|
friend FMT_CONSTEXPR auto fg(detail::color_type foreground) noexcept
|
||||||
|
-> text_style;
|
||||||
|
|
||||||
friend FMT_CONSTEXPR text_style bg(detail::color_type background) noexcept;
|
friend FMT_CONSTEXPR auto bg(detail::color_type background) noexcept
|
||||||
|
-> text_style;
|
||||||
|
|
||||||
detail::color_type foreground_color;
|
uint64_t style_ = 0;
|
||||||
detail::color_type background_color;
|
|
||||||
bool set_foreground_color;
|
|
||||||
bool set_background_color;
|
|
||||||
emphasis ems;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Creates a text style from the foreground (text) color. */
|
/// Creates a text style from the foreground (text) color.
|
||||||
FMT_CONSTEXPR inline text_style fg(detail::color_type foreground) noexcept {
|
FMT_CONSTEXPR inline auto fg(detail::color_type foreground) noexcept
|
||||||
return text_style(true, foreground);
|
-> text_style {
|
||||||
|
return foreground.value_;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Creates a text style from the background color. */
|
/// Creates a text style from the background color.
|
||||||
FMT_CONSTEXPR inline text_style bg(detail::color_type background) noexcept {
|
FMT_CONSTEXPR inline auto bg(detail::color_type background) noexcept
|
||||||
return text_style(false, background);
|
-> text_style {
|
||||||
|
return static_cast<uint64_t>(background.value_) << 27;
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_CONSTEXPR inline text_style operator|(emphasis lhs, emphasis rhs) noexcept {
|
FMT_CONSTEXPR inline auto operator|(emphasis lhs, emphasis rhs) noexcept
|
||||||
|
-> text_style {
|
||||||
return text_style(lhs) | rhs;
|
return text_style(lhs) | rhs;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template <typename Char> struct ansi_color_escape {
|
template <typename Char> struct ansi_color_escape {
|
||||||
FMT_CONSTEXPR ansi_color_escape(detail::color_type text_color,
|
FMT_CONSTEXPR ansi_color_escape(color_type text_color,
|
||||||
const char* esc) noexcept {
|
const char* esc) noexcept {
|
||||||
// If we have a terminal color, we need to output another escape code
|
// If we have a terminal color, we need to output another escape code
|
||||||
// sequence.
|
// sequence.
|
||||||
if (!text_color.is_rgb) {
|
if (text_color.is_terminal_color()) {
|
||||||
bool is_background = esc == string_view("\x1b[48;2;");
|
bool is_background = esc == string_view("\x1b[48;2;");
|
||||||
uint32_t value = text_color.value.term_color;
|
uint32_t value = text_color.value();
|
||||||
// Background ASCII codes are the same as the foreground ones but with
|
// Background ASCII codes are the same as the foreground ones but with
|
||||||
// 10 more.
|
// 10 more.
|
||||||
if (is_background) value += 10u;
|
if (is_background) value += 10u;
|
||||||
|
|
||||||
size_t index = 0;
|
buffer[size++] = static_cast<Char>('\x1b');
|
||||||
buffer[index++] = static_cast<Char>('\x1b');
|
buffer[size++] = static_cast<Char>('[');
|
||||||
buffer[index++] = static_cast<Char>('[');
|
|
||||||
|
|
||||||
if (value >= 100u) {
|
if (value >= 100u) {
|
||||||
buffer[index++] = static_cast<Char>('1');
|
buffer[size++] = static_cast<Char>('1');
|
||||||
value %= 100u;
|
value %= 100u;
|
||||||
}
|
}
|
||||||
buffer[index++] = static_cast<Char>('0' + value / 10u);
|
buffer[size++] = static_cast<Char>('0' + value / 10u);
|
||||||
buffer[index++] = static_cast<Char>('0' + value % 10u);
|
buffer[size++] = static_cast<Char>('0' + value % 10u);
|
||||||
|
|
||||||
buffer[index++] = static_cast<Char>('m');
|
buffer[size++] = static_cast<Char>('m');
|
||||||
buffer[index++] = static_cast<Char>('\0');
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < 7; i++) {
|
for (int i = 0; i < 7; i++) {
|
||||||
buffer[i] = static_cast<Char>(esc[i]);
|
buffer[i] = static_cast<Char>(esc[i]);
|
||||||
}
|
}
|
||||||
rgb color(text_color.value.rgb_color);
|
rgb color(text_color.value());
|
||||||
to_esc(color.r, buffer + 7, ';');
|
to_esc(color.r, buffer + 7, ';');
|
||||||
to_esc(color.g, buffer + 11, ';');
|
to_esc(color.g, buffer + 11, ';');
|
||||||
to_esc(color.b, buffer + 15, 'm');
|
to_esc(color.b, buffer + 15, 'm');
|
||||||
buffer[19] = static_cast<Char>(0);
|
size = 19;
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR ansi_color_escape(emphasis em) noexcept {
|
FMT_CONSTEXPR ansi_color_escape(emphasis em) noexcept {
|
||||||
uint8_t em_codes[num_emphases] = {};
|
uint8_t em_codes[num_emphases] = {};
|
||||||
|
|
@ -372,26 +409,28 @@ template <typename Char> struct ansi_color_escape {
|
||||||
if (has_emphasis(em, emphasis::conceal)) em_codes[6] = 8;
|
if (has_emphasis(em, emphasis::conceal)) em_codes[6] = 8;
|
||||||
if (has_emphasis(em, emphasis::strikethrough)) em_codes[7] = 9;
|
if (has_emphasis(em, emphasis::strikethrough)) em_codes[7] = 9;
|
||||||
|
|
||||||
size_t index = 0;
|
buffer[size++] = static_cast<Char>('\x1b');
|
||||||
|
buffer[size++] = static_cast<Char>('[');
|
||||||
|
|
||||||
for (size_t i = 0; i < num_emphases; ++i) {
|
for (size_t i = 0; i < num_emphases; ++i) {
|
||||||
if (!em_codes[i]) continue;
|
if (!em_codes[i]) continue;
|
||||||
buffer[index++] = static_cast<Char>('\x1b');
|
buffer[size++] = static_cast<Char>('0' + em_codes[i]);
|
||||||
buffer[index++] = static_cast<Char>('[');
|
buffer[size++] = static_cast<Char>(';');
|
||||||
buffer[index++] = static_cast<Char>('0' + em_codes[i]);
|
|
||||||
buffer[index++] = static_cast<Char>('m');
|
|
||||||
}
|
}
|
||||||
buffer[index++] = static_cast<Char>(0);
|
|
||||||
|
buffer[size - 1] = static_cast<Char>('m');
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR operator const Char*() const noexcept { return buffer; }
|
FMT_CONSTEXPR operator const Char*() const noexcept { return buffer; }
|
||||||
|
|
||||||
FMT_CONSTEXPR const Char* begin() const noexcept { return buffer; }
|
FMT_CONSTEXPR auto begin() const noexcept -> const Char* { return buffer; }
|
||||||
FMT_CONSTEXPR_CHAR_TRAITS const Char* end() const noexcept {
|
FMT_CONSTEXPR auto end() const noexcept -> const Char* {
|
||||||
return buffer + std::char_traits<Char>::length(buffer);
|
return buffer + size;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static constexpr size_t num_emphases = 8;
|
static constexpr size_t num_emphases = 8;
|
||||||
Char buffer[7u + 3u * num_emphases + 1u];
|
Char buffer[7u + 4u * num_emphases] = {};
|
||||||
|
size_t size = 0;
|
||||||
|
|
||||||
static FMT_CONSTEXPR void to_esc(uint8_t c, Char* out,
|
static FMT_CONSTEXPR void to_esc(uint8_t c, Char* out,
|
||||||
char delimiter) noexcept {
|
char delimiter) noexcept {
|
||||||
|
|
@ -400,25 +439,27 @@ template <typename Char> struct ansi_color_escape {
|
||||||
out[2] = static_cast<Char>('0' + c % 10);
|
out[2] = static_cast<Char>('0' + c % 10);
|
||||||
out[3] = static_cast<Char>(delimiter);
|
out[3] = static_cast<Char>(delimiter);
|
||||||
}
|
}
|
||||||
static FMT_CONSTEXPR bool has_emphasis(emphasis em, emphasis mask) noexcept {
|
static FMT_CONSTEXPR auto has_emphasis(emphasis em, emphasis mask) noexcept
|
||||||
|
-> bool {
|
||||||
return static_cast<uint8_t>(em) & static_cast<uint8_t>(mask);
|
return static_cast<uint8_t>(em) & static_cast<uint8_t>(mask);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char> make_foreground_color(
|
FMT_CONSTEXPR auto make_foreground_color(color_type foreground) noexcept
|
||||||
detail::color_type foreground) noexcept {
|
-> ansi_color_escape<Char> {
|
||||||
return ansi_color_escape<Char>(foreground, "\x1b[38;2;");
|
return ansi_color_escape<Char>(foreground, "\x1b[38;2;");
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char> make_background_color(
|
FMT_CONSTEXPR auto make_background_color(color_type background) noexcept
|
||||||
detail::color_type background) noexcept {
|
-> ansi_color_escape<Char> {
|
||||||
return ansi_color_escape<Char>(background, "\x1b[48;2;");
|
return ansi_color_escape<Char>(background, "\x1b[48;2;");
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char> make_emphasis(emphasis em) noexcept {
|
FMT_CONSTEXPR auto make_emphasis(emphasis em) noexcept
|
||||||
|
-> ansi_color_escape<Char> {
|
||||||
return ansi_color_escape<Char>(em);
|
return ansi_color_escape<Char>(em);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -427,149 +468,116 @@ template <typename Char> inline void reset_color(buffer<Char>& buffer) {
|
||||||
buffer.append(reset_color.begin(), reset_color.end());
|
buffer.append(reset_color.begin(), reset_color.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T> struct styled_arg {
|
template <typename T> struct styled_arg : view {
|
||||||
const T& value;
|
const T& value;
|
||||||
text_style style;
|
text_style style;
|
||||||
|
styled_arg(const T& v, text_style s) : value(v), style(s) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
void vformat_to(buffer<Char>& buf, const text_style& ts,
|
void vformat_to(buffer<Char>& buf, text_style ts, basic_string_view<Char> fmt,
|
||||||
basic_string_view<Char> format_str,
|
basic_format_args<buffered_context<Char>> args) {
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
|
||||||
bool has_style = false;
|
|
||||||
if (ts.has_emphasis()) {
|
if (ts.has_emphasis()) {
|
||||||
has_style = true;
|
auto emphasis = make_emphasis<Char>(ts.get_emphasis());
|
||||||
auto emphasis = detail::make_emphasis<Char>(ts.get_emphasis());
|
|
||||||
buf.append(emphasis.begin(), emphasis.end());
|
buf.append(emphasis.begin(), emphasis.end());
|
||||||
}
|
}
|
||||||
if (ts.has_foreground()) {
|
if (ts.has_foreground()) {
|
||||||
has_style = true;
|
auto foreground = make_foreground_color<Char>(ts.get_foreground());
|
||||||
auto foreground = detail::make_foreground_color<Char>(ts.get_foreground());
|
|
||||||
buf.append(foreground.begin(), foreground.end());
|
buf.append(foreground.begin(), foreground.end());
|
||||||
}
|
}
|
||||||
if (ts.has_background()) {
|
if (ts.has_background()) {
|
||||||
has_style = true;
|
auto background = make_background_color<Char>(ts.get_background());
|
||||||
auto background = detail::make_background_color<Char>(ts.get_background());
|
|
||||||
buf.append(background.begin(), background.end());
|
buf.append(background.begin(), background.end());
|
||||||
}
|
}
|
||||||
detail::vformat_to(buf, format_str, args, {});
|
vformat_to(buf, fmt, args);
|
||||||
if (has_style) detail::reset_color<Char>(buf);
|
if (ts != text_style()) reset_color<Char>(buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
inline void vprint(std::FILE* f, const text_style& ts, string_view fmt,
|
inline void vprint(FILE* f, text_style ts, string_view fmt, format_args args) {
|
||||||
format_args args) {
|
|
||||||
// Legacy wide streams are not supported.
|
|
||||||
auto buf = memory_buffer();
|
auto buf = memory_buffer();
|
||||||
detail::vformat_to(buf, ts, fmt, args);
|
detail::vformat_to(buf, ts, fmt, args);
|
||||||
if (detail::is_utf8()) {
|
print(f, FMT_STRING("{}"), string_view(buf.begin(), buf.size()));
|
||||||
detail::print(f, string_view(buf.begin(), buf.size()));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
buf.push_back('\0');
|
|
||||||
int result = std::fputs(buf.data(), f);
|
|
||||||
if (result < 0)
|
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot write to file")));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats a string and prints it to the specified file stream using ANSI
|
||||||
Formats a string and prints it to the specified file stream using ANSI
|
* escape sequences to specify text formatting.
|
||||||
escape sequences to specify text formatting.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
||||||
fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
* "Elapsed time: {0:.2f} seconds", 1.23);
|
||||||
"Elapsed time: {0:.2f} seconds", 1.23);
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename S, typename... Args,
|
template <typename... T>
|
||||||
FMT_ENABLE_IF(detail::is_string<S>::value)>
|
void print(FILE* f, text_style ts, format_string<T...> fmt, T&&... args) {
|
||||||
void print(std::FILE* f, const text_style& ts, const S& format_str,
|
vprint(f, ts, fmt.str, vargs<T...>{{args...}});
|
||||||
const Args&... args) {
|
|
||||||
vprint(f, ts, format_str,
|
|
||||||
fmt::make_format_args<buffer_context<char_t<S>>>(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats a string and prints it to stdout using ANSI escape sequences to
|
||||||
Formats a string and prints it to stdout using ANSI escape sequences to
|
* specify text formatting.
|
||||||
specify text formatting.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
||||||
fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
* "Elapsed time: {0:.2f} seconds", 1.23);
|
||||||
"Elapsed time: {0:.2f} seconds", 1.23);
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename S, typename... Args,
|
template <typename... T>
|
||||||
FMT_ENABLE_IF(detail::is_string<S>::value)>
|
void print(text_style ts, format_string<T...> fmt, T&&... args) {
|
||||||
void print(const text_style& ts, const S& format_str, const Args&... args) {
|
return print(stdout, ts, fmt, std::forward<T>(args)...);
|
||||||
return print(stdout, ts, format_str, args...);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename Char = char_t<S>>
|
inline auto vformat(text_style ts, string_view fmt, format_args args)
|
||||||
inline std::basic_string<Char> vformat(
|
-> std::string {
|
||||||
const text_style& ts, const S& format_str,
|
auto buf = memory_buffer();
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
detail::vformat_to(buf, ts, fmt, args);
|
||||||
basic_memory_buffer<Char> buf;
|
|
||||||
detail::vformat_to(buf, ts, detail::to_string_view(format_str), args);
|
|
||||||
return fmt::to_string(buf);
|
return fmt::to_string(buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats arguments and returns the result as a string using ANSI escape
|
||||||
Formats arguments and returns the result as a string using ANSI
|
* sequences to specify text formatting.
|
||||||
escape sequences to specify text formatting.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* ```
|
||||||
#include <fmt/color.h>
|
* #include <fmt/color.h>
|
||||||
std::string message = fmt::format(fmt::emphasis::bold | fg(fmt::color::red),
|
* std::string message = fmt::format(fmt::emphasis::bold | fg(fmt::color::red),
|
||||||
"The answer is {}", 42);
|
* "The answer is {}", 42);
|
||||||
\endrst
|
* ```
|
||||||
*/
|
*/
|
||||||
template <typename S, typename... Args, typename Char = char_t<S>>
|
template <typename... T>
|
||||||
inline std::basic_string<Char> format(const text_style& ts, const S& format_str,
|
inline auto format(text_style ts, format_string<T...> fmt, T&&... args)
|
||||||
const Args&... args) {
|
-> std::string {
|
||||||
return fmt::vformat(ts, detail::to_string_view(format_str),
|
return fmt::vformat(ts, fmt.str, vargs<T...>{{args...}});
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/// Formats a string with the given text_style and writes the output to `out`.
|
||||||
Formats a string with the given text_style and writes the output to ``out``.
|
template <typename OutputIt,
|
||||||
*/
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, char>::value)>
|
||||||
template <typename OutputIt, typename Char,
|
auto vformat_to(OutputIt out, text_style ts, string_view fmt, format_args args)
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value)>
|
-> OutputIt {
|
||||||
OutputIt vformat_to(
|
auto&& buf = detail::get_buffer<char>(out);
|
||||||
OutputIt out, const text_style& ts, basic_string_view<Char> format_str,
|
detail::vformat_to(buf, ts, fmt, args);
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
|
||||||
auto&& buf = detail::get_buffer<Char>(out);
|
|
||||||
detail::vformat_to(buf, ts, format_str, args);
|
|
||||||
return detail::get_iterator(buf, out);
|
return detail::get_iterator(buf, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats arguments with the given text style, writes the result to the output
|
||||||
Formats arguments with the given text_style, writes the result to the output
|
* iterator `out` and returns the iterator past the end of the output range.
|
||||||
iterator ``out`` and returns the iterator past the end of the output range.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* std::vector<char> out;
|
||||||
std::vector<char> out;
|
* fmt::format_to(std::back_inserter(out),
|
||||||
fmt::format_to(std::back_inserter(out),
|
* fmt::emphasis::bold | fg(fmt::color::red), "{}", 42);
|
||||||
fmt::emphasis::bold | fg(fmt::color::red), "{}", 42);
|
*/
|
||||||
\endrst
|
template <typename OutputIt, typename... T,
|
||||||
*/
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, char>::value)>
|
||||||
template <typename OutputIt, typename S, typename... Args,
|
inline auto format_to(OutputIt out, text_style ts, format_string<T...> fmt,
|
||||||
bool enable = detail::is_output_iterator<OutputIt, char_t<S>>::value&&
|
T&&... args) -> OutputIt {
|
||||||
detail::is_string<S>::value>
|
return vformat_to(out, ts, fmt.str, vargs<T...>{{args...}});
|
||||||
inline auto format_to(OutputIt out, const text_style& ts, const S& format_str,
|
|
||||||
Args&&... args) ->
|
|
||||||
typename std::enable_if<enable, OutputIt>::type {
|
|
||||||
return vformat_to(out, ts, detail::to_string_view(format_str),
|
|
||||||
fmt::make_format_args<buffer_context<char_t<S>>>(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
|
|
@ -578,47 +586,44 @@ struct formatter<detail::styled_arg<T>, Char> : formatter<T, Char> {
|
||||||
auto format(const detail::styled_arg<T>& arg, FormatContext& ctx) const
|
auto format(const detail::styled_arg<T>& arg, FormatContext& ctx) const
|
||||||
-> decltype(ctx.out()) {
|
-> decltype(ctx.out()) {
|
||||||
const auto& ts = arg.style;
|
const auto& ts = arg.style;
|
||||||
const auto& value = arg.value;
|
|
||||||
auto out = ctx.out();
|
auto out = ctx.out();
|
||||||
|
|
||||||
bool has_style = false;
|
bool has_style = false;
|
||||||
if (ts.has_emphasis()) {
|
if (ts.has_emphasis()) {
|
||||||
has_style = true;
|
has_style = true;
|
||||||
auto emphasis = detail::make_emphasis<Char>(ts.get_emphasis());
|
auto emphasis = detail::make_emphasis<Char>(ts.get_emphasis());
|
||||||
out = std::copy(emphasis.begin(), emphasis.end(), out);
|
out = detail::copy<Char>(emphasis.begin(), emphasis.end(), out);
|
||||||
}
|
}
|
||||||
if (ts.has_foreground()) {
|
if (ts.has_foreground()) {
|
||||||
has_style = true;
|
has_style = true;
|
||||||
auto foreground =
|
auto foreground =
|
||||||
detail::make_foreground_color<Char>(ts.get_foreground());
|
detail::make_foreground_color<Char>(ts.get_foreground());
|
||||||
out = std::copy(foreground.begin(), foreground.end(), out);
|
out = detail::copy<Char>(foreground.begin(), foreground.end(), out);
|
||||||
}
|
}
|
||||||
if (ts.has_background()) {
|
if (ts.has_background()) {
|
||||||
has_style = true;
|
has_style = true;
|
||||||
auto background =
|
auto background =
|
||||||
detail::make_background_color<Char>(ts.get_background());
|
detail::make_background_color<Char>(ts.get_background());
|
||||||
out = std::copy(background.begin(), background.end(), out);
|
out = detail::copy<Char>(background.begin(), background.end(), out);
|
||||||
}
|
}
|
||||||
out = formatter<T, Char>::format(value, ctx);
|
out = formatter<T, Char>::format(arg.value, ctx);
|
||||||
if (has_style) {
|
if (has_style) {
|
||||||
auto reset_color = string_view("\x1b[0m");
|
auto reset_color = string_view("\x1b[0m");
|
||||||
out = std::copy(reset_color.begin(), reset_color.end(), out);
|
out = detail::copy<Char>(reset_color.begin(), reset_color.end(), out);
|
||||||
}
|
}
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Returns an argument that will be formatted using ANSI escape sequences,
|
||||||
Returns an argument that will be formatted using ANSI escape sequences,
|
* to be used in a formatting function.
|
||||||
to be used in a formatting function.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print("Elapsed time: {0:.2f} seconds",
|
||||||
fmt::print("Elapsed time: {0:.2f} seconds",
|
* fmt::styled(1.23, fmt::fg(fmt::color::green) |
|
||||||
fmt::styled(1.23, fmt::fg(fmt::color::green) |
|
* fmt::bg(fmt::color::blue)));
|
||||||
fmt::bg(fmt::color::blue)));
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename T>
|
template <typename T>
|
||||||
FMT_CONSTEXPR auto styled(const T& value, text_style ts)
|
FMT_CONSTEXPR auto styled(const T& value, text_style ts)
|
||||||
|
|
|
||||||
356
deps/fmt/include/fmt/compile.h
vendored
356
deps/fmt/include/fmt/compile.h
vendored
|
|
@ -8,16 +8,14 @@
|
||||||
#ifndef FMT_COMPILE_H_
|
#ifndef FMT_COMPILE_H_
|
||||||
#define FMT_COMPILE_H_
|
#define FMT_COMPILE_H_
|
||||||
|
|
||||||
|
#ifndef FMT_MODULE
|
||||||
|
# include <iterator> // std::back_inserter
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
namespace detail {
|
FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
template <typename Char, typename InputIt>
|
|
||||||
FMT_CONSTEXPR inline counting_iterator copy_str(InputIt begin, InputIt end,
|
|
||||||
counting_iterator it) {
|
|
||||||
return it + (end - begin);
|
|
||||||
}
|
|
||||||
|
|
||||||
// A compile-time string which is compiled into fast formatting code.
|
// A compile-time string which is compiled into fast formatting code.
|
||||||
class compiled_string {};
|
class compiled_string {};
|
||||||
|
|
@ -26,48 +24,58 @@ template <typename S>
|
||||||
struct is_compiled_string : std::is_base_of<compiled_string, S> {};
|
struct is_compiled_string : std::is_base_of<compiled_string, S> {};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Converts a string literal `s` into a format string that will be parsed at
|
||||||
Converts a string literal *s* into a format string that will be parsed at
|
* compile time and converted into efficient formatting code. Requires C++17
|
||||||
compile time and converted into efficient formatting code. Requires C++17
|
* `constexpr if` compiler support.
|
||||||
``constexpr if`` compiler support.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* // Converts 42 into std::string using the most efficient method and no
|
||||||
// Converts 42 into std::string using the most efficient method and no
|
* // runtime format string processing.
|
||||||
// runtime format string processing.
|
* std::string s = fmt::format(FMT_COMPILE("{}"), 42);
|
||||||
std::string s = fmt::format(FMT_COMPILE("{}"), 42);
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
||||||
# define FMT_COMPILE(s) \
|
# define FMT_COMPILE(s) FMT_STRING_IMPL(s, fmt::compiled_string)
|
||||||
FMT_STRING_IMPL(s, fmt::detail::compiled_string, explicit)
|
|
||||||
#else
|
#else
|
||||||
# define FMT_COMPILE(s) FMT_STRING(s)
|
# define FMT_COMPILE(s) FMT_STRING(s)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a string literal into a format string that will be parsed at
|
||||||
|
* compile time and converted into efficient formatting code. Requires support
|
||||||
|
* for class types in constant template parameters (a C++20 feature).
|
||||||
|
*
|
||||||
|
* **Example**:
|
||||||
|
*
|
||||||
|
* // Converts 42 into std::string using the most efficient method and no
|
||||||
|
* // runtime format string processing.
|
||||||
|
* using namespace fmt::literals;
|
||||||
|
* std::string s = fmt::format("{}"_cf, 42);
|
||||||
|
*/
|
||||||
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
||||||
template <typename Char, size_t N,
|
inline namespace literals {
|
||||||
fmt::detail_exported::fixed_string<Char, N> Str>
|
template <detail::fixed_string Str> constexpr auto operator""_cf() {
|
||||||
struct udl_compiled_string : compiled_string {
|
return FMT_COMPILE(Str.data);
|
||||||
using char_type = Char;
|
}
|
||||||
explicit constexpr operator basic_string_view<char_type>() const {
|
} // namespace literals
|
||||||
return {Str.data, N - 1};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
FMT_END_EXPORT
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
template <typename T, typename... Tail>
|
template <typename T, typename... Tail>
|
||||||
const T& first(const T& value, const Tail&...) {
|
constexpr auto first(const T& value, const Tail&...) -> const T& {
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
||||||
template <typename... Args> struct type_list {};
|
template <typename... T> struct type_list {};
|
||||||
|
|
||||||
// Returns a reference to the argument at index N from [first, rest...].
|
// Returns a reference to the argument at index N from [first, rest...].
|
||||||
template <int N, typename T, typename... Args>
|
template <int N, typename T, typename... Args>
|
||||||
constexpr const auto& get([[maybe_unused]] const T& first,
|
constexpr auto get([[maybe_unused]] const T& first,
|
||||||
[[maybe_unused]] const Args&... rest) {
|
[[maybe_unused]] const Args&... rest) -> const auto& {
|
||||||
static_assert(N < 1 + sizeof...(Args), "index is out of bounds");
|
static_assert(N < 1 + sizeof...(Args), "index is out of bounds");
|
||||||
if constexpr (N == 0)
|
if constexpr (N == 0)
|
||||||
return first;
|
return first;
|
||||||
|
|
@ -75,9 +83,32 @@ constexpr const auto& get([[maybe_unused]] const T& first,
|
||||||
return detail::get<N - 1>(rest...);
|
return detail::get<N - 1>(rest...);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
||||||
|
template <int N, typename T, typename... Args, typename Char>
|
||||||
|
constexpr auto get_arg_index_by_name(basic_string_view<Char> name) -> int {
|
||||||
|
if constexpr (is_static_named_arg<T>()) {
|
||||||
|
if (name == T::name) return N;
|
||||||
|
}
|
||||||
|
if constexpr (sizeof...(Args) > 0)
|
||||||
|
return get_arg_index_by_name<N + 1, Args...>(name);
|
||||||
|
(void)name; // Workaround an MSVC bug about "unused" parameter.
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
# endif
|
||||||
|
|
||||||
|
template <typename... Args, typename Char>
|
||||||
|
FMT_CONSTEXPR auto get_arg_index_by_name(basic_string_view<Char> name) -> int {
|
||||||
|
# if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
||||||
|
if constexpr (sizeof...(Args) > 0)
|
||||||
|
return get_arg_index_by_name<0, Args...>(name);
|
||||||
|
# endif
|
||||||
|
(void)name;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Char, typename... Args>
|
template <typename Char, typename... Args>
|
||||||
constexpr int get_arg_index_by_name(basic_string_view<Char> name,
|
constexpr auto get_arg_index_by_name(basic_string_view<Char> name,
|
||||||
type_list<Args...>) {
|
type_list<Args...>) -> int {
|
||||||
return get_arg_index_by_name<Args...>(name);
|
return get_arg_index_by_name<Args...>(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -97,8 +128,8 @@ template <typename Char> struct text {
|
||||||
basic_string_view<Char> data;
|
basic_string_view<Char> data;
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&...) const {
|
constexpr auto format(OutputIt out, const T&...) const -> OutputIt {
|
||||||
return write<Char>(out, data);
|
return write<Char>(out, data);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -107,8 +138,8 @@ template <typename Char>
|
||||||
struct is_compiled_format<text<Char>> : std::true_type {};
|
struct is_compiled_format<text<Char>> : std::true_type {};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
constexpr text<Char> make_text(basic_string_view<Char> s, size_t pos,
|
constexpr auto make_text(basic_string_view<Char> s, size_t pos, size_t size)
|
||||||
size_t size) {
|
-> text<Char> {
|
||||||
return {{&s[pos], size}};
|
return {{&s[pos], size}};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -116,8 +147,8 @@ template <typename Char> struct code_unit {
|
||||||
Char value;
|
Char value;
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&...) const {
|
constexpr auto format(OutputIt out, const T&...) const -> OutputIt {
|
||||||
*out++ = value;
|
*out++ = value;
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
@ -125,7 +156,7 @@ template <typename Char> struct code_unit {
|
||||||
|
|
||||||
// This ensures that the argument type is convertible to `const T&`.
|
// This ensures that the argument type is convertible to `const T&`.
|
||||||
template <typename T, int N, typename... Args>
|
template <typename T, int N, typename... Args>
|
||||||
constexpr const T& get_arg_checked(const Args&... args) {
|
constexpr auto get_arg_checked(const Args&... args) -> const T& {
|
||||||
const auto& arg = detail::get<N>(args...);
|
const auto& arg = detail::get<N>(args...);
|
||||||
if constexpr (detail::is_named_arg<remove_cvref_t<decltype(arg)>>()) {
|
if constexpr (detail::is_named_arg<remove_cvref_t<decltype(arg)>>()) {
|
||||||
return arg.value;
|
return arg.value;
|
||||||
|
|
@ -138,17 +169,18 @@ template <typename Char>
|
||||||
struct is_compiled_format<code_unit<Char>> : std::true_type {};
|
struct is_compiled_format<code_unit<Char>> : std::true_type {};
|
||||||
|
|
||||||
// A replacement field that refers to argument N.
|
// A replacement field that refers to argument N.
|
||||||
template <typename Char, typename T, int N> struct field {
|
template <typename Char, typename V, int N> struct field {
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&... args) const {
|
constexpr auto format(OutputIt out, const T&... args) const -> OutputIt {
|
||||||
const T& arg = get_arg_checked<T, N>(args...);
|
const V& arg = get_arg_checked<V, N>(args...);
|
||||||
if constexpr (std::is_convertible_v<T, basic_string_view<Char>>) {
|
if constexpr (std::is_convertible<V, basic_string_view<Char>>::value) {
|
||||||
auto s = basic_string_view<Char>(arg);
|
auto s = basic_string_view<Char>(arg);
|
||||||
return copy_str<Char>(s.begin(), s.end(), out);
|
return copy<Char>(s.begin(), s.end(), out);
|
||||||
|
} else {
|
||||||
|
return write<Char>(out, arg);
|
||||||
}
|
}
|
||||||
return write<Char>(out, arg);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -161,10 +193,10 @@ template <typename Char> struct runtime_named_field {
|
||||||
basic_string_view<Char> name;
|
basic_string_view<Char> name;
|
||||||
|
|
||||||
template <typename OutputIt, typename T>
|
template <typename OutputIt, typename T>
|
||||||
constexpr static bool try_format_argument(
|
constexpr static auto try_format_argument(
|
||||||
OutputIt& out,
|
OutputIt& out,
|
||||||
// [[maybe_unused]] due to unused-but-set-parameter warning in GCC 7,8,9
|
// [[maybe_unused]] due to unused-but-set-parameter warning in GCC 7,8,9
|
||||||
[[maybe_unused]] basic_string_view<Char> arg_name, const T& arg) {
|
[[maybe_unused]] basic_string_view<Char> arg_name, const T& arg) -> bool {
|
||||||
if constexpr (is_named_arg<typename std::remove_cv<T>::type>::value) {
|
if constexpr (is_named_arg<typename std::remove_cv<T>::type>::value) {
|
||||||
if (arg_name == arg.name) {
|
if (arg_name == arg.name) {
|
||||||
out = write<Char>(out, arg.value);
|
out = write<Char>(out, arg.value);
|
||||||
|
|
@ -174,8 +206,8 @@ template <typename Char> struct runtime_named_field {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&... args) const {
|
constexpr auto format(OutputIt out, const T&... args) const -> OutputIt {
|
||||||
bool found = (try_format_argument(out, name, args) || ...);
|
bool found = (try_format_argument(out, name, args) || ...);
|
||||||
if (!found) {
|
if (!found) {
|
||||||
FMT_THROW(format_error("argument with specified name is not found"));
|
FMT_THROW(format_error("argument with specified name is not found"));
|
||||||
|
|
@ -188,17 +220,17 @@ template <typename Char>
|
||||||
struct is_compiled_format<runtime_named_field<Char>> : std::true_type {};
|
struct is_compiled_format<runtime_named_field<Char>> : std::true_type {};
|
||||||
|
|
||||||
// A replacement field that refers to argument N and has format specifiers.
|
// A replacement field that refers to argument N and has format specifiers.
|
||||||
template <typename Char, typename T, int N> struct spec_field {
|
template <typename Char, typename V, int N> struct spec_field {
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
formatter<T, Char> fmt;
|
formatter<V, Char> fmt;
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr FMT_INLINE OutputIt format(OutputIt out,
|
constexpr FMT_INLINE auto format(OutputIt out, const T&... args) const
|
||||||
const Args&... args) const {
|
-> OutputIt {
|
||||||
const auto& vargs =
|
const auto& vargs =
|
||||||
fmt::make_format_args<basic_format_context<OutputIt, Char>>(args...);
|
fmt::make_format_args<basic_format_context<OutputIt, Char>>(args...);
|
||||||
basic_format_context<OutputIt, Char> ctx(out, vargs);
|
basic_format_context<OutputIt, Char> ctx(out, vargs);
|
||||||
return fmt.format(get_arg_checked<T, N>(args...), ctx);
|
return fmt.format(get_arg_checked<V, N>(args...), ctx);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -210,8 +242,8 @@ template <typename L, typename R> struct concat {
|
||||||
R rhs;
|
R rhs;
|
||||||
using char_type = typename L::char_type;
|
using char_type = typename L::char_type;
|
||||||
|
|
||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... T>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&... args) const {
|
constexpr auto format(OutputIt out, const T&... args) const -> OutputIt {
|
||||||
out = lhs.format(out, args...);
|
out = lhs.format(out, args...);
|
||||||
return rhs.format(out, args...);
|
return rhs.format(out, args...);
|
||||||
}
|
}
|
||||||
|
|
@ -221,14 +253,14 @@ template <typename L, typename R>
|
||||||
struct is_compiled_format<concat<L, R>> : std::true_type {};
|
struct is_compiled_format<concat<L, R>> : std::true_type {};
|
||||||
|
|
||||||
template <typename L, typename R>
|
template <typename L, typename R>
|
||||||
constexpr concat<L, R> make_concat(L lhs, R rhs) {
|
constexpr auto make_concat(L lhs, R rhs) -> concat<L, R> {
|
||||||
return {lhs, rhs};
|
return {lhs, rhs};
|
||||||
}
|
}
|
||||||
|
|
||||||
struct unknown_format {};
|
struct unknown_format {};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
constexpr size_t parse_text(basic_string_view<Char> str, size_t pos) {
|
constexpr auto parse_text(basic_string_view<Char> str, size_t pos) -> size_t {
|
||||||
for (size_t size = str.size(); pos != size; ++pos) {
|
for (size_t size = str.size(); pos != size; ++pos) {
|
||||||
if (str[pos] == '{' || str[pos] == '}') break;
|
if (str[pos] == '{' || str[pos] == '}') break;
|
||||||
}
|
}
|
||||||
|
|
@ -236,13 +268,12 @@ constexpr size_t parse_text(basic_string_view<Char> str, size_t pos) {
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Args, size_t POS, int ID, typename S>
|
template <typename Args, size_t POS, int ID, typename S>
|
||||||
constexpr auto compile_format_string(S format_str);
|
constexpr auto compile_format_string(S fmt);
|
||||||
|
|
||||||
template <typename Args, size_t POS, int ID, typename T, typename S>
|
template <typename Args, size_t POS, int ID, typename T, typename S>
|
||||||
constexpr auto parse_tail(T head, S format_str) {
|
constexpr auto parse_tail(T head, S fmt) {
|
||||||
if constexpr (POS !=
|
if constexpr (POS != basic_string_view<typename S::char_type>(fmt).size()) {
|
||||||
basic_string_view<typename S::char_type>(format_str).size()) {
|
constexpr auto tail = compile_format_string<Args, POS, ID>(fmt);
|
||||||
constexpr auto tail = compile_format_string<Args, POS, ID>(format_str);
|
|
||||||
if constexpr (std::is_same<remove_cvref_t<decltype(tail)>,
|
if constexpr (std::is_same<remove_cvref_t<decltype(tail)>,
|
||||||
unknown_format>())
|
unknown_format>())
|
||||||
return tail;
|
return tail;
|
||||||
|
|
@ -262,8 +293,8 @@ template <typename T, typename Char> struct parse_specs_result {
|
||||||
enum { manual_indexing_id = -1 };
|
enum { manual_indexing_id = -1 };
|
||||||
|
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
constexpr parse_specs_result<T, Char> parse_specs(basic_string_view<Char> str,
|
constexpr auto parse_specs(basic_string_view<Char> str, size_t pos,
|
||||||
size_t pos, int next_arg_id) {
|
int next_arg_id) -> parse_specs_result<T, Char> {
|
||||||
str.remove_prefix(pos);
|
str.remove_prefix(pos);
|
||||||
auto ctx =
|
auto ctx =
|
||||||
compile_parse_context<Char>(str, max_value<int>(), nullptr, next_arg_id);
|
compile_parse_context<Char>(str, max_value<int>(), nullptr, next_arg_id);
|
||||||
|
|
@ -274,32 +305,36 @@ constexpr parse_specs_result<T, Char> parse_specs(basic_string_view<Char> str,
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char> struct arg_id_handler {
|
template <typename Char> struct arg_id_handler {
|
||||||
|
arg_id_kind kind;
|
||||||
arg_ref<Char> arg_id;
|
arg_ref<Char> arg_id;
|
||||||
|
|
||||||
constexpr int on_auto() {
|
constexpr auto on_auto() -> int {
|
||||||
FMT_ASSERT(false, "handler cannot be used with automatic indexing");
|
FMT_ASSERT(false, "handler cannot be used with automatic indexing");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
constexpr int on_index(int id) {
|
constexpr auto on_index(int id) -> int {
|
||||||
|
kind = arg_id_kind::index;
|
||||||
arg_id = arg_ref<Char>(id);
|
arg_id = arg_ref<Char>(id);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
constexpr int on_name(basic_string_view<Char> id) {
|
constexpr auto on_name(basic_string_view<Char> id) -> int {
|
||||||
|
kind = arg_id_kind::name;
|
||||||
arg_id = arg_ref<Char>(id);
|
arg_id = arg_ref<Char>(id);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char> struct parse_arg_id_result {
|
template <typename Char> struct parse_arg_id_result {
|
||||||
|
arg_id_kind kind;
|
||||||
arg_ref<Char> arg_id;
|
arg_ref<Char> arg_id;
|
||||||
const Char* arg_id_end;
|
const Char* arg_id_end;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <int ID, typename Char>
|
template <int ID, typename Char>
|
||||||
constexpr auto parse_arg_id(const Char* begin, const Char* end) {
|
constexpr auto parse_arg_id(const Char* begin, const Char* end) {
|
||||||
auto handler = arg_id_handler<Char>{arg_ref<Char>{}};
|
auto handler = arg_id_handler<Char>{arg_id_kind::none, arg_ref<Char>{}};
|
||||||
auto arg_id_end = parse_arg_id(begin, end, handler);
|
auto arg_id_end = parse_arg_id(begin, end, handler);
|
||||||
return parse_arg_id_result<Char>{handler.arg_id, arg_id_end};
|
return parse_arg_id_result<Char>{handler.kind, handler.arg_id, arg_id_end};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, typename Enable = void> struct field_type {
|
template <typename T, typename Enable = void> struct field_type {
|
||||||
|
|
@ -313,14 +348,13 @@ struct field_type<T, enable_if_t<detail::is_named_arg<T>::value>> {
|
||||||
|
|
||||||
template <typename T, typename Args, size_t END_POS, int ARG_INDEX, int NEXT_ID,
|
template <typename T, typename Args, size_t END_POS, int ARG_INDEX, int NEXT_ID,
|
||||||
typename S>
|
typename S>
|
||||||
constexpr auto parse_replacement_field_then_tail(S format_str) {
|
constexpr auto parse_replacement_field_then_tail(S fmt) {
|
||||||
using char_type = typename S::char_type;
|
using char_type = typename S::char_type;
|
||||||
constexpr auto str = basic_string_view<char_type>(format_str);
|
constexpr auto str = basic_string_view<char_type>(fmt);
|
||||||
constexpr char_type c = END_POS != str.size() ? str[END_POS] : char_type();
|
constexpr char_type c = END_POS != str.size() ? str[END_POS] : char_type();
|
||||||
if constexpr (c == '}') {
|
if constexpr (c == '}') {
|
||||||
return parse_tail<Args, END_POS + 1, NEXT_ID>(
|
return parse_tail<Args, END_POS + 1, NEXT_ID>(
|
||||||
field<char_type, typename field_type<T>::type, ARG_INDEX>(),
|
field<char_type, typename field_type<T>::type, ARG_INDEX>(), fmt);
|
||||||
format_str);
|
|
||||||
} else if constexpr (c != ':') {
|
} else if constexpr (c != ':') {
|
||||||
FMT_THROW(format_error("expected ':'"));
|
FMT_THROW(format_error("expected ':'"));
|
||||||
} else {
|
} else {
|
||||||
|
|
@ -333,7 +367,7 @@ constexpr auto parse_replacement_field_then_tail(S format_str) {
|
||||||
return parse_tail<Args, result.end + 1, result.next_arg_id>(
|
return parse_tail<Args, result.end + 1, result.next_arg_id>(
|
||||||
spec_field<char_type, typename field_type<T>::type, ARG_INDEX>{
|
spec_field<char_type, typename field_type<T>::type, ARG_INDEX>{
|
||||||
result.fmt},
|
result.fmt},
|
||||||
format_str);
|
fmt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -341,22 +375,21 @@ constexpr auto parse_replacement_field_then_tail(S format_str) {
|
||||||
// Compiles a non-empty format string and returns the compiled representation
|
// Compiles a non-empty format string and returns the compiled representation
|
||||||
// or unknown_format() on unrecognized input.
|
// or unknown_format() on unrecognized input.
|
||||||
template <typename Args, size_t POS, int ID, typename S>
|
template <typename Args, size_t POS, int ID, typename S>
|
||||||
constexpr auto compile_format_string(S format_str) {
|
constexpr auto compile_format_string(S fmt) {
|
||||||
using char_type = typename S::char_type;
|
using char_type = typename S::char_type;
|
||||||
constexpr auto str = basic_string_view<char_type>(format_str);
|
constexpr auto str = basic_string_view<char_type>(fmt);
|
||||||
if constexpr (str[POS] == '{') {
|
if constexpr (str[POS] == '{') {
|
||||||
if constexpr (POS + 1 == str.size())
|
if constexpr (POS + 1 == str.size())
|
||||||
FMT_THROW(format_error("unmatched '{' in format string"));
|
FMT_THROW(format_error("unmatched '{' in format string"));
|
||||||
if constexpr (str[POS + 1] == '{') {
|
if constexpr (str[POS + 1] == '{') {
|
||||||
return parse_tail<Args, POS + 2, ID>(make_text(str, POS, 1), format_str);
|
return parse_tail<Args, POS + 2, ID>(make_text(str, POS, 1), fmt);
|
||||||
} else if constexpr (str[POS + 1] == '}' || str[POS + 1] == ':') {
|
} else if constexpr (str[POS + 1] == '}' || str[POS + 1] == ':') {
|
||||||
static_assert(ID != manual_indexing_id,
|
static_assert(ID != manual_indexing_id,
|
||||||
"cannot switch from manual to automatic argument indexing");
|
"cannot switch from manual to automatic argument indexing");
|
||||||
constexpr auto next_id =
|
constexpr auto next_id =
|
||||||
ID != manual_indexing_id ? ID + 1 : manual_indexing_id;
|
ID != manual_indexing_id ? ID + 1 : manual_indexing_id;
|
||||||
return parse_replacement_field_then_tail<get_type<ID, Args>, Args,
|
return parse_replacement_field_then_tail<get_type<ID, Args>, Args,
|
||||||
POS + 1, ID, next_id>(
|
POS + 1, ID, next_id>(fmt);
|
||||||
format_str);
|
|
||||||
} else {
|
} else {
|
||||||
constexpr auto arg_id_result =
|
constexpr auto arg_id_result =
|
||||||
parse_arg_id<ID>(str.data() + POS + 1, str.data() + str.size());
|
parse_arg_id<ID>(str.data() + POS + 1, str.data() + str.size());
|
||||||
|
|
@ -364,28 +397,27 @@ constexpr auto compile_format_string(S format_str) {
|
||||||
constexpr char_type c =
|
constexpr char_type c =
|
||||||
arg_id_end_pos != str.size() ? str[arg_id_end_pos] : char_type();
|
arg_id_end_pos != str.size() ? str[arg_id_end_pos] : char_type();
|
||||||
static_assert(c == '}' || c == ':', "missing '}' in format string");
|
static_assert(c == '}' || c == ':', "missing '}' in format string");
|
||||||
if constexpr (arg_id_result.arg_id.kind == arg_id_kind::index) {
|
if constexpr (arg_id_result.kind == arg_id_kind::index) {
|
||||||
static_assert(
|
static_assert(
|
||||||
ID == manual_indexing_id || ID == 0,
|
ID == manual_indexing_id || ID == 0,
|
||||||
"cannot switch from automatic to manual argument indexing");
|
"cannot switch from automatic to manual argument indexing");
|
||||||
constexpr auto arg_index = arg_id_result.arg_id.val.index;
|
constexpr auto arg_index = arg_id_result.arg_id.index;
|
||||||
return parse_replacement_field_then_tail<get_type<arg_index, Args>,
|
return parse_replacement_field_then_tail<get_type<arg_index, Args>,
|
||||||
Args, arg_id_end_pos,
|
Args, arg_id_end_pos,
|
||||||
arg_index, manual_indexing_id>(
|
arg_index, manual_indexing_id>(
|
||||||
format_str);
|
fmt);
|
||||||
} else if constexpr (arg_id_result.arg_id.kind == arg_id_kind::name) {
|
} else if constexpr (arg_id_result.kind == arg_id_kind::name) {
|
||||||
constexpr auto arg_index =
|
constexpr auto arg_index =
|
||||||
get_arg_index_by_name(arg_id_result.arg_id.val.name, Args{});
|
get_arg_index_by_name(arg_id_result.arg_id.name, Args{});
|
||||||
if constexpr (arg_index >= 0) {
|
if constexpr (arg_index >= 0) {
|
||||||
constexpr auto next_id =
|
constexpr auto next_id =
|
||||||
ID != manual_indexing_id ? ID + 1 : manual_indexing_id;
|
ID != manual_indexing_id ? ID + 1 : manual_indexing_id;
|
||||||
return parse_replacement_field_then_tail<
|
return parse_replacement_field_then_tail<
|
||||||
decltype(get_type<arg_index, Args>::value), Args, arg_id_end_pos,
|
decltype(get_type<arg_index, Args>::value), Args, arg_id_end_pos,
|
||||||
arg_index, next_id>(format_str);
|
arg_index, next_id>(fmt);
|
||||||
} else if constexpr (c == '}') {
|
} else if constexpr (c == '}') {
|
||||||
return parse_tail<Args, arg_id_end_pos + 1, ID>(
|
return parse_tail<Args, arg_id_end_pos + 1, ID>(
|
||||||
runtime_named_field<char_type>{arg_id_result.arg_id.val.name},
|
runtime_named_field<char_type>{arg_id_result.arg_id.name}, fmt);
|
||||||
format_str);
|
|
||||||
} else if constexpr (c == ':') {
|
} else if constexpr (c == ':') {
|
||||||
return unknown_format(); // no type info for specs parsing
|
return unknown_format(); // no type info for specs parsing
|
||||||
}
|
}
|
||||||
|
|
@ -394,29 +426,26 @@ constexpr auto compile_format_string(S format_str) {
|
||||||
} else if constexpr (str[POS] == '}') {
|
} else if constexpr (str[POS] == '}') {
|
||||||
if constexpr (POS + 1 == str.size())
|
if constexpr (POS + 1 == str.size())
|
||||||
FMT_THROW(format_error("unmatched '}' in format string"));
|
FMT_THROW(format_error("unmatched '}' in format string"));
|
||||||
return parse_tail<Args, POS + 2, ID>(make_text(str, POS, 1), format_str);
|
return parse_tail<Args, POS + 2, ID>(make_text(str, POS, 1), fmt);
|
||||||
} else {
|
} else {
|
||||||
constexpr auto end = parse_text(str, POS + 1);
|
constexpr auto end = parse_text(str, POS + 1);
|
||||||
if constexpr (end - POS > 1) {
|
if constexpr (end - POS > 1) {
|
||||||
return parse_tail<Args, end, ID>(make_text(str, POS, end - POS),
|
return parse_tail<Args, end, ID>(make_text(str, POS, end - POS), fmt);
|
||||||
format_str);
|
|
||||||
} else {
|
} else {
|
||||||
return parse_tail<Args, end, ID>(code_unit<char_type>{str[POS]},
|
return parse_tail<Args, end, ID>(code_unit<char_type>{str[POS]}, fmt);
|
||||||
format_str);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... Args, typename S,
|
template <typename... Args, typename S,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
constexpr auto compile(S format_str) {
|
constexpr auto compile(S fmt) {
|
||||||
constexpr auto str = basic_string_view<typename S::char_type>(format_str);
|
constexpr auto str = basic_string_view<typename S::char_type>(fmt);
|
||||||
if constexpr (str.size() == 0) {
|
if constexpr (str.size() == 0) {
|
||||||
return detail::make_text(str, 0, 0);
|
return detail::make_text(str, 0, 0);
|
||||||
} else {
|
} else {
|
||||||
constexpr auto result =
|
constexpr auto result =
|
||||||
detail::compile_format_string<detail::type_list<Args...>, 0, 0>(
|
detail::compile_format_string<detail::type_list<Args...>, 0, 0>(fmt);
|
||||||
format_str);
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -427,27 +456,28 @@ FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
#if defined(__cpp_if_constexpr) && defined(__cpp_return_type_deduction)
|
||||||
|
|
||||||
template <typename CompiledFormat, typename... Args,
|
template <typename CompiledFormat, typename... T,
|
||||||
typename Char = typename CompiledFormat::char_type,
|
typename Char = typename CompiledFormat::char_type,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_format<CompiledFormat>::value)>
|
FMT_ENABLE_IF(detail::is_compiled_format<CompiledFormat>::value)>
|
||||||
FMT_INLINE std::basic_string<Char> format(const CompiledFormat& cf,
|
FMT_INLINE FMT_CONSTEXPR_STRING auto format(const CompiledFormat& cf,
|
||||||
const Args&... args) {
|
const T&... args)
|
||||||
|
-> std::basic_string<Char> {
|
||||||
auto s = std::basic_string<Char>();
|
auto s = std::basic_string<Char>();
|
||||||
cf.format(std::back_inserter(s), args...);
|
cf.format(std::back_inserter(s), args...);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename CompiledFormat, typename... Args,
|
template <typename OutputIt, typename CompiledFormat, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_format<CompiledFormat>::value)>
|
FMT_ENABLE_IF(detail::is_compiled_format<CompiledFormat>::value)>
|
||||||
constexpr FMT_INLINE OutputIt format_to(OutputIt out, const CompiledFormat& cf,
|
constexpr FMT_INLINE auto format_to(OutputIt out, const CompiledFormat& cf,
|
||||||
const Args&... args) {
|
const T&... args) -> OutputIt {
|
||||||
return cf.format(out, args...);
|
return cf.format(out, args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
FMT_INLINE std::basic_string<typename S::char_type> format(const S&,
|
FMT_INLINE FMT_CONSTEXPR_STRING auto format(const S&, T&&... args)
|
||||||
Args&&... args) {
|
-> std::basic_string<typename S::char_type> {
|
||||||
if constexpr (std::is_same<typename S::char_type, char>::value) {
|
if constexpr (std::is_same<typename S::char_type, char>::value) {
|
||||||
constexpr auto str = basic_string_view<typename S::char_type>(S());
|
constexpr auto str = basic_string_view<typename S::char_type>(S());
|
||||||
if constexpr (str.size() == 2 && str[0] == '{' && str[1] == '}') {
|
if constexpr (str.size() == 2 && str[0] == '{' && str[1] == '}') {
|
||||||
|
|
@ -460,73 +490,97 @@ FMT_INLINE std::basic_string<typename S::char_type> format(const S&,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
constexpr auto compiled = detail::compile<Args...>(S());
|
constexpr auto compiled = detail::compile<T...>(S());
|
||||||
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
||||||
detail::unknown_format>()) {
|
detail::unknown_format>()) {
|
||||||
return fmt::format(
|
return fmt::format(
|
||||||
static_cast<basic_string_view<typename S::char_type>>(S()),
|
static_cast<basic_string_view<typename S::char_type>>(S()),
|
||||||
std::forward<Args>(args)...);
|
std::forward<T>(args)...);
|
||||||
} else {
|
} else {
|
||||||
return fmt::format(compiled, std::forward<Args>(args)...);
|
return fmt::format(compiled, std::forward<T>(args)...);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... Args,
|
template <typename OutputIt, typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
FMT_CONSTEXPR OutputIt format_to(OutputIt out, const S&, Args&&... args) {
|
FMT_CONSTEXPR auto format_to(OutputIt out, const S&, T&&... args) -> OutputIt {
|
||||||
constexpr auto compiled = detail::compile<Args...>(S());
|
constexpr auto compiled = detail::compile<T...>(S());
|
||||||
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
||||||
detail::unknown_format>()) {
|
detail::unknown_format>()) {
|
||||||
return fmt::format_to(
|
return fmt::format_to(
|
||||||
out, static_cast<basic_string_view<typename S::char_type>>(S()),
|
out, static_cast<basic_string_view<typename S::char_type>>(S()),
|
||||||
std::forward<Args>(args)...);
|
std::forward<T>(args)...);
|
||||||
} else {
|
} else {
|
||||||
return fmt::format_to(out, compiled, std::forward<Args>(args)...);
|
return fmt::format_to(out, compiled, std::forward<T>(args)...);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... Args,
|
template <typename OutputIt, typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
format_to_n_result<OutputIt> format_to_n(OutputIt out, size_t n,
|
auto format_to_n(OutputIt out, size_t n, const S& fmt, T&&... args)
|
||||||
const S& format_str, Args&&... args) {
|
-> format_to_n_result<OutputIt> {
|
||||||
using traits = detail::fixed_buffer_traits;
|
using traits = detail::fixed_buffer_traits;
|
||||||
auto buf = detail::iterator_buffer<OutputIt, char, traits>(out, n);
|
auto buf = detail::iterator_buffer<OutputIt, char, traits>(out, n);
|
||||||
format_to(std::back_inserter(buf), format_str, std::forward<Args>(args)...);
|
fmt::format_to(std::back_inserter(buf), fmt, std::forward<T>(args)...);
|
||||||
return {buf.out(), buf.count()};
|
return {buf.out(), buf.count()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
FMT_CONSTEXPR20 size_t formatted_size(const S& format_str,
|
FMT_CONSTEXPR20 auto formatted_size(const S& fmt, T&&... args) -> size_t {
|
||||||
const Args&... args) {
|
auto buf = detail::counting_buffer<>();
|
||||||
return fmt::format_to(detail::counting_iterator(), format_str, args...)
|
fmt::format_to(appender(buf), fmt, std::forward<T>(args)...);
|
||||||
.count();
|
return buf.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
void print(std::FILE* f, const S& format_str, const Args&... args) {
|
void print(std::FILE* f, const S& fmt, T&&... args) {
|
||||||
memory_buffer buffer;
|
auto buf = memory_buffer();
|
||||||
fmt::format_to(std::back_inserter(buffer), format_str, args...);
|
fmt::format_to(appender(buf), fmt, std::forward<T>(args)...);
|
||||||
detail::print(f, {buffer.data(), buffer.size()});
|
detail::print(f, {buf.data(), buf.size()});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
void print(const S& format_str, const Args&... args) {
|
void print(const S& fmt, T&&... args) {
|
||||||
print(stdout, format_str, args...);
|
print(stdout, fmt, std::forward<T>(args)...);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
template <size_t N> class static_format_result {
|
||||||
inline namespace literals {
|
private:
|
||||||
template <detail_exported::fixed_string Str> constexpr auto operator""_cf() {
|
char data[N];
|
||||||
using char_t = remove_cvref_t<decltype(Str.data[0])>;
|
|
||||||
return detail::udl_compiled_string<char_t, sizeof(Str.data) / sizeof(char_t),
|
public:
|
||||||
Str>();
|
template <typename S, typename... T,
|
||||||
}
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
} // namespace literals
|
explicit FMT_CONSTEXPR static_format_result(const S& fmt, T&&... args) {
|
||||||
#endif
|
*fmt::format_to(data, fmt, std::forward<T>(args)...) = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
auto str() const -> fmt::string_view { return {data, N - 1}; }
|
||||||
|
auto c_str() const -> const char* { return data; }
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formats arguments according to the format string `fmt_str` and produces
|
||||||
|
* a string of the exact required size at compile time. Both the format string
|
||||||
|
* and the arguments must be compile-time expressions.
|
||||||
|
*
|
||||||
|
* The resulting string can be accessed as a C string via `c_str()` or as
|
||||||
|
* a `fmt::string_view` via `str()`.
|
||||||
|
*
|
||||||
|
* **Example**:
|
||||||
|
*
|
||||||
|
* // Produces the static string "42" at compile time.
|
||||||
|
* static constexpr auto result = FMT_STATIC_FORMAT("{}", 42);
|
||||||
|
* const char* s = result.c_str();
|
||||||
|
*/
|
||||||
|
#define FMT_STATIC_FORMAT(fmt_str, ...) \
|
||||||
|
fmt::static_format_result< \
|
||||||
|
fmt::formatted_size(FMT_COMPILE(fmt_str), __VA_ARGS__) + 1>( \
|
||||||
|
FMT_COMPILE(fmt_str), __VA_ARGS__)
|
||||||
|
|
||||||
FMT_END_EXPORT
|
FMT_END_EXPORT
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
|
||||||
2925
deps/fmt/include/fmt/core.h
vendored
2925
deps/fmt/include/fmt/core.h
vendored
File diff suppressed because it is too large
Load diff
574
deps/fmt/include/fmt/format-inl.h
vendored
574
deps/fmt/include/fmt/format-inl.h
vendored
|
|
@ -8,37 +8,67 @@
|
||||||
#ifndef FMT_FORMAT_INL_H_
|
#ifndef FMT_FORMAT_INL_H_
|
||||||
#define FMT_FORMAT_INL_H_
|
#define FMT_FORMAT_INL_H_
|
||||||
|
|
||||||
#include <algorithm>
|
#ifndef FMT_MODULE
|
||||||
#include <cerrno> // errno
|
# include <algorithm>
|
||||||
#include <climits>
|
# include <cerrno> // errno
|
||||||
#include <cmath>
|
# include <climits>
|
||||||
#include <exception>
|
# include <cmath>
|
||||||
|
# include <exception>
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
|
||||||
# include <locale>
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32) && !defined(FMT_USE_WRITE_CONSOLE)
|
||||||
# include <io.h> // _isatty
|
# include <io.h> // _isatty
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
#if FMT_USE_LOCALE && !defined(FMT_MODULE)
|
||||||
namespace detail {
|
# include <locale>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef FMT_FUNC
|
||||||
|
# define FMT_FUNC
|
||||||
|
#endif
|
||||||
|
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
|
#ifndef FMT_CUSTOM_ASSERT_FAIL
|
||||||
FMT_FUNC void assert_fail(const char* file, int line, const char* message) {
|
FMT_FUNC void assert_fail(const char* file, int line, const char* message) {
|
||||||
// Use unchecked std::fprintf to avoid triggering another assertion when
|
// Use unchecked std::fprintf to avoid triggering another assertion when
|
||||||
// writing to stderr fails
|
// writing to stderr fails.
|
||||||
std::fprintf(stderr, "%s:%d: assertion failed: %s", file, line, message);
|
std::fprintf(stderr, "%s:%d: assertion failed: %s", file, line, message);
|
||||||
// Chosen instead of std::abort to satisfy Clang in CUDA mode during device
|
abort();
|
||||||
// code pass.
|
}
|
||||||
std::terminate();
|
#endif
|
||||||
|
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
|
namespace detail {
|
||||||
|
using std::locale;
|
||||||
|
using std::numpunct;
|
||||||
|
using std::use_facet;
|
||||||
|
} // namespace detail
|
||||||
|
#else
|
||||||
|
namespace detail {
|
||||||
|
struct locale {};
|
||||||
|
template <typename Char> struct numpunct {
|
||||||
|
auto grouping() const -> std::string { return "\03"; }
|
||||||
|
auto thousands_sep() const -> Char { return ','; }
|
||||||
|
auto decimal_point() const -> Char { return '.'; }
|
||||||
|
};
|
||||||
|
template <typename Facet> Facet use_facet(locale) { return {}; }
|
||||||
|
} // namespace detail
|
||||||
|
#endif // FMT_USE_LOCALE
|
||||||
|
|
||||||
|
template <typename Locale> auto locale_ref::get() const -> Locale {
|
||||||
|
using namespace detail;
|
||||||
|
static_assert(std::is_same<Locale, locale>::value, "");
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
|
if (locale_) return *static_cast<const locale*>(locale_);
|
||||||
|
#endif
|
||||||
|
return locale();
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_FUNC void throw_format_error(const char* message) {
|
namespace detail {
|
||||||
FMT_THROW(format_error(message));
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void format_error_code(detail::buffer<char>& out, int error_code,
|
FMT_FUNC void format_error_code(detail::buffer<char>& out, int error_code,
|
||||||
string_view message) noexcept {
|
string_view message) noexcept {
|
||||||
|
|
@ -56,112 +86,101 @@ FMT_FUNC void format_error_code(detail::buffer<char>& out, int error_code,
|
||||||
++error_code_size;
|
++error_code_size;
|
||||||
}
|
}
|
||||||
error_code_size += detail::to_unsigned(detail::count_digits(abs_value));
|
error_code_size += detail::to_unsigned(detail::count_digits(abs_value));
|
||||||
auto it = buffer_appender<char>(out);
|
auto it = appender(out);
|
||||||
if (message.size() <= inline_buffer_size - error_code_size)
|
if (message.size() <= inline_buffer_size - error_code_size)
|
||||||
format_to(it, FMT_STRING("{}{}"), message, SEP);
|
fmt::format_to(it, FMT_STRING("{}{}"), message, SEP);
|
||||||
format_to(it, FMT_STRING("{}{}"), ERROR_STR, error_code);
|
fmt::format_to(it, FMT_STRING("{}{}"), ERROR_STR, error_code);
|
||||||
FMT_ASSERT(out.size() <= inline_buffer_size, "");
|
FMT_ASSERT(out.size() <= inline_buffer_size, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_FUNC void report_error(format_func func, int error_code,
|
FMT_FUNC void do_report_error(format_func func, int error_code,
|
||||||
const char* message) noexcept {
|
const char* message) noexcept {
|
||||||
memory_buffer full_message;
|
memory_buffer full_message;
|
||||||
func(full_message, error_code, message);
|
func(full_message, error_code, message);
|
||||||
// Don't use fwrite_fully because the latter may throw.
|
// Don't use fwrite_all because the latter may throw.
|
||||||
if (std::fwrite(full_message.data(), full_message.size(), 1, stderr) > 0)
|
if (std::fwrite(full_message.data(), full_message.size(), 1, stderr) > 0)
|
||||||
std::fputc('\n', stderr);
|
std::fputc('\n', stderr);
|
||||||
}
|
}
|
||||||
|
|
||||||
// A wrapper around fwrite that throws on error.
|
// A wrapper around fwrite that throws on error.
|
||||||
inline void fwrite_fully(const void* ptr, size_t size, size_t count,
|
inline void fwrite_all(const void* ptr, size_t count, FILE* stream) {
|
||||||
FILE* stream) {
|
size_t written = std::fwrite(ptr, 1, count, stream);
|
||||||
size_t written = std::fwrite(ptr, size, count, stream);
|
|
||||||
if (written < count)
|
if (written < count)
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot write to file")));
|
FMT_THROW(system_error(errno, FMT_STRING("cannot write to file")));
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
|
||||||
template <typename Locale>
|
|
||||||
locale_ref::locale_ref(const Locale& loc) : locale_(&loc) {
|
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Locale> Locale locale_ref::get() const {
|
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
|
||||||
return locale_ ? *static_cast<const std::locale*>(locale_) : std::locale();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_FUNC auto thousands_sep_impl(locale_ref loc) -> thousands_sep_result<Char> {
|
FMT_FUNC auto thousands_sep_impl(locale_ref loc) -> thousands_sep_result<Char> {
|
||||||
auto& facet = std::use_facet<std::numpunct<Char>>(loc.get<std::locale>());
|
auto&& facet = use_facet<numpunct<Char>>(loc.get<locale>());
|
||||||
auto grouping = facet.grouping();
|
auto grouping = facet.grouping();
|
||||||
auto thousands_sep = grouping.empty() ? Char() : facet.thousands_sep();
|
auto thousands_sep = grouping.empty() ? Char() : facet.thousands_sep();
|
||||||
return {std::move(grouping), thousands_sep};
|
return {std::move(grouping), thousands_sep};
|
||||||
}
|
}
|
||||||
template <typename Char> FMT_FUNC Char decimal_point_impl(locale_ref loc) {
|
|
||||||
return std::use_facet<std::numpunct<Char>>(loc.get<std::locale>())
|
|
||||||
.decimal_point();
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_FUNC auto thousands_sep_impl(locale_ref) -> thousands_sep_result<Char> {
|
FMT_FUNC auto decimal_point_impl(locale_ref loc) -> Char {
|
||||||
return {"\03", FMT_STATIC_THOUSANDS_SEPARATOR};
|
return use_facet<numpunct<Char>>(loc.get<locale>()).decimal_point();
|
||||||
}
|
}
|
||||||
template <typename Char> FMT_FUNC Char decimal_point_impl(locale_ref) {
|
|
||||||
return '.';
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
FMT_FUNC auto write_loc(appender out, loc_value value,
|
FMT_FUNC auto write_loc(appender out, loc_value value,
|
||||||
const format_specs<>& specs, locale_ref loc) -> bool {
|
const format_specs& specs, locale_ref loc) -> bool {
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
|
||||||
auto locale = loc.get<std::locale>();
|
auto locale = loc.get<std::locale>();
|
||||||
// We cannot use the num_put<char> facet because it may produce output in
|
// We cannot use the num_put<char> facet because it may produce output in
|
||||||
// a wrong encoding.
|
// a wrong encoding.
|
||||||
using facet = format_facet<std::locale>;
|
using facet = format_facet<std::locale>;
|
||||||
if (std::has_facet<facet>(locale))
|
if (std::has_facet<facet>(locale))
|
||||||
return std::use_facet<facet>(locale).put(out, value, specs);
|
return use_facet<facet>(locale).put(out, value, specs);
|
||||||
return facet(locale).put(out, value, specs);
|
return facet(locale).put(out, value, specs);
|
||||||
#endif
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
|
FMT_FUNC void report_error(const char* message) {
|
||||||
|
#if FMT_MSC_VERSION || defined(__NVCC__)
|
||||||
|
// Silence unreachable code warnings in MSVC and NVCC because these
|
||||||
|
// are nearly impossible to fix in a generic code.
|
||||||
|
volatile bool b = true;
|
||||||
|
if (!b) return;
|
||||||
|
#endif
|
||||||
|
FMT_THROW(format_error(message));
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Locale> typename Locale::id format_facet<Locale>::id;
|
template <typename Locale> typename Locale::id format_facet<Locale>::id;
|
||||||
|
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
|
||||||
template <typename Locale> format_facet<Locale>::format_facet(Locale& loc) {
|
template <typename Locale> format_facet<Locale>::format_facet(Locale& loc) {
|
||||||
auto& numpunct = std::use_facet<std::numpunct<char>>(loc);
|
auto& np = detail::use_facet<detail::numpunct<char>>(loc);
|
||||||
grouping_ = numpunct.grouping();
|
grouping_ = np.grouping();
|
||||||
if (!grouping_.empty()) separator_ = std::string(1, numpunct.thousands_sep());
|
if (!grouping_.empty()) separator_ = std::string(1, np.thousands_sep());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
template <>
|
template <>
|
||||||
FMT_API FMT_FUNC auto format_facet<std::locale>::do_put(
|
FMT_API FMT_FUNC auto format_facet<std::locale>::do_put(
|
||||||
appender out, loc_value val, const format_specs<>& specs) const -> bool {
|
appender out, loc_value val, const format_specs& specs) const -> bool {
|
||||||
return val.visit(
|
return val.visit(
|
||||||
detail::loc_writer<>{out, specs, separator_, grouping_, decimal_point_});
|
detail::loc_writer<>{out, specs, separator_, grouping_, decimal_point_});
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
FMT_FUNC std::system_error vsystem_error(int error_code, string_view fmt,
|
FMT_FUNC auto vsystem_error(int error_code, string_view fmt, format_args args)
|
||||||
format_args args) {
|
-> std::system_error {
|
||||||
auto ec = std::error_code(error_code, std::generic_category());
|
auto ec = std::error_code(error_code, std::generic_category());
|
||||||
return std::system_error(ec, vformat(fmt, args));
|
return std::system_error(ec, vformat(fmt, args));
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template <typename F> inline bool operator==(basic_fp<F> x, basic_fp<F> y) {
|
template <typename F>
|
||||||
|
inline auto operator==(basic_fp<F> x, basic_fp<F> y) -> bool {
|
||||||
return x.f == y.f && x.e == y.e;
|
return x.f == y.f && x.e == y.e;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compilers should be able to optimize this into the ror instruction.
|
// Compilers should be able to optimize this into the ror instruction.
|
||||||
FMT_CONSTEXPR inline uint32_t rotr(uint32_t n, uint32_t r) noexcept {
|
FMT_INLINE auto rotr(uint32_t n, uint32_t r) noexcept -> uint32_t {
|
||||||
r &= 31;
|
r &= 31;
|
||||||
return (n >> r) | (n << (32 - r));
|
return (n >> r) | (n << (32 - r));
|
||||||
}
|
}
|
||||||
FMT_CONSTEXPR inline uint64_t rotr(uint64_t n, uint32_t r) noexcept {
|
FMT_INLINE auto rotr(uint64_t n, uint32_t r) noexcept -> uint64_t {
|
||||||
r &= 63;
|
r &= 63;
|
||||||
return (n >> r) | (n << (64 - r));
|
return (n >> r) | (n << (64 - r));
|
||||||
}
|
}
|
||||||
|
|
@ -170,14 +189,14 @@ FMT_CONSTEXPR inline uint64_t rotr(uint64_t n, uint32_t r) noexcept {
|
||||||
namespace dragonbox {
|
namespace dragonbox {
|
||||||
// Computes upper 64 bits of multiplication of a 32-bit unsigned integer and a
|
// Computes upper 64 bits of multiplication of a 32-bit unsigned integer and a
|
||||||
// 64-bit unsigned integer.
|
// 64-bit unsigned integer.
|
||||||
inline uint64_t umul96_upper64(uint32_t x, uint64_t y) noexcept {
|
inline auto umul96_upper64(uint32_t x, uint64_t y) noexcept -> uint64_t {
|
||||||
return umul128_upper64(static_cast<uint64_t>(x) << 32, y);
|
return umul128_upper64(static_cast<uint64_t>(x) << 32, y);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Computes lower 128 bits of multiplication of a 64-bit unsigned integer and a
|
// Computes lower 128 bits of multiplication of a 64-bit unsigned integer and a
|
||||||
// 128-bit unsigned integer.
|
// 128-bit unsigned integer.
|
||||||
inline uint128_fallback umul192_lower128(uint64_t x,
|
inline auto umul192_lower128(uint64_t x, uint128_fallback y) noexcept
|
||||||
uint128_fallback y) noexcept {
|
-> uint128_fallback {
|
||||||
uint64_t high = x * y.high();
|
uint64_t high = x * y.high();
|
||||||
uint128_fallback high_low = umul128(x, y.low());
|
uint128_fallback high_low = umul128(x, y.low());
|
||||||
return {high + high_low.high(), high_low.low()};
|
return {high + high_low.high(), high_low.low()};
|
||||||
|
|
@ -185,17 +204,17 @@ inline uint128_fallback umul192_lower128(uint64_t x,
|
||||||
|
|
||||||
// Computes lower 64 bits of multiplication of a 32-bit unsigned integer and a
|
// Computes lower 64 bits of multiplication of a 32-bit unsigned integer and a
|
||||||
// 64-bit unsigned integer.
|
// 64-bit unsigned integer.
|
||||||
inline uint64_t umul96_lower64(uint32_t x, uint64_t y) noexcept {
|
inline auto umul96_lower64(uint32_t x, uint64_t y) noexcept -> uint64_t {
|
||||||
return x * y;
|
return x * y;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Various fast log computations.
|
// Various fast log computations.
|
||||||
inline int floor_log10_pow2_minus_log10_4_over_3(int e) noexcept {
|
inline auto floor_log10_pow2_minus_log10_4_over_3(int e) noexcept -> int {
|
||||||
FMT_ASSERT(e <= 2936 && e >= -2985, "too large exponent");
|
FMT_ASSERT(e <= 2936 && e >= -2985, "too large exponent");
|
||||||
return (e * 631305 - 261663) >> 21;
|
return (e * 631305 - 261663) >> 21;
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_INLINE_VARIABLE constexpr struct {
|
FMT_INLINE_VARIABLE constexpr struct div_small_pow10_infos_struct {
|
||||||
uint32_t divisor;
|
uint32_t divisor;
|
||||||
int shift_amount;
|
int shift_amount;
|
||||||
} div_small_pow10_infos[] = {{10, 16}, {100, 16}};
|
} div_small_pow10_infos[] = {{10, 16}, {100, 16}};
|
||||||
|
|
@ -204,7 +223,7 @@ FMT_INLINE_VARIABLE constexpr struct {
|
||||||
// divisible by pow(10, N).
|
// divisible by pow(10, N).
|
||||||
// Precondition: n <= pow(10, N + 1).
|
// Precondition: n <= pow(10, N + 1).
|
||||||
template <int N>
|
template <int N>
|
||||||
bool check_divisibility_and_divide_by_pow10(uint32_t& n) noexcept {
|
auto check_divisibility_and_divide_by_pow10(uint32_t& n) noexcept -> bool {
|
||||||
// The numbers below are chosen such that:
|
// The numbers below are chosen such that:
|
||||||
// 1. floor(n/d) = floor(nm / 2^k) where d=10 or d=100,
|
// 1. floor(n/d) = floor(nm / 2^k) where d=10 or d=100,
|
||||||
// 2. nm mod 2^k < m if and only if n is divisible by d,
|
// 2. nm mod 2^k < m if and only if n is divisible by d,
|
||||||
|
|
@ -229,7 +248,7 @@ bool check_divisibility_and_divide_by_pow10(uint32_t& n) noexcept {
|
||||||
|
|
||||||
// Computes floor(n / pow(10, N)) for small n and N.
|
// Computes floor(n / pow(10, N)) for small n and N.
|
||||||
// Precondition: n <= pow(10, N + 1).
|
// Precondition: n <= pow(10, N + 1).
|
||||||
template <int N> uint32_t small_division_by_pow10(uint32_t n) noexcept {
|
template <int N> auto small_division_by_pow10(uint32_t n) noexcept -> uint32_t {
|
||||||
constexpr auto info = div_small_pow10_infos[N - 1];
|
constexpr auto info = div_small_pow10_infos[N - 1];
|
||||||
FMT_ASSERT(n <= info.divisor * 10, "n is too large");
|
FMT_ASSERT(n <= info.divisor * 10, "n is too large");
|
||||||
constexpr uint32_t magic_number =
|
constexpr uint32_t magic_number =
|
||||||
|
|
@ -238,12 +257,12 @@ template <int N> uint32_t small_division_by_pow10(uint32_t n) noexcept {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Computes floor(n / 10^(kappa + 1)) (float)
|
// Computes floor(n / 10^(kappa + 1)) (float)
|
||||||
inline uint32_t divide_by_10_to_kappa_plus_1(uint32_t n) noexcept {
|
inline auto divide_by_10_to_kappa_plus_1(uint32_t n) noexcept -> uint32_t {
|
||||||
// 1374389535 = ceil(2^37/100)
|
// 1374389535 = ceil(2^37/100)
|
||||||
return static_cast<uint32_t>((static_cast<uint64_t>(n) * 1374389535) >> 37);
|
return static_cast<uint32_t>((static_cast<uint64_t>(n) * 1374389535) >> 37);
|
||||||
}
|
}
|
||||||
// Computes floor(n / 10^(kappa + 1)) (double)
|
// Computes floor(n / 10^(kappa + 1)) (double)
|
||||||
inline uint64_t divide_by_10_to_kappa_plus_1(uint64_t n) noexcept {
|
inline auto divide_by_10_to_kappa_plus_1(uint64_t n) noexcept -> uint64_t {
|
||||||
// 2361183241434822607 = ceil(2^(64+7)/1000)
|
// 2361183241434822607 = ceil(2^(64+7)/1000)
|
||||||
return umul128_upper64(n, 2361183241434822607ull) >> 7;
|
return umul128_upper64(n, 2361183241434822607ull) >> 7;
|
||||||
}
|
}
|
||||||
|
|
@ -255,10 +274,10 @@ template <> struct cache_accessor<float> {
|
||||||
using carrier_uint = float_info<float>::carrier_uint;
|
using carrier_uint = float_info<float>::carrier_uint;
|
||||||
using cache_entry_type = uint64_t;
|
using cache_entry_type = uint64_t;
|
||||||
|
|
||||||
static uint64_t get_cached_power(int k) noexcept {
|
static auto get_cached_power(int k) noexcept -> uint64_t {
|
||||||
FMT_ASSERT(k >= float_info<float>::min_k && k <= float_info<float>::max_k,
|
FMT_ASSERT(k >= float_info<float>::min_k && k <= float_info<float>::max_k,
|
||||||
"k is out of range");
|
"k is out of range");
|
||||||
static constexpr const uint64_t pow10_significands[] = {
|
static constexpr uint64_t pow10_significands[] = {
|
||||||
0x81ceb32c4b43fcf5, 0xa2425ff75e14fc32, 0xcad2f7f5359a3b3f,
|
0x81ceb32c4b43fcf5, 0xa2425ff75e14fc32, 0xcad2f7f5359a3b3f,
|
||||||
0xfd87b5f28300ca0e, 0x9e74d1b791e07e49, 0xc612062576589ddb,
|
0xfd87b5f28300ca0e, 0x9e74d1b791e07e49, 0xc612062576589ddb,
|
||||||
0xf79687aed3eec552, 0x9abe14cd44753b53, 0xc16d9a0095928a28,
|
0xf79687aed3eec552, 0x9abe14cd44753b53, 0xc16d9a0095928a28,
|
||||||
|
|
@ -297,20 +316,23 @@ template <> struct cache_accessor<float> {
|
||||||
bool is_integer;
|
bool is_integer;
|
||||||
};
|
};
|
||||||
|
|
||||||
static compute_mul_result compute_mul(
|
static auto compute_mul(carrier_uint u,
|
||||||
carrier_uint u, const cache_entry_type& cache) noexcept {
|
const cache_entry_type& cache) noexcept
|
||||||
|
-> compute_mul_result {
|
||||||
auto r = umul96_upper64(u, cache);
|
auto r = umul96_upper64(u, cache);
|
||||||
return {static_cast<carrier_uint>(r >> 32),
|
return {static_cast<carrier_uint>(r >> 32),
|
||||||
static_cast<carrier_uint>(r) == 0};
|
static_cast<carrier_uint>(r) == 0};
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32_t compute_delta(const cache_entry_type& cache,
|
static auto compute_delta(const cache_entry_type& cache, int beta) noexcept
|
||||||
int beta) noexcept {
|
-> uint32_t {
|
||||||
return static_cast<uint32_t>(cache >> (64 - 1 - beta));
|
return static_cast<uint32_t>(cache >> (64 - 1 - beta));
|
||||||
}
|
}
|
||||||
|
|
||||||
static compute_mul_parity_result compute_mul_parity(
|
static auto compute_mul_parity(carrier_uint two_f,
|
||||||
carrier_uint two_f, const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache,
|
||||||
|
int beta) noexcept
|
||||||
|
-> compute_mul_parity_result {
|
||||||
FMT_ASSERT(beta >= 1, "");
|
FMT_ASSERT(beta >= 1, "");
|
||||||
FMT_ASSERT(beta < 64, "");
|
FMT_ASSERT(beta < 64, "");
|
||||||
|
|
||||||
|
|
@ -319,22 +341,22 @@ template <> struct cache_accessor<float> {
|
||||||
static_cast<uint32_t>(r >> (32 - beta)) == 0};
|
static_cast<uint32_t>(r >> (32 - beta)) == 0};
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_left_endpoint_for_shorter_interval_case(
|
static auto compute_left_endpoint_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return static_cast<carrier_uint>(
|
return static_cast<carrier_uint>(
|
||||||
(cache - (cache >> (num_significand_bits<float>() + 2))) >>
|
(cache - (cache >> (num_significand_bits<float>() + 2))) >>
|
||||||
(64 - num_significand_bits<float>() - 1 - beta));
|
(64 - num_significand_bits<float>() - 1 - beta));
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_right_endpoint_for_shorter_interval_case(
|
static auto compute_right_endpoint_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return static_cast<carrier_uint>(
|
return static_cast<carrier_uint>(
|
||||||
(cache + (cache >> (num_significand_bits<float>() + 1))) >>
|
(cache + (cache >> (num_significand_bits<float>() + 1))) >>
|
||||||
(64 - num_significand_bits<float>() - 1 - beta));
|
(64 - num_significand_bits<float>() - 1 - beta));
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_round_up_for_shorter_interval_case(
|
static auto compute_round_up_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return (static_cast<carrier_uint>(
|
return (static_cast<carrier_uint>(
|
||||||
cache >> (64 - num_significand_bits<float>() - 2 - beta)) +
|
cache >> (64 - num_significand_bits<float>() - 2 - beta)) +
|
||||||
1) /
|
1) /
|
||||||
|
|
@ -346,11 +368,11 @@ template <> struct cache_accessor<double> {
|
||||||
using carrier_uint = float_info<double>::carrier_uint;
|
using carrier_uint = float_info<double>::carrier_uint;
|
||||||
using cache_entry_type = uint128_fallback;
|
using cache_entry_type = uint128_fallback;
|
||||||
|
|
||||||
static uint128_fallback get_cached_power(int k) noexcept {
|
static auto get_cached_power(int k) noexcept -> uint128_fallback {
|
||||||
FMT_ASSERT(k >= float_info<double>::min_k && k <= float_info<double>::max_k,
|
FMT_ASSERT(k >= float_info<double>::min_k && k <= float_info<double>::max_k,
|
||||||
"k is out of range");
|
"k is out of range");
|
||||||
|
|
||||||
static constexpr const uint128_fallback pow10_significands[] = {
|
static constexpr uint128_fallback pow10_significands[] = {
|
||||||
#if FMT_USE_FULL_CACHE_DRAGONBOX
|
#if FMT_USE_FULL_CACHE_DRAGONBOX
|
||||||
{0xff77b1fcbebcdc4f, 0x25e8e89c13bb0f7b},
|
{0xff77b1fcbebcdc4f, 0x25e8e89c13bb0f7b},
|
||||||
{0x9faacf3df73609b1, 0x77b191618c54e9ad},
|
{0x9faacf3df73609b1, 0x77b191618c54e9ad},
|
||||||
|
|
@ -985,8 +1007,7 @@ template <> struct cache_accessor<double> {
|
||||||
{0xe0accfa875af45a7, 0x93eb1b80a33b8606},
|
{0xe0accfa875af45a7, 0x93eb1b80a33b8606},
|
||||||
{0x8c6c01c9498d8b88, 0xbc72f130660533c4},
|
{0x8c6c01c9498d8b88, 0xbc72f130660533c4},
|
||||||
{0xaf87023b9bf0ee6a, 0xeb8fad7c7f8680b5},
|
{0xaf87023b9bf0ee6a, 0xeb8fad7c7f8680b5},
|
||||||
{ 0xdb68c2ca82ed2a05,
|
{0xdb68c2ca82ed2a05, 0xa67398db9f6820e2},
|
||||||
0xa67398db9f6820e2 }
|
|
||||||
#else
|
#else
|
||||||
{0xff77b1fcbebcdc4f, 0x25e8e89c13bb0f7b},
|
{0xff77b1fcbebcdc4f, 0x25e8e89c13bb0f7b},
|
||||||
{0xce5d73ff402d98e3, 0xfb0a3d212dc81290},
|
{0xce5d73ff402d98e3, 0xfb0a3d212dc81290},
|
||||||
|
|
@ -1018,7 +1039,7 @@ template <> struct cache_accessor<double> {
|
||||||
#if FMT_USE_FULL_CACHE_DRAGONBOX
|
#if FMT_USE_FULL_CACHE_DRAGONBOX
|
||||||
return pow10_significands[k - float_info<double>::min_k];
|
return pow10_significands[k - float_info<double>::min_k];
|
||||||
#else
|
#else
|
||||||
static constexpr const uint64_t powers_of_5_64[] = {
|
static constexpr uint64_t powers_of_5_64[] = {
|
||||||
0x0000000000000001, 0x0000000000000005, 0x0000000000000019,
|
0x0000000000000001, 0x0000000000000005, 0x0000000000000019,
|
||||||
0x000000000000007d, 0x0000000000000271, 0x0000000000000c35,
|
0x000000000000007d, 0x0000000000000271, 0x0000000000000c35,
|
||||||
0x0000000000003d09, 0x000000000001312d, 0x000000000005f5e1,
|
0x0000000000003d09, 0x000000000001312d, 0x000000000005f5e1,
|
||||||
|
|
@ -1071,19 +1092,22 @@ template <> struct cache_accessor<double> {
|
||||||
bool is_integer;
|
bool is_integer;
|
||||||
};
|
};
|
||||||
|
|
||||||
static compute_mul_result compute_mul(
|
static auto compute_mul(carrier_uint u,
|
||||||
carrier_uint u, const cache_entry_type& cache) noexcept {
|
const cache_entry_type& cache) noexcept
|
||||||
|
-> compute_mul_result {
|
||||||
auto r = umul192_upper128(u, cache);
|
auto r = umul192_upper128(u, cache);
|
||||||
return {r.high(), r.low() == 0};
|
return {r.high(), r.low() == 0};
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32_t compute_delta(cache_entry_type const& cache,
|
static auto compute_delta(const cache_entry_type& cache, int beta) noexcept
|
||||||
int beta) noexcept {
|
-> uint32_t {
|
||||||
return static_cast<uint32_t>(cache.high() >> (64 - 1 - beta));
|
return static_cast<uint32_t>(cache.high() >> (64 - 1 - beta));
|
||||||
}
|
}
|
||||||
|
|
||||||
static compute_mul_parity_result compute_mul_parity(
|
static auto compute_mul_parity(carrier_uint two_f,
|
||||||
carrier_uint two_f, const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache,
|
||||||
|
int beta) noexcept
|
||||||
|
-> compute_mul_parity_result {
|
||||||
FMT_ASSERT(beta >= 1, "");
|
FMT_ASSERT(beta >= 1, "");
|
||||||
FMT_ASSERT(beta < 64, "");
|
FMT_ASSERT(beta < 64, "");
|
||||||
|
|
||||||
|
|
@ -1092,47 +1116,47 @@ template <> struct cache_accessor<double> {
|
||||||
((r.high() << beta) | (r.low() >> (64 - beta))) == 0};
|
((r.high() << beta) | (r.low() >> (64 - beta))) == 0};
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_left_endpoint_for_shorter_interval_case(
|
static auto compute_left_endpoint_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return (cache.high() -
|
return (cache.high() -
|
||||||
(cache.high() >> (num_significand_bits<double>() + 2))) >>
|
(cache.high() >> (num_significand_bits<double>() + 2))) >>
|
||||||
(64 - num_significand_bits<double>() - 1 - beta);
|
(64 - num_significand_bits<double>() - 1 - beta);
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_right_endpoint_for_shorter_interval_case(
|
static auto compute_right_endpoint_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return (cache.high() +
|
return (cache.high() +
|
||||||
(cache.high() >> (num_significand_bits<double>() + 1))) >>
|
(cache.high() >> (num_significand_bits<double>() + 1))) >>
|
||||||
(64 - num_significand_bits<double>() - 1 - beta);
|
(64 - num_significand_bits<double>() - 1 - beta);
|
||||||
}
|
}
|
||||||
|
|
||||||
static carrier_uint compute_round_up_for_shorter_interval_case(
|
static auto compute_round_up_for_shorter_interval_case(
|
||||||
const cache_entry_type& cache, int beta) noexcept {
|
const cache_entry_type& cache, int beta) noexcept -> carrier_uint {
|
||||||
return ((cache.high() >> (64 - num_significand_bits<double>() - 2 - beta)) +
|
return ((cache.high() >> (64 - num_significand_bits<double>() - 2 - beta)) +
|
||||||
1) /
|
1) /
|
||||||
2;
|
2;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_FUNC uint128_fallback get_cached_power(int k) noexcept {
|
FMT_FUNC auto get_cached_power(int k) noexcept -> uint128_fallback {
|
||||||
return cache_accessor<double>::get_cached_power(k);
|
return cache_accessor<double>::get_cached_power(k);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Various integer checks
|
// Various integer checks
|
||||||
template <typename T>
|
template <typename T>
|
||||||
bool is_left_endpoint_integer_shorter_interval(int exponent) noexcept {
|
auto is_left_endpoint_integer_shorter_interval(int exponent) noexcept -> bool {
|
||||||
const int case_shorter_interval_left_endpoint_lower_threshold = 2;
|
const int case_shorter_interval_left_endpoint_lower_threshold = 2;
|
||||||
const int case_shorter_interval_left_endpoint_upper_threshold = 3;
|
const int case_shorter_interval_left_endpoint_upper_threshold = 3;
|
||||||
return exponent >= case_shorter_interval_left_endpoint_lower_threshold &&
|
return exponent >= case_shorter_interval_left_endpoint_lower_threshold &&
|
||||||
exponent <= case_shorter_interval_left_endpoint_upper_threshold;
|
exponent <= case_shorter_interval_left_endpoint_upper_threshold;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove trailing zeros from n and return the number of zeros removed (float)
|
// Remove trailing zeros from n and return the number of zeros removed (float).
|
||||||
FMT_INLINE int remove_trailing_zeros(uint32_t& n, int s = 0) noexcept {
|
FMT_INLINE auto remove_trailing_zeros(uint32_t& n, int s = 0) noexcept -> int {
|
||||||
FMT_ASSERT(n != 0, "");
|
FMT_ASSERT(n != 0, "");
|
||||||
// Modular inverse of 5 (mod 2^32): (mod_inv_5 * 5) mod 2^32 = 1.
|
// Modular inverse of 5 (mod 2^32): (mod_inv_5 * 5) mod 2^32 = 1.
|
||||||
constexpr uint32_t mod_inv_5 = 0xcccccccd;
|
constexpr uint32_t mod_inv_5 = 0xcccccccd;
|
||||||
constexpr uint32_t mod_inv_25 = 0xc28f5c29; // = mod_inv_5 * mod_inv_5
|
constexpr uint32_t mod_inv_25 = 0xc28f5c29; // = mod_inv_5 * mod_inv_5
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
auto q = rotr(n * mod_inv_25, 2);
|
auto q = rotr(n * mod_inv_25, 2);
|
||||||
|
|
@ -1148,27 +1172,24 @@ FMT_INLINE int remove_trailing_zeros(uint32_t& n, int s = 0) noexcept {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Removes trailing zeros and returns the number of zeros removed (double)
|
// Removes trailing zeros and returns the number of zeros removed (double).
|
||||||
FMT_INLINE int remove_trailing_zeros(uint64_t& n) noexcept {
|
FMT_INLINE auto remove_trailing_zeros(uint64_t& n) noexcept -> int {
|
||||||
FMT_ASSERT(n != 0, "");
|
FMT_ASSERT(n != 0, "");
|
||||||
|
|
||||||
// This magic number is ceil(2^90 / 10^8).
|
|
||||||
constexpr uint64_t magic_number = 12379400392853802749ull;
|
|
||||||
auto nm = umul128(n, magic_number);
|
|
||||||
|
|
||||||
// Is n is divisible by 10^8?
|
// Is n is divisible by 10^8?
|
||||||
if ((nm.high() & ((1ull << (90 - 64)) - 1)) == 0 && nm.low() < magic_number) {
|
constexpr uint32_t ten_pow_8 = 100000000u;
|
||||||
|
if ((n % ten_pow_8) == 0) {
|
||||||
// If yes, work with the quotient...
|
// If yes, work with the quotient...
|
||||||
auto n32 = static_cast<uint32_t>(nm.high() >> (90 - 64));
|
auto n32 = static_cast<uint32_t>(n / ten_pow_8);
|
||||||
// ... and use the 32 bit variant of the function
|
// ... and use the 32 bit variant of the function
|
||||||
int s = remove_trailing_zeros(n32, 8);
|
int num_zeros = remove_trailing_zeros(n32, 8);
|
||||||
n = n32;
|
n = n32;
|
||||||
return s;
|
return num_zeros;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If n is not divisible by 10^8, work with n itself.
|
// If n is not divisible by 10^8, work with n itself.
|
||||||
constexpr uint64_t mod_inv_5 = 0xcccccccccccccccd;
|
constexpr uint64_t mod_inv_5 = 0xcccccccccccccccd;
|
||||||
constexpr uint64_t mod_inv_25 = 0x8f5c28f5c28f5c29; // = mod_inv_5 * mod_inv_5
|
constexpr uint64_t mod_inv_25 = 0x8f5c28f5c28f5c29; // mod_inv_5 * mod_inv_5
|
||||||
|
|
||||||
int s = 0;
|
int s = 0;
|
||||||
while (true) {
|
while (true) {
|
||||||
|
|
@ -1188,7 +1209,7 @@ FMT_INLINE int remove_trailing_zeros(uint64_t& n) noexcept {
|
||||||
|
|
||||||
// The main algorithm for shorter interval case
|
// The main algorithm for shorter interval case
|
||||||
template <typename T>
|
template <typename T>
|
||||||
FMT_INLINE decimal_fp<T> shorter_interval_case(int exponent) noexcept {
|
FMT_INLINE auto shorter_interval_case(int exponent) noexcept -> decimal_fp<T> {
|
||||||
decimal_fp<T> ret_value;
|
decimal_fp<T> ret_value;
|
||||||
// Compute k and beta
|
// Compute k and beta
|
||||||
const int minus_k = floor_log10_pow2_minus_log10_4_over_3(exponent);
|
const int minus_k = floor_log10_pow2_minus_log10_4_over_3(exponent);
|
||||||
|
|
@ -1234,7 +1255,7 @@ FMT_INLINE decimal_fp<T> shorter_interval_case(int exponent) noexcept {
|
||||||
return ret_value;
|
return ret_value;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T> decimal_fp<T> to_decimal(T x) noexcept {
|
template <typename T> auto to_decimal(T x) noexcept -> decimal_fp<T> {
|
||||||
// Step 1: integer promotion & Schubfach multiplier calculation.
|
// Step 1: integer promotion & Schubfach multiplier calculation.
|
||||||
|
|
||||||
using carrier_uint = typename float_info<T>::carrier_uint;
|
using carrier_uint = typename float_info<T>::carrier_uint;
|
||||||
|
|
@ -1373,15 +1394,15 @@ template <> struct formatter<detail::bigint> {
|
||||||
for (auto i = n.bigits_.size(); i > 0; --i) {
|
for (auto i = n.bigits_.size(); i > 0; --i) {
|
||||||
auto value = n.bigits_[i - 1u];
|
auto value = n.bigits_[i - 1u];
|
||||||
if (first) {
|
if (first) {
|
||||||
out = format_to(out, FMT_STRING("{:x}"), value);
|
out = fmt::format_to(out, FMT_STRING("{:x}"), value);
|
||||||
first = false;
|
first = false;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
out = format_to(out, FMT_STRING("{:08x}"), value);
|
out = fmt::format_to(out, FMT_STRING("{:08x}"), value);
|
||||||
}
|
}
|
||||||
if (n.exp_ > 0)
|
if (n.exp_ > 0)
|
||||||
out = format_to(out, FMT_STRING("p{}"),
|
out = fmt::format_to(out, FMT_STRING("p{}"),
|
||||||
n.exp_ * detail::bigint::bigit_bits);
|
n.exp_ * detail::bigint::bigit_bits);
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -1405,7 +1426,7 @@ FMT_FUNC void format_system_error(detail::buffer<char>& out, int error_code,
|
||||||
const char* message) noexcept {
|
const char* message) noexcept {
|
||||||
FMT_TRY {
|
FMT_TRY {
|
||||||
auto ec = std::error_code(error_code, std::generic_category());
|
auto ec = std::error_code(error_code, std::generic_category());
|
||||||
write(std::back_inserter(out), std::system_error(ec, message).what());
|
detail::write(appender(out), std::system_error(ec, message).what());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
FMT_CATCH(...) {}
|
FMT_CATCH(...) {}
|
||||||
|
|
@ -1414,10 +1435,10 @@ FMT_FUNC void format_system_error(detail::buffer<char>& out, int error_code,
|
||||||
|
|
||||||
FMT_FUNC void report_system_error(int error_code,
|
FMT_FUNC void report_system_error(int error_code,
|
||||||
const char* message) noexcept {
|
const char* message) noexcept {
|
||||||
report_error(format_system_error, error_code, message);
|
do_report_error(format_system_error, error_code, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_FUNC std::string vformat(string_view fmt, format_args args) {
|
FMT_FUNC auto vformat(string_view fmt, format_args args) -> std::string {
|
||||||
// Don't optimize the "{}" case to keep the binary size small and because it
|
// Don't optimize the "{}" case to keep the binary size small and because it
|
||||||
// can be better optimized in fmt::format anyway.
|
// can be better optimized in fmt::format anyway.
|
||||||
auto buffer = memory_buffer();
|
auto buffer = memory_buffer();
|
||||||
|
|
@ -1426,42 +1447,307 @@ FMT_FUNC std::string vformat(string_view fmt, format_args args) {
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
#ifndef _WIN32
|
|
||||||
FMT_FUNC bool write_console(std::FILE*, string_view) { return false; }
|
FMT_FUNC void vformat_to(buffer<char>& buf, string_view fmt, format_args args,
|
||||||
|
locale_ref loc) {
|
||||||
|
auto out = appender(buf);
|
||||||
|
if (fmt.size() == 2 && equal2(fmt.data(), "{}"))
|
||||||
|
return args.get(0).visit(default_arg_formatter<char>{out});
|
||||||
|
parse_format_string(fmt,
|
||||||
|
format_handler<>{parse_context<>(fmt), {out, args, loc}});
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T> struct span {
|
||||||
|
T* data;
|
||||||
|
size_t size;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename F> auto flockfile(F* f) -> decltype(_lock_file(f)) {
|
||||||
|
_lock_file(f);
|
||||||
|
}
|
||||||
|
template <typename F> auto funlockfile(F* f) -> decltype(_unlock_file(f)) {
|
||||||
|
_unlock_file(f);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef getc_unlocked
|
||||||
|
template <typename F> auto getc_unlocked(F* f) -> decltype(_fgetc_nolock(f)) {
|
||||||
|
return _fgetc_nolock(f);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template <typename F = FILE, typename Enable = void>
|
||||||
|
struct has_flockfile : std::false_type {};
|
||||||
|
|
||||||
|
template <typename F>
|
||||||
|
struct has_flockfile<F, void_t<decltype(flockfile(&std::declval<F&>()))>>
|
||||||
|
: std::true_type {};
|
||||||
|
|
||||||
|
// A FILE wrapper. F is FILE defined as a template parameter to make system API
|
||||||
|
// detection work.
|
||||||
|
template <typename F> class file_base {
|
||||||
|
public:
|
||||||
|
F* file_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
file_base(F* file) : file_(file) {}
|
||||||
|
operator F*() const { return file_; }
|
||||||
|
|
||||||
|
// Reads a code unit from the stream.
|
||||||
|
auto get() -> int {
|
||||||
|
int result = getc_unlocked(file_);
|
||||||
|
if (result == EOF && ferror(file_) != 0)
|
||||||
|
FMT_THROW(system_error(errno, FMT_STRING("getc failed")));
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Puts the code unit back into the stream buffer.
|
||||||
|
void unget(char c) {
|
||||||
|
if (ungetc(c, file_) == EOF)
|
||||||
|
FMT_THROW(system_error(errno, FMT_STRING("ungetc failed")));
|
||||||
|
}
|
||||||
|
|
||||||
|
void flush() { fflush(this->file_); }
|
||||||
|
};
|
||||||
|
|
||||||
|
// A FILE wrapper for glibc.
|
||||||
|
template <typename F> class glibc_file : public file_base<F> {
|
||||||
|
private:
|
||||||
|
enum {
|
||||||
|
line_buffered = 0x200, // _IO_LINE_BUF
|
||||||
|
unbuffered = 2 // _IO_UNBUFFERED
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
using file_base<F>::file_base;
|
||||||
|
|
||||||
|
auto is_buffered() const -> bool {
|
||||||
|
return (this->file_->_flags & unbuffered) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_buffer() {
|
||||||
|
if (this->file_->_IO_write_ptr < this->file_->_IO_write_end) return;
|
||||||
|
// Force buffer initialization by placing and removing a char in a buffer.
|
||||||
|
putc_unlocked(0, this->file_);
|
||||||
|
--this->file_->_IO_write_ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the file's read buffer.
|
||||||
|
auto get_read_buffer() const -> span<const char> {
|
||||||
|
auto ptr = this->file_->_IO_read_ptr;
|
||||||
|
return {ptr, to_unsigned(this->file_->_IO_read_end - ptr)};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the file's write buffer.
|
||||||
|
auto get_write_buffer() const -> span<char> {
|
||||||
|
auto ptr = this->file_->_IO_write_ptr;
|
||||||
|
return {ptr, to_unsigned(this->file_->_IO_buf_end - ptr)};
|
||||||
|
}
|
||||||
|
|
||||||
|
void advance_write_buffer(size_t size) { this->file_->_IO_write_ptr += size; }
|
||||||
|
|
||||||
|
auto needs_flush() const -> bool {
|
||||||
|
if ((this->file_->_flags & line_buffered) == 0) return false;
|
||||||
|
char* end = this->file_->_IO_write_end;
|
||||||
|
auto size = max_of<ptrdiff_t>(this->file_->_IO_write_ptr - end, 0);
|
||||||
|
return memchr(end, '\n', static_cast<size_t>(size));
|
||||||
|
}
|
||||||
|
|
||||||
|
void flush() { fflush_unlocked(this->file_); }
|
||||||
|
};
|
||||||
|
|
||||||
|
// A FILE wrapper for Apple's libc.
|
||||||
|
template <typename F> class apple_file : public file_base<F> {
|
||||||
|
private:
|
||||||
|
enum {
|
||||||
|
line_buffered = 1, // __SNBF
|
||||||
|
unbuffered = 2 // __SLBF
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
using file_base<F>::file_base;
|
||||||
|
|
||||||
|
auto is_buffered() const -> bool {
|
||||||
|
return (this->file_->_flags & unbuffered) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void init_buffer() {
|
||||||
|
if (this->file_->_p) return;
|
||||||
|
// Force buffer initialization by placing and removing a char in a buffer.
|
||||||
|
if (!FMT_CLANG_ANALYZER) putc_unlocked(0, this->file_);
|
||||||
|
--this->file_->_p;
|
||||||
|
++this->file_->_w;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto get_read_buffer() const -> span<const char> {
|
||||||
|
return {reinterpret_cast<char*>(this->file_->_p),
|
||||||
|
to_unsigned(this->file_->_r)};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto get_write_buffer() const -> span<char> {
|
||||||
|
return {reinterpret_cast<char*>(this->file_->_p),
|
||||||
|
to_unsigned(this->file_->_bf._base + this->file_->_bf._size -
|
||||||
|
this->file_->_p)};
|
||||||
|
}
|
||||||
|
|
||||||
|
void advance_write_buffer(size_t size) {
|
||||||
|
this->file_->_p += size;
|
||||||
|
this->file_->_w -= size;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto needs_flush() const -> bool {
|
||||||
|
if ((this->file_->_flags & line_buffered) == 0) return false;
|
||||||
|
return memchr(this->file_->_p + this->file_->_w, '\n',
|
||||||
|
to_unsigned(-this->file_->_w));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// A fallback FILE wrapper.
|
||||||
|
template <typename F> class fallback_file : public file_base<F> {
|
||||||
|
private:
|
||||||
|
char next_; // The next unconsumed character in the buffer.
|
||||||
|
bool has_next_ = false;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using file_base<F>::file_base;
|
||||||
|
|
||||||
|
auto is_buffered() const -> bool { return false; }
|
||||||
|
auto needs_flush() const -> bool { return false; }
|
||||||
|
void init_buffer() {}
|
||||||
|
|
||||||
|
auto get_read_buffer() const -> span<const char> {
|
||||||
|
return {&next_, has_next_ ? 1u : 0u};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto get_write_buffer() const -> span<char> { return {nullptr, 0}; }
|
||||||
|
|
||||||
|
void advance_write_buffer(size_t) {}
|
||||||
|
|
||||||
|
auto get() -> int {
|
||||||
|
has_next_ = false;
|
||||||
|
return file_base<F>::get();
|
||||||
|
}
|
||||||
|
|
||||||
|
void unget(char c) {
|
||||||
|
file_base<F>::unget(c);
|
||||||
|
next_ = c;
|
||||||
|
has_next_ = true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifndef FMT_USE_FALLBACK_FILE
|
||||||
|
# define FMT_USE_FALLBACK_FILE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template <typename F,
|
||||||
|
FMT_ENABLE_IF(sizeof(F::_p) != 0 && !FMT_USE_FALLBACK_FILE)>
|
||||||
|
auto get_file(F* f, int) -> apple_file<F> {
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
template <typename F,
|
||||||
|
FMT_ENABLE_IF(sizeof(F::_IO_read_ptr) != 0 && !FMT_USE_FALLBACK_FILE)>
|
||||||
|
inline auto get_file(F* f, int) -> glibc_file<F> {
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline auto get_file(FILE* f, ...) -> fallback_file<FILE> { return f; }
|
||||||
|
|
||||||
|
using file_ref = decltype(get_file(static_cast<FILE*>(nullptr), 0));
|
||||||
|
|
||||||
|
template <typename F = FILE, typename Enable = void>
|
||||||
|
class file_print_buffer : public buffer<char> {
|
||||||
|
public:
|
||||||
|
explicit file_print_buffer(F*) : buffer(nullptr, size_t()) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename F>
|
||||||
|
class file_print_buffer<F, enable_if_t<has_flockfile<F>::value>>
|
||||||
|
: public buffer<char> {
|
||||||
|
private:
|
||||||
|
file_ref file_;
|
||||||
|
|
||||||
|
static void grow(buffer<char>& base, size_t) {
|
||||||
|
auto& self = static_cast<file_print_buffer&>(base);
|
||||||
|
self.file_.advance_write_buffer(self.size());
|
||||||
|
if (self.file_.get_write_buffer().size == 0) self.file_.flush();
|
||||||
|
auto buf = self.file_.get_write_buffer();
|
||||||
|
FMT_ASSERT(buf.size > 0, "");
|
||||||
|
self.set(buf.data, buf.size);
|
||||||
|
self.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit file_print_buffer(F* f) : buffer(grow, size_t()), file_(f) {
|
||||||
|
flockfile(f);
|
||||||
|
file_.init_buffer();
|
||||||
|
auto buf = file_.get_write_buffer();
|
||||||
|
set(buf.data, buf.size);
|
||||||
|
}
|
||||||
|
~file_print_buffer() {
|
||||||
|
file_.advance_write_buffer(size());
|
||||||
|
bool flush = file_.needs_flush();
|
||||||
|
F* f = file_; // Make funlockfile depend on the template parameter F
|
||||||
|
funlockfile(f); // for the system API detection to work.
|
||||||
|
if (flush) fflush(file_);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#if !defined(_WIN32) || defined(FMT_USE_WRITE_CONSOLE)
|
||||||
|
FMT_FUNC auto write_console(int, string_view) -> bool { return false; }
|
||||||
#else
|
#else
|
||||||
using dword = conditional_t<sizeof(long) == 4, unsigned long, unsigned>;
|
using dword = conditional_t<sizeof(long) == 4, unsigned long, unsigned>;
|
||||||
extern "C" __declspec(dllimport) int __stdcall WriteConsoleW( //
|
extern "C" __declspec(dllimport) int __stdcall WriteConsoleW( //
|
||||||
void*, const void*, dword, dword*, void*);
|
void*, const void*, dword, dword*, void*);
|
||||||
|
|
||||||
FMT_FUNC bool write_console(std::FILE* f, string_view text) {
|
FMT_FUNC bool write_console(int fd, string_view text) {
|
||||||
auto fd = _fileno(f);
|
|
||||||
if (!_isatty(fd)) return false;
|
|
||||||
auto u16 = utf8_to_utf16(text);
|
auto u16 = utf8_to_utf16(text);
|
||||||
auto written = dword();
|
|
||||||
return WriteConsoleW(reinterpret_cast<void*>(_get_osfhandle(fd)), u16.c_str(),
|
return WriteConsoleW(reinterpret_cast<void*>(_get_osfhandle(fd)), u16.c_str(),
|
||||||
static_cast<uint32_t>(u16.size()), &written, nullptr) != 0;
|
static_cast<dword>(u16.size()), nullptr, nullptr) != 0;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
// Print assuming legacy (non-Unicode) encoding.
|
// Print assuming legacy (non-Unicode) encoding.
|
||||||
FMT_FUNC void vprint_mojibake(std::FILE* f, string_view fmt, format_args args) {
|
FMT_FUNC void vprint_mojibake(std::FILE* f, string_view fmt, format_args args,
|
||||||
|
bool newline) {
|
||||||
auto buffer = memory_buffer();
|
auto buffer = memory_buffer();
|
||||||
detail::vformat_to(buffer, fmt,
|
detail::vformat_to(buffer, fmt, args);
|
||||||
basic_format_args<buffer_context<char>>(args));
|
if (newline) buffer.push_back('\n');
|
||||||
fwrite_fully(buffer.data(), 1, buffer.size(), f);
|
fwrite_all(buffer.data(), buffer.size(), f);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
FMT_FUNC void print(std::FILE* f, string_view text) {
|
FMT_FUNC void print(std::FILE* f, string_view text) {
|
||||||
if (!write_console(f, text)) fwrite_fully(text.data(), 1, text.size(), f);
|
#if defined(_WIN32) && !defined(FMT_USE_WRITE_CONSOLE)
|
||||||
|
int fd = _fileno(f);
|
||||||
|
if (_isatty(fd)) {
|
||||||
|
std::fflush(f);
|
||||||
|
if (write_console(fd, text)) return;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
fwrite_all(text.data(), text.size(), f);
|
||||||
}
|
}
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
FMT_FUNC void vprint(std::FILE* f, string_view fmt, format_args args) {
|
FMT_FUNC void vprint_buffered(std::FILE* f, string_view fmt, format_args args) {
|
||||||
auto buffer = memory_buffer();
|
auto buffer = memory_buffer();
|
||||||
detail::vformat_to(buffer, fmt, args);
|
detail::vformat_to(buffer, fmt, args);
|
||||||
detail::print(f, {buffer.data(), buffer.size()});
|
detail::print(f, {buffer.data(), buffer.size()});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FMT_FUNC void vprint(std::FILE* f, string_view fmt, format_args args) {
|
||||||
|
if (!detail::file_ref(f).is_buffered() || !detail::has_flockfile<>())
|
||||||
|
return vprint_buffered(f, fmt, args);
|
||||||
|
auto&& buffer = detail::file_print_buffer<>(f);
|
||||||
|
return detail::vformat_to(buffer, fmt, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
FMT_FUNC void vprintln(std::FILE* f, string_view fmt, format_args args) {
|
||||||
|
auto buffer = memory_buffer();
|
||||||
|
detail::vformat_to(buffer, fmt, args);
|
||||||
|
buffer.push_back('\n');
|
||||||
|
detail::print(f, {buffer.data(), buffer.size()});
|
||||||
|
}
|
||||||
|
|
||||||
FMT_FUNC void vprint(string_view fmt, format_args args) {
|
FMT_FUNC void vprint(string_view fmt, format_args args) {
|
||||||
vprint(stdout, fmt, args);
|
vprint(stdout, fmt, args);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
3841
deps/fmt/include/fmt/format.h
vendored
3841
deps/fmt/include/fmt/format.h
vendored
File diff suppressed because it is too large
Load diff
298
deps/fmt/include/fmt/os.h
vendored
298
deps/fmt/include/fmt/os.h
vendored
|
|
@ -8,17 +8,19 @@
|
||||||
#ifndef FMT_OS_H_
|
#ifndef FMT_OS_H_
|
||||||
#define FMT_OS_H_
|
#define FMT_OS_H_
|
||||||
|
|
||||||
#include <cerrno>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <system_error> // std::system_error
|
|
||||||
|
|
||||||
#if defined __APPLE__ || defined(__FreeBSD__)
|
|
||||||
# include <xlocale.h> // for LC_NUMERIC_MASK on OS X
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
|
#ifndef FMT_MODULE
|
||||||
|
# include <cerrno>
|
||||||
|
# include <cstddef>
|
||||||
|
# include <cstdio>
|
||||||
|
# include <system_error> // std::system_error
|
||||||
|
|
||||||
|
# if FMT_HAS_INCLUDE(<xlocale.h>)
|
||||||
|
# include <xlocale.h> // LC_NUMERIC_MASK on macOS
|
||||||
|
# endif
|
||||||
|
#endif // FMT_MODULE
|
||||||
|
|
||||||
#ifndef FMT_USE_FCNTL
|
#ifndef FMT_USE_FCNTL
|
||||||
// UWP doesn't provide _pipe.
|
// UWP doesn't provide _pipe.
|
||||||
# if FMT_HAS_INCLUDE("winapifamily.h")
|
# if FMT_HAS_INCLUDE("winapifamily.h")
|
||||||
|
|
@ -27,7 +29,8 @@
|
||||||
# if (FMT_HAS_INCLUDE(<fcntl.h>) || defined(__APPLE__) || \
|
# if (FMT_HAS_INCLUDE(<fcntl.h>) || defined(__APPLE__) || \
|
||||||
defined(__linux__)) && \
|
defined(__linux__)) && \
|
||||||
(!defined(WINAPI_FAMILY) || \
|
(!defined(WINAPI_FAMILY) || \
|
||||||
(WINAPI_FAMILY == WINAPI_FAMILY_DESKTOP_APP))
|
(WINAPI_FAMILY == WINAPI_FAMILY_DESKTOP_APP)) && \
|
||||||
|
!defined(__wasm__)
|
||||||
# include <fcntl.h> // for O_RDONLY
|
# include <fcntl.h> // for O_RDONLY
|
||||||
# define FMT_USE_FCNTL 1
|
# define FMT_USE_FCNTL 1
|
||||||
# else
|
# else
|
||||||
|
|
@ -46,6 +49,7 @@
|
||||||
|
|
||||||
// Calls to system functions are wrapped in FMT_SYSTEM for testability.
|
// Calls to system functions are wrapped in FMT_SYSTEM for testability.
|
||||||
#ifdef FMT_SYSTEM
|
#ifdef FMT_SYSTEM
|
||||||
|
# define FMT_HAS_SYSTEM
|
||||||
# define FMT_POSIX_CALL(call) FMT_SYSTEM(call)
|
# define FMT_POSIX_CALL(call) FMT_SYSTEM(call)
|
||||||
#else
|
#else
|
||||||
# define FMT_SYSTEM(call) ::call
|
# define FMT_SYSTEM(call) ::call
|
||||||
|
|
@ -74,47 +78,34 @@ FMT_BEGIN_NAMESPACE
|
||||||
FMT_BEGIN_EXPORT
|
FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* A reference to a null-terminated string. It can be constructed from a C
|
||||||
A reference to a null-terminated string. It can be constructed from a C
|
* string or `std::string`.
|
||||||
string or ``std::string``.
|
*
|
||||||
|
* You can use one of the following type aliases for common character types:
|
||||||
You can use one of the following type aliases for common character types:
|
*
|
||||||
|
* +---------------+-----------------------------+
|
||||||
+---------------+-----------------------------+
|
* | Type | Definition |
|
||||||
| Type | Definition |
|
* +===============+=============================+
|
||||||
+===============+=============================+
|
* | cstring_view | basic_cstring_view<char> |
|
||||||
| cstring_view | basic_cstring_view<char> |
|
* +---------------+-----------------------------+
|
||||||
+---------------+-----------------------------+
|
* | wcstring_view | basic_cstring_view<wchar_t> |
|
||||||
| wcstring_view | basic_cstring_view<wchar_t> |
|
* +---------------+-----------------------------+
|
||||||
+---------------+-----------------------------+
|
*
|
||||||
|
* This class is most useful as a parameter type for functions that wrap C APIs.
|
||||||
This class is most useful as a parameter type to allow passing
|
|
||||||
different types of strings to a function, for example::
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string format(cstring_view format_str, const Args & ... args);
|
|
||||||
|
|
||||||
format("{}", 42);
|
|
||||||
format(std::string("{}"), 42);
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename Char> class basic_cstring_view {
|
template <typename Char> class basic_cstring_view {
|
||||||
private:
|
private:
|
||||||
const Char* data_;
|
const Char* data_;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/** Constructs a string reference object from a C string. */
|
/// Constructs a string reference object from a C string.
|
||||||
basic_cstring_view(const Char* s) : data_(s) {}
|
basic_cstring_view(const Char* s) : data_(s) {}
|
||||||
|
|
||||||
/**
|
/// Constructs a string reference from an `std::string` object.
|
||||||
\rst
|
|
||||||
Constructs a string reference from an ``std::string`` object.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
basic_cstring_view(const std::basic_string<Char>& s) : data_(s.c_str()) {}
|
basic_cstring_view(const std::basic_string<Char>& s) : data_(s.c_str()) {}
|
||||||
|
|
||||||
/** Returns the pointer to a C string. */
|
/// Returns the pointer to a C string.
|
||||||
const Char* c_str() const { return data_; }
|
auto c_str() const -> const Char* { return data_; }
|
||||||
};
|
};
|
||||||
|
|
||||||
using cstring_view = basic_cstring_view<char>;
|
using cstring_view = basic_cstring_view<char>;
|
||||||
|
|
@ -128,48 +119,44 @@ FMT_API void format_windows_error(buffer<char>& out, int error_code,
|
||||||
const char* message) noexcept;
|
const char* message) noexcept;
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_API std::system_error vwindows_error(int error_code, string_view format_str,
|
FMT_API std::system_error vwindows_error(int error_code, string_view fmt,
|
||||||
format_args args);
|
format_args args);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Constructs a `std::system_error` object with the description of the form
|
||||||
Constructs a :class:`std::system_error` object with the description
|
*
|
||||||
of the form
|
* <message>: <system-message>
|
||||||
|
*
|
||||||
.. parsed-literal::
|
* where `<message>` is the formatted message and `<system-message>` is the
|
||||||
*<message>*: *<system-message>*
|
* system message corresponding to the error code.
|
||||||
|
* `error_code` is a Windows error code as given by `GetLastError`.
|
||||||
where *<message>* is the formatted message and *<system-message>* is the
|
* If `error_code` is not a valid error code such as -1, the system message
|
||||||
system message corresponding to the error code.
|
* will look like "error -1".
|
||||||
*error_code* is a Windows error code as given by ``GetLastError``.
|
*
|
||||||
If *error_code* is not a valid error code such as -1, the system message
|
* **Example**:
|
||||||
will look like "error -1".
|
*
|
||||||
|
* // This throws a system_error with the description
|
||||||
**Example**::
|
* // cannot open file 'foo': The system cannot find the file specified.
|
||||||
|
* // or similar (system message may vary) if the file doesn't exist.
|
||||||
// This throws a system_error with the description
|
* const char *filename = "foo";
|
||||||
// cannot open file 'madeup': The system cannot find the file specified.
|
* LPOFSTRUCT of = LPOFSTRUCT();
|
||||||
// or similar (system message may vary).
|
* HFILE file = OpenFile(filename, &of, OF_READ);
|
||||||
const char *filename = "madeup";
|
* if (file == HFILE_ERROR) {
|
||||||
LPOFSTRUCT of = LPOFSTRUCT();
|
* throw fmt::windows_error(GetLastError(),
|
||||||
HFILE file = OpenFile(filename, &of, OF_READ);
|
* "cannot open file '{}'", filename);
|
||||||
if (file == HFILE_ERROR) {
|
* }
|
||||||
throw fmt::windows_error(GetLastError(),
|
*/
|
||||||
"cannot open file '{}'", filename);
|
template <typename... T>
|
||||||
}
|
auto windows_error(int error_code, string_view message, const T&... args)
|
||||||
\endrst
|
-> std::system_error {
|
||||||
*/
|
return vwindows_error(error_code, message, vargs<T...>{{args...}});
|
||||||
template <typename... Args>
|
|
||||||
std::system_error windows_error(int error_code, string_view message,
|
|
||||||
const Args&... args) {
|
|
||||||
return vwindows_error(error_code, message, fmt::make_format_args(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reports a Windows error without throwing an exception.
|
// Reports a Windows error without throwing an exception.
|
||||||
// Can be used to report errors from destructors.
|
// Can be used to report errors from destructors.
|
||||||
FMT_API void report_windows_error(int error_code, const char* message) noexcept;
|
FMT_API void report_windows_error(int error_code, const char* message) noexcept;
|
||||||
#else
|
#else
|
||||||
inline const std::error_category& system_category() noexcept {
|
inline auto system_category() noexcept -> const std::error_category& {
|
||||||
return std::system_category();
|
return std::system_category();
|
||||||
}
|
}
|
||||||
#endif // _WIN32
|
#endif // _WIN32
|
||||||
|
|
@ -177,8 +164,8 @@ inline const std::error_category& system_category() noexcept {
|
||||||
// std::system is not available on some platforms such as iOS (#2248).
|
// std::system is not available on some platforms such as iOS (#2248).
|
||||||
#ifdef __OSX__
|
#ifdef __OSX__
|
||||||
template <typename S, typename... Args, typename Char = char_t<S>>
|
template <typename S, typename... Args, typename Char = char_t<S>>
|
||||||
void say(const S& format_str, Args&&... args) {
|
void say(const S& fmt, Args&&... args) {
|
||||||
std::system(format("say \"{}\"", format(format_str, args...)).c_str());
|
std::system(format("say \"{}\"", format(fmt, args...)).c_str());
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
@ -189,24 +176,24 @@ class buffered_file {
|
||||||
|
|
||||||
friend class file;
|
friend class file;
|
||||||
|
|
||||||
explicit buffered_file(FILE* f) : file_(f) {}
|
inline explicit buffered_file(FILE* f) : file_(f) {}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
buffered_file(const buffered_file&) = delete;
|
buffered_file(const buffered_file&) = delete;
|
||||||
void operator=(const buffered_file&) = delete;
|
void operator=(const buffered_file&) = delete;
|
||||||
|
|
||||||
// Constructs a buffered_file object which doesn't represent any file.
|
// Constructs a buffered_file object which doesn't represent any file.
|
||||||
buffered_file() noexcept : file_(nullptr) {}
|
inline buffered_file() noexcept : file_(nullptr) {}
|
||||||
|
|
||||||
// Destroys the object closing the file it represents if any.
|
// Destroys the object closing the file it represents if any.
|
||||||
FMT_API ~buffered_file() noexcept;
|
FMT_API ~buffered_file() noexcept;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
buffered_file(buffered_file&& other) noexcept : file_(other.file_) {
|
inline buffered_file(buffered_file&& other) noexcept : file_(other.file_) {
|
||||||
other.file_ = nullptr;
|
other.file_ = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
buffered_file& operator=(buffered_file&& other) {
|
inline auto operator=(buffered_file&& other) -> buffered_file& {
|
||||||
close();
|
close();
|
||||||
file_ = other.file_;
|
file_ = other.file_;
|
||||||
other.file_ = nullptr;
|
other.file_ = nullptr;
|
||||||
|
|
@ -220,21 +207,20 @@ class buffered_file {
|
||||||
FMT_API void close();
|
FMT_API void close();
|
||||||
|
|
||||||
// Returns the pointer to a FILE object representing this file.
|
// Returns the pointer to a FILE object representing this file.
|
||||||
FILE* get() const noexcept { return file_; }
|
inline auto get() const noexcept -> FILE* { return file_; }
|
||||||
|
|
||||||
FMT_API int descriptor() const;
|
FMT_API auto descriptor() const -> int;
|
||||||
|
|
||||||
void vprint(string_view format_str, format_args args) {
|
template <typename... T>
|
||||||
fmt::vprint(file_, format_str, args);
|
inline void print(string_view fmt, const T&... args) {
|
||||||
}
|
fmt::vargs<T...> vargs = {{args...}};
|
||||||
|
detail::is_locking<T...>() ? fmt::vprint_buffered(file_, fmt, vargs)
|
||||||
template <typename... Args>
|
: fmt::vprint(file_, fmt, vargs);
|
||||||
inline void print(string_view format_str, const Args&... args) {
|
|
||||||
vprint(format_str, fmt::make_format_args(args...));
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
#if FMT_USE_FCNTL
|
#if FMT_USE_FCNTL
|
||||||
|
|
||||||
// A file. Closed file is represented by a file object with descriptor -1.
|
// A file. Closed file is represented by a file object with descriptor -1.
|
||||||
// Methods that are not declared with noexcept may throw
|
// Methods that are not declared with noexcept may throw
|
||||||
// fmt::system_error in case of failure. Note that some errors such as
|
// fmt::system_error in case of failure. Note that some errors such as
|
||||||
|
|
@ -248,6 +234,8 @@ class FMT_API file {
|
||||||
// Constructs a file object with a given descriptor.
|
// Constructs a file object with a given descriptor.
|
||||||
explicit file(int fd) : fd_(fd) {}
|
explicit file(int fd) : fd_(fd) {}
|
||||||
|
|
||||||
|
friend struct pipe;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Possible values for the oflag argument to the constructor.
|
// Possible values for the oflag argument to the constructor.
|
||||||
enum {
|
enum {
|
||||||
|
|
@ -260,7 +248,7 @@ class FMT_API file {
|
||||||
};
|
};
|
||||||
|
|
||||||
// Constructs a file object which doesn't represent any file.
|
// Constructs a file object which doesn't represent any file.
|
||||||
file() noexcept : fd_(-1) {}
|
inline file() noexcept : fd_(-1) {}
|
||||||
|
|
||||||
// Opens a file and constructs a file object representing this file.
|
// Opens a file and constructs a file object representing this file.
|
||||||
file(cstring_view path, int oflag);
|
file(cstring_view path, int oflag);
|
||||||
|
|
@ -269,10 +257,10 @@ class FMT_API file {
|
||||||
file(const file&) = delete;
|
file(const file&) = delete;
|
||||||
void operator=(const file&) = delete;
|
void operator=(const file&) = delete;
|
||||||
|
|
||||||
file(file&& other) noexcept : fd_(other.fd_) { other.fd_ = -1; }
|
inline file(file&& other) noexcept : fd_(other.fd_) { other.fd_ = -1; }
|
||||||
|
|
||||||
// Move assignment is not noexcept because close may throw.
|
// Move assignment is not noexcept because close may throw.
|
||||||
file& operator=(file&& other) {
|
inline auto operator=(file&& other) -> file& {
|
||||||
close();
|
close();
|
||||||
fd_ = other.fd_;
|
fd_ = other.fd_;
|
||||||
other.fd_ = -1;
|
other.fd_ = -1;
|
||||||
|
|
@ -283,24 +271,24 @@ class FMT_API file {
|
||||||
~file() noexcept;
|
~file() noexcept;
|
||||||
|
|
||||||
// Returns the file descriptor.
|
// Returns the file descriptor.
|
||||||
int descriptor() const noexcept { return fd_; }
|
inline auto descriptor() const noexcept -> int { return fd_; }
|
||||||
|
|
||||||
// Closes the file.
|
// Closes the file.
|
||||||
void close();
|
void close();
|
||||||
|
|
||||||
// Returns the file size. The size has signed type for consistency with
|
// Returns the file size. The size has signed type for consistency with
|
||||||
// stat::st_size.
|
// stat::st_size.
|
||||||
long long size() const;
|
auto size() const -> long long;
|
||||||
|
|
||||||
// Attempts to read count bytes from the file into the specified buffer.
|
// Attempts to read count bytes from the file into the specified buffer.
|
||||||
size_t read(void* buffer, size_t count);
|
auto read(void* buffer, size_t count) -> size_t;
|
||||||
|
|
||||||
// Attempts to write count bytes from the specified buffer to the file.
|
// Attempts to write count bytes from the specified buffer to the file.
|
||||||
size_t write(const void* buffer, size_t count);
|
auto write(const void* buffer, size_t count) -> size_t;
|
||||||
|
|
||||||
// Duplicates a file descriptor with the dup function and returns
|
// Duplicates a file descriptor with the dup function and returns
|
||||||
// the duplicate as a file object.
|
// the duplicate as a file object.
|
||||||
static file dup(int fd);
|
static auto dup(int fd) -> file;
|
||||||
|
|
||||||
// Makes fd be the copy of this file descriptor, closing fd first if
|
// Makes fd be the copy of this file descriptor, closing fd first if
|
||||||
// necessary.
|
// necessary.
|
||||||
|
|
@ -310,13 +298,9 @@ class FMT_API file {
|
||||||
// necessary.
|
// necessary.
|
||||||
void dup2(int fd, std::error_code& ec) noexcept;
|
void dup2(int fd, std::error_code& ec) noexcept;
|
||||||
|
|
||||||
// Creates a pipe setting up read_end and write_end file objects for reading
|
|
||||||
// and writing respectively.
|
|
||||||
static void pipe(file& read_end, file& write_end);
|
|
||||||
|
|
||||||
// Creates a buffered_file object associated with this file and detaches
|
// Creates a buffered_file object associated with this file and detaches
|
||||||
// this file object from the file.
|
// this file object from the file.
|
||||||
buffered_file fdopen(const char* mode);
|
auto fdopen(const char* mode) -> buffered_file;
|
||||||
|
|
||||||
# if defined(_WIN32) && !defined(__MINGW32__)
|
# if defined(_WIN32) && !defined(__MINGW32__)
|
||||||
// Opens a file and constructs a file object representing this file by
|
// Opens a file and constructs a file object representing this file by
|
||||||
|
|
@ -325,15 +309,24 @@ class FMT_API file {
|
||||||
# endif
|
# endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct FMT_API pipe {
|
||||||
|
file read_end;
|
||||||
|
file write_end;
|
||||||
|
|
||||||
|
// Creates a pipe setting up read_end and write_end file objects for reading
|
||||||
|
// and writing respectively.
|
||||||
|
pipe();
|
||||||
|
};
|
||||||
|
|
||||||
// Returns the memory page size.
|
// Returns the memory page size.
|
||||||
long getpagesize();
|
auto getpagesize() -> long;
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
struct buffer_size {
|
struct buffer_size {
|
||||||
buffer_size() = default;
|
constexpr buffer_size() = default;
|
||||||
size_t value = 0;
|
size_t value = 0;
|
||||||
buffer_size operator=(size_t val) const {
|
FMT_CONSTEXPR auto operator=(size_t val) const -> buffer_size {
|
||||||
auto bs = buffer_size();
|
auto bs = buffer_size();
|
||||||
bs.value = val;
|
bs.value = val;
|
||||||
return bs;
|
return bs;
|
||||||
|
|
@ -344,7 +337,7 @@ struct ostream_params {
|
||||||
int oflag = file::WRONLY | file::CREATE | file::TRUNC;
|
int oflag = file::WRONLY | file::CREATE | file::TRUNC;
|
||||||
size_t buffer_size = BUFSIZ > 32768 ? BUFSIZ : 32768;
|
size_t buffer_size = BUFSIZ > 32768 ? BUFSIZ : 32768;
|
||||||
|
|
||||||
ostream_params() {}
|
constexpr ostream_params() {}
|
||||||
|
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
ostream_params(T... params, int new_oflag) : ostream_params(params...) {
|
ostream_params(T... params, int new_oflag) : ostream_params(params...) {
|
||||||
|
|
@ -365,82 +358,65 @@ struct ostream_params {
|
||||||
# endif
|
# endif
|
||||||
};
|
};
|
||||||
|
|
||||||
class file_buffer final : public buffer<char> {
|
} // namespace detail
|
||||||
|
|
||||||
|
FMT_INLINE_VARIABLE constexpr auto buffer_size = detail::buffer_size();
|
||||||
|
|
||||||
|
/// A fast buffered output stream for writing from a single thread. Writing from
|
||||||
|
/// multiple threads without external synchronization may result in a data race.
|
||||||
|
class ostream : private detail::buffer<char> {
|
||||||
|
private:
|
||||||
file file_;
|
file file_;
|
||||||
|
|
||||||
FMT_API void grow(size_t) override;
|
FMT_API ostream(cstring_view path, const detail::ostream_params& params);
|
||||||
|
|
||||||
|
FMT_API static void grow(buffer<char>& buf, size_t);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_API file_buffer(cstring_view path, const ostream_params& params);
|
FMT_API ostream(ostream&& other) noexcept;
|
||||||
FMT_API file_buffer(file_buffer&& other);
|
FMT_API ~ostream();
|
||||||
FMT_API ~file_buffer();
|
|
||||||
|
|
||||||
void flush() {
|
operator writer() {
|
||||||
|
detail::buffer<char>& buf = *this;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void flush() {
|
||||||
if (size() == 0) return;
|
if (size() == 0) return;
|
||||||
file_.write(data(), size() * sizeof(data()[0]));
|
file_.write(data(), size() * sizeof(data()[0]));
|
||||||
clear();
|
clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void close() {
|
template <typename... T>
|
||||||
|
friend auto output_file(cstring_view path, T... params) -> ostream;
|
||||||
|
|
||||||
|
inline void close() {
|
||||||
flush();
|
flush();
|
||||||
file_.close();
|
file_.close();
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace detail
|
/// Formats `args` according to specifications in `fmt` and writes the
|
||||||
|
/// output to the file.
|
||||||
// Added {} below to work around default constructor error known to
|
|
||||||
// occur in Xcode versions 7.2.1 and 8.2.1.
|
|
||||||
constexpr detail::buffer_size buffer_size{};
|
|
||||||
|
|
||||||
/** A fast output stream which is not thread-safe. */
|
|
||||||
class FMT_API ostream {
|
|
||||||
private:
|
|
||||||
FMT_MSC_WARNING(suppress : 4251)
|
|
||||||
detail::file_buffer buffer_;
|
|
||||||
|
|
||||||
ostream(cstring_view path, const detail::ostream_params& params)
|
|
||||||
: buffer_(path, params) {}
|
|
||||||
|
|
||||||
public:
|
|
||||||
ostream(ostream&& other) : buffer_(std::move(other.buffer_)) {}
|
|
||||||
|
|
||||||
~ostream();
|
|
||||||
|
|
||||||
void flush() { buffer_.flush(); }
|
|
||||||
|
|
||||||
template <typename... T>
|
|
||||||
friend ostream output_file(cstring_view path, T... params);
|
|
||||||
|
|
||||||
void close() { buffer_.close(); }
|
|
||||||
|
|
||||||
/**
|
|
||||||
Formats ``args`` according to specifications in ``fmt`` and writes the
|
|
||||||
output to the file.
|
|
||||||
*/
|
|
||||||
template <typename... T> void print(format_string<T...> fmt, T&&... args) {
|
template <typename... T> void print(format_string<T...> fmt, T&&... args) {
|
||||||
vformat_to(detail::buffer_appender<char>(buffer_), fmt,
|
vformat_to(appender(*this), fmt.str, vargs<T...>{{args...}});
|
||||||
fmt::make_format_args(args...));
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Opens a file for writing. Supported parameters passed in `params`:
|
||||||
Opens a file for writing. Supported parameters passed in *params*:
|
*
|
||||||
|
* - `<integer>`: Flags passed to [open](
|
||||||
* ``<integer>``: Flags passed to `open
|
* https://pubs.opengroup.org/onlinepubs/007904875/functions/open.html)
|
||||||
<https://pubs.opengroup.org/onlinepubs/007904875/functions/open.html>`_
|
* (`file::WRONLY | file::CREATE | file::TRUNC` by default)
|
||||||
(``file::WRONLY | file::CREATE | file::TRUNC`` by default)
|
* - `buffer_size=<integer>`: Output buffer size
|
||||||
* ``buffer_size=<integer>``: Output buffer size
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* auto out = fmt::output_file("guide.txt");
|
||||||
auto out = fmt::output_file("guide.txt");
|
* out.print("Don't {}", "Panic");
|
||||||
out.print("Don't {}", "Panic");
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
inline ostream output_file(cstring_view path, T... params) {
|
inline auto output_file(cstring_view path, T... params) -> ostream {
|
||||||
return {path, detail::ostream_params(params...)};
|
return {path, detail::ostream_params(params...)};
|
||||||
}
|
}
|
||||||
#endif // FMT_USE_FCNTL
|
#endif // FMT_USE_FCNTL
|
||||||
|
|
|
||||||
198
deps/fmt/include/fmt/ostream.h
vendored
198
deps/fmt/include/fmt/ostream.h
vendored
|
|
@ -8,23 +8,33 @@
|
||||||
#ifndef FMT_OSTREAM_H_
|
#ifndef FMT_OSTREAM_H_
|
||||||
#define FMT_OSTREAM_H_
|
#define FMT_OSTREAM_H_
|
||||||
|
|
||||||
#include <fstream> // std::filebuf
|
#ifndef FMT_MODULE
|
||||||
|
# include <fstream> // std::filebuf
|
||||||
#if defined(_WIN32) && defined(__GLIBCXX__)
|
|
||||||
# include <ext/stdio_filebuf.h>
|
|
||||||
# include <ext/stdio_sync_filebuf.h>
|
|
||||||
#elif defined(_WIN32) && defined(_LIBCPP_VERSION)
|
|
||||||
# include <__std_stream>
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#ifdef _WIN32
|
||||||
|
# ifdef __GLIBCXX__
|
||||||
|
# include <ext/stdio_filebuf.h>
|
||||||
|
# include <ext/stdio_sync_filebuf.h>
|
||||||
|
# endif
|
||||||
|
# include <io.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "chrono.h" // formatbuf
|
||||||
|
|
||||||
|
#ifdef _MSVC_STL_UPDATE
|
||||||
|
# define FMT_MSVC_STL_UPDATE _MSVC_STL_UPDATE
|
||||||
|
#elif defined(_MSC_VER) && _MSC_VER < 1912 // VS 15.5
|
||||||
|
# define FMT_MSVC_STL_UPDATE _MSVC_LANG
|
||||||
|
#else
|
||||||
|
# define FMT_MSVC_STL_UPDATE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
// Generate a unique explicit instantion in every translation unit using a tag
|
// Generate a unique explicit instantiation in every translation unit using a
|
||||||
// type in an anonymous namespace.
|
// tag type in an anonymous namespace.
|
||||||
namespace {
|
namespace {
|
||||||
struct file_access_tag {};
|
struct file_access_tag {};
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
@ -33,49 +43,18 @@ class file_access {
|
||||||
friend auto get_file(BufType& obj) -> FILE* { return obj.*FileMemberPtr; }
|
friend auto get_file(BufType& obj) -> FILE* { return obj.*FileMemberPtr; }
|
||||||
};
|
};
|
||||||
|
|
||||||
#if FMT_MSC_VERSION
|
#if FMT_MSVC_STL_UPDATE
|
||||||
template class file_access<file_access_tag, std::filebuf,
|
template class file_access<file_access_tag, std::filebuf,
|
||||||
&std::filebuf::_Myfile>;
|
&std::filebuf::_Myfile>;
|
||||||
auto get_file(std::filebuf&) -> FILE*;
|
auto get_file(std::filebuf&) -> FILE*;
|
||||||
#elif defined(_WIN32) && defined(_LIBCPP_VERSION)
|
|
||||||
template class file_access<file_access_tag, std::__stdoutbuf<char>,
|
|
||||||
&std::__stdoutbuf<char>::__file_>;
|
|
||||||
auto get_file(std::__stdoutbuf<char>&) -> FILE*;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
inline bool write_ostream_unicode(std::ostream& os, fmt::string_view data) {
|
|
||||||
#if FMT_MSC_VERSION
|
|
||||||
if (auto* buf = dynamic_cast<std::filebuf*>(os.rdbuf()))
|
|
||||||
if (FILE* f = get_file(*buf)) return write_console(f, data);
|
|
||||||
#elif defined(_WIN32) && defined(__GLIBCXX__)
|
|
||||||
auto* rdbuf = os.rdbuf();
|
|
||||||
FILE* c_file;
|
|
||||||
if (auto* sfbuf = dynamic_cast<__gnu_cxx::stdio_sync_filebuf<char>*>(rdbuf))
|
|
||||||
c_file = sfbuf->file();
|
|
||||||
else if (auto* fbuf = dynamic_cast<__gnu_cxx::stdio_filebuf<char>*>(rdbuf))
|
|
||||||
c_file = fbuf->file();
|
|
||||||
else
|
|
||||||
return false;
|
|
||||||
if (c_file) return write_console(c_file, data);
|
|
||||||
#elif defined(_WIN32) && defined(_LIBCPP_VERSION)
|
|
||||||
if (auto* buf = dynamic_cast<std::__stdoutbuf<char>*>(os.rdbuf()))
|
|
||||||
if (FILE* f = get_file(*buf)) return write_console(f, data);
|
|
||||||
#else
|
|
||||||
ignore_unused(os, data);
|
|
||||||
#endif
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
inline bool write_ostream_unicode(std::wostream&,
|
|
||||||
fmt::basic_string_view<wchar_t>) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write the content of buf to os.
|
// Write the content of buf to os.
|
||||||
// It is a separate function rather than a part of vprint to simplify testing.
|
// It is a separate function rather than a part of vprint to simplify testing.
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
void write_buffer(std::basic_ostream<Char>& os, buffer<Char>& buf) {
|
void write_buffer(std::basic_ostream<Char>& os, buffer<Char>& buf) {
|
||||||
const Char* buf_data = buf.data();
|
const Char* buf_data = buf.data();
|
||||||
using unsigned_streamsize = std::make_unsigned<std::streamsize>::type;
|
using unsigned_streamsize = make_unsigned_t<std::streamsize>;
|
||||||
unsigned_streamsize size = buf.size();
|
unsigned_streamsize size = buf.size();
|
||||||
unsigned_streamsize max_size = to_unsigned(max_value<std::streamsize>());
|
unsigned_streamsize max_size = to_unsigned(max_value<std::streamsize>());
|
||||||
do {
|
do {
|
||||||
|
|
@ -86,20 +65,9 @@ void write_buffer(std::basic_ostream<Char>& os, buffer<Char>& buf) {
|
||||||
} while (size != 0);
|
} while (size != 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char, typename T>
|
template <typename T> struct streamed_view {
|
||||||
void format_value(buffer<Char>& buf, const T& value,
|
const T& value;
|
||||||
locale_ref loc = locale_ref()) {
|
};
|
||||||
auto&& format_buf = formatbuf<std::basic_streambuf<Char>>(buf);
|
|
||||||
auto&& output = std::basic_ostream<Char>(&format_buf);
|
|
||||||
#if !defined(FMT_STATIC_THOUSANDS_SEPARATOR)
|
|
||||||
if (loc) output.imbue(loc.get<std::locale>());
|
|
||||||
#endif
|
|
||||||
output << value;
|
|
||||||
output.exceptions(std::ios_base::failbit | std::ios_base::badbit);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T> struct streamed_view { const T& value; };
|
|
||||||
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
// Formats an object of type T that has an overloaded ostream operator<<.
|
// Formats an object of type T that has an overloaded ostream operator<<.
|
||||||
|
|
@ -107,11 +75,14 @@ template <typename Char>
|
||||||
struct basic_ostream_formatter : formatter<basic_string_view<Char>, Char> {
|
struct basic_ostream_formatter : formatter<basic_string_view<Char>, Char> {
|
||||||
void set_debug_format() = delete;
|
void set_debug_format() = delete;
|
||||||
|
|
||||||
template <typename T, typename OutputIt>
|
template <typename T, typename Context>
|
||||||
auto format(const T& value, basic_format_context<OutputIt, Char>& ctx) const
|
auto format(const T& value, Context& ctx) const -> decltype(ctx.out()) {
|
||||||
-> OutputIt {
|
|
||||||
auto buffer = basic_memory_buffer<Char>();
|
auto buffer = basic_memory_buffer<Char>();
|
||||||
detail::format_value(buffer, value, ctx.locale());
|
auto&& formatbuf = detail::formatbuf<std::basic_streambuf<Char>>(buffer);
|
||||||
|
auto&& output = std::basic_ostream<Char>(&formatbuf);
|
||||||
|
output.imbue(std::locale::classic()); // The default is always unlocalized.
|
||||||
|
output << value;
|
||||||
|
output.exceptions(std::ios_base::failbit | std::ios_base::badbit);
|
||||||
return formatter<basic_string_view<Char>, Char>::format(
|
return formatter<basic_string_view<Char>, Char>::format(
|
||||||
{buffer.data(), buffer.size()}, ctx);
|
{buffer.data(), buffer.size()}, ctx);
|
||||||
}
|
}
|
||||||
|
|
@ -122,86 +93,73 @@ using ostream_formatter = basic_ostream_formatter<char>;
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
struct formatter<detail::streamed_view<T>, Char>
|
struct formatter<detail::streamed_view<T>, Char>
|
||||||
: basic_ostream_formatter<Char> {
|
: basic_ostream_formatter<Char> {
|
||||||
template <typename OutputIt>
|
template <typename Context>
|
||||||
auto format(detail::streamed_view<T> view,
|
auto format(detail::streamed_view<T> view, Context& ctx) const
|
||||||
basic_format_context<OutputIt, Char>& ctx) const -> OutputIt {
|
-> decltype(ctx.out()) {
|
||||||
return basic_ostream_formatter<Char>::format(view.value, ctx);
|
return basic_ostream_formatter<Char>::format(view.value, ctx);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Returns a view that formats `value` via an ostream `operator<<`.
|
||||||
Returns a view that formats `value` via an ostream ``operator<<``.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print("Current thread id: {}\n",
|
||||||
fmt::print("Current thread id: {}\n",
|
* fmt::streamed(std::this_thread::get_id()));
|
||||||
fmt::streamed(std::this_thread::get_id()));
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename T>
|
template <typename T>
|
||||||
auto streamed(const T& value) -> detail::streamed_view<T> {
|
constexpr auto streamed(const T& value) -> detail::streamed_view<T> {
|
||||||
return {value};
|
return {value};
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace detail {
|
inline void vprint(std::ostream& os, string_view fmt, format_args args) {
|
||||||
|
|
||||||
inline void vprint_directly(std::ostream& os, string_view format_str,
|
|
||||||
format_args args) {
|
|
||||||
auto buffer = memory_buffer();
|
auto buffer = memory_buffer();
|
||||||
detail::vformat_to(buffer, format_str, args);
|
detail::vformat_to(buffer, fmt, args);
|
||||||
detail::write_buffer(os, buffer);
|
FILE* f = nullptr;
|
||||||
}
|
#if FMT_MSVC_STL_UPDATE && FMT_USE_RTTI
|
||||||
|
if (auto* buf = dynamic_cast<std::filebuf*>(os.rdbuf()))
|
||||||
} // namespace detail
|
f = detail::get_file(*buf);
|
||||||
|
#elif defined(_WIN32) && defined(__GLIBCXX__) && FMT_USE_RTTI
|
||||||
FMT_EXPORT template <typename Char>
|
auto* rdbuf = os.rdbuf();
|
||||||
void vprint(std::basic_ostream<Char>& os,
|
if (auto* sfbuf = dynamic_cast<__gnu_cxx::stdio_sync_filebuf<char>*>(rdbuf))
|
||||||
basic_string_view<type_identity_t<Char>> format_str,
|
f = sfbuf->file();
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
else if (auto* fbuf = dynamic_cast<__gnu_cxx::stdio_filebuf<char>*>(rdbuf))
|
||||||
auto buffer = basic_memory_buffer<Char>();
|
f = fbuf->file();
|
||||||
detail::vformat_to(buffer, format_str, args);
|
#endif
|
||||||
if (detail::write_ostream_unicode(os, {buffer.data(), buffer.size()})) return;
|
#ifdef _WIN32
|
||||||
|
if (f) {
|
||||||
|
int fd = _fileno(f);
|
||||||
|
if (_isatty(fd)) {
|
||||||
|
os.flush();
|
||||||
|
if (detail::write_console(fd, {buffer.data(), buffer.size()})) return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
detail::ignore_unused(f);
|
||||||
detail::write_buffer(os, buffer);
|
detail::write_buffer(os, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Prints formatted data to the stream `os`.
|
||||||
Prints formatted data to the stream *os*.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print(cerr, "Don't {}!", "panic");
|
||||||
fmt::print(cerr, "Don't {}!", "panic");
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
FMT_EXPORT template <typename... T>
|
FMT_EXPORT template <typename... T>
|
||||||
void print(std::ostream& os, format_string<T...> fmt, T&&... args) {
|
void print(std::ostream& os, format_string<T...> fmt, T&&... args) {
|
||||||
const auto& vargs = fmt::make_format_args(args...);
|
fmt::vargs<T...> vargs = {{args...}};
|
||||||
if (detail::is_utf8())
|
if (detail::const_check(detail::use_utf8)) return vprint(os, fmt.str, vargs);
|
||||||
vprint(os, fmt, vargs);
|
auto buffer = memory_buffer();
|
||||||
else
|
detail::vformat_to(buffer, fmt.str, vargs);
|
||||||
detail::vprint_directly(os, fmt, vargs);
|
detail::write_buffer(os, buffer);
|
||||||
}
|
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename... Args>
|
|
||||||
void print(std::wostream& os,
|
|
||||||
basic_format_string<wchar_t, type_identity_t<Args>...> fmt,
|
|
||||||
Args&&... args) {
|
|
||||||
vprint(os, fmt, fmt::make_format_args<buffer_context<wchar_t>>(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_EXPORT template <typename... T>
|
FMT_EXPORT template <typename... T>
|
||||||
void println(std::ostream& os, format_string<T...> fmt, T&&... args) {
|
void println(std::ostream& os, format_string<T...> fmt, T&&... args) {
|
||||||
fmt::print(os, "{}\n", fmt::format(fmt, std::forward<T>(args)...));
|
fmt::print(os, FMT_STRING("{}\n"),
|
||||||
}
|
fmt::format(fmt, std::forward<T>(args)...));
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename... Args>
|
|
||||||
void println(std::wostream& os,
|
|
||||||
basic_format_string<wchar_t, type_identity_t<Args>...> fmt,
|
|
||||||
Args&&... args) {
|
|
||||||
print(os, L"{}\n", fmt::format(fmt, std::forward<Args>(args)...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
|
||||||
455
deps/fmt/include/fmt/printf.h
vendored
455
deps/fmt/include/fmt/printf.h
vendored
|
|
@ -8,60 +8,71 @@
|
||||||
#ifndef FMT_PRINTF_H_
|
#ifndef FMT_PRINTF_H_
|
||||||
#define FMT_PRINTF_H_
|
#define FMT_PRINTF_H_
|
||||||
|
|
||||||
#include <algorithm> // std::max
|
#ifndef FMT_MODULE
|
||||||
#include <limits> // std::numeric_limits
|
# include <algorithm> // std::find
|
||||||
|
# include <limits> // std::numeric_limits
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
FMT_BEGIN_EXPORT
|
FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
template <typename T> struct printf_formatter { printf_formatter() = delete; };
|
|
||||||
|
|
||||||
template <typename Char> class basic_printf_context {
|
template <typename Char> class basic_printf_context {
|
||||||
private:
|
private:
|
||||||
detail::buffer_appender<Char> out_;
|
basic_appender<Char> out_;
|
||||||
basic_format_args<basic_printf_context> args_;
|
basic_format_args<basic_printf_context> args_;
|
||||||
|
|
||||||
|
static_assert(std::is_same<Char, char>::value ||
|
||||||
|
std::is_same<Char, wchar_t>::value,
|
||||||
|
"Unsupported code unit type.");
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
using parse_context_type = basic_format_parse_context<Char>;
|
enum { builtin_types = 1 };
|
||||||
template <typename T> using formatter_type = printf_formatter<T>;
|
|
||||||
|
|
||||||
/**
|
/// Constructs a `printf_context` object. References to the arguments are
|
||||||
\rst
|
/// stored in the context object so make sure they have appropriate lifetimes.
|
||||||
Constructs a ``printf_context`` object. References to the arguments are
|
basic_printf_context(basic_appender<Char> out,
|
||||||
stored in the context object so make sure they have appropriate lifetimes.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
basic_printf_context(detail::buffer_appender<Char> out,
|
|
||||||
basic_format_args<basic_printf_context> args)
|
basic_format_args<basic_printf_context> args)
|
||||||
: out_(out), args_(args) {}
|
: out_(out), args_(args) {}
|
||||||
|
|
||||||
auto out() -> detail::buffer_appender<Char> { return out_; }
|
auto out() -> basic_appender<Char> { return out_; }
|
||||||
void advance_to(detail::buffer_appender<Char>) {}
|
void advance_to(basic_appender<Char>) {}
|
||||||
|
|
||||||
auto locale() -> detail::locale_ref { return {}; }
|
auto locale() -> locale_ref { return {}; }
|
||||||
|
|
||||||
auto arg(int id) const -> basic_format_arg<basic_printf_context> {
|
auto arg(int id) const -> basic_format_arg<basic_printf_context> {
|
||||||
return args_.get(id);
|
return args_.get(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_CONSTEXPR void on_error(const char* message) {
|
|
||||||
detail::error_handler().on_error(message);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
|
// Return the result via the out param to workaround gcc bug 77539.
|
||||||
|
template <bool IS_CONSTEXPR, typename T, typename Ptr = const T*>
|
||||||
|
FMT_CONSTEXPR auto find(Ptr first, Ptr last, T value, Ptr& out) -> bool {
|
||||||
|
for (out = first; out != last; ++out) {
|
||||||
|
if (*out == value) return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <>
|
||||||
|
inline auto find<false, char>(const char* first, const char* last, char value,
|
||||||
|
const char*& out) -> bool {
|
||||||
|
out =
|
||||||
|
static_cast<const char*>(memchr(first, value, to_unsigned(last - first)));
|
||||||
|
return out != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
// Checks if a value fits in int - used to avoid warnings about comparing
|
// Checks if a value fits in int - used to avoid warnings about comparing
|
||||||
// signed and unsigned integers.
|
// signed and unsigned integers.
|
||||||
template <bool IsSigned> struct int_checker {
|
template <bool IS_SIGNED> struct int_checker {
|
||||||
template <typename T> static auto fits_in_int(T value) -> bool {
|
template <typename T> static auto fits_in_int(T value) -> bool {
|
||||||
unsigned max = max_value<int>();
|
return value <= to_unsigned(max_value<int>());
|
||||||
return value <= max;
|
|
||||||
}
|
}
|
||||||
static auto fits_in_int(bool) -> bool { return true; }
|
inline static auto fits_in_int(bool) -> bool { return true; }
|
||||||
};
|
};
|
||||||
|
|
||||||
template <> struct int_checker<true> {
|
template <> struct int_checker<true> {
|
||||||
|
|
@ -69,20 +80,20 @@ template <> struct int_checker<true> {
|
||||||
return value >= (std::numeric_limits<int>::min)() &&
|
return value >= (std::numeric_limits<int>::min)() &&
|
||||||
value <= max_value<int>();
|
value <= max_value<int>();
|
||||||
}
|
}
|
||||||
static auto fits_in_int(int) -> bool { return true; }
|
inline static auto fits_in_int(int) -> bool { return true; }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct printf_precision_handler {
|
struct printf_precision_handler {
|
||||||
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
||||||
auto operator()(T value) -> int {
|
auto operator()(T value) -> int {
|
||||||
if (!int_checker<std::numeric_limits<T>::is_signed>::fits_in_int(value))
|
if (!int_checker<std::numeric_limits<T>::is_signed>::fits_in_int(value))
|
||||||
throw_format_error("number is too big");
|
report_error("number is too big");
|
||||||
return (std::max)(static_cast<int>(value), 0);
|
return max_of(static_cast<int>(value), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
||||||
auto operator()(T) -> int {
|
auto operator()(T) -> int {
|
||||||
throw_format_error("precision is not integer");
|
report_error("precision is not integer");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -102,7 +113,9 @@ struct is_zero_int {
|
||||||
|
|
||||||
template <typename T> struct make_unsigned_or_bool : std::make_unsigned<T> {};
|
template <typename T> struct make_unsigned_or_bool : std::make_unsigned<T> {};
|
||||||
|
|
||||||
template <> struct make_unsigned_or_bool<bool> { using type = bool; };
|
template <> struct make_unsigned_or_bool<bool> {
|
||||||
|
using type = bool;
|
||||||
|
};
|
||||||
|
|
||||||
template <typename T, typename Context> class arg_converter {
|
template <typename T, typename Context> class arg_converter {
|
||||||
private:
|
private:
|
||||||
|
|
@ -125,25 +138,19 @@ template <typename T, typename Context> class arg_converter {
|
||||||
using target_type = conditional_t<std::is_same<T, void>::value, U, T>;
|
using target_type = conditional_t<std::is_same<T, void>::value, U, T>;
|
||||||
if (const_check(sizeof(target_type) <= sizeof(int))) {
|
if (const_check(sizeof(target_type) <= sizeof(int))) {
|
||||||
// Extra casts are used to silence warnings.
|
// Extra casts are used to silence warnings.
|
||||||
if (is_signed) {
|
using unsigned_type = typename make_unsigned_or_bool<target_type>::type;
|
||||||
auto n = static_cast<int>(static_cast<target_type>(value));
|
if (is_signed)
|
||||||
arg_ = detail::make_arg<Context>(n);
|
arg_ = static_cast<int>(static_cast<target_type>(value));
|
||||||
} else {
|
else
|
||||||
using unsigned_type = typename make_unsigned_or_bool<target_type>::type;
|
arg_ = static_cast<unsigned>(static_cast<unsigned_type>(value));
|
||||||
auto n = static_cast<unsigned>(static_cast<unsigned_type>(value));
|
|
||||||
arg_ = detail::make_arg<Context>(n);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (is_signed) {
|
// glibc's printf doesn't sign extend arguments of smaller types:
|
||||||
// glibc's printf doesn't sign extend arguments of smaller types:
|
// std::printf("%lld", -42); // prints "4294967254"
|
||||||
// std::printf("%lld", -42); // prints "4294967254"
|
// but we don't have to do the same because it's a UB.
|
||||||
// but we don't have to do the same because it's a UB.
|
if (is_signed)
|
||||||
auto n = static_cast<long long>(value);
|
arg_ = static_cast<long long>(value);
|
||||||
arg_ = detail::make_arg<Context>(n);
|
else
|
||||||
} else {
|
arg_ = static_cast<typename make_unsigned_or_bool<U>::type>(value);
|
||||||
auto n = static_cast<typename make_unsigned_or_bool<U>::type>(value);
|
|
||||||
arg_ = detail::make_arg<Context>(n);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -157,7 +164,7 @@ template <typename T, typename Context> class arg_converter {
|
||||||
// unsigned).
|
// unsigned).
|
||||||
template <typename T, typename Context, typename Char>
|
template <typename T, typename Context, typename Char>
|
||||||
void convert_arg(basic_format_arg<Context>& arg, Char type) {
|
void convert_arg(basic_format_arg<Context>& arg, Char type) {
|
||||||
visit_format_arg(arg_converter<T, Context>(arg, type), arg);
|
arg.visit(arg_converter<T, Context>(arg, type));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Converts an integer argument to char for printf.
|
// Converts an integer argument to char for printf.
|
||||||
|
|
@ -170,8 +177,7 @@ template <typename Context> class char_converter {
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
||||||
void operator()(T value) {
|
void operator()(T value) {
|
||||||
auto c = static_cast<typename Context::char_type>(value);
|
arg_ = static_cast<typename Context::char_type>(value);
|
||||||
arg_ = detail::make_arg<Context>(c);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
||||||
|
|
@ -187,28 +193,28 @@ template <typename Char> struct get_cstring {
|
||||||
|
|
||||||
// Checks if an argument is a valid printf width specifier and sets
|
// Checks if an argument is a valid printf width specifier and sets
|
||||||
// left alignment if it is negative.
|
// left alignment if it is negative.
|
||||||
template <typename Char> class printf_width_handler {
|
class printf_width_handler {
|
||||||
private:
|
private:
|
||||||
format_specs<Char>& specs_;
|
format_specs& specs_;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit printf_width_handler(format_specs<Char>& specs) : specs_(specs) {}
|
inline explicit printf_width_handler(format_specs& specs) : specs_(specs) {}
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(std::is_integral<T>::value)>
|
||||||
auto operator()(T value) -> unsigned {
|
auto operator()(T value) -> unsigned {
|
||||||
auto width = static_cast<uint32_or_64_or_128_t<T>>(value);
|
auto width = static_cast<uint32_or_64_or_128_t<T>>(value);
|
||||||
if (detail::is_negative(value)) {
|
if (detail::is_negative(value)) {
|
||||||
specs_.align = align::left;
|
specs_.set_align(align::left);
|
||||||
width = 0 - width;
|
width = 0 - width;
|
||||||
}
|
}
|
||||||
unsigned int_max = max_value<int>();
|
unsigned int_max = to_unsigned(max_value<int>());
|
||||||
if (width > int_max) throw_format_error("number is too big");
|
if (width > int_max) report_error("number is too big");
|
||||||
return static_cast<unsigned>(width);
|
return static_cast<unsigned>(width);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(!std::is_integral<T>::value)>
|
||||||
auto operator()(T) -> unsigned {
|
auto operator()(T) -> unsigned {
|
||||||
throw_format_error("width is not integer");
|
report_error("width is not integer");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -216,12 +222,12 @@ template <typename Char> class printf_width_handler {
|
||||||
// Workaround for a bug with the XL compiler when initializing
|
// Workaround for a bug with the XL compiler when initializing
|
||||||
// printf_arg_formatter's base class.
|
// printf_arg_formatter's base class.
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
auto make_arg_formatter(buffer_appender<Char> iter, format_specs<Char>& s)
|
auto make_arg_formatter(basic_appender<Char> iter, format_specs& s)
|
||||||
-> arg_formatter<Char> {
|
-> arg_formatter<Char> {
|
||||||
return {iter, s, locale_ref()};
|
return {iter, s, locale_ref()};
|
||||||
}
|
}
|
||||||
|
|
||||||
// The ``printf`` argument formatter.
|
// The `printf` argument formatter.
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
class printf_arg_formatter : public arg_formatter<Char> {
|
class printf_arg_formatter : public arg_formatter<Char> {
|
||||||
private:
|
private:
|
||||||
|
|
@ -232,105 +238,96 @@ class printf_arg_formatter : public arg_formatter<Char> {
|
||||||
|
|
||||||
void write_null_pointer(bool is_string = false) {
|
void write_null_pointer(bool is_string = false) {
|
||||||
auto s = this->specs;
|
auto s = this->specs;
|
||||||
s.type = presentation_type::none;
|
s.set_type(presentation_type::none);
|
||||||
write_bytes(this->out, is_string ? "(null)" : "(nil)", s);
|
write_bytes<Char>(this->out, is_string ? "(null)" : "(nil)", s);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T> void write(T value) {
|
||||||
|
detail::write<Char>(this->out, value, this->specs, this->locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
printf_arg_formatter(buffer_appender<Char> iter, format_specs<Char>& s,
|
printf_arg_formatter(basic_appender<Char> iter, format_specs& s,
|
||||||
context_type& ctx)
|
context_type& ctx)
|
||||||
: base(make_arg_formatter(iter, s)), context_(ctx) {}
|
: base(make_arg_formatter(iter, s)), context_(ctx) {}
|
||||||
|
|
||||||
void operator()(monostate value) { base::operator()(value); }
|
void operator()(monostate value) { write(value); }
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(detail::is_integral<T>::value)>
|
template <typename T, FMT_ENABLE_IF(detail::is_integral<T>::value)>
|
||||||
void operator()(T value) {
|
void operator()(T value) {
|
||||||
// MSVC2013 fails to compile separate overloads for bool and Char so use
|
// MSVC2013 fails to compile separate overloads for bool and Char so use
|
||||||
// std::is_same instead.
|
// std::is_same instead.
|
||||||
if (!std::is_same<T, Char>::value) {
|
if (!std::is_same<T, Char>::value) {
|
||||||
base::operator()(value);
|
write(value);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
format_specs<Char> fmt_specs = this->specs;
|
format_specs s = this->specs;
|
||||||
if (fmt_specs.type != presentation_type::none &&
|
if (s.type() != presentation_type::none &&
|
||||||
fmt_specs.type != presentation_type::chr) {
|
s.type() != presentation_type::chr) {
|
||||||
return (*this)(static_cast<int>(value));
|
return (*this)(static_cast<int>(value));
|
||||||
}
|
}
|
||||||
fmt_specs.sign = sign::none;
|
s.set_sign(sign::none);
|
||||||
fmt_specs.alt = false;
|
s.clear_alt();
|
||||||
fmt_specs.fill[0] = ' '; // Ignore '0' flag for char types.
|
s.set_fill(' '); // Ignore '0' flag for char types.
|
||||||
// align::numeric needs to be overwritten here since the '0' flag is
|
// align::numeric needs to be overwritten here since the '0' flag is
|
||||||
// ignored for non-numeric types
|
// ignored for non-numeric types
|
||||||
if (fmt_specs.align == align::none || fmt_specs.align == align::numeric)
|
if (s.align() == align::none || s.align() == align::numeric)
|
||||||
fmt_specs.align = align::right;
|
s.set_align(align::right);
|
||||||
write<Char>(this->out, static_cast<Char>(value), fmt_specs);
|
detail::write<Char>(this->out, static_cast<Char>(value), s);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
|
template <typename T, FMT_ENABLE_IF(std::is_floating_point<T>::value)>
|
||||||
void operator()(T value) {
|
void operator()(T value) {
|
||||||
base::operator()(value);
|
write(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Formats a null-terminated C string. */
|
|
||||||
void operator()(const char* value) {
|
void operator()(const char* value) {
|
||||||
if (value)
|
if (value)
|
||||||
base::operator()(value);
|
write(value);
|
||||||
else
|
else
|
||||||
write_null_pointer(this->specs.type != presentation_type::pointer);
|
write_null_pointer(this->specs.type() != presentation_type::pointer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Formats a null-terminated wide C string. */
|
|
||||||
void operator()(const wchar_t* value) {
|
void operator()(const wchar_t* value) {
|
||||||
if (value)
|
if (value)
|
||||||
base::operator()(value);
|
write(value);
|
||||||
else
|
else
|
||||||
write_null_pointer(this->specs.type != presentation_type::pointer);
|
write_null_pointer(this->specs.type() != presentation_type::pointer);
|
||||||
}
|
}
|
||||||
|
|
||||||
void operator()(basic_string_view<Char> value) { base::operator()(value); }
|
void operator()(basic_string_view<Char> value) { write(value); }
|
||||||
|
|
||||||
/** Formats a pointer. */
|
|
||||||
void operator()(const void* value) {
|
void operator()(const void* value) {
|
||||||
if (value)
|
if (value)
|
||||||
base::operator()(value);
|
write(value);
|
||||||
else
|
else
|
||||||
write_null_pointer();
|
write_null_pointer();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Formats an argument of a custom (user-defined) type. */
|
|
||||||
void operator()(typename basic_format_arg<context_type>::handle handle) {
|
void operator()(typename basic_format_arg<context_type>::handle handle) {
|
||||||
auto parse_ctx = basic_format_parse_context<Char>({});
|
auto parse_ctx = parse_context<Char>({});
|
||||||
handle.format(parse_ctx, context_);
|
handle.format(parse_ctx, context_);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
void parse_flags(format_specs<Char>& specs, const Char*& it, const Char* end) {
|
void parse_flags(format_specs& specs, const Char*& it, const Char* end) {
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
switch (*it) {
|
switch (*it) {
|
||||||
case '-':
|
case '-': specs.set_align(align::left); break;
|
||||||
specs.align = align::left;
|
case '+': specs.set_sign(sign::plus); break;
|
||||||
break;
|
case '0': specs.set_fill('0'); break;
|
||||||
case '+':
|
|
||||||
specs.sign = sign::plus;
|
|
||||||
break;
|
|
||||||
case '0':
|
|
||||||
specs.fill[0] = '0';
|
|
||||||
break;
|
|
||||||
case ' ':
|
case ' ':
|
||||||
if (specs.sign != sign::plus) specs.sign = sign::space;
|
if (specs.sign() != sign::plus) specs.set_sign(sign::space);
|
||||||
break;
|
break;
|
||||||
case '#':
|
case '#': specs.set_alt(); break;
|
||||||
specs.alt = true;
|
default: return;
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char, typename GetArg>
|
template <typename Char, typename GetArg>
|
||||||
auto parse_header(const Char*& it, const Char* end, format_specs<Char>& specs,
|
auto parse_header(const Char*& it, const Char* end, format_specs& specs,
|
||||||
GetArg get_arg) -> int {
|
GetArg get_arg) -> int {
|
||||||
int arg_index = -1;
|
int arg_index = -1;
|
||||||
Char c = *it;
|
Char c = *it;
|
||||||
|
|
@ -342,11 +339,11 @@ auto parse_header(const Char*& it, const Char* end, format_specs<Char>& specs,
|
||||||
++it;
|
++it;
|
||||||
arg_index = value != -1 ? value : max_value<int>();
|
arg_index = value != -1 ? value : max_value<int>();
|
||||||
} else {
|
} else {
|
||||||
if (c == '0') specs.fill[0] = '0';
|
if (c == '0') specs.set_fill('0');
|
||||||
if (value != 0) {
|
if (value != 0) {
|
||||||
// Nonzero value means that we parsed width and don't need to
|
// Nonzero value means that we parsed width and don't need to
|
||||||
// parse it or flags again, so return now.
|
// parse it or flags again, so return now.
|
||||||
if (value == -1) throw_format_error("number is too big");
|
if (value == -1) report_error("number is too big");
|
||||||
specs.width = value;
|
specs.width = value;
|
||||||
return arg_index;
|
return arg_index;
|
||||||
}
|
}
|
||||||
|
|
@ -357,63 +354,47 @@ auto parse_header(const Char*& it, const Char* end, format_specs<Char>& specs,
|
||||||
if (it != end) {
|
if (it != end) {
|
||||||
if (*it >= '0' && *it <= '9') {
|
if (*it >= '0' && *it <= '9') {
|
||||||
specs.width = parse_nonnegative_int(it, end, -1);
|
specs.width = parse_nonnegative_int(it, end, -1);
|
||||||
if (specs.width == -1) throw_format_error("number is too big");
|
if (specs.width == -1) report_error("number is too big");
|
||||||
} else if (*it == '*') {
|
} else if (*it == '*') {
|
||||||
++it;
|
++it;
|
||||||
specs.width = static_cast<int>(visit_format_arg(
|
specs.width = static_cast<int>(
|
||||||
detail::printf_width_handler<Char>(specs), get_arg(-1)));
|
get_arg(-1).visit(detail::printf_width_handler(specs)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return arg_index;
|
return arg_index;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline auto parse_printf_presentation_type(char c, type t)
|
inline auto parse_printf_presentation_type(char c, type t, bool& upper)
|
||||||
-> presentation_type {
|
-> presentation_type {
|
||||||
using pt = presentation_type;
|
using pt = presentation_type;
|
||||||
constexpr auto integral_set = sint_set | uint_set | bool_set | char_set;
|
constexpr auto integral_set = sint_set | uint_set | bool_set | char_set;
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 'd':
|
case 'd': return in(t, integral_set) ? pt::dec : pt::none;
|
||||||
return in(t, integral_set) ? pt::dec : pt::none;
|
case 'o': return in(t, integral_set) ? pt::oct : pt::none;
|
||||||
case 'o':
|
case 'X': upper = true; FMT_FALLTHROUGH;
|
||||||
return in(t, integral_set) ? pt::oct : pt::none;
|
case 'x': return in(t, integral_set) ? pt::hex : pt::none;
|
||||||
case 'x':
|
case 'E': upper = true; FMT_FALLTHROUGH;
|
||||||
return in(t, integral_set) ? pt::hex_lower : pt::none;
|
case 'e': return in(t, float_set) ? pt::exp : pt::none;
|
||||||
case 'X':
|
case 'F': upper = true; FMT_FALLTHROUGH;
|
||||||
return in(t, integral_set) ? pt::hex_upper : pt::none;
|
case 'f': return in(t, float_set) ? pt::fixed : pt::none;
|
||||||
case 'a':
|
case 'G': upper = true; FMT_FALLTHROUGH;
|
||||||
return in(t, float_set) ? pt::hexfloat_lower : pt::none;
|
case 'g': return in(t, float_set) ? pt::general : pt::none;
|
||||||
case 'A':
|
case 'A': upper = true; FMT_FALLTHROUGH;
|
||||||
return in(t, float_set) ? pt::hexfloat_upper : pt::none;
|
case 'a': return in(t, float_set) ? pt::hexfloat : pt::none;
|
||||||
case 'e':
|
case 'c': return in(t, integral_set) ? pt::chr : pt::none;
|
||||||
return in(t, float_set) ? pt::exp_lower : pt::none;
|
case 's': return in(t, string_set | cstring_set) ? pt::string : pt::none;
|
||||||
case 'E':
|
case 'p': return in(t, pointer_set | cstring_set) ? pt::pointer : pt::none;
|
||||||
return in(t, float_set) ? pt::exp_upper : pt::none;
|
default: return pt::none;
|
||||||
case 'f':
|
|
||||||
return in(t, float_set) ? pt::fixed_lower : pt::none;
|
|
||||||
case 'F':
|
|
||||||
return in(t, float_set) ? pt::fixed_upper : pt::none;
|
|
||||||
case 'g':
|
|
||||||
return in(t, float_set) ? pt::general_lower : pt::none;
|
|
||||||
case 'G':
|
|
||||||
return in(t, float_set) ? pt::general_upper : pt::none;
|
|
||||||
case 'c':
|
|
||||||
return in(t, integral_set) ? pt::chr : pt::none;
|
|
||||||
case 's':
|
|
||||||
return in(t, string_set | cstring_set) ? pt::string : pt::none;
|
|
||||||
case 'p':
|
|
||||||
return in(t, pointer_set | cstring_set) ? pt::pointer : pt::none;
|
|
||||||
default:
|
|
||||||
return pt::none;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char, typename Context>
|
template <typename Char, typename Context>
|
||||||
void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
basic_format_args<Context> args) {
|
basic_format_args<Context> args) {
|
||||||
using iterator = buffer_appender<Char>;
|
using iterator = basic_appender<Char>;
|
||||||
auto out = iterator(buf);
|
auto out = iterator(buf);
|
||||||
auto context = basic_printf_context<Char>(out, args);
|
auto context = basic_printf_context<Char>(out, args);
|
||||||
auto parse_ctx = basic_format_parse_context<Char>(format);
|
auto parse_ctx = parse_context<Char>(format);
|
||||||
|
|
||||||
// Returns the argument with specified index or, if arg_index is -1, the next
|
// Returns the argument with specified index or, if arg_index is -1, the next
|
||||||
// argument.
|
// argument.
|
||||||
|
|
@ -422,7 +403,9 @@ void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
arg_index = parse_ctx.next_arg_id();
|
arg_index = parse_ctx.next_arg_id();
|
||||||
else
|
else
|
||||||
parse_ctx.check_arg_id(--arg_index);
|
parse_ctx.check_arg_id(--arg_index);
|
||||||
return detail::get_arg(context, arg_index);
|
auto arg = context.arg(arg_index);
|
||||||
|
if (!arg) report_error("argument not found");
|
||||||
|
return arg;
|
||||||
};
|
};
|
||||||
|
|
||||||
const Char* start = parse_ctx.begin();
|
const Char* start = parse_ctx.begin();
|
||||||
|
|
@ -441,12 +424,12 @@ void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
}
|
}
|
||||||
write(out, basic_string_view<Char>(start, to_unsigned(it - 1 - start)));
|
write(out, basic_string_view<Char>(start, to_unsigned(it - 1 - start)));
|
||||||
|
|
||||||
auto specs = format_specs<Char>();
|
auto specs = format_specs();
|
||||||
specs.align = align::right;
|
specs.set_align(align::right);
|
||||||
|
|
||||||
// Parse argument index, flags and width.
|
// Parse argument index, flags and width.
|
||||||
int arg_index = parse_header(it, end, specs, get_arg);
|
int arg_index = parse_header(it, end, specs, get_arg);
|
||||||
if (arg_index == 0) throw_format_error("argument not found");
|
if (arg_index == 0) report_error("argument not found");
|
||||||
|
|
||||||
// Parse precision.
|
// Parse precision.
|
||||||
if (it != end && *it == '.') {
|
if (it != end && *it == '.') {
|
||||||
|
|
@ -456,8 +439,8 @@ void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
specs.precision = parse_nonnegative_int(it, end, 0);
|
specs.precision = parse_nonnegative_int(it, end, 0);
|
||||||
} else if (c == '*') {
|
} else if (c == '*') {
|
||||||
++it;
|
++it;
|
||||||
specs.precision = static_cast<int>(
|
specs.precision =
|
||||||
visit_format_arg(printf_precision_handler(), get_arg(-1)));
|
static_cast<int>(get_arg(-1).visit(printf_precision_handler()));
|
||||||
} else {
|
} else {
|
||||||
specs.precision = 0;
|
specs.precision = 0;
|
||||||
}
|
}
|
||||||
|
|
@ -466,25 +449,26 @@ void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
auto arg = get_arg(arg_index);
|
auto arg = get_arg(arg_index);
|
||||||
// For d, i, o, u, x, and X conversion specifiers, if a precision is
|
// For d, i, o, u, x, and X conversion specifiers, if a precision is
|
||||||
// specified, the '0' flag is ignored
|
// specified, the '0' flag is ignored
|
||||||
if (specs.precision >= 0 && arg.is_integral()) {
|
if (specs.precision >= 0 && is_integral_type(arg.type())) {
|
||||||
// Ignore '0' for non-numeric types or if '-' present.
|
// Ignore '0' for non-numeric types or if '-' present.
|
||||||
specs.fill[0] = ' ';
|
specs.set_fill(' ');
|
||||||
}
|
}
|
||||||
if (specs.precision >= 0 && arg.type() == type::cstring_type) {
|
if (specs.precision >= 0 && arg.type() == type::cstring_type) {
|
||||||
auto str = visit_format_arg(get_cstring<Char>(), arg);
|
auto str = arg.visit(get_cstring<Char>());
|
||||||
auto str_end = str + specs.precision;
|
auto str_end = str + specs.precision;
|
||||||
auto nul = std::find(str, str_end, Char());
|
auto nul = std::find(str, str_end, Char());
|
||||||
auto sv = basic_string_view<Char>(
|
auto sv = basic_string_view<Char>(
|
||||||
str, to_unsigned(nul != str_end ? nul - str : specs.precision));
|
str, to_unsigned(nul != str_end ? nul - str : specs.precision));
|
||||||
arg = make_arg<basic_printf_context<Char>>(sv);
|
arg = sv;
|
||||||
}
|
}
|
||||||
if (specs.alt && visit_format_arg(is_zero_int(), arg)) specs.alt = false;
|
if (specs.alt() && arg.visit(is_zero_int())) specs.clear_alt();
|
||||||
if (specs.fill[0] == '0') {
|
if (specs.fill_unit<Char>() == '0') {
|
||||||
if (arg.is_arithmetic() && specs.align != align::left)
|
if (is_arithmetic_type(arg.type()) && specs.align() != align::left) {
|
||||||
specs.align = align::numeric;
|
specs.set_align(align::numeric);
|
||||||
else
|
} else {
|
||||||
specs.fill[0] = ' '; // Ignore '0' flag for non-numeric types or if '-'
|
// Ignore '0' flag for non-numeric types or if '-' flag is also present.
|
||||||
// flag is also present.
|
specs.set_fill(' ');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse length and convert the argument to the required type.
|
// Parse length and convert the argument to the required type.
|
||||||
|
|
@ -509,47 +493,39 @@ void vprintf(buffer<Char>& buf, basic_string_view<Char> format,
|
||||||
convert_arg<long>(arg, t);
|
convert_arg<long>(arg, t);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'j':
|
case 'j': convert_arg<intmax_t>(arg, t); break;
|
||||||
convert_arg<intmax_t>(arg, t);
|
case 'z': convert_arg<size_t>(arg, t); break;
|
||||||
break;
|
case 't': convert_arg<std::ptrdiff_t>(arg, t); break;
|
||||||
case 'z':
|
|
||||||
convert_arg<size_t>(arg, t);
|
|
||||||
break;
|
|
||||||
case 't':
|
|
||||||
convert_arg<std::ptrdiff_t>(arg, t);
|
|
||||||
break;
|
|
||||||
case 'L':
|
case 'L':
|
||||||
// printf produces garbage when 'L' is omitted for long double, no
|
// printf produces garbage when 'L' is omitted for long double, no
|
||||||
// need to do the same.
|
// need to do the same.
|
||||||
break;
|
break;
|
||||||
default:
|
default: --it; convert_arg<void>(arg, c);
|
||||||
--it;
|
|
||||||
convert_arg<void>(arg, c);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse type.
|
// Parse type.
|
||||||
if (it == end) throw_format_error("invalid format string");
|
if (it == end) report_error("invalid format string");
|
||||||
char type = static_cast<char>(*it++);
|
char type = static_cast<char>(*it++);
|
||||||
if (arg.is_integral()) {
|
if (is_integral_type(arg.type())) {
|
||||||
// Normalize type.
|
// Normalize type.
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'i':
|
case 'i':
|
||||||
case 'u':
|
case 'u': type = 'd'; break;
|
||||||
type = 'd';
|
|
||||||
break;
|
|
||||||
case 'c':
|
case 'c':
|
||||||
visit_format_arg(char_converter<basic_printf_context<Char>>(arg), arg);
|
arg.visit(char_converter<basic_printf_context<Char>>(arg));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
specs.type = parse_printf_presentation_type(type, arg.type());
|
bool upper = false;
|
||||||
if (specs.type == presentation_type::none)
|
specs.set_type(parse_printf_presentation_type(type, arg.type(), upper));
|
||||||
throw_format_error("invalid format specifier");
|
if (specs.type() == presentation_type::none)
|
||||||
|
report_error("invalid format specifier");
|
||||||
|
if (upper) specs.set_upper();
|
||||||
|
|
||||||
start = it;
|
start = it;
|
||||||
|
|
||||||
// Format argument.
|
// Format argument.
|
||||||
visit_format_arg(printf_arg_formatter<Char>(out, specs, context), arg);
|
arg.visit(printf_arg_formatter<Char>(out, specs, context));
|
||||||
}
|
}
|
||||||
write(out, basic_string_view<Char>(start, to_unsigned(it - start)));
|
write(out, basic_string_view<Char>(start, to_unsigned(it - start)));
|
||||||
}
|
}
|
||||||
|
|
@ -561,56 +537,48 @@ using wprintf_context = basic_printf_context<wchar_t>;
|
||||||
using printf_args = basic_format_args<printf_context>;
|
using printf_args = basic_format_args<printf_context>;
|
||||||
using wprintf_args = basic_format_args<wprintf_context>;
|
using wprintf_args = basic_format_args<wprintf_context>;
|
||||||
|
|
||||||
/**
|
/// Constructs an `format_arg_store` object that contains references to
|
||||||
\rst
|
/// arguments and can be implicitly converted to `printf_args`.
|
||||||
Constructs an `~fmt::format_arg_store` object that contains references to
|
template <typename Char = char, typename... T>
|
||||||
arguments and can be implicitly converted to `~fmt::printf_args`.
|
inline auto make_printf_args(T&... args)
|
||||||
\endrst
|
-> decltype(fmt::make_format_args<basic_printf_context<Char>>(args...)) {
|
||||||
*/
|
return fmt::make_format_args<basic_printf_context<Char>>(args...);
|
||||||
template <typename... T>
|
|
||||||
inline auto make_printf_args(const T&... args)
|
|
||||||
-> format_arg_store<printf_context, T...> {
|
|
||||||
return {args...};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DEPRECATED!
|
template <typename Char> struct vprintf_args {
|
||||||
template <typename... T>
|
using type = basic_format_args<basic_printf_context<Char>>;
|
||||||
inline auto make_wprintf_args(const T&... args)
|
};
|
||||||
-> format_arg_store<wprintf_context, T...> {
|
|
||||||
return {args...};
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
inline auto vsprintf(
|
inline auto vsprintf(basic_string_view<Char> fmt,
|
||||||
basic_string_view<Char> fmt,
|
typename vprintf_args<Char>::type args)
|
||||||
basic_format_args<basic_printf_context<type_identity_t<Char>>> args)
|
|
||||||
-> std::basic_string<Char> {
|
-> std::basic_string<Char> {
|
||||||
auto buf = basic_memory_buffer<Char>();
|
auto buf = basic_memory_buffer<Char>();
|
||||||
detail::vprintf(buf, fmt, args);
|
detail::vprintf(buf, fmt, args);
|
||||||
return to_string(buf);
|
return {buf.data(), buf.size()};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats `args` according to specifications in `fmt` and returns the result
|
||||||
Formats arguments and returns the result as a string.
|
* as as string.
|
||||||
|
*
|
||||||
**Example**::
|
* **Example**:
|
||||||
|
*
|
||||||
std::string message = fmt::sprintf("The answer is %d", 42);
|
* std::string message = fmt::sprintf("The answer is %d", 42);
|
||||||
\endrst
|
*/
|
||||||
*/
|
template <typename... T>
|
||||||
template <typename S, typename... T,
|
inline auto sprintf(string_view fmt, const T&... args) -> std::string {
|
||||||
typename Char = enable_if_t<detail::is_string<S>::value, char_t<S>>>
|
return vsprintf(fmt, make_printf_args(args...));
|
||||||
inline auto sprintf(const S& fmt, const T&... args) -> std::basic_string<Char> {
|
}
|
||||||
return vsprintf(detail::to_string_view(fmt),
|
template <typename... T>
|
||||||
fmt::make_format_args<basic_printf_context<Char>>(args...));
|
FMT_DEPRECATED auto sprintf(basic_string_view<wchar_t> fmt, const T&... args)
|
||||||
|
-> std::wstring {
|
||||||
|
return vsprintf(fmt, make_printf_args<wchar_t>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
inline auto vfprintf(
|
auto vfprintf(std::FILE* f, basic_string_view<Char> fmt,
|
||||||
std::FILE* f, basic_string_view<Char> fmt,
|
typename vprintf_args<Char>::type args) -> int {
|
||||||
basic_format_args<basic_printf_context<type_identity_t<Char>>> args)
|
|
||||||
-> int {
|
|
||||||
auto buf = basic_memory_buffer<Char>();
|
auto buf = basic_memory_buffer<Char>();
|
||||||
detail::vprintf(buf, fmt, args);
|
detail::vprintf(buf, fmt, args);
|
||||||
size_t size = buf.size();
|
size_t size = buf.size();
|
||||||
|
|
@ -620,46 +588,35 @@ inline auto vfprintf(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats `args` according to specifications in `fmt` and writes the output
|
||||||
Prints formatted data to the file *f*.
|
* to `f`.
|
||||||
|
*
|
||||||
**Example**::
|
* **Example**:
|
||||||
|
*
|
||||||
fmt::fprintf(stderr, "Don't %s!", "panic");
|
* fmt::fprintf(stderr, "Don't %s!", "panic");
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename S, typename... T, typename Char = char_t<S>>
|
template <typename... T>
|
||||||
inline auto fprintf(std::FILE* f, const S& fmt, const T&... args) -> int {
|
inline auto fprintf(std::FILE* f, string_view fmt, const T&... args) -> int {
|
||||||
return vfprintf(f, detail::to_string_view(fmt),
|
return vfprintf(f, fmt, make_printf_args(args...));
|
||||||
fmt::make_format_args<basic_printf_context<Char>>(args...));
|
|
||||||
}
|
}
|
||||||
|
template <typename... T>
|
||||||
template <typename Char>
|
FMT_DEPRECATED auto fprintf(std::FILE* f, basic_string_view<wchar_t> fmt,
|
||||||
FMT_DEPRECATED inline auto vprintf(
|
const T&... args) -> int {
|
||||||
basic_string_view<Char> fmt,
|
return vfprintf(f, fmt, make_printf_args<wchar_t>(args...));
|
||||||
basic_format_args<basic_printf_context<type_identity_t<Char>>> args)
|
|
||||||
-> int {
|
|
||||||
return vfprintf(stdout, fmt, args);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Formats `args` according to specifications in `fmt` and writes the output
|
||||||
Prints formatted data to ``stdout``.
|
* to `stdout`.
|
||||||
|
*
|
||||||
**Example**::
|
* **Example**:
|
||||||
|
*
|
||||||
fmt::printf("Elapsed time: %.2f seconds", 1.23);
|
* fmt::printf("Elapsed time: %.2f seconds", 1.23);
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
inline auto printf(string_view fmt, const T&... args) -> int {
|
inline auto printf(string_view fmt, const T&... args) -> int {
|
||||||
return vfprintf(stdout, fmt, make_printf_args(args...));
|
return vfprintf(stdout, fmt, make_printf_args(args...));
|
||||||
}
|
}
|
||||||
template <typename... T>
|
|
||||||
FMT_DEPRECATED inline auto printf(basic_string_view<wchar_t> fmt,
|
|
||||||
const T&... args) -> int {
|
|
||||||
return vfprintf(stdout, fmt, make_wprintf_args(args...));
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_END_EXPORT
|
FMT_END_EXPORT
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
|
||||||
696
deps/fmt/include/fmt/ranges.h
vendored
696
deps/fmt/include/fmt/ranges.h
vendored
|
|
@ -1,78 +1,44 @@
|
||||||
// Formatting library for C++ - experimental range support
|
// Formatting library for C++ - range and tuple support
|
||||||
//
|
//
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
// Copyright (c) 2012 - present, Victor Zverovich and {fmt} contributors
|
||||||
// All rights reserved.
|
// All rights reserved.
|
||||||
//
|
//
|
||||||
// For the license information refer to format.h.
|
// For the license information refer to format.h.
|
||||||
//
|
|
||||||
// Copyright (c) 2018 - present, Remotion (Igor Schulz)
|
|
||||||
// All Rights Reserved
|
|
||||||
// {fmt} support for ranges, containers and types tuple interface.
|
|
||||||
|
|
||||||
#ifndef FMT_RANGES_H_
|
#ifndef FMT_RANGES_H_
|
||||||
#define FMT_RANGES_H_
|
#define FMT_RANGES_H_
|
||||||
|
|
||||||
#include <initializer_list>
|
#ifndef FMT_MODULE
|
||||||
#include <tuple>
|
# include <initializer_list>
|
||||||
#include <type_traits>
|
# include <iterator>
|
||||||
|
# include <tuple>
|
||||||
|
# include <type_traits>
|
||||||
|
# include <utility>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
|
#if FMT_HAS_CPP_ATTRIBUTE(clang::lifetimebound)
|
||||||
|
# define FMT_LIFETIMEBOUND [[clang::lifetimebound]]
|
||||||
|
#else
|
||||||
|
# define FMT_LIFETIMEBOUND
|
||||||
|
#endif
|
||||||
|
FMT_PRAGMA_CLANG(diagnostic error "-Wreturn-stack-address")
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
enum class range_format { disabled, map, set, sequence, string, debug_string };
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template <typename Range, typename OutputIt>
|
|
||||||
auto copy(const Range& range, OutputIt out) -> OutputIt {
|
|
||||||
for (auto it = range.begin(), end = range.end(); it != end; ++it)
|
|
||||||
*out++ = *it;
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt>
|
|
||||||
auto copy(const char* str, OutputIt out) -> OutputIt {
|
|
||||||
while (*str) *out++ = *str++;
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt> auto copy(char ch, OutputIt out) -> OutputIt {
|
|
||||||
*out++ = ch;
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt> auto copy(wchar_t ch, OutputIt out) -> OutputIt {
|
|
||||||
*out++ = ch;
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns true if T has a std::string-like interface, like std::string_view.
|
|
||||||
template <typename T> class is_std_string_like {
|
|
||||||
template <typename U>
|
|
||||||
static auto check(U* p)
|
|
||||||
-> decltype((void)p->find('a'), p->length(), (void)p->data(), int());
|
|
||||||
template <typename> static void check(...);
|
|
||||||
|
|
||||||
public:
|
|
||||||
static constexpr const bool value =
|
|
||||||
is_string<T>::value ||
|
|
||||||
std::is_convertible<T, std_string_view<char>>::value ||
|
|
||||||
!std::is_void<decltype(check<T>(nullptr))>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct is_std_string_like<fmt::basic_string_view<Char>> : std::true_type {};
|
|
||||||
|
|
||||||
template <typename T> class is_map {
|
template <typename T> class is_map {
|
||||||
template <typename U> static auto check(U*) -> typename U::mapped_type;
|
template <typename U> static auto check(U*) -> typename U::mapped_type;
|
||||||
template <typename> static void check(...);
|
template <typename> static void check(...);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
#ifdef FMT_FORMAT_MAP_AS_LIST // DEPRECATED!
|
static constexpr bool value =
|
||||||
static constexpr const bool value = false;
|
|
||||||
#else
|
|
||||||
static constexpr const bool value =
|
|
||||||
!std::is_void<decltype(check<T>(nullptr))>::value;
|
!std::is_void<decltype(check<T>(nullptr))>::value;
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T> class is_set {
|
template <typename T> class is_set {
|
||||||
|
|
@ -80,33 +46,16 @@ template <typename T> class is_set {
|
||||||
template <typename> static void check(...);
|
template <typename> static void check(...);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
#ifdef FMT_FORMAT_SET_AS_LIST // DEPRECATED!
|
static constexpr bool value =
|
||||||
static constexpr const bool value = false;
|
|
||||||
#else
|
|
||||||
static constexpr const bool value =
|
|
||||||
!std::is_void<decltype(check<T>(nullptr))>::value && !is_map<T>::value;
|
!std::is_void<decltype(check<T>(nullptr))>::value && !is_map<T>::value;
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename... Ts> struct conditional_helper {};
|
|
||||||
|
|
||||||
template <typename T, typename _ = void> struct is_range_ : std::false_type {};
|
|
||||||
|
|
||||||
#if !FMT_MSC_VERSION || FMT_MSC_VERSION > 1800
|
|
||||||
|
|
||||||
# define FMT_DECLTYPE_RETURN(val) \
|
|
||||||
->decltype(val) { return val; } \
|
|
||||||
static_assert( \
|
|
||||||
true, "") // This makes it so that a semicolon is required after the
|
|
||||||
// macro, which helps clang-format handle the formatting.
|
|
||||||
|
|
||||||
// C array overload
|
// C array overload
|
||||||
template <typename T, std::size_t N>
|
template <typename T, size_t N>
|
||||||
auto range_begin(const T (&arr)[N]) -> const T* {
|
auto range_begin(const T (&arr)[N]) -> const T* {
|
||||||
return arr;
|
return arr;
|
||||||
}
|
}
|
||||||
template <typename T, std::size_t N>
|
template <typename T, size_t N> auto range_end(const T (&arr)[N]) -> const T* {
|
||||||
auto range_end(const T (&arr)[N]) -> const T* {
|
|
||||||
return arr + N;
|
return arr + N;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -114,17 +63,21 @@ template <typename T, typename Enable = void>
|
||||||
struct has_member_fn_begin_end_t : std::false_type {};
|
struct has_member_fn_begin_end_t : std::false_type {};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct has_member_fn_begin_end_t<T, void_t<decltype(std::declval<T>().begin()),
|
struct has_member_fn_begin_end_t<T, void_t<decltype(*std::declval<T>().begin()),
|
||||||
decltype(std::declval<T>().end())>>
|
decltype(std::declval<T>().end())>>
|
||||||
: std::true_type {};
|
: std::true_type {};
|
||||||
|
|
||||||
// Member function overload
|
// Member function overloads.
|
||||||
template <typename T>
|
template <typename T>
|
||||||
auto range_begin(T&& rng) FMT_DECLTYPE_RETURN(static_cast<T&&>(rng).begin());
|
auto range_begin(T&& rng) -> decltype(static_cast<T&&>(rng).begin()) {
|
||||||
|
return static_cast<T&&>(rng).begin();
|
||||||
|
}
|
||||||
template <typename T>
|
template <typename T>
|
||||||
auto range_end(T&& rng) FMT_DECLTYPE_RETURN(static_cast<T&&>(rng).end());
|
auto range_end(T&& rng) -> decltype(static_cast<T&&>(rng).end()) {
|
||||||
|
return static_cast<T&&>(rng).end();
|
||||||
|
}
|
||||||
|
|
||||||
// ADL overload. Only participates in overload resolution if member functions
|
// ADL overloads. Only participate in overload resolution if member functions
|
||||||
// are not found.
|
// are not found.
|
||||||
template <typename T>
|
template <typename T>
|
||||||
auto range_begin(T&& rng)
|
auto range_begin(T&& rng)
|
||||||
|
|
@ -145,35 +98,34 @@ struct has_mutable_begin_end : std::false_type {};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct has_const_begin_end<
|
struct has_const_begin_end<
|
||||||
T,
|
T, void_t<decltype(*detail::range_begin(
|
||||||
void_t<
|
std::declval<const remove_cvref_t<T>&>())),
|
||||||
decltype(detail::range_begin(std::declval<const remove_cvref_t<T>&>())),
|
decltype(detail::range_end(
|
||||||
decltype(detail::range_end(std::declval<const remove_cvref_t<T>&>()))>>
|
std::declval<const remove_cvref_t<T>&>()))>>
|
||||||
: std::true_type {};
|
: std::true_type {};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct has_mutable_begin_end<
|
struct has_mutable_begin_end<
|
||||||
T, void_t<decltype(detail::range_begin(std::declval<T>())),
|
T, void_t<decltype(*detail::range_begin(std::declval<T&>())),
|
||||||
decltype(detail::range_end(std::declval<T>())),
|
decltype(detail::range_end(std::declval<T&>())),
|
||||||
// the extra int here is because older versions of MSVC don't
|
// the extra int here is because older versions of MSVC don't
|
||||||
// SFINAE properly unless there are distinct types
|
// SFINAE properly unless there are distinct types
|
||||||
int>> : std::true_type {};
|
int>> : std::true_type {};
|
||||||
|
|
||||||
|
template <typename T, typename _ = void> struct is_range_ : std::false_type {};
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct is_range_<T, void>
|
struct is_range_<T, void>
|
||||||
: std::integral_constant<bool, (has_const_begin_end<T>::value ||
|
: std::integral_constant<bool, (has_const_begin_end<T>::value ||
|
||||||
has_mutable_begin_end<T>::value)> {};
|
has_mutable_begin_end<T>::value)> {};
|
||||||
# undef FMT_DECLTYPE_RETURN
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// tuple_size and tuple_element check.
|
// tuple_size and tuple_element check.
|
||||||
template <typename T> class is_tuple_like_ {
|
template <typename T> class is_tuple_like_ {
|
||||||
template <typename U>
|
template <typename U, typename V = typename std::remove_cv<U>::type>
|
||||||
static auto check(U* p) -> decltype(std::tuple_size<U>::value, int());
|
static auto check(U* p) -> decltype(std::tuple_size<V>::value, 0);
|
||||||
template <typename> static void check(...);
|
template <typename> static void check(...);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr const bool value =
|
static constexpr bool value =
|
||||||
!std::is_void<decltype(check<T>(nullptr))>::value;
|
!std::is_void<decltype(check<T>(nullptr))>::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -187,7 +139,7 @@ template <size_t N> using make_index_sequence = std::make_index_sequence<N>;
|
||||||
template <typename T, T... N> struct integer_sequence {
|
template <typename T, T... N> struct integer_sequence {
|
||||||
using value_type = T;
|
using value_type = T;
|
||||||
|
|
||||||
static FMT_CONSTEXPR size_t size() { return sizeof...(N); }
|
static FMT_CONSTEXPR auto size() -> size_t { return sizeof...(N); }
|
||||||
};
|
};
|
||||||
|
|
||||||
template <size_t... N> using index_sequence = integer_sequence<size_t, N...>;
|
template <size_t... N> using index_sequence = integer_sequence<size_t, N...>;
|
||||||
|
|
@ -207,22 +159,23 @@ using tuple_index_sequence = make_index_sequence<std::tuple_size<T>::value>;
|
||||||
template <typename T, typename C, bool = is_tuple_like_<T>::value>
|
template <typename T, typename C, bool = is_tuple_like_<T>::value>
|
||||||
class is_tuple_formattable_ {
|
class is_tuple_formattable_ {
|
||||||
public:
|
public:
|
||||||
static constexpr const bool value = false;
|
static constexpr bool value = false;
|
||||||
};
|
};
|
||||||
template <typename T, typename C> class is_tuple_formattable_<T, C, true> {
|
template <typename T, typename C> class is_tuple_formattable_<T, C, true> {
|
||||||
template <std::size_t... Is>
|
template <size_t... Is>
|
||||||
static std::true_type check2(index_sequence<Is...>,
|
static auto all_true(index_sequence<Is...>,
|
||||||
integer_sequence<bool, (Is == Is)...>);
|
integer_sequence<bool, (Is >= 0)...>) -> std::true_type;
|
||||||
static std::false_type check2(...);
|
static auto all_true(...) -> std::false_type;
|
||||||
template <std::size_t... Is>
|
|
||||||
static decltype(check2(
|
template <size_t... Is>
|
||||||
|
static auto check(index_sequence<Is...>) -> decltype(all_true(
|
||||||
index_sequence<Is...>{},
|
index_sequence<Is...>{},
|
||||||
integer_sequence<
|
integer_sequence<bool,
|
||||||
bool, (is_formattable<typename std::tuple_element<Is, T>::type,
|
(is_formattable<typename std::tuple_element<Is, T>::type,
|
||||||
C>::value)...>{})) check(index_sequence<Is...>);
|
C>::value)...>{}));
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr const bool value =
|
static constexpr bool value =
|
||||||
decltype(check(tuple_index_sequence<T>{}))::value;
|
decltype(check(tuple_index_sequence<T>{}))::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -260,7 +213,7 @@ template <typename Char, typename... T>
|
||||||
using result_t = std::tuple<formatter<remove_cvref_t<T>, Char>...>;
|
using result_t = std::tuple<formatter<remove_cvref_t<T>, Char>...>;
|
||||||
|
|
||||||
using std::get;
|
using std::get;
|
||||||
template <typename Tuple, typename Char, std::size_t... Is>
|
template <typename Tuple, typename Char, size_t... Is>
|
||||||
auto get_formatters(index_sequence<Is...>)
|
auto get_formatters(index_sequence<Is...>)
|
||||||
-> result_t<Char, decltype(get<Is>(std::declval<Tuple>()))...>;
|
-> result_t<Char, decltype(get<Is>(std::declval<Tuple>()))...>;
|
||||||
} // namespace tuple
|
} // namespace tuple
|
||||||
|
|
@ -271,7 +224,7 @@ template <typename R> struct range_reference_type_impl {
|
||||||
using type = decltype(*detail::range_begin(std::declval<R&>()));
|
using type = decltype(*detail::range_begin(std::declval<R&>()));
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, std::size_t N> struct range_reference_type_impl<T[N]> {
|
template <typename T, size_t N> struct range_reference_type_impl<T[N]> {
|
||||||
using type = T&;
|
using type = T&;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -288,29 +241,32 @@ using range_reference_type =
|
||||||
template <typename Range>
|
template <typename Range>
|
||||||
using uncvref_type = remove_cvref_t<range_reference_type<Range>>;
|
using uncvref_type = remove_cvref_t<range_reference_type<Range>>;
|
||||||
|
|
||||||
template <typename Formatter>
|
template <typename T>
|
||||||
FMT_CONSTEXPR auto maybe_set_debug_format(Formatter& f, bool set)
|
struct range_format_kind_
|
||||||
-> decltype(f.set_debug_format(set)) {
|
: std::integral_constant<range_format,
|
||||||
f.set_debug_format(set);
|
std::is_same<uncvref_type<T>, T>::value
|
||||||
}
|
? range_format::disabled
|
||||||
template <typename Formatter>
|
: is_map<T>::value ? range_format::map
|
||||||
FMT_CONSTEXPR void maybe_set_debug_format(Formatter&, ...) {}
|
: is_set<T>::value ? range_format::set
|
||||||
|
: range_format::sequence> {};
|
||||||
|
|
||||||
|
template <range_format K>
|
||||||
|
using range_format_constant = std::integral_constant<range_format, K>;
|
||||||
|
|
||||||
// These are not generic lambdas for compatibility with C++11.
|
// These are not generic lambdas for compatibility with C++11.
|
||||||
template <typename ParseContext> struct parse_empty_specs {
|
template <typename Char> struct parse_empty_specs {
|
||||||
template <typename Formatter> FMT_CONSTEXPR void operator()(Formatter& f) {
|
template <typename Formatter> FMT_CONSTEXPR void operator()(Formatter& f) {
|
||||||
f.parse(ctx);
|
f.parse(ctx);
|
||||||
detail::maybe_set_debug_format(f, true);
|
detail::maybe_set_debug_format(f, true);
|
||||||
}
|
}
|
||||||
ParseContext& ctx;
|
parse_context<Char>& ctx;
|
||||||
};
|
};
|
||||||
template <typename FormatContext> struct format_tuple_element {
|
template <typename FormatContext> struct format_tuple_element {
|
||||||
using char_type = typename FormatContext::char_type;
|
using char_type = typename FormatContext::char_type;
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void operator()(const formatter<T, char_type>& f, const T& v) {
|
void operator()(const formatter<T, char_type>& f, const T& v) {
|
||||||
if (i > 0)
|
if (i > 0) ctx.advance_to(detail::copy<char_type>(separator, ctx.out()));
|
||||||
ctx.advance_to(detail::copy_str<char_type>(separator, ctx.out()));
|
|
||||||
ctx.advance_to(f.format(v, ctx));
|
ctx.advance_to(f.format(v, ctx));
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
|
@ -322,14 +278,15 @@ template <typename FormatContext> struct format_tuple_element {
|
||||||
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
template <typename T> struct is_tuple_like {
|
template <typename T> struct is_tuple_like {
|
||||||
static constexpr const bool value =
|
static constexpr bool value =
|
||||||
detail::is_tuple_like_<T>::value && !detail::is_range_<T>::value;
|
detail::is_tuple_like_<T>::value && !detail::is_range_<T>::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
template <typename T, typename C> struct is_tuple_formattable {
|
template <typename T, typename C> struct is_tuple_formattable {
|
||||||
static constexpr const bool value =
|
static constexpr bool value = detail::is_tuple_formattable_<T, C>::value;
|
||||||
detail::is_tuple_formattable_<T, C>::value;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Tuple, typename Char>
|
template <typename Tuple, typename Char>
|
||||||
|
|
@ -359,68 +316,58 @@ struct formatter<Tuple, Char,
|
||||||
closing_bracket_ = close;
|
closing_bracket_ = close;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ParseContext>
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
auto it = ctx.begin();
|
auto it = ctx.begin();
|
||||||
if (it != ctx.end() && *it != '}')
|
auto end = ctx.end();
|
||||||
FMT_THROW(format_error("invalid format specifier"));
|
if (it != end && detail::to_ascii(*it) == 'n') {
|
||||||
detail::for_each(formatters_, detail::parse_empty_specs<ParseContext>{ctx});
|
++it;
|
||||||
|
set_brackets({}, {});
|
||||||
|
set_separator({});
|
||||||
|
}
|
||||||
|
if (it != end && *it != '}') report_error("invalid format specifier");
|
||||||
|
ctx.advance_to(it);
|
||||||
|
detail::for_each(formatters_, detail::parse_empty_specs<Char>{ctx});
|
||||||
return it;
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const Tuple& value, FormatContext& ctx) const
|
auto format(const Tuple& value, FormatContext& ctx) const
|
||||||
-> decltype(ctx.out()) {
|
-> decltype(ctx.out()) {
|
||||||
ctx.advance_to(detail::copy_str<Char>(opening_bracket_, ctx.out()));
|
ctx.advance_to(detail::copy<Char>(opening_bracket_, ctx.out()));
|
||||||
detail::for_each2(
|
detail::for_each2(
|
||||||
formatters_, value,
|
formatters_, value,
|
||||||
detail::format_tuple_element<FormatContext>{0, ctx, separator_});
|
detail::format_tuple_element<FormatContext>{0, ctx, separator_});
|
||||||
return detail::copy_str<Char>(closing_bracket_, ctx.out());
|
return detail::copy<Char>(closing_bracket_, ctx.out());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
template <typename T, typename Char> struct is_range {
|
template <typename T, typename Char> struct is_range {
|
||||||
static constexpr const bool value =
|
static constexpr bool value =
|
||||||
detail::is_range_<T>::value && !detail::is_std_string_like<T>::value &&
|
detail::is_range_<T>::value && !detail::has_to_string_view<T>::value;
|
||||||
!std::is_convertible<T, std::basic_string<Char>>::value &&
|
|
||||||
!std::is_convertible<T, detail::std_string_view<Char>>::value;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
template <typename Context> struct range_mapper {
|
|
||||||
using mapper = arg_mapper<Context>;
|
|
||||||
|
|
||||||
template <typename T,
|
|
||||||
FMT_ENABLE_IF(has_formatter<remove_cvref_t<T>, Context>::value)>
|
|
||||||
static auto map(T&& value) -> T&& {
|
|
||||||
return static_cast<T&&>(value);
|
|
||||||
}
|
|
||||||
template <typename T,
|
|
||||||
FMT_ENABLE_IF(!has_formatter<remove_cvref_t<T>, Context>::value)>
|
|
||||||
static auto map(T&& value)
|
|
||||||
-> decltype(mapper().map(static_cast<T&&>(value))) {
|
|
||||||
return mapper().map(static_cast<T&&>(value));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char, typename Element>
|
template <typename Char, typename Element>
|
||||||
using range_formatter_type =
|
using range_formatter_type = formatter<remove_cvref_t<Element>, Char>;
|
||||||
formatter<remove_cvref_t<decltype(range_mapper<buffer_context<Char>>{}.map(
|
|
||||||
std::declval<Element>()))>,
|
|
||||||
Char>;
|
|
||||||
|
|
||||||
template <typename R>
|
template <typename R>
|
||||||
using maybe_const_range =
|
using maybe_const_range =
|
||||||
conditional_t<has_const_begin_end<R>::value, const R, R>;
|
conditional_t<has_const_begin_end<R>::value, const R, R>;
|
||||||
|
|
||||||
// Workaround a bug in MSVC 2015 and earlier.
|
|
||||||
#if !FMT_MSC_VERSION || FMT_MSC_VERSION >= 1910
|
|
||||||
template <typename R, typename Char>
|
template <typename R, typename Char>
|
||||||
struct is_formattable_delayed
|
struct is_formattable_delayed
|
||||||
: is_formattable<uncvref_type<maybe_const_range<R>>, Char> {};
|
: is_formattable<uncvref_type<maybe_const_range<R>>, Char> {};
|
||||||
#endif
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
|
template <typename...> struct conjunction : std::true_type {};
|
||||||
|
template <typename P> struct conjunction<P> : P {};
|
||||||
|
template <typename P1, typename... Pn>
|
||||||
|
struct conjunction<P1, Pn...>
|
||||||
|
: conditional_t<bool(P1::value), conjunction<Pn...>, P1> {};
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
template <typename T, typename Char, typename Enable = void>
|
template <typename T, typename Char, typename Enable = void>
|
||||||
struct range_formatter;
|
struct range_formatter;
|
||||||
|
|
||||||
|
|
@ -436,6 +383,24 @@ struct range_formatter<
|
||||||
detail::string_literal<Char, '['>{};
|
detail::string_literal<Char, '['>{};
|
||||||
basic_string_view<Char> closing_bracket_ =
|
basic_string_view<Char> closing_bracket_ =
|
||||||
detail::string_literal<Char, ']'>{};
|
detail::string_literal<Char, ']'>{};
|
||||||
|
bool is_debug = false;
|
||||||
|
|
||||||
|
template <typename Output, typename It, typename Sentinel, typename U = T,
|
||||||
|
FMT_ENABLE_IF(std::is_same<U, Char>::value)>
|
||||||
|
auto write_debug_string(Output& out, It it, Sentinel end) const -> Output {
|
||||||
|
auto buf = basic_memory_buffer<Char>();
|
||||||
|
for (; it != end; ++it) buf.push_back(*it);
|
||||||
|
auto specs = format_specs();
|
||||||
|
specs.set_type(presentation_type::debug);
|
||||||
|
return detail::write<Char>(
|
||||||
|
out, basic_string_view<Char>(buf.data(), buf.size()), specs);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Output, typename It, typename Sentinel, typename U = T,
|
||||||
|
FMT_ENABLE_IF(!std::is_same<U, Char>::value)>
|
||||||
|
auto write_debug_string(Output& out, It, Sentinel) const -> Output {
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR range_formatter() {}
|
FMT_CONSTEXPR range_formatter() {}
|
||||||
|
|
@ -454,21 +419,40 @@ struct range_formatter<
|
||||||
closing_bracket_ = close;
|
closing_bracket_ = close;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ParseContext>
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
auto it = ctx.begin();
|
auto it = ctx.begin();
|
||||||
auto end = ctx.end();
|
auto end = ctx.end();
|
||||||
|
detail::maybe_set_debug_format(underlying_, true);
|
||||||
|
if (it == end) return underlying_.parse(ctx);
|
||||||
|
|
||||||
if (it != end && *it == 'n') {
|
switch (detail::to_ascii(*it)) {
|
||||||
|
case 'n':
|
||||||
set_brackets({}, {});
|
set_brackets({}, {});
|
||||||
++it;
|
++it;
|
||||||
|
break;
|
||||||
|
case '?':
|
||||||
|
is_debug = true;
|
||||||
|
set_brackets({}, {});
|
||||||
|
++it;
|
||||||
|
if (it == end || *it != 's') report_error("invalid format specifier");
|
||||||
|
FMT_FALLTHROUGH;
|
||||||
|
case 's':
|
||||||
|
if (!std::is_same<T, Char>::value)
|
||||||
|
report_error("invalid format specifier");
|
||||||
|
if (!is_debug) {
|
||||||
|
set_brackets(detail::string_literal<Char, '"'>{},
|
||||||
|
detail::string_literal<Char, '"'>{});
|
||||||
|
set_separator({});
|
||||||
|
detail::maybe_set_debug_format(underlying_, false);
|
||||||
|
}
|
||||||
|
++it;
|
||||||
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (it != end && *it != '}') {
|
if (it != end && *it != '}') {
|
||||||
if (*it != ':') FMT_THROW(format_error("invalid format specifier"));
|
if (*it != ':') report_error("invalid format specifier");
|
||||||
|
detail::maybe_set_debug_format(underlying_, false);
|
||||||
++it;
|
++it;
|
||||||
} else {
|
|
||||||
detail::maybe_set_debug_format(underlying_, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.advance_to(it);
|
ctx.advance_to(it);
|
||||||
|
|
@ -477,79 +461,26 @@ struct range_formatter<
|
||||||
|
|
||||||
template <typename R, typename FormatContext>
|
template <typename R, typename FormatContext>
|
||||||
auto format(R&& range, FormatContext& ctx) const -> decltype(ctx.out()) {
|
auto format(R&& range, FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
detail::range_mapper<buffer_context<Char>> mapper;
|
|
||||||
auto out = ctx.out();
|
auto out = ctx.out();
|
||||||
out = detail::copy_str<Char>(opening_bracket_, out);
|
|
||||||
int i = 0;
|
|
||||||
auto it = detail::range_begin(range);
|
auto it = detail::range_begin(range);
|
||||||
auto end = detail::range_end(range);
|
auto end = detail::range_end(range);
|
||||||
|
if (is_debug) return write_debug_string(out, std::move(it), end);
|
||||||
|
|
||||||
|
out = detail::copy<Char>(opening_bracket_, out);
|
||||||
|
int i = 0;
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
if (i > 0) out = detail::copy_str<Char>(separator_, out);
|
if (i > 0) out = detail::copy<Char>(separator_, out);
|
||||||
ctx.advance_to(out);
|
ctx.advance_to(out);
|
||||||
out = underlying_.format(mapper.map(*it), ctx);
|
auto&& item = *it; // Need an lvalue
|
||||||
|
out = underlying_.format(item, ctx);
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
out = detail::copy_str<Char>(closing_bracket_, out);
|
out = detail::copy<Char>(closing_bracket_, out);
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class range_format { disabled, map, set, sequence, string, debug_string };
|
FMT_EXPORT
|
||||||
|
|
||||||
namespace detail {
|
|
||||||
template <typename T>
|
|
||||||
struct range_format_kind_
|
|
||||||
: std::integral_constant<range_format,
|
|
||||||
std::is_same<uncvref_type<T>, T>::value
|
|
||||||
? range_format::disabled
|
|
||||||
: is_map<T>::value ? range_format::map
|
|
||||||
: is_set<T>::value ? range_format::set
|
|
||||||
: range_format::sequence> {};
|
|
||||||
|
|
||||||
template <range_format K, typename R, typename Char, typename Enable = void>
|
|
||||||
struct range_default_formatter;
|
|
||||||
|
|
||||||
template <range_format K>
|
|
||||||
using range_format_constant = std::integral_constant<range_format, K>;
|
|
||||||
|
|
||||||
template <range_format K, typename R, typename Char>
|
|
||||||
struct range_default_formatter<
|
|
||||||
K, R, Char,
|
|
||||||
enable_if_t<(K == range_format::sequence || K == range_format::map ||
|
|
||||||
K == range_format::set)>> {
|
|
||||||
using range_type = detail::maybe_const_range<R>;
|
|
||||||
range_formatter<detail::uncvref_type<range_type>, Char> underlying_;
|
|
||||||
|
|
||||||
FMT_CONSTEXPR range_default_formatter() { init(range_format_constant<K>()); }
|
|
||||||
|
|
||||||
FMT_CONSTEXPR void init(range_format_constant<range_format::set>) {
|
|
||||||
underlying_.set_brackets(detail::string_literal<Char, '{'>{},
|
|
||||||
detail::string_literal<Char, '}'>{});
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR void init(range_format_constant<range_format::map>) {
|
|
||||||
underlying_.set_brackets(detail::string_literal<Char, '{'>{},
|
|
||||||
detail::string_literal<Char, '}'>{});
|
|
||||||
underlying_.underlying().set_brackets({}, {});
|
|
||||||
underlying_.underlying().set_separator(
|
|
||||||
detail::string_literal<Char, ':', ' '>{});
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR void init(range_format_constant<range_format::sequence>) {}
|
|
||||||
|
|
||||||
template <typename ParseContext>
|
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
return underlying_.parse(ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
auto format(range_type& range, FormatContext& ctx) const
|
|
||||||
-> decltype(ctx.out()) {
|
|
||||||
return underlying_.format(range, ctx);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
template <typename T, typename Char, typename Enable = void>
|
template <typename T, typename Char, typename Enable = void>
|
||||||
struct range_format_kind
|
struct range_format_kind
|
||||||
: conditional_t<
|
: conditional_t<
|
||||||
|
|
@ -559,23 +490,192 @@ struct range_format_kind
|
||||||
template <typename R, typename Char>
|
template <typename R, typename Char>
|
||||||
struct formatter<
|
struct formatter<
|
||||||
R, Char,
|
R, Char,
|
||||||
enable_if_t<conjunction<bool_constant<range_format_kind<R, Char>::value !=
|
enable_if_t<conjunction<
|
||||||
range_format::disabled>
|
bool_constant<
|
||||||
// Workaround a bug in MSVC 2015 and earlier.
|
range_format_kind<R, Char>::value != range_format::disabled &&
|
||||||
#if !FMT_MSC_VERSION || FMT_MSC_VERSION >= 1910
|
range_format_kind<R, Char>::value != range_format::map &&
|
||||||
,
|
range_format_kind<R, Char>::value != range_format::string &&
|
||||||
detail::is_formattable_delayed<R, Char>
|
range_format_kind<R, Char>::value != range_format::debug_string>,
|
||||||
#endif
|
detail::is_formattable_delayed<R, Char>>::value>> {
|
||||||
>::value>>
|
private:
|
||||||
: detail::range_default_formatter<range_format_kind<R, Char>::value, R,
|
using range_type = detail::maybe_const_range<R>;
|
||||||
Char> {
|
range_formatter<detail::uncvref_type<range_type>, Char> range_formatter_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using nonlocking = void;
|
||||||
|
|
||||||
|
FMT_CONSTEXPR formatter() {
|
||||||
|
if (detail::const_check(range_format_kind<R, Char>::value !=
|
||||||
|
range_format::set))
|
||||||
|
return;
|
||||||
|
range_formatter_.set_brackets(detail::string_literal<Char, '{'>{},
|
||||||
|
detail::string_literal<Char, '}'>{});
|
||||||
|
}
|
||||||
|
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
return range_formatter_.parse(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(range_type& range, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
return range_formatter_.format(range, ctx);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char, typename... T> struct tuple_join_view : detail::view {
|
// A map formatter.
|
||||||
const std::tuple<T...>& tuple;
|
template <typename R, typename Char>
|
||||||
|
struct formatter<
|
||||||
|
R, Char,
|
||||||
|
enable_if_t<conjunction<
|
||||||
|
bool_constant<range_format_kind<R, Char>::value == range_format::map>,
|
||||||
|
detail::is_formattable_delayed<R, Char>>::value>> {
|
||||||
|
private:
|
||||||
|
using map_type = detail::maybe_const_range<R>;
|
||||||
|
using element_type = detail::uncvref_type<map_type>;
|
||||||
|
|
||||||
|
decltype(detail::tuple::get_formatters<element_type, Char>(
|
||||||
|
detail::tuple_index_sequence<element_type>())) formatters_;
|
||||||
|
bool no_delimiters_ = false;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR formatter() {}
|
||||||
|
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
auto it = ctx.begin();
|
||||||
|
auto end = ctx.end();
|
||||||
|
if (it != end) {
|
||||||
|
if (detail::to_ascii(*it) == 'n') {
|
||||||
|
no_delimiters_ = true;
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
if (it != end && *it != '}') {
|
||||||
|
if (*it != ':') report_error("invalid format specifier");
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
ctx.advance_to(it);
|
||||||
|
}
|
||||||
|
detail::for_each(formatters_, detail::parse_empty_specs<Char>{ctx});
|
||||||
|
return it;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(map_type& map, FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
basic_string_view<Char> open = detail::string_literal<Char, '{'>{};
|
||||||
|
if (!no_delimiters_) out = detail::copy<Char>(open, out);
|
||||||
|
int i = 0;
|
||||||
|
basic_string_view<Char> sep = detail::string_literal<Char, ',', ' '>{};
|
||||||
|
for (auto&& value : map) {
|
||||||
|
if (i > 0) out = detail::copy<Char>(sep, out);
|
||||||
|
ctx.advance_to(out);
|
||||||
|
detail::for_each2(formatters_, value,
|
||||||
|
detail::format_tuple_element<FormatContext>{
|
||||||
|
0, ctx, detail::string_literal<Char, ':', ' '>{}});
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
basic_string_view<Char> close = detail::string_literal<Char, '}'>{};
|
||||||
|
if (!no_delimiters_) out = detail::copy<Char>(close, out);
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// A (debug_)string formatter.
|
||||||
|
template <typename R, typename Char>
|
||||||
|
struct formatter<
|
||||||
|
R, Char,
|
||||||
|
enable_if_t<range_format_kind<R, Char>::value == range_format::string ||
|
||||||
|
range_format_kind<R, Char>::value ==
|
||||||
|
range_format::debug_string>> {
|
||||||
|
private:
|
||||||
|
using range_type = detail::maybe_const_range<R>;
|
||||||
|
using string_type =
|
||||||
|
conditional_t<std::is_constructible<
|
||||||
|
detail::std_string_view<Char>,
|
||||||
|
decltype(detail::range_begin(std::declval<R>())),
|
||||||
|
decltype(detail::range_end(std::declval<R>()))>::value,
|
||||||
|
detail::std_string_view<Char>, std::basic_string<Char>>;
|
||||||
|
|
||||||
|
formatter<string_type, Char> underlying_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
return underlying_.parse(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(range_type& range, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
if (detail::const_check(range_format_kind<R, Char>::value ==
|
||||||
|
range_format::debug_string))
|
||||||
|
*out++ = '"';
|
||||||
|
out = underlying_.format(
|
||||||
|
string_type{detail::range_begin(range), detail::range_end(range)}, ctx);
|
||||||
|
if (detail::const_check(range_format_kind<R, Char>::value ==
|
||||||
|
range_format::debug_string))
|
||||||
|
*out++ = '"';
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename It, typename Sentinel, typename Char = char>
|
||||||
|
struct join_view : detail::view {
|
||||||
|
It begin;
|
||||||
|
Sentinel end;
|
||||||
basic_string_view<Char> sep;
|
basic_string_view<Char> sep;
|
||||||
|
|
||||||
tuple_join_view(const std::tuple<T...>& t, basic_string_view<Char> s)
|
join_view(It b, Sentinel e, basic_string_view<Char> s)
|
||||||
|
: begin(std::move(b)), end(e), sep(s) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename It, typename Sentinel, typename Char>
|
||||||
|
struct formatter<join_view<It, Sentinel, Char>, Char> {
|
||||||
|
private:
|
||||||
|
using value_type =
|
||||||
|
#ifdef __cpp_lib_ranges
|
||||||
|
std::iter_value_t<It>;
|
||||||
|
#else
|
||||||
|
typename std::iterator_traits<It>::value_type;
|
||||||
|
#endif
|
||||||
|
formatter<remove_cvref_t<value_type>, Char> value_formatter_;
|
||||||
|
|
||||||
|
using view = conditional_t<std::is_copy_constructible<It>::value,
|
||||||
|
const join_view<It, Sentinel, Char>,
|
||||||
|
join_view<It, Sentinel, Char>>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using nonlocking = void;
|
||||||
|
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
return value_formatter_.parse(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(view& value, FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
|
using iter =
|
||||||
|
conditional_t<std::is_copy_constructible<view>::value, It, It&>;
|
||||||
|
iter it = value.begin;
|
||||||
|
auto out = ctx.out();
|
||||||
|
if (it == value.end) return out;
|
||||||
|
out = value_formatter_.format(*it, ctx);
|
||||||
|
++it;
|
||||||
|
while (it != value.end) {
|
||||||
|
out = detail::copy<Char>(value.sep.begin(), value.sep.end(), out);
|
||||||
|
ctx.advance_to(out);
|
||||||
|
out = value_formatter_.format(*it, ctx);
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
template <typename Tuple, typename Char> struct tuple_join_view : detail::view {
|
||||||
|
const Tuple& tuple;
|
||||||
|
basic_string_view<Char> sep;
|
||||||
|
|
||||||
|
tuple_join_view(const Tuple& t, basic_string_view<Char> s)
|
||||||
: tuple(t), sep{s} {}
|
: tuple(t), sep{s} {}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -586,65 +686,64 @@ template <typename Char, typename... T> struct tuple_join_view : detail::view {
|
||||||
# define FMT_TUPLE_JOIN_SPECIFIERS 0
|
# define FMT_TUPLE_JOIN_SPECIFIERS 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
template <typename Char, typename... T>
|
template <typename Tuple, typename Char>
|
||||||
struct formatter<tuple_join_view<Char, T...>, Char> {
|
struct formatter<tuple_join_view<Tuple, Char>, Char,
|
||||||
template <typename ParseContext>
|
enable_if_t<is_tuple_like<Tuple>::value>> {
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
return do_parse(ctx, std::integral_constant<size_t, sizeof...(T)>());
|
return do_parse(ctx, std::tuple_size<Tuple>());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const tuple_join_view<Char, T...>& value,
|
auto format(const tuple_join_view<Tuple, Char>& value,
|
||||||
FormatContext& ctx) const -> typename FormatContext::iterator {
|
FormatContext& ctx) const -> typename FormatContext::iterator {
|
||||||
return do_format(value, ctx,
|
return do_format(value, ctx, std::tuple_size<Tuple>());
|
||||||
std::integral_constant<size_t, sizeof...(T)>());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::tuple<formatter<typename std::decay<T>::type, Char>...> formatters_;
|
decltype(detail::tuple::get_formatters<Tuple, Char>(
|
||||||
|
detail::tuple_index_sequence<Tuple>())) formatters_;
|
||||||
|
|
||||||
template <typename ParseContext>
|
FMT_CONSTEXPR auto do_parse(parse_context<Char>& ctx,
|
||||||
FMT_CONSTEXPR auto do_parse(ParseContext& ctx,
|
|
||||||
std::integral_constant<size_t, 0>)
|
std::integral_constant<size_t, 0>)
|
||||||
-> decltype(ctx.begin()) {
|
-> const Char* {
|
||||||
return ctx.begin();
|
return ctx.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ParseContext, size_t N>
|
template <size_t N>
|
||||||
FMT_CONSTEXPR auto do_parse(ParseContext& ctx,
|
FMT_CONSTEXPR auto do_parse(parse_context<Char>& ctx,
|
||||||
std::integral_constant<size_t, N>)
|
std::integral_constant<size_t, N>)
|
||||||
-> decltype(ctx.begin()) {
|
-> const Char* {
|
||||||
auto end = ctx.begin();
|
auto end = ctx.begin();
|
||||||
#if FMT_TUPLE_JOIN_SPECIFIERS
|
#if FMT_TUPLE_JOIN_SPECIFIERS
|
||||||
end = std::get<sizeof...(T) - N>(formatters_).parse(ctx);
|
end = std::get<std::tuple_size<Tuple>::value - N>(formatters_).parse(ctx);
|
||||||
if (N > 1) {
|
if (N > 1) {
|
||||||
auto end1 = do_parse(ctx, std::integral_constant<size_t, N - 1>());
|
auto end1 = do_parse(ctx, std::integral_constant<size_t, N - 1>());
|
||||||
if (end != end1)
|
if (end != end1)
|
||||||
FMT_THROW(format_error("incompatible format specs for tuple elements"));
|
report_error("incompatible format specs for tuple elements");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
return end;
|
return end;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto do_format(const tuple_join_view<Char, T...>&, FormatContext& ctx,
|
auto do_format(const tuple_join_view<Tuple, Char>&, FormatContext& ctx,
|
||||||
std::integral_constant<size_t, 0>) const ->
|
std::integral_constant<size_t, 0>) const ->
|
||||||
typename FormatContext::iterator {
|
typename FormatContext::iterator {
|
||||||
return ctx.out();
|
return ctx.out();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext, size_t N>
|
template <typename FormatContext, size_t N>
|
||||||
auto do_format(const tuple_join_view<Char, T...>& value, FormatContext& ctx,
|
auto do_format(const tuple_join_view<Tuple, Char>& value, FormatContext& ctx,
|
||||||
std::integral_constant<size_t, N>) const ->
|
std::integral_constant<size_t, N>) const ->
|
||||||
typename FormatContext::iterator {
|
typename FormatContext::iterator {
|
||||||
auto out = std::get<sizeof...(T) - N>(formatters_)
|
using std::get;
|
||||||
.format(std::get<sizeof...(T) - N>(value.tuple), ctx);
|
auto out =
|
||||||
if (N > 1) {
|
std::get<std::tuple_size<Tuple>::value - N>(formatters_)
|
||||||
out = std::copy(value.sep.begin(), value.sep.end(), out);
|
.format(get<std::tuple_size<Tuple>::value - N>(value.tuple), ctx);
|
||||||
ctx.advance_to(out);
|
if (N <= 1) return out;
|
||||||
return do_format(value, ctx, std::integral_constant<size_t, N - 1>());
|
out = detail::copy<Char>(value.sep, out);
|
||||||
}
|
ctx.advance_to(out);
|
||||||
return out;
|
return do_format(value, ctx, std::integral_constant<size_t, N - 1>());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -656,7 +755,7 @@ template <typename T> class is_container_adaptor_like {
|
||||||
template <typename> static void check(...);
|
template <typename> static void check(...);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr const bool value =
|
static constexpr bool value =
|
||||||
!std::is_void<decltype(check<T>(nullptr))>::value;
|
!std::is_void<decltype(check<T>(nullptr))>::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -676,52 +775,69 @@ struct formatter<
|
||||||
: formatter<detail::all<typename T::container_type>, Char> {
|
: formatter<detail::all<typename T::container_type>, Char> {
|
||||||
using all = detail::all<typename T::container_type>;
|
using all = detail::all<typename T::container_type>;
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const T& t, FormatContext& ctx) const -> decltype(ctx.out()) {
|
auto format(const T& value, FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
struct getter : T {
|
struct getter : T {
|
||||||
static auto get(const T& t) -> all {
|
static auto get(const T& v) -> all {
|
||||||
return {t.*(&getter::c)}; // Access c through the derived class.
|
return {v.*(&getter::c)}; // Access c through the derived class.
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
return formatter<all>::format(getter::get(t), ctx);
|
return formatter<all>::format(getter::get(value), ctx);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_BEGIN_EXPORT
|
FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
|
/// Returns a view that formats the iterator range `[begin, end)` with elements
|
||||||
|
/// separated by `sep`.
|
||||||
|
template <typename It, typename Sentinel>
|
||||||
|
auto join(It begin, Sentinel end, string_view sep) -> join_view<It, Sentinel> {
|
||||||
|
return {std::move(begin), end, sep};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Returns a view that formats `range` with elements separated by `sep`.
|
||||||
Returns an object that formats `tuple` with elements separated by `sep`.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* auto v = std::vector<int>{1, 2, 3};
|
||||||
std::tuple<int, char> t = {1, 'a'};
|
* fmt::print("{}", fmt::join(v, ", "));
|
||||||
fmt::print("{}", fmt::join(t, ", "));
|
* // Output: 1, 2, 3
|
||||||
// Output: "1, a"
|
*
|
||||||
\endrst
|
* `fmt::join` applies passed format specifiers to the range elements:
|
||||||
|
*
|
||||||
|
* fmt::print("{:02}", fmt::join(v, ", "));
|
||||||
|
* // Output: 01, 02, 03
|
||||||
*/
|
*/
|
||||||
template <typename... T>
|
template <typename Range, FMT_ENABLE_IF(!is_tuple_like<Range>::value)>
|
||||||
FMT_CONSTEXPR auto join(const std::tuple<T...>& tuple, string_view sep)
|
auto join(Range&& r, string_view sep)
|
||||||
-> tuple_join_view<char, T...> {
|
-> join_view<decltype(detail::range_begin(r)),
|
||||||
return {tuple, sep};
|
decltype(detail::range_end(r))> {
|
||||||
|
return {detail::range_begin(r), detail::range_end(r), sep};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... T>
|
/**
|
||||||
FMT_CONSTEXPR auto join(const std::tuple<T...>& tuple,
|
* Returns an object that formats `std::tuple` with elements separated by `sep`.
|
||||||
basic_string_view<wchar_t> sep)
|
*
|
||||||
-> tuple_join_view<wchar_t, T...> {
|
* **Example**:
|
||||||
|
*
|
||||||
|
* auto t = std::tuple<int, char>(1, 'a');
|
||||||
|
* fmt::print("{}", fmt::join(t, ", "));
|
||||||
|
* // Output: 1, a
|
||||||
|
*/
|
||||||
|
template <typename Tuple, FMT_ENABLE_IF(is_tuple_like<Tuple>::value)>
|
||||||
|
FMT_CONSTEXPR auto join(const Tuple& tuple FMT_LIFETIMEBOUND, string_view sep)
|
||||||
|
-> tuple_join_view<Tuple, char> {
|
||||||
return {tuple, sep};
|
return {tuple, sep};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\rst
|
* Returns an object that formats `std::initializer_list` with elements
|
||||||
Returns an object that formats `initializer_list` with elements separated by
|
* separated by `sep`.
|
||||||
`sep`.
|
*
|
||||||
|
* **Example**:
|
||||||
**Example**::
|
*
|
||||||
|
* fmt::print("{}", fmt::join({1, 2, 3}, ", "));
|
||||||
fmt::print("{}", fmt::join({1, 2, 3}, ", "));
|
* // Output: "1, 2, 3"
|
||||||
// Output: "1, 2, 3"
|
|
||||||
\endrst
|
|
||||||
*/
|
*/
|
||||||
template <typename T>
|
template <typename T>
|
||||||
auto join(std::initializer_list<T> list, string_view sep)
|
auto join(std::initializer_list<T> list, string_view sep)
|
||||||
|
|
|
||||||
758
deps/fmt/include/fmt/std.h
vendored
758
deps/fmt/include/fmt/std.h
vendored
|
|
@ -8,35 +8,47 @@
|
||||||
#ifndef FMT_STD_H_
|
#ifndef FMT_STD_H_
|
||||||
#define FMT_STD_H_
|
#define FMT_STD_H_
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <bitset>
|
|
||||||
#include <cstdlib>
|
|
||||||
#include <exception>
|
|
||||||
#include <memory>
|
|
||||||
#include <thread>
|
|
||||||
#include <type_traits>
|
|
||||||
#include <typeinfo>
|
|
||||||
#include <utility>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
#include "ostream.h"
|
#include "ostream.h"
|
||||||
|
|
||||||
|
#ifndef FMT_MODULE
|
||||||
|
# include <atomic>
|
||||||
|
# include <bitset>
|
||||||
|
# include <complex>
|
||||||
|
# include <exception>
|
||||||
|
# include <functional> // std::reference_wrapper
|
||||||
|
# include <memory>
|
||||||
|
# include <thread>
|
||||||
|
# include <type_traits>
|
||||||
|
# include <typeinfo> // std::type_info
|
||||||
|
# include <utility> // std::make_index_sequence
|
||||||
|
|
||||||
|
// Check FMT_CPLUSPLUS to suppress a bogus warning in MSVC.
|
||||||
|
# if FMT_CPLUSPLUS >= 201703L
|
||||||
|
# if FMT_HAS_INCLUDE(<filesystem>) && \
|
||||||
|
(!defined(FMT_CPP_LIB_FILESYSTEM) || FMT_CPP_LIB_FILESYSTEM != 0)
|
||||||
|
# include <filesystem>
|
||||||
|
# endif
|
||||||
|
# if FMT_HAS_INCLUDE(<variant>)
|
||||||
|
# include <variant>
|
||||||
|
# endif
|
||||||
|
# if FMT_HAS_INCLUDE(<optional>)
|
||||||
|
# include <optional>
|
||||||
|
# endif
|
||||||
|
# endif
|
||||||
|
// Use > instead of >= in the version check because <source_location> may be
|
||||||
|
// available after C++17 but before C++20 is marked as implemented.
|
||||||
|
# if FMT_CPLUSPLUS > 201703L && FMT_HAS_INCLUDE(<source_location>)
|
||||||
|
# include <source_location>
|
||||||
|
# endif
|
||||||
|
# if FMT_CPLUSPLUS > 202002L && FMT_HAS_INCLUDE(<expected>)
|
||||||
|
# include <expected>
|
||||||
|
# endif
|
||||||
|
#endif // FMT_MODULE
|
||||||
|
|
||||||
#if FMT_HAS_INCLUDE(<version>)
|
#if FMT_HAS_INCLUDE(<version>)
|
||||||
# include <version>
|
# include <version>
|
||||||
#endif
|
#endif
|
||||||
// Checking FMT_CPLUSPLUS for warning suppression in MSVC.
|
|
||||||
#if FMT_CPLUSPLUS >= 201703L
|
|
||||||
# if FMT_HAS_INCLUDE(<filesystem>)
|
|
||||||
# include <filesystem>
|
|
||||||
# endif
|
|
||||||
# if FMT_HAS_INCLUDE(<variant>)
|
|
||||||
# include <variant>
|
|
||||||
# endif
|
|
||||||
# if FMT_HAS_INCLUDE(<optional>)
|
|
||||||
# include <optional>
|
|
||||||
# endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// GCC 4 does not support FMT_HAS_INCLUDE.
|
// GCC 4 does not support FMT_HAS_INCLUDE.
|
||||||
#if FMT_HAS_INCLUDE(<cxxabi.h>) || defined(__GLIBCXX__)
|
#if FMT_HAS_INCLUDE(<cxxabi.h>) || defined(__GLIBCXX__)
|
||||||
|
|
@ -48,139 +60,341 @@
|
||||||
# endif
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Check if typeid is available.
|
#ifdef FMT_CPP_LIB_FILESYSTEM
|
||||||
#ifndef FMT_USE_TYPEID
|
// Use the provided definition.
|
||||||
// __RTTI is for EDG compilers. In MSVC typeid is available without RTTI.
|
#elif defined(__cpp_lib_filesystem)
|
||||||
# if defined(__GXX_RTTI) || FMT_HAS_FEATURE(cxx_rtti) || FMT_MSC_VERSION || \
|
# define FMT_CPP_LIB_FILESYSTEM __cpp_lib_filesystem
|
||||||
defined(__INTEL_RTTI__) || defined(__RTTI)
|
#else
|
||||||
# define FMT_USE_TYPEID 1
|
# define FMT_CPP_LIB_FILESYSTEM 0
|
||||||
# else
|
|
||||||
# define FMT_USE_TYPEID 0
|
|
||||||
# endif
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __cpp_lib_filesystem
|
#ifdef FMT_CPP_LIB_VARIANT
|
||||||
FMT_BEGIN_NAMESPACE
|
// Use the provided definition.
|
||||||
|
#elif defined(__cpp_lib_variant)
|
||||||
|
# define FMT_CPP_LIB_VARIANT __cpp_lib_variant
|
||||||
|
#else
|
||||||
|
# define FMT_CPP_LIB_VARIANT 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template <typename Char> auto get_path_string(const std::filesystem::path& p) {
|
#if FMT_CPP_LIB_FILESYSTEM
|
||||||
return p.string<Char>();
|
|
||||||
|
template <typename Char, typename PathChar>
|
||||||
|
auto get_path_string(const std::filesystem::path& p,
|
||||||
|
const std::basic_string<PathChar>& native) {
|
||||||
|
if constexpr (std::is_same_v<Char, char> && std::is_same_v<PathChar, wchar_t>)
|
||||||
|
return to_utf8<wchar_t>(native, to_utf8_error_policy::replace);
|
||||||
|
else
|
||||||
|
return p.string<Char>();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char, typename PathChar>
|
||||||
void write_escaped_path(basic_memory_buffer<Char>& quoted,
|
void write_escaped_path(basic_memory_buffer<Char>& quoted,
|
||||||
const std::filesystem::path& p) {
|
const std::filesystem::path& p,
|
||||||
write_escaped_string<Char>(std::back_inserter(quoted), p.string<Char>());
|
const std::basic_string<PathChar>& native) {
|
||||||
|
if constexpr (std::is_same_v<Char, char> &&
|
||||||
|
std::is_same_v<PathChar, wchar_t>) {
|
||||||
|
auto buf = basic_memory_buffer<wchar_t>();
|
||||||
|
write_escaped_string<wchar_t>(std::back_inserter(buf), native);
|
||||||
|
bool valid = to_utf8<wchar_t>::convert(quoted, {buf.data(), buf.size()});
|
||||||
|
FMT_ASSERT(valid, "invalid utf16");
|
||||||
|
} else if constexpr (std::is_same_v<Char, PathChar>) {
|
||||||
|
write_escaped_string<std::filesystem::path::value_type>(
|
||||||
|
std::back_inserter(quoted), native);
|
||||||
|
} else {
|
||||||
|
write_escaped_string<Char>(std::back_inserter(quoted), p.string<Char>());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
# ifdef _WIN32
|
#endif // FMT_CPP_LIB_FILESYSTEM
|
||||||
template <>
|
|
||||||
inline auto get_path_string<char>(const std::filesystem::path& p) {
|
#if defined(__cpp_lib_expected) || FMT_CPP_LIB_VARIANT
|
||||||
return to_utf8<wchar_t>(p.native(), to_utf8_error_policy::replace);
|
|
||||||
|
template <typename Char, typename OutputIt, typename T, typename FormatContext>
|
||||||
|
auto write_escaped_alternative(OutputIt out, const T& v, FormatContext& ctx)
|
||||||
|
-> OutputIt {
|
||||||
|
if constexpr (has_to_string_view<T>::value)
|
||||||
|
return write_escaped_string<Char>(out, detail::to_string_view(v));
|
||||||
|
if constexpr (std::is_same_v<T, Char>) return write_escaped_char(out, v);
|
||||||
|
|
||||||
|
formatter<std::remove_cv_t<T>, Char> underlying;
|
||||||
|
maybe_set_debug_format(underlying, true);
|
||||||
|
return underlying.format(v, ctx);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if FMT_CPP_LIB_VARIANT
|
||||||
|
|
||||||
|
template <typename> struct is_variant_like_ : std::false_type {};
|
||||||
|
template <typename... Types>
|
||||||
|
struct is_variant_like_<std::variant<Types...>> : std::true_type {};
|
||||||
|
|
||||||
|
template <typename Variant, typename Char> class is_variant_formattable {
|
||||||
|
template <size_t... Is>
|
||||||
|
static auto check(std::index_sequence<Is...>) -> std::conjunction<
|
||||||
|
is_formattable<std::variant_alternative_t<Is, Variant>, Char>...>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
static constexpr bool value = decltype(check(
|
||||||
|
std::make_index_sequence<std::variant_size<Variant>::value>()))::value;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // FMT_CPP_LIB_VARIANT
|
||||||
|
|
||||||
|
#if FMT_USE_RTTI
|
||||||
|
inline auto normalize_libcxx_inline_namespaces(string_view demangled_name_view,
|
||||||
|
char* begin) -> string_view {
|
||||||
|
// Normalization of stdlib inline namespace names.
|
||||||
|
// libc++ inline namespaces.
|
||||||
|
// std::__1::* -> std::*
|
||||||
|
// std::__1::__fs::* -> std::*
|
||||||
|
// libstdc++ inline namespaces.
|
||||||
|
// std::__cxx11::* -> std::*
|
||||||
|
// std::filesystem::__cxx11::* -> std::filesystem::*
|
||||||
|
if (demangled_name_view.starts_with("std::")) {
|
||||||
|
char* to = begin + 5; // std::
|
||||||
|
for (const char *from = to, *end = begin + demangled_name_view.size();
|
||||||
|
from < end;) {
|
||||||
|
// This is safe, because demangled_name is NUL-terminated.
|
||||||
|
if (from[0] == '_' && from[1] == '_') {
|
||||||
|
const char* next = from + 1;
|
||||||
|
while (next < end && *next != ':') next++;
|
||||||
|
if (next[0] == ':' && next[1] == ':') {
|
||||||
|
from = next + 2;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*to++ = *from++;
|
||||||
|
}
|
||||||
|
demangled_name_view = {begin, detail::to_unsigned(to - begin)};
|
||||||
|
}
|
||||||
|
return demangled_name_view;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template <class OutputIt>
|
||||||
inline void write_escaped_path<char>(memory_buffer& quoted,
|
auto normalize_msvc_abi_name(string_view abi_name_view, OutputIt out)
|
||||||
const std::filesystem::path& p) {
|
-> OutputIt {
|
||||||
auto buf = basic_memory_buffer<wchar_t>();
|
const string_view demangled_name(abi_name_view);
|
||||||
write_escaped_string<wchar_t>(std::back_inserter(buf), p.native());
|
for (size_t i = 0; i < demangled_name.size(); ++i) {
|
||||||
bool valid = to_utf8<wchar_t>::convert(quoted, {buf.data(), buf.size()});
|
auto sub = demangled_name;
|
||||||
FMT_ASSERT(valid, "invalid utf16");
|
sub.remove_prefix(i);
|
||||||
|
if (sub.starts_with("enum ")) {
|
||||||
|
i += 4;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (sub.starts_with("class ") || sub.starts_with("union ")) {
|
||||||
|
i += 5;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (sub.starts_with("struct ")) {
|
||||||
|
i += 6;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (*sub.begin() != ' ') *out++ = *sub.begin();
|
||||||
|
}
|
||||||
|
return out;
|
||||||
}
|
}
|
||||||
# endif // _WIN32
|
|
||||||
|
|
||||||
template <>
|
template <typename OutputIt>
|
||||||
inline void write_escaped_path<std::filesystem::path::value_type>(
|
auto write_demangled_name(OutputIt out, const std::type_info& ti) -> OutputIt {
|
||||||
basic_memory_buffer<std::filesystem::path::value_type>& quoted,
|
# ifdef FMT_HAS_ABI_CXA_DEMANGLE
|
||||||
const std::filesystem::path& p) {
|
int status = 0;
|
||||||
write_escaped_string<std::filesystem::path::value_type>(
|
size_t size = 0;
|
||||||
std::back_inserter(quoted), p.native());
|
std::unique_ptr<char, void (*)(void*)> demangled_name_ptr(
|
||||||
|
abi::__cxa_demangle(ti.name(), nullptr, &size, &status), &free);
|
||||||
|
|
||||||
|
string_view demangled_name_view;
|
||||||
|
if (demangled_name_ptr) {
|
||||||
|
demangled_name_view = normalize_libcxx_inline_namespaces(
|
||||||
|
demangled_name_ptr.get(), demangled_name_ptr.get());
|
||||||
|
} else {
|
||||||
|
demangled_name_view = string_view(ti.name());
|
||||||
|
}
|
||||||
|
return detail::write_bytes<char>(out, demangled_name_view);
|
||||||
|
# elif FMT_MSC_VERSION && defined(_MSVC_STL_UPDATE)
|
||||||
|
return normalize_msvc_abi_name(ti.name(), out);
|
||||||
|
# elif FMT_MSC_VERSION && defined(_LIBCPP_VERSION)
|
||||||
|
const string_view demangled_name = ti.name();
|
||||||
|
std::string name_copy(demangled_name.size(), '\0');
|
||||||
|
// normalize_msvc_abi_name removes class, struct, union etc that MSVC has in
|
||||||
|
// front of types
|
||||||
|
name_copy.erase(normalize_msvc_abi_name(demangled_name, name_copy.begin()),
|
||||||
|
name_copy.end());
|
||||||
|
// normalize_libcxx_inline_namespaces removes the inline __1, __2, etc
|
||||||
|
// namespaces libc++ uses for ABI versioning On MSVC ABI + libc++
|
||||||
|
// environments, we need to eliminate both of them.
|
||||||
|
const string_view normalized_name =
|
||||||
|
normalize_libcxx_inline_namespaces(name_copy, name_copy.data());
|
||||||
|
return detail::write_bytes<char>(out, normalized_name);
|
||||||
|
# else
|
||||||
|
return detail::write_bytes<char>(out, string_view(ti.name()));
|
||||||
|
# endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#endif // FMT_USE_RTTI
|
||||||
|
|
||||||
|
template <typename T, typename Enable = void>
|
||||||
|
struct has_flip : std::false_type {};
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct has_flip<T, void_t<decltype(std::declval<T>().flip())>>
|
||||||
|
: std::true_type {};
|
||||||
|
|
||||||
|
template <typename T> struct is_bit_reference_like {
|
||||||
|
static constexpr bool value = std::is_convertible<T, bool>::value &&
|
||||||
|
std::is_nothrow_assignable<T, bool>::value &&
|
||||||
|
has_flip<T>::value;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Workaround for libc++ incompatibility with C++ standard.
|
||||||
|
// According to the Standard, `bitset::operator[] const` returns bool.
|
||||||
|
#if defined(_LIBCPP_VERSION) && !defined(FMT_IMPORT_STD)
|
||||||
|
template <typename C>
|
||||||
|
struct is_bit_reference_like<std::__bit_const_reference<C>> {
|
||||||
|
static constexpr bool value = true;
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template <typename T, typename Enable = void>
|
||||||
|
struct has_format_as : std::false_type {};
|
||||||
|
template <typename T>
|
||||||
|
struct has_format_as<T, void_t<decltype(format_as(std::declval<const T&>()))>>
|
||||||
|
: std::true_type {};
|
||||||
|
|
||||||
|
template <typename T, typename Enable = void>
|
||||||
|
struct has_format_as_member : std::false_type {};
|
||||||
|
template <typename T>
|
||||||
|
struct has_format_as_member<
|
||||||
|
T, void_t<decltype(formatter<T>::format_as(std::declval<const T&>()))>>
|
||||||
|
: std::true_type {};
|
||||||
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
FMT_EXPORT
|
template <typename T, typename Deleter>
|
||||||
|
auto ptr(const std::unique_ptr<T, Deleter>& p) -> const void* {
|
||||||
|
return p.get();
|
||||||
|
}
|
||||||
|
template <typename T> auto ptr(const std::shared_ptr<T>& p) -> const void* {
|
||||||
|
return p.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
#if FMT_CPP_LIB_FILESYSTEM
|
||||||
|
|
||||||
template <typename Char> struct formatter<std::filesystem::path, Char> {
|
template <typename Char> struct formatter<std::filesystem::path, Char> {
|
||||||
private:
|
private:
|
||||||
format_specs<Char> specs_;
|
format_specs specs_;
|
||||||
detail::arg_ref<Char> width_ref_;
|
detail::arg_ref<Char> width_ref_;
|
||||||
bool debug_ = false;
|
bool debug_ = false;
|
||||||
|
char path_type_ = 0;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR void set_debug_format(bool set = true) { debug_ = set; }
|
FMT_CONSTEXPR void set_debug_format(bool set = true) { debug_ = set; }
|
||||||
|
|
||||||
template <typename ParseContext> FMT_CONSTEXPR auto parse(ParseContext& ctx) {
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) {
|
||||||
auto it = ctx.begin(), end = ctx.end();
|
auto it = ctx.begin(), end = ctx.end();
|
||||||
if (it == end) return it;
|
if (it == end) return it;
|
||||||
|
|
||||||
it = detail::parse_align(it, end, specs_);
|
it = detail::parse_align(it, end, specs_);
|
||||||
if (it == end) return it;
|
if (it == end) return it;
|
||||||
|
|
||||||
it = detail::parse_dynamic_spec(it, end, specs_.width, width_ref_, ctx);
|
Char c = *it;
|
||||||
|
if ((c >= '0' && c <= '9') || c == '{')
|
||||||
|
it = detail::parse_width(it, end, specs_, width_ref_, ctx);
|
||||||
if (it != end && *it == '?') {
|
if (it != end && *it == '?') {
|
||||||
debug_ = true;
|
debug_ = true;
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
|
if (it != end && (*it == 'g')) path_type_ = detail::to_ascii(*it++);
|
||||||
return it;
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const std::filesystem::path& p, FormatContext& ctx) const {
|
auto format(const std::filesystem::path& p, FormatContext& ctx) const {
|
||||||
auto specs = specs_;
|
auto specs = specs_;
|
||||||
detail::handle_dynamic_spec<detail::width_checker>(specs.width, width_ref_,
|
auto path_string =
|
||||||
ctx);
|
!path_type_ ? p.native()
|
||||||
|
: p.generic_string<std::filesystem::path::value_type>();
|
||||||
|
|
||||||
|
detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
|
||||||
|
ctx);
|
||||||
if (!debug_) {
|
if (!debug_) {
|
||||||
auto s = detail::get_path_string<Char>(p);
|
auto s = detail::get_path_string<Char>(p, path_string);
|
||||||
return detail::write(ctx.out(), basic_string_view<Char>(s), specs);
|
return detail::write(ctx.out(), basic_string_view<Char>(s), specs);
|
||||||
}
|
}
|
||||||
auto quoted = basic_memory_buffer<Char>();
|
auto quoted = basic_memory_buffer<Char>();
|
||||||
detail::write_escaped_path(quoted, p);
|
detail::write_escaped_path(quoted, p, path_string);
|
||||||
return detail::write(ctx.out(),
|
return detail::write(ctx.out(),
|
||||||
basic_string_view<Char>(quoted.data(), quoted.size()),
|
basic_string_view<Char>(quoted.data(), quoted.size()),
|
||||||
specs);
|
specs);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
FMT_END_NAMESPACE
|
|
||||||
#endif
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
class path : public std::filesystem::path {
|
||||||
FMT_EXPORT
|
public:
|
||||||
|
auto display_string() const -> std::string {
|
||||||
|
const std::filesystem::path& base = *this;
|
||||||
|
return fmt::format(FMT_STRING("{}"), base);
|
||||||
|
}
|
||||||
|
auto system_string() const -> std::string { return string(); }
|
||||||
|
|
||||||
|
auto generic_display_string() const -> std::string {
|
||||||
|
const std::filesystem::path& base = *this;
|
||||||
|
return fmt::format(FMT_STRING("{:g}"), base);
|
||||||
|
}
|
||||||
|
auto generic_system_string() const -> std::string { return generic_string(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // FMT_CPP_LIB_FILESYSTEM
|
||||||
|
|
||||||
|
template <size_t N, typename Char>
|
||||||
|
struct formatter<std::bitset<N>, Char>
|
||||||
|
: nested_formatter<basic_string_view<Char>, Char> {
|
||||||
|
private:
|
||||||
|
// This is a functor because C++11 doesn't support generic lambdas.
|
||||||
|
struct writer {
|
||||||
|
const std::bitset<N>& bs;
|
||||||
|
|
||||||
|
template <typename OutputIt>
|
||||||
|
FMT_CONSTEXPR auto operator()(OutputIt out) -> OutputIt {
|
||||||
|
for (auto pos = N; pos > 0; --pos)
|
||||||
|
out = detail::write<Char>(out, bs[pos - 1] ? Char('1') : Char('0'));
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(const std::bitset<N>& bs, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
return this->write_padded(ctx, writer{bs});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
struct formatter<std::thread::id, Char> : basic_ostream_formatter<Char> {};
|
struct formatter<std::thread::id, Char> : basic_ostream_formatter<Char> {};
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#ifdef __cpp_lib_optional
|
#ifdef __cpp_lib_optional
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
struct formatter<std::optional<T>, Char,
|
struct formatter<std::optional<T>, Char,
|
||||||
std::enable_if_t<is_formattable<T, Char>::value>> {
|
std::enable_if_t<is_formattable<T, Char>::value>> {
|
||||||
private:
|
private:
|
||||||
formatter<T, Char> underlying_;
|
formatter<std::remove_cv_t<T>, Char> underlying_;
|
||||||
static constexpr basic_string_view<Char> optional =
|
static constexpr basic_string_view<Char> optional =
|
||||||
detail::string_literal<Char, 'o', 'p', 't', 'i', 'o', 'n', 'a', 'l',
|
detail::string_literal<Char, 'o', 'p', 't', 'i', 'o', 'n', 'a', 'l',
|
||||||
'('>{};
|
'('>{};
|
||||||
static constexpr basic_string_view<Char> none =
|
static constexpr basic_string_view<Char> none =
|
||||||
detail::string_literal<Char, 'n', 'o', 'n', 'e'>{};
|
detail::string_literal<Char, 'n', 'o', 'n', 'e'>{};
|
||||||
|
|
||||||
template <class U>
|
|
||||||
FMT_CONSTEXPR static auto maybe_set_debug_format(U& u, bool set)
|
|
||||||
-> decltype(u.set_debug_format(set)) {
|
|
||||||
u.set_debug_format(set);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <class U>
|
|
||||||
FMT_CONSTEXPR static void maybe_set_debug_format(U&, ...) {}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
template <typename ParseContext> FMT_CONSTEXPR auto parse(ParseContext& ctx) {
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) {
|
||||||
maybe_set_debug_format(underlying_, true);
|
detail::maybe_set_debug_format(underlying_, true);
|
||||||
return underlying_.parse(ctx);
|
return underlying_.parse(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(std::optional<T> const& opt, FormatContext& ctx) const
|
auto format(const std::optional<T>& opt, FormatContext& ctx) const
|
||||||
-> decltype(ctx.out()) {
|
-> decltype(ctx.out()) {
|
||||||
if (!opt) return detail::write<Char>(ctx.out(), none);
|
if (!opt) return detail::write<Char>(ctx.out(), none);
|
||||||
|
|
||||||
|
|
@ -191,58 +405,65 @@ struct formatter<std::optional<T>, Char,
|
||||||
return detail::write(out, ')');
|
return detail::write(out, ')');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
FMT_END_NAMESPACE
|
|
||||||
#endif // __cpp_lib_optional
|
#endif // __cpp_lib_optional
|
||||||
|
|
||||||
#ifdef __cpp_lib_variant
|
#ifdef __cpp_lib_expected
|
||||||
FMT_BEGIN_NAMESPACE
|
template <typename T, typename E, typename Char>
|
||||||
namespace detail {
|
struct formatter<std::expected<T, E>, Char,
|
||||||
|
std::enable_if_t<(std::is_void<T>::value ||
|
||||||
|
is_formattable<T, Char>::value) &&
|
||||||
|
is_formattable<E, Char>::value>> {
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
return ctx.begin();
|
||||||
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename FormatContext>
|
||||||
using variant_index_sequence =
|
auto format(const std::expected<T, E>& value, FormatContext& ctx) const
|
||||||
std::make_index_sequence<std::variant_size<T>::value>;
|
-> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
|
||||||
template <typename> struct is_variant_like_ : std::false_type {};
|
if (value.has_value()) {
|
||||||
template <typename... Types>
|
out = detail::write<Char>(out, "expected(");
|
||||||
struct is_variant_like_<std::variant<Types...>> : std::true_type {};
|
if constexpr (!std::is_void<T>::value)
|
||||||
|
out = detail::write_escaped_alternative<Char>(out, *value, ctx);
|
||||||
// formattable element check.
|
} else {
|
||||||
template <typename T, typename C> class is_variant_formattable_ {
|
out = detail::write<Char>(out, "unexpected(");
|
||||||
template <std::size_t... Is>
|
out = detail::write_escaped_alternative<Char>(out, value.error(), ctx);
|
||||||
static std::conjunction<
|
}
|
||||||
is_formattable<std::variant_alternative_t<Is, T>, C>...>
|
*out++ = ')';
|
||||||
check(std::index_sequence<Is...>);
|
return out;
|
||||||
|
}
|
||||||
public:
|
|
||||||
static constexpr const bool value =
|
|
||||||
decltype(check(variant_index_sequence<T>{}))::value;
|
|
||||||
};
|
};
|
||||||
|
#endif // __cpp_lib_expected
|
||||||
|
|
||||||
template <typename Char, typename OutputIt, typename T>
|
#ifdef __cpp_lib_source_location
|
||||||
auto write_variant_alternative(OutputIt out, const T& v) -> OutputIt {
|
template <> struct formatter<std::source_location> {
|
||||||
if constexpr (is_string<T>::value)
|
FMT_CONSTEXPR auto parse(parse_context<>& ctx) { return ctx.begin(); }
|
||||||
return write_escaped_string<Char>(out, detail::to_string_view(v));
|
|
||||||
else if constexpr (std::is_same_v<T, Char>)
|
|
||||||
return write_escaped_char(out, v);
|
|
||||||
else
|
|
||||||
return write<Char>(out, v);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace detail
|
template <typename FormatContext>
|
||||||
|
auto format(const std::source_location& loc, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
out = detail::write(out, loc.file_name());
|
||||||
|
out = detail::write(out, ':');
|
||||||
|
out = detail::write<char>(out, loc.line());
|
||||||
|
out = detail::write(out, ':');
|
||||||
|
out = detail::write<char>(out, loc.column());
|
||||||
|
out = detail::write(out, ": ");
|
||||||
|
out = detail::write(out, loc.function_name());
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if FMT_CPP_LIB_VARIANT
|
||||||
|
|
||||||
template <typename T> struct is_variant_like {
|
template <typename T> struct is_variant_like {
|
||||||
static constexpr const bool value = detail::is_variant_like_<T>::value;
|
static constexpr bool value = detail::is_variant_like_<T>::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename C> struct is_variant_formattable {
|
|
||||||
static constexpr const bool value =
|
|
||||||
detail::is_variant_formattable_<T, C>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename Char> struct formatter<std::monostate, Char> {
|
template <typename Char> struct formatter<std::monostate, Char> {
|
||||||
template <typename ParseContext>
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
return ctx.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -253,14 +474,12 @@ template <typename Char> struct formatter<std::monostate, Char> {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename Variant, typename Char>
|
template <typename Variant, typename Char>
|
||||||
struct formatter<
|
struct formatter<Variant, Char,
|
||||||
Variant, Char,
|
std::enable_if_t<std::conjunction_v<
|
||||||
std::enable_if_t<std::conjunction_v<
|
is_variant_like<Variant>,
|
||||||
is_variant_like<Variant>, is_variant_formattable<Variant, Char>>>> {
|
detail::is_variant_formattable<Variant, Char>>>> {
|
||||||
template <typename ParseContext>
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
return ctx.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -273,7 +492,7 @@ struct formatter<
|
||||||
FMT_TRY {
|
FMT_TRY {
|
||||||
std::visit(
|
std::visit(
|
||||||
[&](const auto& v) {
|
[&](const auto& v) {
|
||||||
out = detail::write_variant_alternative<Char>(out, v);
|
out = detail::write_escaped_alternative<Char>(out, v, ctx);
|
||||||
},
|
},
|
||||||
value);
|
value);
|
||||||
}
|
}
|
||||||
|
|
@ -284,147 +503,115 @@ struct formatter<
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
FMT_END_NAMESPACE
|
|
||||||
#endif // __cpp_lib_variant
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
#endif // FMT_CPP_LIB_VARIANT
|
||||||
FMT_EXPORT
|
|
||||||
template <typename Char> struct formatter<std::error_code, Char> {
|
template <> struct formatter<std::error_code> {
|
||||||
template <typename ParseContext>
|
private:
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
format_specs specs_;
|
||||||
return ctx.begin();
|
detail::arg_ref<char> width_ref_;
|
||||||
|
bool debug_ = false;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR void set_debug_format(bool set = true) { debug_ = set; }
|
||||||
|
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<>& ctx) -> const char* {
|
||||||
|
auto it = ctx.begin(), end = ctx.end();
|
||||||
|
if (it == end) return it;
|
||||||
|
|
||||||
|
it = detail::parse_align(it, end, specs_);
|
||||||
|
|
||||||
|
char c = *it;
|
||||||
|
if (it != end && ((c >= '0' && c <= '9') || c == '{'))
|
||||||
|
it = detail::parse_width(it, end, specs_, width_ref_, ctx);
|
||||||
|
|
||||||
|
if (it != end && *it == '?') {
|
||||||
|
debug_ = true;
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
if (it != end && *it == 's') {
|
||||||
|
specs_.set_type(presentation_type::string);
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
FMT_CONSTEXPR auto format(const std::error_code& ec, FormatContext& ctx) const
|
FMT_CONSTEXPR20 auto format(const std::error_code& ec,
|
||||||
-> decltype(ctx.out()) {
|
FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
auto out = ctx.out();
|
auto specs = specs_;
|
||||||
out = detail::write_bytes(out, ec.category().name(), format_specs<Char>());
|
detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
|
||||||
out = detail::write<Char>(out, Char(':'));
|
ctx);
|
||||||
out = detail::write<Char>(out, ec.value());
|
auto buf = memory_buffer();
|
||||||
return out;
|
if (specs_.type() == presentation_type::string) {
|
||||||
|
buf.append(ec.message());
|
||||||
|
} else {
|
||||||
|
buf.append(string_view(ec.category().name()));
|
||||||
|
buf.push_back(':');
|
||||||
|
detail::write<char>(appender(buf), ec.value());
|
||||||
|
}
|
||||||
|
auto quoted = memory_buffer();
|
||||||
|
auto str = string_view(buf.data(), buf.size());
|
||||||
|
if (debug_) {
|
||||||
|
detail::write_escaped_string<char>(std::back_inserter(quoted), str);
|
||||||
|
str = string_view(quoted.data(), quoted.size());
|
||||||
|
}
|
||||||
|
return detail::write<char>(ctx.out(), str, specs);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_EXPORT
|
#if FMT_USE_RTTI
|
||||||
template <typename T, typename Char>
|
template <> struct formatter<std::type_info> {
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<>& ctx) -> const char* {
|
||||||
|
return ctx.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Context>
|
||||||
|
auto format(const std::type_info& ti, Context& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
return detail::write_demangled_name(ctx.out(), ti);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif // FMT_USE_RTTI
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
struct formatter<
|
struct formatter<
|
||||||
T, Char,
|
T, char,
|
||||||
typename std::enable_if<std::is_base_of<std::exception, T>::value>::type> {
|
typename std::enable_if<std::is_base_of<std::exception, T>::value>::type> {
|
||||||
private:
|
private:
|
||||||
bool with_typename_ = false;
|
bool with_typename_ = false;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR auto parse(basic_format_parse_context<Char>& ctx)
|
FMT_CONSTEXPR auto parse(parse_context<>& ctx) -> const char* {
|
||||||
-> decltype(ctx.begin()) {
|
|
||||||
auto it = ctx.begin();
|
auto it = ctx.begin();
|
||||||
auto end = ctx.end();
|
auto end = ctx.end();
|
||||||
if (it == end || *it == '}') return it;
|
if (it == end || *it == '}') return it;
|
||||||
if (*it == 't') {
|
if (*it == 't') {
|
||||||
++it;
|
++it;
|
||||||
with_typename_ = FMT_USE_TYPEID != 0;
|
with_typename_ = FMT_USE_RTTI != 0;
|
||||||
}
|
}
|
||||||
return it;
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt>
|
template <typename Context>
|
||||||
auto format(const std::exception& ex,
|
auto format(const std::exception& ex, Context& ctx) const
|
||||||
basic_format_context<OutputIt, Char>& ctx) const -> OutputIt {
|
-> decltype(ctx.out()) {
|
||||||
format_specs<Char> spec;
|
|
||||||
auto out = ctx.out();
|
auto out = ctx.out();
|
||||||
if (!with_typename_)
|
#if FMT_USE_RTTI
|
||||||
return detail::write_bytes(out, string_view(ex.what()), spec);
|
if (with_typename_) {
|
||||||
|
out = detail::write_demangled_name(out, typeid(ex));
|
||||||
#if FMT_USE_TYPEID
|
*out++ = ':';
|
||||||
const std::type_info& ti = typeid(ex);
|
*out++ = ' ';
|
||||||
# ifdef FMT_HAS_ABI_CXA_DEMANGLE
|
|
||||||
int status = 0;
|
|
||||||
std::size_t size = 0;
|
|
||||||
std::unique_ptr<char, decltype(&std::free)> demangled_name_ptr(
|
|
||||||
abi::__cxa_demangle(ti.name(), nullptr, &size, &status), &std::free);
|
|
||||||
|
|
||||||
string_view demangled_name_view;
|
|
||||||
if (demangled_name_ptr) {
|
|
||||||
demangled_name_view = demangled_name_ptr.get();
|
|
||||||
|
|
||||||
// Normalization of stdlib inline namespace names.
|
|
||||||
// libc++ inline namespaces.
|
|
||||||
// std::__1::* -> std::*
|
|
||||||
// std::__1::__fs::* -> std::*
|
|
||||||
// libstdc++ inline namespaces.
|
|
||||||
// std::__cxx11::* -> std::*
|
|
||||||
// std::filesystem::__cxx11::* -> std::filesystem::*
|
|
||||||
if (demangled_name_view.starts_with("std::")) {
|
|
||||||
char* begin = demangled_name_ptr.get();
|
|
||||||
char* to = begin + 5; // std::
|
|
||||||
for (char *from = to, *end = begin + demangled_name_view.size();
|
|
||||||
from < end;) {
|
|
||||||
// This is safe, because demangled_name is NUL-terminated.
|
|
||||||
if (from[0] == '_' && from[1] == '_') {
|
|
||||||
char* next = from + 1;
|
|
||||||
while (next < end && *next != ':') next++;
|
|
||||||
if (next[0] == ':' && next[1] == ':') {
|
|
||||||
from = next + 2;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*to++ = *from++;
|
|
||||||
}
|
|
||||||
demangled_name_view = {begin, detail::to_unsigned(to - begin)};
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
demangled_name_view = string_view(ti.name());
|
|
||||||
}
|
}
|
||||||
out = detail::write_bytes(out, demangled_name_view, spec);
|
|
||||||
# elif FMT_MSC_VERSION
|
|
||||||
string_view demangled_name_view(ti.name());
|
|
||||||
if (demangled_name_view.starts_with("class "))
|
|
||||||
demangled_name_view.remove_prefix(6);
|
|
||||||
else if (demangled_name_view.starts_with("struct "))
|
|
||||||
demangled_name_view.remove_prefix(7);
|
|
||||||
out = detail::write_bytes(out, demangled_name_view, spec);
|
|
||||||
# else
|
|
||||||
out = detail::write_bytes(out, string_view(ti.name()), spec);
|
|
||||||
# endif
|
|
||||||
*out++ = ':';
|
|
||||||
*out++ = ' ';
|
|
||||||
return detail::write_bytes(out, string_view(ex.what()), spec);
|
|
||||||
#endif
|
#endif
|
||||||
|
return detail::write_bytes<char>(out, string_view(ex.what()));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace detail {
|
|
||||||
|
|
||||||
template <typename T, typename Enable = void>
|
|
||||||
struct has_flip : std::false_type {};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct has_flip<T, void_t<decltype(std::declval<T>().flip())>>
|
|
||||||
: std::true_type {};
|
|
||||||
|
|
||||||
template <typename T> struct is_bit_reference_like {
|
|
||||||
static constexpr const bool value =
|
|
||||||
std::is_convertible<T, bool>::value &&
|
|
||||||
std::is_nothrow_assignable<T, bool>::value && has_flip<T>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
#ifdef _LIBCPP_VERSION
|
|
||||||
|
|
||||||
// Workaround for libc++ incompatibility with C++ standard.
|
|
||||||
// According to the Standard, `bitset::operator[] const` returns bool.
|
|
||||||
template <typename C>
|
|
||||||
struct is_bit_reference_like<std::__bit_const_reference<C>> {
|
|
||||||
static constexpr const bool value = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
// We can't use std::vector<bool, Allocator>::reference and
|
// We can't use std::vector<bool, Allocator>::reference and
|
||||||
// std::bitset<N>::reference because the compiler can't deduce Allocator and N
|
// std::bitset<N>::reference because the compiler can't deduce Allocator and N
|
||||||
// in partial specialization.
|
// in partial specialization.
|
||||||
FMT_EXPORT
|
|
||||||
template <typename BitRef, typename Char>
|
template <typename BitRef, typename Char>
|
||||||
struct formatter<BitRef, Char,
|
struct formatter<BitRef, Char,
|
||||||
enable_if_t<detail::is_bit_reference_like<BitRef>::value>>
|
enable_if_t<detail::is_bit_reference_like<BitRef>::value>>
|
||||||
|
|
@ -436,7 +623,6 @@ struct formatter<BitRef, Char,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
struct formatter<std::atomic<T>, Char,
|
struct formatter<std::atomic<T>, Char,
|
||||||
enable_if_t<is_formattable<T, Char>::value>>
|
enable_if_t<is_formattable<T, Char>::value>>
|
||||||
|
|
@ -449,17 +635,93 @@ struct formatter<std::atomic<T>, Char,
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef __cpp_lib_atomic_flag_test
|
#ifdef __cpp_lib_atomic_flag_test
|
||||||
FMT_EXPORT
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
struct formatter<std::atomic_flag, Char>
|
struct formatter<std::atomic_flag, Char> : formatter<bool, Char> {
|
||||||
: formatter<bool, Char> {
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const std::atomic_flag& v, FormatContext& ctx) const
|
auto format(const std::atomic_flag& v, FormatContext& ctx) const
|
||||||
-> decltype(ctx.out()) {
|
-> decltype(ctx.out()) {
|
||||||
return formatter<bool, Char>::format(v.test(), ctx);
|
return formatter<bool, Char>::format(v.test(), ctx);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
#endif // __cpp_lib_atomic_flag_test
|
#endif // __cpp_lib_atomic_flag_test
|
||||||
|
|
||||||
|
template <typename T, typename Char> struct formatter<std::complex<T>, Char> {
|
||||||
|
private:
|
||||||
|
detail::dynamic_format_specs<Char> specs_;
|
||||||
|
|
||||||
|
template <typename FormatContext, typename OutputIt>
|
||||||
|
FMT_CONSTEXPR auto do_format(const std::complex<T>& c,
|
||||||
|
detail::dynamic_format_specs<Char>& specs,
|
||||||
|
FormatContext& ctx, OutputIt out) const
|
||||||
|
-> OutputIt {
|
||||||
|
if (c.real() != 0) {
|
||||||
|
*out++ = Char('(');
|
||||||
|
out = detail::write<Char>(out, c.real(), specs, ctx.locale());
|
||||||
|
specs.set_sign(sign::plus);
|
||||||
|
out = detail::write<Char>(out, c.imag(), specs, ctx.locale());
|
||||||
|
if (!detail::isfinite(c.imag())) *out++ = Char(' ');
|
||||||
|
*out++ = Char('i');
|
||||||
|
*out++ = Char(')');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
out = detail::write<Char>(out, c.imag(), specs, ctx.locale());
|
||||||
|
if (!detail::isfinite(c.imag())) *out++ = Char(' ');
|
||||||
|
*out++ = Char('i');
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
if (ctx.begin() == ctx.end() || *ctx.begin() == '}') return ctx.begin();
|
||||||
|
return parse_format_specs(ctx.begin(), ctx.end(), specs_, ctx,
|
||||||
|
detail::type_constant<T, Char>::value);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(const std::complex<T>& c, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
auto specs = specs_;
|
||||||
|
if (specs.dynamic()) {
|
||||||
|
detail::handle_dynamic_spec(specs.dynamic_width(), specs.width,
|
||||||
|
specs.width_ref, ctx);
|
||||||
|
detail::handle_dynamic_spec(specs.dynamic_precision(), specs.precision,
|
||||||
|
specs.precision_ref, ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (specs.width == 0) return do_format(c, specs, ctx, ctx.out());
|
||||||
|
auto buf = basic_memory_buffer<Char>();
|
||||||
|
|
||||||
|
auto outer_specs = format_specs();
|
||||||
|
outer_specs.width = specs.width;
|
||||||
|
outer_specs.copy_fill_from(specs);
|
||||||
|
outer_specs.set_align(specs.align());
|
||||||
|
|
||||||
|
specs.width = 0;
|
||||||
|
specs.set_fill({});
|
||||||
|
specs.set_align(align::none);
|
||||||
|
|
||||||
|
do_format(c, specs, ctx, basic_appender<Char>(buf));
|
||||||
|
return detail::write<Char>(ctx.out(),
|
||||||
|
basic_string_view<Char>(buf.data(), buf.size()),
|
||||||
|
outer_specs);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T, typename Char>
|
||||||
|
struct formatter<std::reference_wrapper<T>, Char,
|
||||||
|
// Guard against format_as because reference_wrapper is
|
||||||
|
// implicitly convertible to T&.
|
||||||
|
enable_if_t<is_formattable<remove_cvref_t<T>, Char>::value &&
|
||||||
|
!detail::has_format_as<T>::value &&
|
||||||
|
!detail::has_format_as_member<T>::value>>
|
||||||
|
: formatter<remove_cvref_t<T>, Char> {
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(std::reference_wrapper<T> ref, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
return formatter<remove_cvref_t<T>, Char>::format(ref.get(), ctx);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
||||||
#endif // FMT_STD_H_
|
#endif // FMT_STD_H_
|
||||||
|
|
|
||||||
282
deps/fmt/include/fmt/xchar.h
vendored
282
deps/fmt/include/fmt/xchar.h
vendored
|
|
@ -8,12 +8,16 @@
|
||||||
#ifndef FMT_XCHAR_H_
|
#ifndef FMT_XCHAR_H_
|
||||||
#define FMT_XCHAR_H_
|
#define FMT_XCHAR_H_
|
||||||
|
|
||||||
#include <cwchar>
|
#include "color.h"
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
#include "ostream.h"
|
||||||
|
#include "ranges.h"
|
||||||
|
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
#ifndef FMT_MODULE
|
||||||
# include <locale>
|
# include <cwchar>
|
||||||
|
# if FMT_USE_LOCALE
|
||||||
|
# include <locale>
|
||||||
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
@ -22,10 +26,26 @@ namespace detail {
|
||||||
template <typename T>
|
template <typename T>
|
||||||
using is_exotic_char = bool_constant<!std::is_same<T, char>::value>;
|
using is_exotic_char = bool_constant<!std::is_same<T, char>::value>;
|
||||||
|
|
||||||
inline auto write_loc(std::back_insert_iterator<detail::buffer<wchar_t>> out,
|
template <typename S, typename = void> struct format_string_char {};
|
||||||
loc_value value, const format_specs<wchar_t>& specs,
|
|
||||||
locale_ref loc) -> bool {
|
template <typename S>
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
struct format_string_char<
|
||||||
|
S, void_t<decltype(sizeof(detail::to_string_view(std::declval<S>())))>> {
|
||||||
|
using type = char_t<S>;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename S>
|
||||||
|
struct format_string_char<
|
||||||
|
S, enable_if_t<std::is_base_of<detail::compile_string, S>::value>> {
|
||||||
|
using type = typename S::char_type;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename S>
|
||||||
|
using format_string_char_t = typename format_string_char<S>::type;
|
||||||
|
|
||||||
|
inline auto write_loc(basic_appender<wchar_t> out, loc_value value,
|
||||||
|
const format_specs& specs, locale_ref loc) -> bool {
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
auto& numpunct =
|
auto& numpunct =
|
||||||
std::use_facet<std::numpunct<wchar_t>>(loc.get<std::locale>());
|
std::use_facet<std::numpunct<wchar_t>>(loc.get<std::locale>());
|
||||||
auto separator = std::wstring();
|
auto separator = std::wstring();
|
||||||
|
|
@ -35,46 +55,86 @@ inline auto write_loc(std::back_insert_iterator<detail::buffer<wchar_t>> out,
|
||||||
#endif
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Char>
|
||||||
|
void vformat_to(buffer<Char>& buf, basic_string_view<Char> fmt,
|
||||||
|
basic_format_args<buffered_context<Char>> args,
|
||||||
|
locale_ref loc = {}) {
|
||||||
|
static_assert(!std::is_same<Char, char>::value, "");
|
||||||
|
auto out = basic_appender<Char>(buf);
|
||||||
|
parse_format_string(
|
||||||
|
fmt, format_handler<Char>{parse_context<Char>(fmt), {out, args, loc}});
|
||||||
|
}
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
FMT_BEGIN_EXPORT
|
FMT_BEGIN_EXPORT
|
||||||
|
|
||||||
using wstring_view = basic_string_view<wchar_t>;
|
using wstring_view = basic_string_view<wchar_t>;
|
||||||
using wformat_parse_context = basic_format_parse_context<wchar_t>;
|
using wformat_parse_context = parse_context<wchar_t>;
|
||||||
using wformat_context = buffer_context<wchar_t>;
|
using wformat_context = buffered_context<wchar_t>;
|
||||||
using wformat_args = basic_format_args<wformat_context>;
|
using wformat_args = basic_format_args<wformat_context>;
|
||||||
using wmemory_buffer = basic_memory_buffer<wchar_t>;
|
using wmemory_buffer = basic_memory_buffer<wchar_t>;
|
||||||
|
|
||||||
#if FMT_GCC_VERSION && FMT_GCC_VERSION < 409
|
template <typename Char, typename... T> struct basic_fstring {
|
||||||
// Workaround broken conversion on older gcc.
|
private:
|
||||||
template <typename... Args> using wformat_string = wstring_view;
|
basic_string_view<Char> str_;
|
||||||
inline auto runtime(wstring_view s) -> wstring_view { return s; }
|
|
||||||
#else
|
static constexpr int num_static_named_args =
|
||||||
template <typename... Args>
|
detail::count_static_named_args<T...>();
|
||||||
using wformat_string = basic_format_string<wchar_t, type_identity_t<Args>...>;
|
|
||||||
|
using checker = detail::format_string_checker<
|
||||||
|
Char, static_cast<int>(sizeof...(T)), num_static_named_args,
|
||||||
|
num_static_named_args != detail::count_named_args<T...>()>;
|
||||||
|
|
||||||
|
using arg_pack = detail::arg_pack<T...>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using t = basic_fstring;
|
||||||
|
|
||||||
|
template <typename S,
|
||||||
|
FMT_ENABLE_IF(
|
||||||
|
std::is_convertible<const S&, basic_string_view<Char>>::value)>
|
||||||
|
FMT_CONSTEVAL FMT_ALWAYS_INLINE basic_fstring(const S& s) : str_(s) {
|
||||||
|
if (FMT_USE_CONSTEVAL)
|
||||||
|
detail::parse_format_string<Char>(s, checker(s, arg_pack()));
|
||||||
|
}
|
||||||
|
template <typename S,
|
||||||
|
FMT_ENABLE_IF(std::is_base_of<detail::compile_string, S>::value&&
|
||||||
|
std::is_same<typename S::char_type, Char>::value)>
|
||||||
|
FMT_ALWAYS_INLINE basic_fstring(const S&) : str_(S()) {
|
||||||
|
FMT_CONSTEXPR auto sv = basic_string_view<Char>(S());
|
||||||
|
FMT_CONSTEXPR int ignore =
|
||||||
|
(parse_format_string(sv, checker(sv, arg_pack())), 0);
|
||||||
|
detail::ignore_unused(ignore);
|
||||||
|
}
|
||||||
|
basic_fstring(runtime_format_string<Char> fmt) : str_(fmt.str) {}
|
||||||
|
|
||||||
|
operator basic_string_view<Char>() const { return str_; }
|
||||||
|
auto get() const -> basic_string_view<Char> { return str_; }
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename Char, typename... T>
|
||||||
|
using basic_format_string = basic_fstring<Char, T...>;
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
using wformat_string = typename basic_format_string<wchar_t, T...>::t;
|
||||||
inline auto runtime(wstring_view s) -> runtime_format_string<wchar_t> {
|
inline auto runtime(wstring_view s) -> runtime_format_string<wchar_t> {
|
||||||
return {{s}};
|
return {{s}};
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
template <> struct is_char<wchar_t> : std::true_type {};
|
|
||||||
template <> struct is_char<detail::char8_type> : std::true_type {};
|
|
||||||
template <> struct is_char<char16_t> : std::true_type {};
|
|
||||||
template <> struct is_char<char32_t> : std::true_type {};
|
|
||||||
|
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
constexpr format_arg_store<wformat_context, T...> make_wformat_args(
|
constexpr auto make_wformat_args(T&... args)
|
||||||
const T&... args) {
|
-> decltype(fmt::make_format_args<wformat_context>(args...)) {
|
||||||
return {args...};
|
return fmt::make_format_args<wformat_context>(args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if !FMT_USE_NONTYPE_TEMPLATE_ARGS
|
||||||
inline namespace literals {
|
inline namespace literals {
|
||||||
#if FMT_USE_USER_DEFINED_LITERALS && !FMT_USE_NONTYPE_TEMPLATE_ARGS
|
inline auto operator""_a(const wchar_t* s, size_t) -> detail::udl_arg<wchar_t> {
|
||||||
constexpr detail::udl_arg<wchar_t> operator"" _a(const wchar_t* s, size_t) {
|
|
||||||
return {s};
|
return {s};
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
} // namespace literals
|
} // namespace literals
|
||||||
|
#endif
|
||||||
|
|
||||||
template <typename It, typename Sentinel>
|
template <typename It, typename Sentinel>
|
||||||
auto join(It begin, Sentinel end, wstring_view sep)
|
auto join(It begin, Sentinel end, wstring_view sep)
|
||||||
|
|
@ -82,9 +142,9 @@ auto join(It begin, Sentinel end, wstring_view sep)
|
||||||
return {begin, end, sep};
|
return {begin, end, sep};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Range>
|
template <typename Range, FMT_ENABLE_IF(!is_tuple_like<Range>::value)>
|
||||||
auto join(Range&& range, wstring_view sep)
|
auto join(Range&& range, wstring_view sep)
|
||||||
-> join_view<detail::iterator_t<Range>, detail::sentinel_t<Range>,
|
-> join_view<decltype(std::begin(range)), decltype(std::end(range)),
|
||||||
wchar_t> {
|
wchar_t> {
|
||||||
return join(std::begin(range), std::end(range), sep);
|
return join(std::begin(range), std::end(range), sep);
|
||||||
}
|
}
|
||||||
|
|
@ -95,13 +155,19 @@ auto join(std::initializer_list<T> list, wstring_view sep)
|
||||||
return join(std::begin(list), std::end(list), sep);
|
return join(std::begin(list), std::end(list), sep);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Tuple, FMT_ENABLE_IF(is_tuple_like<Tuple>::value)>
|
||||||
|
auto join(const Tuple& tuple, basic_string_view<wchar_t> sep)
|
||||||
|
-> tuple_join_view<Tuple, wchar_t> {
|
||||||
|
return {tuple, sep};
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Char, FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
|
template <typename Char, FMT_ENABLE_IF(!std::is_same<Char, char>::value)>
|
||||||
auto vformat(basic_string_view<Char> format_str,
|
auto vformat(basic_string_view<Char> fmt,
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
basic_format_args<buffered_context<Char>> args)
|
||||||
-> std::basic_string<Char> {
|
-> std::basic_string<Char> {
|
||||||
auto buf = basic_memory_buffer<Char>();
|
auto buf = basic_memory_buffer<Char>();
|
||||||
detail::vformat_to(buf, format_str, args);
|
detail::vformat_to(buf, fmt, args);
|
||||||
return to_string(buf);
|
return {buf.data(), buf.size()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
|
|
@ -109,110 +175,115 @@ auto format(wformat_string<T...> fmt, T&&... args) -> std::wstring {
|
||||||
return vformat(fmt::wstring_view(fmt), fmt::make_wformat_args(args...));
|
return vformat(fmt::wstring_view(fmt), fmt::make_wformat_args(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename OutputIt, typename... T>
|
||||||
|
auto format_to(OutputIt out, wformat_string<T...> fmt, T&&... args)
|
||||||
|
-> OutputIt {
|
||||||
|
return vformat_to(out, fmt::wstring_view(fmt),
|
||||||
|
fmt::make_wformat_args(args...));
|
||||||
|
}
|
||||||
|
|
||||||
// Pass char_t as a default template parameter instead of using
|
// Pass char_t as a default template parameter instead of using
|
||||||
// std::basic_string<char_t<S>> to reduce the symbol size.
|
// std::basic_string<char_t<S>> to reduce the symbol size.
|
||||||
template <typename S, typename... T, typename Char = char_t<S>,
|
template <typename S, typename... T,
|
||||||
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(!std::is_same<Char, char>::value &&
|
FMT_ENABLE_IF(!std::is_same<Char, char>::value &&
|
||||||
!std::is_same<Char, wchar_t>::value)>
|
!std::is_same<Char, wchar_t>::value)>
|
||||||
auto format(const S& format_str, T&&... args) -> std::basic_string<Char> {
|
auto format(const S& fmt, T&&... args) -> std::basic_string<Char> {
|
||||||
return vformat(detail::to_string_view(format_str),
|
return vformat(detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Locale, typename S, typename Char = char_t<S>,
|
template <typename S, typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
FMT_ENABLE_IF(detail::is_exotic_char<Char>::value)>
|
||||||
detail::is_exotic_char<Char>::value)>
|
inline auto vformat(locale_ref loc, const S& fmt,
|
||||||
inline auto vformat(
|
basic_format_args<buffered_context<Char>> args)
|
||||||
const Locale& loc, const S& format_str,
|
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
|
||||||
-> std::basic_string<Char> {
|
-> std::basic_string<Char> {
|
||||||
return detail::vformat(loc, detail::to_string_view(format_str), args);
|
auto buf = basic_memory_buffer<Char>();
|
||||||
|
detail::vformat_to(buf, detail::to_string_view(fmt), args, loc);
|
||||||
|
return {buf.data(), buf.size()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Locale, typename S, typename... T, typename Char = char_t<S>,
|
template <typename S, typename... T,
|
||||||
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
typename Char = detail::format_string_char_t<S>,
|
||||||
detail::is_exotic_char<Char>::value)>
|
FMT_ENABLE_IF(detail::is_exotic_char<Char>::value)>
|
||||||
inline auto format(const Locale& loc, const S& format_str, T&&... args)
|
inline auto format(locale_ref loc, const S& fmt, T&&... args)
|
||||||
-> std::basic_string<Char> {
|
-> std::basic_string<Char> {
|
||||||
return detail::vformat(loc, detail::to_string_view(format_str),
|
return vformat(loc, detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename Char = char_t<S>,
|
template <typename OutputIt, typename S,
|
||||||
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
auto vformat_to(OutputIt out, const S& format_str,
|
auto vformat_to(OutputIt out, const S& fmt,
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
basic_format_args<buffered_context<Char>> args) -> OutputIt {
|
||||||
-> OutputIt {
|
|
||||||
auto&& buf = detail::get_buffer<Char>(out);
|
auto&& buf = detail::get_buffer<Char>(out);
|
||||||
detail::vformat_to(buf, detail::to_string_view(format_str), args);
|
detail::vformat_to(buf, detail::to_string_view(fmt), args);
|
||||||
return detail::get_iterator(buf, out);
|
return detail::get_iterator(buf, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... T,
|
template <typename OutputIt, typename S, typename... T,
|
||||||
typename Char = char_t<S>,
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value &&
|
||||||
detail::is_exotic_char<Char>::value)>
|
!std::is_same<Char, char>::value &&
|
||||||
|
!std::is_same<Char, wchar_t>::value)>
|
||||||
inline auto format_to(OutputIt out, const S& fmt, T&&... args) -> OutputIt {
|
inline auto format_to(OutputIt out, const S& fmt, T&&... args) -> OutputIt {
|
||||||
return vformat_to(out, detail::to_string_view(fmt),
|
return vformat_to(out, detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Locale, typename S, typename OutputIt, typename... Args,
|
template <typename S, typename OutputIt, typename... Args,
|
||||||
typename Char = char_t<S>,
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
||||||
detail::is_locale<Locale>::value&&
|
detail::is_exotic_char<Char>::value)>
|
||||||
detail::is_exotic_char<Char>::value)>
|
inline auto vformat_to(OutputIt out, locale_ref loc, const S& fmt,
|
||||||
inline auto vformat_to(
|
basic_format_args<buffered_context<Char>> args)
|
||||||
OutputIt out, const Locale& loc, const S& format_str,
|
-> OutputIt {
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) -> OutputIt {
|
|
||||||
auto&& buf = detail::get_buffer<Char>(out);
|
auto&& buf = detail::get_buffer<Char>(out);
|
||||||
vformat_to(buf, detail::to_string_view(format_str), args,
|
vformat_to(buf, detail::to_string_view(fmt), args, loc);
|
||||||
detail::locale_ref(loc));
|
|
||||||
return detail::get_iterator(buf, out);
|
return detail::get_iterator(buf, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <
|
template <typename OutputIt, typename S, typename... T,
|
||||||
typename OutputIt, typename Locale, typename S, typename... T,
|
typename Char = detail::format_string_char_t<S>,
|
||||||
typename Char = char_t<S>,
|
bool enable = detail::is_output_iterator<OutputIt, Char>::value &&
|
||||||
bool enable = detail::is_output_iterator<OutputIt, Char>::value&&
|
detail::is_exotic_char<Char>::value>
|
||||||
detail::is_locale<Locale>::value&& detail::is_exotic_char<Char>::value>
|
inline auto format_to(OutputIt out, locale_ref loc, const S& fmt, T&&... args)
|
||||||
inline auto format_to(OutputIt out, const Locale& loc, const S& format_str,
|
-> typename std::enable_if<enable, OutputIt>::type {
|
||||||
T&&... args) ->
|
return vformat_to(out, loc, detail::to_string_view(fmt),
|
||||||
typename std::enable_if<enable, OutputIt>::type {
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
return vformat_to(out, loc, detail::to_string_view(format_str),
|
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename Char, typename... Args,
|
template <typename OutputIt, typename Char, typename... Args,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
inline auto vformat_to_n(
|
inline auto vformat_to_n(OutputIt out, size_t n, basic_string_view<Char> fmt,
|
||||||
OutputIt out, size_t n, basic_string_view<Char> format_str,
|
basic_format_args<buffered_context<Char>> args)
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
|
||||||
-> format_to_n_result<OutputIt> {
|
-> format_to_n_result<OutputIt> {
|
||||||
using traits = detail::fixed_buffer_traits;
|
using traits = detail::fixed_buffer_traits;
|
||||||
auto buf = detail::iterator_buffer<OutputIt, Char, traits>(out, n);
|
auto buf = detail::iterator_buffer<OutputIt, Char, traits>(out, n);
|
||||||
detail::vformat_to(buf, format_str, args);
|
detail::vformat_to(buf, fmt, args);
|
||||||
return {buf.out(), buf.count()};
|
return {buf.out(), buf.count()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... T,
|
template <typename OutputIt, typename S, typename... T,
|
||||||
typename Char = char_t<S>,
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
inline auto format_to_n(OutputIt out, size_t n, const S& fmt, T&&... args)
|
inline auto format_to_n(OutputIt out, size_t n, const S& fmt, T&&... args)
|
||||||
-> format_to_n_result<OutputIt> {
|
-> format_to_n_result<OutputIt> {
|
||||||
return vformat_to_n(out, n, detail::to_string_view(fmt),
|
return vformat_to_n(out, n, fmt::basic_string_view<Char>(fmt),
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... T, typename Char = char_t<S>,
|
template <typename S, typename... T,
|
||||||
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_exotic_char<Char>::value)>
|
FMT_ENABLE_IF(detail::is_exotic_char<Char>::value)>
|
||||||
inline auto formatted_size(const S& fmt, T&&... args) -> size_t {
|
inline auto formatted_size(const S& fmt, T&&... args) -> size_t {
|
||||||
auto buf = detail::counting_buffer<Char>();
|
auto buf = detail::counting_buffer<Char>();
|
||||||
detail::vformat_to(buf, detail::to_string_view(fmt),
|
detail::vformat_to(buf, detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<buffer_context<Char>>(args...));
|
fmt::make_format_args<buffered_context<Char>>(args...));
|
||||||
return buf.count();
|
return buf.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -246,9 +317,36 @@ template <typename... T> void println(wformat_string<T...> fmt, T&&... args) {
|
||||||
return print(L"{}\n", fmt::format(fmt, std::forward<T>(args)...));
|
return print(L"{}\n", fmt::format(fmt, std::forward<T>(args)...));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
inline auto vformat(text_style ts, wstring_view fmt, wformat_args args)
|
||||||
Converts *value* to ``std::wstring`` using the default format for type *T*.
|
-> std::wstring {
|
||||||
*/
|
auto buf = wmemory_buffer();
|
||||||
|
detail::vformat_to(buf, ts, fmt, args);
|
||||||
|
return {buf.data(), buf.size()};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
inline auto format(text_style ts, wformat_string<T...> fmt, T&&... args)
|
||||||
|
-> std::wstring {
|
||||||
|
return fmt::vformat(ts, fmt, fmt::make_wformat_args(args...));
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void vprint(std::wostream& os, wstring_view fmt, wformat_args args) {
|
||||||
|
auto buffer = basic_memory_buffer<wchar_t>();
|
||||||
|
detail::vformat_to(buffer, fmt, args);
|
||||||
|
detail::write_buffer(os, buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
void print(std::wostream& os, wformat_string<T...> fmt, T&&... args) {
|
||||||
|
vprint(os, fmt, fmt::make_format_args<buffered_context<wchar_t>>(args...));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
void println(std::wostream& os, wformat_string<T...> fmt, T&&... args) {
|
||||||
|
print(os, L"{}\n", fmt::format(fmt, std::forward<T>(args)...));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Converts `value` to `std::wstring` using the default format for type `T`.
|
||||||
template <typename T> inline auto to_wstring(const T& value) -> std::wstring {
|
template <typename T> inline auto to_wstring(const T& value) -> std::wstring {
|
||||||
return format(FMT_STRING(L"{}"), value);
|
return format(FMT_STRING(L"{}"), value);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
115
deps/fmt/src/fmt.cc
vendored
115
deps/fmt/src/fmt.cc
vendored
|
|
@ -1,38 +1,61 @@
|
||||||
module;
|
module;
|
||||||
|
|
||||||
|
#define FMT_MODULE
|
||||||
|
|
||||||
|
#ifdef _MSVC_LANG
|
||||||
|
# define FMT_CPLUSPLUS _MSVC_LANG
|
||||||
|
#else
|
||||||
|
# define FMT_CPLUSPLUS __cplusplus
|
||||||
|
#endif
|
||||||
|
|
||||||
// Put all implementation-provided headers into the global module fragment
|
// Put all implementation-provided headers into the global module fragment
|
||||||
// to prevent attachment to this module.
|
// to prevent attachment to this module.
|
||||||
#include <algorithm>
|
#ifndef FMT_IMPORT_STD
|
||||||
|
# include <algorithm>
|
||||||
|
# include <bitset>
|
||||||
|
# include <chrono>
|
||||||
|
# include <cmath>
|
||||||
|
# include <complex>
|
||||||
|
# include <cstddef>
|
||||||
|
# include <cstdint>
|
||||||
|
# include <cstdio>
|
||||||
|
# include <cstdlib>
|
||||||
|
# include <cstring>
|
||||||
|
# include <ctime>
|
||||||
|
# include <exception>
|
||||||
|
# if FMT_CPLUSPLUS > 202002L
|
||||||
|
# include <expected>
|
||||||
|
# endif
|
||||||
|
# include <filesystem>
|
||||||
|
# include <fstream>
|
||||||
|
# include <functional>
|
||||||
|
# include <iterator>
|
||||||
|
# include <limits>
|
||||||
|
# include <locale>
|
||||||
|
# include <memory>
|
||||||
|
# include <optional>
|
||||||
|
# include <ostream>
|
||||||
|
# include <source_location>
|
||||||
|
# include <stdexcept>
|
||||||
|
# include <string>
|
||||||
|
# include <string_view>
|
||||||
|
# include <system_error>
|
||||||
|
# include <thread>
|
||||||
|
# include <type_traits>
|
||||||
|
# include <typeinfo>
|
||||||
|
# include <utility>
|
||||||
|
# include <variant>
|
||||||
|
# include <vector>
|
||||||
|
#else
|
||||||
|
# include <limits.h>
|
||||||
|
# include <stdint.h>
|
||||||
|
# include <stdio.h>
|
||||||
|
# include <stdlib.h>
|
||||||
|
# include <string.h>
|
||||||
|
# include <time.h>
|
||||||
|
#endif
|
||||||
#include <cerrno>
|
#include <cerrno>
|
||||||
#include <chrono>
|
|
||||||
#include <climits>
|
#include <climits>
|
||||||
#include <cmath>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <cstdint>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <cstdlib>
|
|
||||||
#include <cstring>
|
|
||||||
#include <ctime>
|
|
||||||
#include <exception>
|
|
||||||
#include <filesystem>
|
|
||||||
#include <fstream>
|
|
||||||
#include <functional>
|
|
||||||
#include <iterator>
|
|
||||||
#include <limits>
|
|
||||||
#include <locale>
|
|
||||||
#include <memory>
|
|
||||||
#include <optional>
|
|
||||||
#include <ostream>
|
|
||||||
#include <stdexcept>
|
|
||||||
#include <string>
|
|
||||||
#include <string_view>
|
|
||||||
#include <system_error>
|
|
||||||
#include <thread>
|
|
||||||
#include <type_traits>
|
|
||||||
#include <typeinfo>
|
|
||||||
#include <utility>
|
|
||||||
#include <variant>
|
|
||||||
#include <vector>
|
|
||||||
#include <version>
|
#include <version>
|
||||||
|
|
||||||
#if __has_include(<cxxabi.h>)
|
#if __has_include(<cxxabi.h>)
|
||||||
|
|
@ -63,8 +86,6 @@ module;
|
||||||
# if defined(__GLIBCXX__)
|
# if defined(__GLIBCXX__)
|
||||||
# include <ext/stdio_filebuf.h>
|
# include <ext/stdio_filebuf.h>
|
||||||
# include <ext/stdio_sync_filebuf.h>
|
# include <ext/stdio_sync_filebuf.h>
|
||||||
# elif defined(_LIBCPP_VERSION)
|
|
||||||
# include <__std_stream>
|
|
||||||
# endif
|
# endif
|
||||||
# define WIN32_LEAN_AND_MEAN
|
# define WIN32_LEAN_AND_MEAN
|
||||||
# include <windows.h>
|
# include <windows.h>
|
||||||
|
|
@ -72,6 +93,10 @@ module;
|
||||||
|
|
||||||
export module fmt;
|
export module fmt;
|
||||||
|
|
||||||
|
#ifdef FMT_IMPORT_STD
|
||||||
|
import std;
|
||||||
|
#endif
|
||||||
|
|
||||||
#define FMT_EXPORT export
|
#define FMT_EXPORT export
|
||||||
#define FMT_BEGIN_EXPORT export {
|
#define FMT_BEGIN_EXPORT export {
|
||||||
#define FMT_END_EXPORT }
|
#define FMT_END_EXPORT }
|
||||||
|
|
@ -85,6 +110,10 @@ export module fmt;
|
||||||
extern "C++" {
|
extern "C++" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef FMT_OS
|
||||||
|
# define FMT_OS 1
|
||||||
|
#endif
|
||||||
|
|
||||||
// All library-provided declarations and definitions must be in the module
|
// All library-provided declarations and definitions must be in the module
|
||||||
// purview to be exported.
|
// purview to be exported.
|
||||||
#include "fmt/args.h"
|
#include "fmt/args.h"
|
||||||
|
|
@ -92,8 +121,12 @@ extern "C++" {
|
||||||
#include "fmt/color.h"
|
#include "fmt/color.h"
|
||||||
#include "fmt/compile.h"
|
#include "fmt/compile.h"
|
||||||
#include "fmt/format.h"
|
#include "fmt/format.h"
|
||||||
#include "fmt/os.h"
|
#if FMT_OS
|
||||||
|
# include "fmt/os.h"
|
||||||
|
#endif
|
||||||
|
#include "fmt/ostream.h"
|
||||||
#include "fmt/printf.h"
|
#include "fmt/printf.h"
|
||||||
|
#include "fmt/ranges.h"
|
||||||
#include "fmt/std.h"
|
#include "fmt/std.h"
|
||||||
#include "fmt/xchar.h"
|
#include "fmt/xchar.h"
|
||||||
|
|
||||||
|
|
@ -103,8 +136,20 @@ extern "C++" {
|
||||||
|
|
||||||
// gcc doesn't yet implement private module fragments
|
// gcc doesn't yet implement private module fragments
|
||||||
#if !FMT_GCC_VERSION
|
#if !FMT_GCC_VERSION
|
||||||
module : private;
|
module :private;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "format.cc"
|
#ifdef FMT_ATTACH_TO_GLOBAL_MODULE
|
||||||
#include "os.cc"
|
extern "C++" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if FMT_HAS_INCLUDE("format.cc")
|
||||||
|
# include "format.cc"
|
||||||
|
#endif
|
||||||
|
#if FMT_OS && FMT_HAS_INCLUDE("os.cc")
|
||||||
|
# include "os.cc"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef FMT_ATTACH_TO_GLOBAL_MODULE
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
|
||||||
16
deps/fmt/src/format.cc
vendored
16
deps/fmt/src/format.cc
vendored
|
|
@ -8,6 +8,12 @@
|
||||||
#include "fmt/format-inl.h"
|
#include "fmt/format-inl.h"
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
|
template FMT_API locale_ref::locale_ref(const std::locale& loc); // DEPRECATED!
|
||||||
|
template FMT_API auto locale_ref::get<std::locale>() const -> std::locale;
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template FMT_API auto dragonbox::to_decimal(float x) noexcept
|
template FMT_API auto dragonbox::to_decimal(float x) noexcept
|
||||||
|
|
@ -15,28 +21,22 @@ template FMT_API auto dragonbox::to_decimal(float x) noexcept
|
||||||
template FMT_API auto dragonbox::to_decimal(double x) noexcept
|
template FMT_API auto dragonbox::to_decimal(double x) noexcept
|
||||||
-> dragonbox::decimal_fp<double>;
|
-> dragonbox::decimal_fp<double>;
|
||||||
|
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
|
||||||
template FMT_API locale_ref::locale_ref(const std::locale& loc);
|
|
||||||
template FMT_API auto locale_ref::get<std::locale>() const -> std::locale;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Explicit instantiations for char.
|
// Explicit instantiations for char.
|
||||||
|
|
||||||
template FMT_API auto thousands_sep_impl(locale_ref)
|
template FMT_API auto thousands_sep_impl(locale_ref)
|
||||||
-> thousands_sep_result<char>;
|
-> thousands_sep_result<char>;
|
||||||
template FMT_API auto decimal_point_impl(locale_ref) -> char;
|
template FMT_API auto decimal_point_impl(locale_ref) -> char;
|
||||||
|
|
||||||
|
// DEPRECATED!
|
||||||
template FMT_API void buffer<char>::append(const char*, const char*);
|
template FMT_API void buffer<char>::append(const char*, const char*);
|
||||||
|
|
||||||
template FMT_API void vformat_to(buffer<char>&, string_view,
|
|
||||||
typename vformat_args<>::type, locale_ref);
|
|
||||||
|
|
||||||
// Explicit instantiations for wchar_t.
|
// Explicit instantiations for wchar_t.
|
||||||
|
|
||||||
template FMT_API auto thousands_sep_impl(locale_ref)
|
template FMT_API auto thousands_sep_impl(locale_ref)
|
||||||
-> thousands_sep_result<wchar_t>;
|
-> thousands_sep_result<wchar_t>;
|
||||||
template FMT_API auto decimal_point_impl(locale_ref) -> wchar_t;
|
template FMT_API auto decimal_point_impl(locale_ref) -> wchar_t;
|
||||||
|
|
||||||
|
// DEPRECATED!
|
||||||
template FMT_API void buffer<wchar_t>::append(const wchar_t*, const wchar_t*);
|
template FMT_API void buffer<wchar_t>::append(const wchar_t*, const wchar_t*);
|
||||||
|
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
|
||||||
172
deps/fmt/src/os.cc
vendored
172
deps/fmt/src/os.cc
vendored
|
|
@ -12,47 +12,51 @@
|
||||||
|
|
||||||
#include "fmt/os.h"
|
#include "fmt/os.h"
|
||||||
|
|
||||||
#include <climits>
|
#ifndef FMT_MODULE
|
||||||
|
# include <climits>
|
||||||
|
|
||||||
#if FMT_USE_FCNTL
|
# if FMT_USE_FCNTL
|
||||||
# include <sys/stat.h>
|
# include <sys/stat.h>
|
||||||
# include <sys/types.h>
|
# include <sys/types.h>
|
||||||
|
|
||||||
# ifdef _WRS_KERNEL // VxWorks7 kernel
|
# ifdef _WRS_KERNEL // VxWorks7 kernel
|
||||||
# include <ioLib.h> // getpagesize
|
# include <ioLib.h> // getpagesize
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# ifndef _WIN32
|
||||||
|
# include <unistd.h>
|
||||||
|
# else
|
||||||
|
# ifndef WIN32_LEAN_AND_MEAN
|
||||||
|
# define WIN32_LEAN_AND_MEAN
|
||||||
|
# endif
|
||||||
|
# include <io.h>
|
||||||
|
# endif // _WIN32
|
||||||
|
# endif // FMT_USE_FCNTL
|
||||||
|
|
||||||
|
# ifdef _WIN32
|
||||||
|
# include <windows.h>
|
||||||
# endif
|
# endif
|
||||||
|
#endif
|
||||||
# ifndef _WIN32
|
|
||||||
# include <unistd.h>
|
|
||||||
# else
|
|
||||||
# ifndef WIN32_LEAN_AND_MEAN
|
|
||||||
# define WIN32_LEAN_AND_MEAN
|
|
||||||
# endif
|
|
||||||
# include <io.h>
|
|
||||||
|
|
||||||
# ifndef S_IRUSR
|
|
||||||
# define S_IRUSR _S_IREAD
|
|
||||||
# endif
|
|
||||||
# ifndef S_IWUSR
|
|
||||||
# define S_IWUSR _S_IWRITE
|
|
||||||
# endif
|
|
||||||
# ifndef S_IRGRP
|
|
||||||
# define S_IRGRP 0
|
|
||||||
# endif
|
|
||||||
# ifndef S_IWGRP
|
|
||||||
# define S_IWGRP 0
|
|
||||||
# endif
|
|
||||||
# ifndef S_IROTH
|
|
||||||
# define S_IROTH 0
|
|
||||||
# endif
|
|
||||||
# ifndef S_IWOTH
|
|
||||||
# define S_IWOTH 0
|
|
||||||
# endif
|
|
||||||
# endif // _WIN32
|
|
||||||
#endif // FMT_USE_FCNTL
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
# include <windows.h>
|
# ifndef S_IRUSR
|
||||||
|
# define S_IRUSR _S_IREAD
|
||||||
|
# endif
|
||||||
|
# ifndef S_IWUSR
|
||||||
|
# define S_IWUSR _S_IWRITE
|
||||||
|
# endif
|
||||||
|
# ifndef S_IRGRP
|
||||||
|
# define S_IRGRP 0
|
||||||
|
# endif
|
||||||
|
# ifndef S_IWGRP
|
||||||
|
# define S_IWGRP 0
|
||||||
|
# endif
|
||||||
|
# ifndef S_IROTH
|
||||||
|
# define S_IROTH 0
|
||||||
|
# endif
|
||||||
|
# ifndef S_IWOTH
|
||||||
|
# define S_IWOTH 0
|
||||||
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
@ -62,14 +66,14 @@ using rwresult = int;
|
||||||
|
|
||||||
// On Windows the count argument to read and write is unsigned, so convert
|
// On Windows the count argument to read and write is unsigned, so convert
|
||||||
// it from size_t preventing integer overflow.
|
// it from size_t preventing integer overflow.
|
||||||
inline unsigned convert_rwcount(std::size_t count) {
|
inline unsigned convert_rwcount(size_t count) {
|
||||||
return count <= UINT_MAX ? static_cast<unsigned>(count) : UINT_MAX;
|
return count <= UINT_MAX ? static_cast<unsigned>(count) : UINT_MAX;
|
||||||
}
|
}
|
||||||
#elif FMT_USE_FCNTL
|
#elif FMT_USE_FCNTL
|
||||||
// Return type of read and write functions.
|
// Return type of read and write functions.
|
||||||
using rwresult = ssize_t;
|
using rwresult = ssize_t;
|
||||||
|
|
||||||
inline std::size_t convert_rwcount(std::size_t count) { return count; }
|
inline auto convert_rwcount(size_t count) -> size_t { return count; }
|
||||||
#endif
|
#endif
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
|
@ -156,7 +160,7 @@ void detail::format_windows_error(detail::buffer<char>& out, int error_code,
|
||||||
}
|
}
|
||||||
|
|
||||||
void report_windows_error(int error_code, const char* message) noexcept {
|
void report_windows_error(int error_code, const char* message) noexcept {
|
||||||
report_error(detail::format_windows_error, error_code, message);
|
do_report_error(detail::format_windows_error, error_code, message);
|
||||||
}
|
}
|
||||||
#endif // _WIN32
|
#endif // _WIN32
|
||||||
|
|
||||||
|
|
@ -181,11 +185,17 @@ void buffered_file::close() {
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot close file")));
|
FMT_THROW(system_error(errno, FMT_STRING("cannot close file")));
|
||||||
}
|
}
|
||||||
|
|
||||||
int buffered_file::descriptor() const {
|
auto buffered_file::descriptor() const -> int {
|
||||||
#ifdef fileno // fileno is a macro on OpenBSD so we cannot use FMT_POSIX_CALL.
|
#ifdef FMT_HAS_SYSTEM
|
||||||
int fd = fileno(file_);
|
// fileno is a macro on OpenBSD.
|
||||||
#else
|
# ifdef fileno
|
||||||
|
# undef fileno
|
||||||
|
# endif
|
||||||
int fd = FMT_POSIX_CALL(fileno(file_));
|
int fd = FMT_POSIX_CALL(fileno(file_));
|
||||||
|
#elif defined(_WIN32)
|
||||||
|
int fd = _fileno(file_);
|
||||||
|
#else
|
||||||
|
int fd = fileno(file_);
|
||||||
#endif
|
#endif
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot get file descriptor")));
|
FMT_THROW(system_error(errno, FMT_STRING("cannot get file descriptor")));
|
||||||
|
|
@ -196,6 +206,7 @@ int buffered_file::descriptor() const {
|
||||||
# ifdef _WIN32
|
# ifdef _WIN32
|
||||||
using mode_t = int;
|
using mode_t = int;
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
constexpr mode_t default_open_mode =
|
constexpr mode_t default_open_mode =
|
||||||
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH;
|
S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH;
|
||||||
|
|
||||||
|
|
@ -229,7 +240,7 @@ void file::close() {
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot close file")));
|
FMT_THROW(system_error(errno, FMT_STRING("cannot close file")));
|
||||||
}
|
}
|
||||||
|
|
||||||
long long file::size() const {
|
auto file::size() const -> long long {
|
||||||
# ifdef _WIN32
|
# ifdef _WIN32
|
||||||
// Use GetFileSize instead of GetFileSizeEx for the case when _WIN32_WINNT
|
// Use GetFileSize instead of GetFileSizeEx for the case when _WIN32_WINNT
|
||||||
// is less than 0x0500 as is the case with some default MinGW builds.
|
// is less than 0x0500 as is the case with some default MinGW builds.
|
||||||
|
|
@ -240,7 +251,7 @@ long long file::size() const {
|
||||||
if (size_lower == INVALID_FILE_SIZE) {
|
if (size_lower == INVALID_FILE_SIZE) {
|
||||||
DWORD error = GetLastError();
|
DWORD error = GetLastError();
|
||||||
if (error != NO_ERROR)
|
if (error != NO_ERROR)
|
||||||
FMT_THROW(windows_error(GetLastError(), "cannot get file size"));
|
FMT_THROW(windows_error(error, "cannot get file size"));
|
||||||
}
|
}
|
||||||
unsigned long long long_size = size_upper;
|
unsigned long long long_size = size_upper;
|
||||||
return (long_size << sizeof(DWORD) * CHAR_BIT) | size_lower;
|
return (long_size << sizeof(DWORD) * CHAR_BIT) | size_lower;
|
||||||
|
|
@ -255,7 +266,7 @@ long long file::size() const {
|
||||||
# endif
|
# endif
|
||||||
}
|
}
|
||||||
|
|
||||||
std::size_t file::read(void* buffer, std::size_t count) {
|
auto file::read(void* buffer, size_t count) -> size_t {
|
||||||
rwresult result = 0;
|
rwresult result = 0;
|
||||||
FMT_RETRY(result, FMT_POSIX_CALL(read(fd_, buffer, convert_rwcount(count))));
|
FMT_RETRY(result, FMT_POSIX_CALL(read(fd_, buffer, convert_rwcount(count))));
|
||||||
if (result < 0)
|
if (result < 0)
|
||||||
|
|
@ -263,7 +274,7 @@ std::size_t file::read(void* buffer, std::size_t count) {
|
||||||
return detail::to_unsigned(result);
|
return detail::to_unsigned(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::size_t file::write(const void* buffer, std::size_t count) {
|
auto file::write(const void* buffer, size_t count) -> size_t {
|
||||||
rwresult result = 0;
|
rwresult result = 0;
|
||||||
FMT_RETRY(result, FMT_POSIX_CALL(write(fd_, buffer, convert_rwcount(count))));
|
FMT_RETRY(result, FMT_POSIX_CALL(write(fd_, buffer, convert_rwcount(count))));
|
||||||
if (result < 0)
|
if (result < 0)
|
||||||
|
|
@ -271,7 +282,7 @@ std::size_t file::write(const void* buffer, std::size_t count) {
|
||||||
return detail::to_unsigned(result);
|
return detail::to_unsigned(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
file file::dup(int fd) {
|
auto file::dup(int fd) -> file {
|
||||||
// Don't retry as dup doesn't return EINTR.
|
// Don't retry as dup doesn't return EINTR.
|
||||||
// http://pubs.opengroup.org/onlinepubs/009695399/functions/dup.html
|
// http://pubs.opengroup.org/onlinepubs/009695399/functions/dup.html
|
||||||
int new_fd = FMT_POSIX_CALL(dup(fd));
|
int new_fd = FMT_POSIX_CALL(dup(fd));
|
||||||
|
|
@ -297,30 +308,7 @@ void file::dup2(int fd, std::error_code& ec) noexcept {
|
||||||
if (result == -1) ec = std::error_code(errno, std::generic_category());
|
if (result == -1) ec = std::error_code(errno, std::generic_category());
|
||||||
}
|
}
|
||||||
|
|
||||||
void file::pipe(file& read_end, file& write_end) {
|
auto file::fdopen(const char* mode) -> buffered_file {
|
||||||
// Close the descriptors first to make sure that assignments don't throw
|
|
||||||
// and there are no leaks.
|
|
||||||
read_end.close();
|
|
||||||
write_end.close();
|
|
||||||
int fds[2] = {};
|
|
||||||
# ifdef _WIN32
|
|
||||||
// Make the default pipe capacity same as on Linux 2.6.11+.
|
|
||||||
enum { DEFAULT_CAPACITY = 65536 };
|
|
||||||
int result = FMT_POSIX_CALL(pipe(fds, DEFAULT_CAPACITY, _O_BINARY));
|
|
||||||
# else
|
|
||||||
// Don't retry as the pipe function doesn't return EINTR.
|
|
||||||
// http://pubs.opengroup.org/onlinepubs/009696799/functions/pipe.html
|
|
||||||
int result = FMT_POSIX_CALL(pipe(fds));
|
|
||||||
# endif
|
|
||||||
if (result != 0)
|
|
||||||
FMT_THROW(system_error(errno, FMT_STRING("cannot create pipe")));
|
|
||||||
// The following assignments don't throw because read_fd and write_fd
|
|
||||||
// are closed.
|
|
||||||
read_end = file(fds[0]);
|
|
||||||
write_end = file(fds[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
buffered_file file::fdopen(const char* mode) {
|
|
||||||
// Don't retry as fdopen doesn't return EINTR.
|
// Don't retry as fdopen doesn't return EINTR.
|
||||||
# if defined(__MINGW32__) && defined(_POSIX_)
|
# if defined(__MINGW32__) && defined(_POSIX_)
|
||||||
FILE* f = ::fdopen(fd_, mode);
|
FILE* f = ::fdopen(fd_, mode);
|
||||||
|
|
@ -348,8 +336,26 @@ file file::open_windows_file(wcstring_view path, int oflag) {
|
||||||
}
|
}
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
|
pipe::pipe() {
|
||||||
|
int fds[2] = {};
|
||||||
|
# ifdef _WIN32
|
||||||
|
// Make the default pipe capacity same as on Linux 2.6.11+.
|
||||||
|
enum { DEFAULT_CAPACITY = 65536 };
|
||||||
|
int result = FMT_POSIX_CALL(pipe(fds, DEFAULT_CAPACITY, _O_BINARY));
|
||||||
|
# else
|
||||||
|
// Don't retry as the pipe function doesn't return EINTR.
|
||||||
|
// http://pubs.opengroup.org/onlinepubs/009696799/functions/pipe.html
|
||||||
|
int result = FMT_POSIX_CALL(pipe(fds));
|
||||||
|
# endif
|
||||||
|
if (result != 0)
|
||||||
|
FMT_THROW(system_error(errno, FMT_STRING("cannot create pipe")));
|
||||||
|
// The following assignments don't throw.
|
||||||
|
read_end = file(fds[0]);
|
||||||
|
write_end = file(fds[1]);
|
||||||
|
}
|
||||||
|
|
||||||
# if !defined(__MSDOS__)
|
# if !defined(__MSDOS__)
|
||||||
long getpagesize() {
|
auto getpagesize() -> long {
|
||||||
# ifdef _WIN32
|
# ifdef _WIN32
|
||||||
SYSTEM_INFO si;
|
SYSTEM_INFO si;
|
||||||
GetSystemInfo(&si);
|
GetSystemInfo(&si);
|
||||||
|
|
@ -368,31 +374,25 @@ long getpagesize() {
|
||||||
}
|
}
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
namespace detail {
|
void ostream::grow(buffer<char>& buf, size_t) {
|
||||||
|
if (buf.size() == buf.capacity()) static_cast<ostream&>(buf).flush();
|
||||||
void file_buffer::grow(size_t) {
|
|
||||||
if (this->size() == this->capacity()) flush();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
file_buffer::file_buffer(cstring_view path,
|
ostream::ostream(cstring_view path, const detail::ostream_params& params)
|
||||||
const detail::ostream_params& params)
|
: buffer<char>(grow), file_(path, params.oflag) {
|
||||||
: file_(path, params.oflag) {
|
|
||||||
set(new char[params.buffer_size], params.buffer_size);
|
set(new char[params.buffer_size], params.buffer_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
file_buffer::file_buffer(file_buffer&& other)
|
ostream::ostream(ostream&& other) noexcept
|
||||||
: detail::buffer<char>(other.data(), other.size(), other.capacity()),
|
: buffer<char>(grow, other.data(), other.size(), other.capacity()),
|
||||||
file_(std::move(other.file_)) {
|
file_(std::move(other.file_)) {
|
||||||
other.clear();
|
other.clear();
|
||||||
other.set(nullptr, 0);
|
other.set(nullptr, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
file_buffer::~file_buffer() {
|
ostream::~ostream() {
|
||||||
flush();
|
flush();
|
||||||
delete[] data();
|
delete[] data();
|
||||||
}
|
}
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
ostream::~ostream() = default;
|
|
||||||
#endif // FMT_USE_FCNTL
|
#endif // FMT_USE_FCNTL
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
|
||||||
4
deps/jemalloc/CMakeLists.txt
vendored
4
deps/jemalloc/CMakeLists.txt
vendored
|
|
@ -10,9 +10,11 @@
|
||||||
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
#
|
#
|
||||||
|
|
||||||
|
include(CheckSymbolExists)
|
||||||
|
|
||||||
if(CMAKE_SYSTEM_NAME MATCHES "Linux" AND NOT NOJEM)
|
if(CMAKE_SYSTEM_NAME MATCHES "Linux" AND NOT NOJEM)
|
||||||
# We need to generate the jemalloc_def.h header based on platform-specific settings
|
# We need to generate the jemalloc_def.h header based on platform-specific settings
|
||||||
CHECK_SYMBOL_EXISTS(MADV_FREE "sys/mman.h" HAVE_MADV_FREE)
|
check_symbol_exists(MADV_FREE "sys/mman.h" HAVE_MADV_FREE)
|
||||||
|
|
||||||
if (PLATFORM EQUAL 32)
|
if (PLATFORM EQUAL 32)
|
||||||
set(JEM_SIZEDEF 2)
|
set(JEM_SIZEDEF 2)
|
||||||
|
|
|
||||||
|
|
@ -69,13 +69,13 @@ public:
|
||||||
bool SetLogLevel(std::string const& name, int32 level, bool isLogger = true);
|
bool SetLogLevel(std::string const& name, int32 level, bool isLogger = true);
|
||||||
|
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
inline void outMessage(std::string const& filter, LogLevel const level, Acore::FormatString<Args...> fmt, Args&&... args)
|
inline void outMessage(std::string const& filter, LogLevel const level, Acore::FormatStringView fmt, Args&&... args)
|
||||||
{
|
{
|
||||||
_outMessage(filter, level, Acore::StringFormat(fmt, std::forward<Args>(args)...));
|
_outMessage(filter, level, Acore::StringFormat(fmt, std::forward<Args>(args)...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
void outCommand(uint32 account, Acore::FormatString<Args...> fmt, Args&&... args)
|
void outCommand(uint32 account, Acore::FormatStringView fmt, Args&&... args)
|
||||||
{
|
{
|
||||||
if (!ShouldLog("commands.gm", LOG_LEVEL_INFO))
|
if (!ShouldLog("commands.gm", LOG_LEVEL_INFO))
|
||||||
{
|
{
|
||||||
|
|
@ -126,19 +126,6 @@ private:
|
||||||
|
|
||||||
#define sLog Log::instance()
|
#define sLog Log::instance()
|
||||||
|
|
||||||
#define LOG_EXCEPTION_FREE(filterType__, level__, ...) \
|
|
||||||
{ \
|
|
||||||
try \
|
|
||||||
{ \
|
|
||||||
sLog->outMessage(filterType__, level__, fmt::format(__VA_ARGS__)); \
|
|
||||||
} \
|
|
||||||
catch (std::exception const& e) \
|
|
||||||
{ \
|
|
||||||
sLog->outMessage("server", LogLevel::LOG_LEVEL_ERROR, "Wrong format occurred ({}) at '{}:{}'", \
|
|
||||||
e.what(), __FILE__, __LINE__); \
|
|
||||||
} \
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef PERFORMANCE_PROFILING
|
#ifdef PERFORMANCE_PROFILING
|
||||||
#define LOG_MESSAGE_BODY(filterType__, level__, ...) ((void)0)
|
#define LOG_MESSAGE_BODY(filterType__, level__, ...) ((void)0)
|
||||||
#else
|
#else
|
||||||
|
|
@ -146,7 +133,7 @@ private:
|
||||||
do \
|
do \
|
||||||
{ \
|
{ \
|
||||||
if (sLog->ShouldLog(filterType__, level__)) \
|
if (sLog->ShouldLog(filterType__, level__)) \
|
||||||
LOG_EXCEPTION_FREE(filterType__, level__, __VA_ARGS__); \
|
sLog->outMessage(filterType__, level__, __VA_ARGS__); \
|
||||||
} while (0)
|
} while (0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -41,29 +41,29 @@ namespace Acore
|
||||||
|
|
||||||
/// Default AC string format function.
|
/// Default AC string format function.
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
inline std::string StringFormat(FormatString<Args...> fmt, Args&&... args)
|
inline std::string StringFormat(FormatStringView fmt, Args&&... args)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return fmt::format(fmt, std::forward<Args>(args)...);
|
return fmt::vformat(fmt, fmt::make_format_args(args...));
|
||||||
}
|
}
|
||||||
catch (std::exception const& e)
|
catch (std::exception const& e)
|
||||||
{
|
{
|
||||||
return fmt::format("Wrong format occurred ({}). Fmt string: '{}'", e.what(), fmt.get());
|
return fmt::format("Wrong format occurred ({}). Fmt string: '{}'", e.what(), fmt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Format directly to an output iterator.
|
/// Format directly to an output iterator.
|
||||||
template<typename OutputIt, typename... Args>
|
template<typename OutputIt, typename... Args>
|
||||||
inline OutputIt StringFormatTo(OutputIt out, FormatString<Args...> fmt, Args&&... args)
|
inline OutputIt StringFormatTo(OutputIt out, FormatStringView fmt, Args&&... args)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return fmt::format_to(out, fmt, std::forward<Args>(args)...);
|
return fmt::vformat_to(out, fmt, fmt::make_format_args(args...));
|
||||||
}
|
}
|
||||||
catch (std::exception const& e)
|
catch (std::exception const& e)
|
||||||
{
|
{
|
||||||
return fmt::format_to(out, "Wrong format occurred ({}). Fmt string: '{}'", e.what(), fmt.get());
|
return fmt::format_to(out, "Wrong format occurred ({}). Fmt string: '{}'", e.what(), fmt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -21,6 +21,7 @@
|
||||||
#include "Util.h"
|
#include "Util.h"
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
#include <memory>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
#include <queue>
|
#include <queue>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
|
||||||
|
|
@ -77,10 +77,10 @@ namespace Acore::Impl::Readline
|
||||||
void utf8print(void* /*arg*/, std::string_view str)
|
void utf8print(void* /*arg*/, std::string_view str)
|
||||||
{
|
{
|
||||||
#if AC_PLATFORM == AC_PLATFORM_WINDOWS
|
#if AC_PLATFORM == AC_PLATFORM_WINDOWS
|
||||||
fmt::print(str);
|
fmt::print("{}", str);
|
||||||
#else
|
#else
|
||||||
{
|
{
|
||||||
fmt::print(str);
|
fmt::print("{}", str);
|
||||||
fflush(stdout);
|
fflush(stdout);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -143,6 +143,11 @@ namespace Acore::ChatCommands
|
||||||
AC_GAME_API ChatCommandResult TryConsume(ChatHandler const* handler, std::string_view args);
|
AC_GAME_API ChatCommandResult TryConsume(ChatHandler const* handler, std::string_view args);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
inline std::string_view format_as(QuotedString const& qs)
|
||||||
|
{
|
||||||
|
return std::string_view(qs);
|
||||||
|
}
|
||||||
|
|
||||||
struct AC_GAME_API AccountIdentifier : Acore::Impl::ChatCommands::ContainerTag
|
struct AC_GAME_API AccountIdentifier : Acore::Impl::ChatCommands::ContainerTag
|
||||||
{
|
{
|
||||||
using value_type = uint32;
|
using value_type = uint32;
|
||||||
|
|
|
||||||
|
|
@ -20,6 +20,7 @@
|
||||||
|
|
||||||
#include "GridCell.h"
|
#include "GridCell.h"
|
||||||
#include "GridReference.h"
|
#include "GridReference.h"
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
class GridTerrainData;
|
class GridTerrainData;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -19,6 +19,7 @@
|
||||||
#include "CellImpl.h"
|
#include "CellImpl.h"
|
||||||
#include "Chat.h"
|
#include "Chat.h"
|
||||||
#include "CommandScript.h"
|
#include "CommandScript.h"
|
||||||
|
#include <fmt/ranges.h>
|
||||||
#include "GameObject.h"
|
#include "GameObject.h"
|
||||||
#include "GridNotifiers.h"
|
#include "GridNotifiers.h"
|
||||||
#include "GridNotifiersImpl.h"
|
#include "GridNotifiersImpl.h"
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue