mirror of
https://github.com/gabime/spdlog.git
synced 2025-04-29 12:03:53 +00:00
Compare commits
87 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
548b264254 | ||
|
847db3375f | ||
|
bb8694b50f | ||
|
cec28bf839 | ||
|
bd0609d7a0 | ||
|
1f4959c832 | ||
|
48bcf39a66 | ||
|
9c58257480 | ||
|
faa0a7a9c5 | ||
|
10320184df | ||
|
3335c380a0 | ||
|
f355b3d58f | ||
|
ac432c3602 | ||
|
3c23c27d2d | ||
|
ae1de0dc8c | ||
|
7cbf2a6967 | ||
|
57505989b7 | ||
|
96a7d2a1d4 | ||
|
d71555306a | ||
|
ad0f31c009 | ||
|
96a8f6250c | ||
|
7f8060d5b2 | ||
|
276ee5f5c0 | ||
|
24dde318fe | ||
|
65e388e82b | ||
|
1e6250e183 | ||
|
951c5b9987 | ||
|
15f539685b | ||
|
43dcb3982d | ||
|
0efef2af24 | ||
|
018d8aa266 | ||
|
35b0417fbe | ||
|
94526fa8e8 | ||
|
633003f40a | ||
|
9edab1b5a1 | ||
|
1245bf8e8a | ||
|
51a0deca2c | ||
|
8e5613379f | ||
|
7cee026baa | ||
|
ebfa906952 | ||
|
68f6ec7af1 | ||
|
d343d413c2 | ||
|
fe4f99527d | ||
|
5673e9e545 | ||
|
63f0875000 | ||
|
5fd32e1a70 | ||
|
35345182f8 | ||
|
3c2e002b51 | ||
|
6192537d08 | ||
|
6c7201553d | ||
|
d939255f0e | ||
|
ecc3881122 | ||
|
bff1a6036a | ||
|
6f2ead1a0e | ||
|
92f9aa32ce | ||
|
64d9b4e263 | ||
|
3d3f71dbe2 | ||
|
3fec1a81b7 | ||
|
984a959883 | ||
|
7ecfb3bc9c | ||
|
614c3a6836 | ||
|
5dc356dcbe | ||
|
a7eb388f84 | ||
|
a5cfbf369d | ||
|
d373093734 | ||
|
7a950e028c | ||
|
9fe79692eb | ||
|
96c9a62bfd | ||
|
85bdab0c18 | ||
|
63d1884215 | ||
|
b6da59447f | ||
|
16e0d2e77c | ||
|
ee16895787 | ||
|
e593f6695c | ||
|
2c76e6101a | ||
|
bdd1dff378 | ||
|
ffd5aa41d6 | ||
|
c1fbafdcef | ||
|
362214a349 | ||
|
2169a6f6ae | ||
|
271f0f3b14 | ||
|
a3a0c9d663 | ||
|
5ebfc92730 | ||
|
885b5473e2 | ||
|
d276069a6e | ||
|
eeb22c13bb | ||
|
c3aed4b683 |
@ -1,9 +1,15 @@
|
|||||||
name: ci
|
name: linux
|
||||||
|
|
||||||
on: [push, pull_request]
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build_linux:
|
# -----------------------------------------------------------------------
|
||||||
|
# Linux build matrix
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
defaults:
|
defaults:
|
||||||
run:
|
run:
|
||||||
@ -16,13 +22,14 @@ jobs:
|
|||||||
- { compiler: gcc, version: 9, build_type: Release, cppstd: 17 }
|
- { compiler: gcc, version: 9, build_type: Release, cppstd: 17 }
|
||||||
- { compiler: gcc, version: 11, build_type: Debug, cppstd: 20 }
|
- { compiler: gcc, version: 11, build_type: Debug, cppstd: 20 }
|
||||||
- { compiler: gcc, version: 12, build_type: Release, cppstd: 20 }
|
- { compiler: gcc, version: 12, build_type: Release, cppstd: 20 }
|
||||||
- { compiler: clang, version: 12, build_type: Debug, cppstd: 17, asan: OFF }
|
- { compiler: gcc, version: 12, build_type: Debug, cppstd: 20, asan: ON }
|
||||||
- { compiler: clang, version: 15, build_type: Release, cppstd: 20, asan: OFF }
|
- { compiler: clang, version: 12, build_type: Debug, cppstd: 17 }
|
||||||
|
- { compiler: clang, version: 15, build_type: Release, cppstd: 20, tsan: ON }
|
||||||
container:
|
container:
|
||||||
image: ${{ matrix.config.compiler == 'clang' && 'teeks99/clang-ubuntu' || matrix.config.compiler }}:${{ matrix.config.version }}
|
image: ${{ matrix.config.compiler == 'clang' && 'teeks99/clang-ubuntu' || matrix.config.compiler }}:${{ matrix.config.version }}
|
||||||
name: "${{ matrix.config.compiler}} ${{ matrix.config.version }} (C++${{ matrix.config.cppstd }}, ${{ matrix.config.build_type }})"
|
name: "${{ matrix.config.compiler}} ${{ matrix.config.version }} (C++${{ matrix.config.cppstd }} ${{ matrix.config.build_type }} ${{ matrix.config.asan == 'ON' && 'ASAN' || '' }}${{ matrix.config.tsan == 'ON' && 'TSAN' || '' }})"
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@main
|
- uses: actions/checkout@v4
|
||||||
- name: Setup
|
- name: Setup
|
||||||
run: |
|
run: |
|
||||||
apt-get update
|
apt-get update
|
||||||
@ -34,10 +41,8 @@ jobs:
|
|||||||
- name: Setup Compiler
|
- name: Setup Compiler
|
||||||
if: matrix.config.compiler == 'clang'
|
if: matrix.config.compiler == 'clang'
|
||||||
run: |
|
run: |
|
||||||
if [[ "${{ matrix.config.version }}" -ge 4 ]]; then
|
|
||||||
scripts/ci_setup_clang.sh "${{ matrix.config.version }}"
|
scripts/ci_setup_clang.sh "${{ matrix.config.version }}"
|
||||||
echo "CXXFLAGS=-stdlib=libc++" >> $GITHUB_ENV
|
echo "CXXFLAGS=-stdlib=libc++" >> $GITHUB_ENV
|
||||||
fi
|
|
||||||
echo "CC=clang-${{ matrix.config.version }}" >> $GITHUB_ENV
|
echo "CC=clang-${{ matrix.config.version }}" >> $GITHUB_ENV
|
||||||
echo "CXX=clang++-${{ matrix.config.version }}" >> $GITHUB_ENV
|
echo "CXX=clang++-${{ matrix.config.version }}" >> $GITHUB_ENV
|
||||||
- name: Build
|
- name: Build
|
||||||
@ -52,15 +57,19 @@ jobs:
|
|||||||
-DSPDLOG_BUILD_BENCH=OFF \
|
-DSPDLOG_BUILD_BENCH=OFF \
|
||||||
-DSPDLOG_BUILD_TESTS=ON \
|
-DSPDLOG_BUILD_TESTS=ON \
|
||||||
-DSPDLOG_BUILD_TESTS_HO=OFF \
|
-DSPDLOG_BUILD_TESTS_HO=OFF \
|
||||||
-DSPDLOG_SANITIZE_ADDRESS=${{ matrix.config.asan || 'ON' }}
|
-DSPDLOG_SANITIZE_ADDRESS=${{ matrix.config.asan || 'OFF' }} \
|
||||||
make -j2
|
-DSPDLOG_SANITIZE_THREAD=${{ matrix.config.tsan || 'OFF' }}
|
||||||
ctest -j2 --output-on-failure
|
make -j 4
|
||||||
|
ctest -j 4 --output-on-failure
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# OS X build matrix
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
build_osx:
|
build_osx:
|
||||||
runs-on: macOS-latest
|
runs-on: macOS-latest
|
||||||
name: "OS X Clang (C++11, Release)"
|
name: "OS X Clang (C++11, Release)"
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@main
|
- uses: actions/checkout@v4
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
mkdir -p build && cd build
|
mkdir -p build && cd build
|
||||||
@ -74,6 +83,5 @@ jobs:
|
|||||||
-DSPDLOG_BUILD_TESTS=ON \
|
-DSPDLOG_BUILD_TESTS=ON \
|
||||||
-DSPDLOG_BUILD_TESTS_HO=OFF \
|
-DSPDLOG_BUILD_TESTS_HO=OFF \
|
||||||
-DSPDLOG_SANITIZE_ADDRESS=OFF
|
-DSPDLOG_SANITIZE_ADDRESS=OFF
|
||||||
make -j2
|
make -j 4
|
||||||
ctest -j2 --output-on-failure
|
ctest -j 4 --output-on-failure
|
||||||
|
|
38
.github/workflows/macos.yml
vendored
Normal file
38
.github/workflows/macos.yml
vendored
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
name: macos
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: macOS-latest
|
||||||
|
name: "macOS Clang (C++11, Release)"
|
||||||
|
strategy:
|
||||||
|
fail-fast: true
|
||||||
|
matrix:
|
||||||
|
config:
|
||||||
|
- USE_STD_FORMAT: 'ON'
|
||||||
|
BUILD_EXAMPLE: 'OFF'
|
||||||
|
- USE_STD_FORMAT: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'ON'
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Build
|
||||||
|
run: |
|
||||||
|
mkdir -p build && cd build
|
||||||
|
cmake .. \
|
||||||
|
-DCMAKE_BUILD_TYPE=Release \
|
||||||
|
-DCMAKE_CXX_STANDARD=11 \
|
||||||
|
-DSPDLOG_BUILD_EXAMPLE=${{ matrix.config.BUILD_EXAMPLE }} \
|
||||||
|
-DSPDLOG_BUILD_EXAMPLE_HO=${{ matrix.config.BUILD_EXAMPLE }} \
|
||||||
|
-DSPDLOG_BUILD_WARNINGS=ON \
|
||||||
|
-DSPDLOG_BUILD_BENCH=OFF \
|
||||||
|
-DSPDLOG_BUILD_TESTS=ON \
|
||||||
|
-DSPDLOG_BUILD_TESTS_HO=OFF \
|
||||||
|
-DSPDLOG_USE_STD_FORMAT=${{ matrix.config.USE_STD_FORMAT }} \
|
||||||
|
-DSPDLOG_SANITIZE_ADDRESS=OFF
|
||||||
|
make -j 4
|
||||||
|
ctest -j 4 --output-on-failure
|
148
.github/workflows/windows.yml
vendored
Normal file
148
.github/workflows/windows.yml
vendored
Normal file
@ -0,0 +1,148 @@
|
|||||||
|
name: windows
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: windows-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: true
|
||||||
|
matrix:
|
||||||
|
config:
|
||||||
|
- GENERATOR: "Visual Studio 17 2022"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'OFF'
|
||||||
|
WCHAR_FILES: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'OFF'
|
||||||
|
USE_STD_FORMAT: 'ON'
|
||||||
|
CXX_STANDARD: 20
|
||||||
|
- GENERATOR: "Visual Studio 17 2022"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'ON'
|
||||||
|
WCHAR_FILES: 'ON'
|
||||||
|
BUILD_EXAMPLE: 'OFF'
|
||||||
|
USE_STD_FORMAT: 'ON'
|
||||||
|
CXX_STANDARD: 20
|
||||||
|
- GENERATOR: "Visual Studio 17 2022"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'OFF'
|
||||||
|
WCHAR_FILES: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'ON'
|
||||||
|
USE_STD_FORMAT: 'OFF'
|
||||||
|
CXX_STANDARD: 17
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: CMake ${{ matrix.config.GENERATOR }} CXX=${{matrix.config.CXX_STANDARD}} WCHAR=${{matrix.config.WCHAR_FILES}} STD_FORMAT=${{matrix.config.USE_STD_FORMAT}}
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake -G "${{ matrix.config.GENERATOR }}" -A x64 `
|
||||||
|
-D CMAKE_BUILD_TYPE=${{ matrix.config.BUILD_TYPE }} `
|
||||||
|
-D BUILD_SHARED_LIBS=${{ matrix.config.BUILD_SHARED }} `
|
||||||
|
-D SPDLOG_WCHAR_SUPPORT=${{ matrix.config.WCHAR }} `
|
||||||
|
-D SPDLOG_WCHAR_FILENAMES=${{ matrix.config.WCHAR_FILES }} `
|
||||||
|
-D SPDLOG_BUILD_EXAMPLE=${{ matrix.config.BUILD_EXAMPLE }} `
|
||||||
|
-D SPDLOG_BUILD_EXAMPLE_HO=${{ matrix.config.BUILD_EXAMPLE }} `
|
||||||
|
-D SPDLOG_BUILD_TESTS=ON `
|
||||||
|
-D SPDLOG_BUILD_TESTS_HO=OFF `
|
||||||
|
-D SPDLOG_BUILD_WARNINGS=${{ matrix.config.FATAL_ERRORS }} `
|
||||||
|
-D SPDLOG_USE_STD_FORMAT=${{ matrix.config.USE_STD_FORMAT }} `
|
||||||
|
-D CMAKE_CXX_STANDARD=${{ matrix.config.CXX_STANDARD }} ..
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
cd build
|
||||||
|
cmake --build . --parallel --config ${{ matrix.config.BUILD_TYPE }}
|
||||||
|
|
||||||
|
- name: Run Tests
|
||||||
|
shell: pwsh
|
||||||
|
env:
|
||||||
|
PATH: ${{ env.PATH }};${{ github.workspace }}\build\_deps\catch2-build\src\${{ matrix.config.BUILD_TYPE }};${{ github.workspace }}\build\${{ matrix.config.BUILD_TYPE }}
|
||||||
|
run: |
|
||||||
|
build\tests\${{ matrix.config.BUILD_TYPE }}\spdlog-utests.exe
|
||||||
|
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
# MSVC 2019 build matrix
|
||||||
|
# -----------------------------------------------------------------------
|
||||||
|
build_2019:
|
||||||
|
runs-on: windows-2019
|
||||||
|
strategy:
|
||||||
|
fail-fast: true
|
||||||
|
matrix:
|
||||||
|
config:
|
||||||
|
- GENERATOR: "Visual Studio 16 2019"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'OFF'
|
||||||
|
WCHAR_FILES: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'ON'
|
||||||
|
USE_STD_FORMAT: 'OFF'
|
||||||
|
CXX_STANDARD: 17
|
||||||
|
- GENERATOR: "Visual Studio 16 2019"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'OFF'
|
||||||
|
WCHAR_FILES: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'ON'
|
||||||
|
USE_STD_FORMAT: 'OFF'
|
||||||
|
CXX_STANDARD: 14
|
||||||
|
- GENERATOR: "Visual Studio 16 2019"
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
BUILD_SHARED: 'ON'
|
||||||
|
FATAL_ERRORS: 'ON'
|
||||||
|
WCHAR: 'OFF'
|
||||||
|
WCHAR_FILES: 'OFF'
|
||||||
|
BUILD_EXAMPLE: 'ON'
|
||||||
|
USE_STD_FORMAT: 'OFF'
|
||||||
|
CXX_STANDARD: 11
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: CMake ${{ matrix.config.GENERATOR }} CXX=${{matrix.config.CXX_STANDARD}} WCHAR=${{matrix.config.WCHAR_FILES}} STD_FORMAT=${{matrix.config.USE_STD_FORMAT}}
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
mkdir build
|
||||||
|
cd build
|
||||||
|
cmake -G "${{ matrix.config.GENERATOR }}" -A x64 `
|
||||||
|
-D CMAKE_BUILD_TYPE=${{ matrix.config.BUILD_TYPE }} `
|
||||||
|
-D BUILD_SHARED_LIBS=${{ matrix.config.BUILD_SHARED }} `
|
||||||
|
-D SPDLOG_WCHAR_SUPPORT=${{ matrix.config.WCHAR }} `
|
||||||
|
-D SPDLOG_WCHAR_FILENAMES=${{ matrix.config.WCHAR_FILES }} `
|
||||||
|
-D SPDLOG_BUILD_EXAMPLE=${{ matrix.config.BUILD_EXAMPLE }} `
|
||||||
|
-D SPDLOG_BUILD_EXAMPLE_HO=${{ matrix.config.BUILD_EXAMPLE }} `
|
||||||
|
-D SPDLOG_BUILD_TESTS=ON `
|
||||||
|
-D SPDLOG_BUILD_TESTS_HO=OFF `
|
||||||
|
-D SPDLOG_BUILD_WARNINGS=${{ matrix.config.FATAL_ERRORS }} `
|
||||||
|
-D SPDLOG_USE_STD_FORMAT=${{ matrix.config.USE_STD_FORMAT }} `
|
||||||
|
-D CMAKE_CXX_STANDARD=${{ matrix.config.CXX_STANDARD }} ..
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
cd build
|
||||||
|
cmake --build . --parallel --config ${{ matrix.config.BUILD_TYPE }}
|
||||||
|
|
||||||
|
- name: Run Tests
|
||||||
|
shell: pwsh
|
||||||
|
env:
|
||||||
|
PATH: ${{ env.PATH }};${{ github.workspace }}\build\_deps\catch2-build\src\${{ matrix.config.BUILD_TYPE }};${{ github.workspace }}\build\${{ matrix.config.BUILD_TYPE }}
|
||||||
|
run: |
|
||||||
|
build\tests\${{ matrix.config.BUILD_TYPE }}\spdlog-utests.exe
|
219
CMakeLists.txt
219
CMakeLists.txt
@ -18,43 +18,39 @@ include(GNUInstallDirs)
|
|||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Set default build to release
|
# Set default build to release
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
|
if (NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES)
|
||||||
set(CMAKE_BUILD_TYPE "Release" CACHE STRING "Choose Release or Debug" FORCE)
|
set(CMAKE_BUILD_TYPE "Release" CACHE STRING "Choose Release or Debug" FORCE)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Compiler config
|
# Compiler config
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(SPDLOG_USE_STD_FORMAT)
|
if (SPDLOG_USE_STD_FORMAT)
|
||||||
set(CMAKE_CXX_STANDARD 20)
|
set(CMAKE_CXX_STANDARD 20)
|
||||||
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
elseif(NOT CMAKE_CXX_STANDARD)
|
elseif (NOT CMAKE_CXX_STANDARD)
|
||||||
set(CMAKE_CXX_STANDARD 11)
|
set(CMAKE_CXX_STANDARD 11)
|
||||||
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# make sure __cplusplus is defined when using msvc and enable parallel build
|
|
||||||
if(MSVC)
|
|
||||||
string(APPEND CMAKE_CXX_FLAGS " /Zc:__cplusplus /MP")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(CMAKE_CXX_EXTENSIONS OFF)
|
set(CMAKE_CXX_EXTENSIONS OFF)
|
||||||
|
|
||||||
if(CMAKE_SYSTEM_NAME MATCHES "CYGWIN" OR CMAKE_SYSTEM_NAME MATCHES "MSYS" OR CMAKE_SYSTEM_NAME MATCHES "MINGW")
|
if (CMAKE_SYSTEM_NAME MATCHES "CYGWIN" OR CMAKE_SYSTEM_NAME MATCHES "MSYS" OR CMAKE_SYSTEM_NAME MATCHES "MINGW")
|
||||||
set(CMAKE_CXX_EXTENSIONS ON)
|
set(CMAKE_CXX_EXTENSIONS ON)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Set SPDLOG_MASTER_PROJECT to ON if we are building spdlog
|
# Set SPDLOG_MASTER_PROJECT to ON if we are building spdlog
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Check if spdlog is being used directly or via add_subdirectory, but allow overriding
|
# Check if spdlog is being used directly or via add_subdirectory, but allow overriding
|
||||||
if(NOT DEFINED SPDLOG_MASTER_PROJECT)
|
if (NOT DEFINED SPDLOG_MASTER_PROJECT)
|
||||||
if(CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR)
|
if (CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR)
|
||||||
set(SPDLOG_MASTER_PROJECT ON)
|
set(SPDLOG_MASTER_PROJECT ON)
|
||||||
else()
|
else ()
|
||||||
set(SPDLOG_MASTER_PROJECT OFF)
|
set(SPDLOG_MASTER_PROJECT OFF)
|
||||||
endif()
|
endif ()
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
option(SPDLOG_BUILD_ALL "Build all artifacts" OFF)
|
option(SPDLOG_BUILD_ALL "Build all artifacts" OFF)
|
||||||
|
|
||||||
@ -80,6 +76,10 @@ option(SPDLOG_BUILD_BENCH "Build benchmarks (Requires https://github.com/google/
|
|||||||
|
|
||||||
# sanitizer options
|
# sanitizer options
|
||||||
option(SPDLOG_SANITIZE_ADDRESS "Enable address sanitizer in tests" OFF)
|
option(SPDLOG_SANITIZE_ADDRESS "Enable address sanitizer in tests" OFF)
|
||||||
|
option(SPDLOG_SANITIZE_THREAD "Enable thread sanitizer in tests" OFF)
|
||||||
|
if (SPDLOG_SANITIZE_ADDRESS AND SPDLOG_SANITIZE_THREAD)
|
||||||
|
message(FATAL_ERROR "SPDLOG_SANITIZE_ADDRESS and SPDLOG_SANITIZE_THREAD are mutually exclusive")
|
||||||
|
endif ()
|
||||||
|
|
||||||
# warning options
|
# warning options
|
||||||
option(SPDLOG_BUILD_WARNINGS "Enable compiler warnings" OFF)
|
option(SPDLOG_BUILD_WARNINGS "Enable compiler warnings" OFF)
|
||||||
@ -92,32 +92,38 @@ option(SPDLOG_FMT_EXTERNAL "Use external fmt library instead of bundled" OFF)
|
|||||||
option(SPDLOG_FMT_EXTERNAL_HO "Use external fmt header-only library instead of bundled" OFF)
|
option(SPDLOG_FMT_EXTERNAL_HO "Use external fmt header-only library instead of bundled" OFF)
|
||||||
option(SPDLOG_NO_EXCEPTIONS "Compile with -fno-exceptions. Call abort() on any spdlog exceptions" OFF)
|
option(SPDLOG_NO_EXCEPTIONS "Compile with -fno-exceptions. Call abort() on any spdlog exceptions" OFF)
|
||||||
|
|
||||||
if(SPDLOG_FMT_EXTERNAL AND SPDLOG_FMT_EXTERNAL_HO)
|
if (SPDLOG_FMT_EXTERNAL AND SPDLOG_FMT_EXTERNAL_HO)
|
||||||
message(FATAL_ERROR "SPDLOG_FMT_EXTERNAL and SPDLOG_FMT_EXTERNAL_HO are mutually exclusive")
|
message(FATAL_ERROR "SPDLOG_FMT_EXTERNAL and SPDLOG_FMT_EXTERNAL_HO are mutually exclusive")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_USE_STD_FORMAT AND SPDLOG_FMT_EXTERNAL_HO)
|
if (SPDLOG_USE_STD_FORMAT AND SPDLOG_FMT_EXTERNAL_HO)
|
||||||
message(FATAL_ERROR "SPDLOG_USE_STD_FORMAT and SPDLOG_FMT_EXTERNAL_HO are mutually exclusive")
|
message(FATAL_ERROR "SPDLOG_USE_STD_FORMAT and SPDLOG_FMT_EXTERNAL_HO are mutually exclusive")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_USE_STD_FORMAT AND SPDLOG_FMT_EXTERNAL)
|
if (SPDLOG_USE_STD_FORMAT AND SPDLOG_FMT_EXTERNAL)
|
||||||
message(FATAL_ERROR "SPDLOG_USE_STD_FORMAT and SPDLOG_FMT_EXTERNAL are mutually exclusive")
|
message(FATAL_ERROR "SPDLOG_USE_STD_FORMAT and SPDLOG_FMT_EXTERNAL are mutually exclusive")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# misc tweakme options
|
# misc tweakme options
|
||||||
if(WIN32)
|
if (WIN32)
|
||||||
option(SPDLOG_WCHAR_SUPPORT "Support wchar api" OFF)
|
option(SPDLOG_WCHAR_SUPPORT "Support wchar api" OFF)
|
||||||
option(SPDLOG_WCHAR_FILENAMES "Support wchar filenames" OFF)
|
option(SPDLOG_WCHAR_FILENAMES "Support wchar filenames" OFF)
|
||||||
else()
|
option(SPDLOG_WCHAR_CONSOLE "Support wchar output to console" OFF)
|
||||||
|
else ()
|
||||||
set(SPDLOG_WCHAR_SUPPORT OFF CACHE BOOL "non supported option" FORCE)
|
set(SPDLOG_WCHAR_SUPPORT OFF CACHE BOOL "non supported option" FORCE)
|
||||||
set(SPDLOG_WCHAR_FILENAMES OFF CACHE BOOL "non supported option" FORCE)
|
set(SPDLOG_WCHAR_FILENAMES OFF CACHE BOOL "non supported option" FORCE)
|
||||||
endif()
|
set(SPDLOG_WCHAR_CONSOLE OFF CACHE BOOL "non supported option" FORCE)
|
||||||
|
endif ()
|
||||||
|
|
||||||
if(${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
|
if (MSVC)
|
||||||
|
option(SPDLOG_MSVC_UTF8 "Enable/disable msvc /utf-8 flag required by fmt lib" ON)
|
||||||
|
endif ()
|
||||||
|
|
||||||
|
if (${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
|
||||||
option(SPDLOG_CLOCK_COARSE "Use CLOCK_REALTIME_COARSE instead of the regular clock," OFF)
|
option(SPDLOG_CLOCK_COARSE "Use CLOCK_REALTIME_COARSE instead of the regular clock," OFF)
|
||||||
else()
|
else ()
|
||||||
set(SPDLOG_CLOCK_COARSE OFF CACHE BOOL "non supported option" FORCE)
|
set(SPDLOG_CLOCK_COARSE OFF CACHE BOOL "non supported option" FORCE)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
option(SPDLOG_PREVENT_CHILD_FD "Prevent from child processes to inherit log file descriptors" OFF)
|
option(SPDLOG_PREVENT_CHILD_FD "Prevent from child processes to inherit log file descriptors" OFF)
|
||||||
option(SPDLOG_NO_THREAD_ID "prevent spdlog from querying the thread id on each log call if thread id is not needed" OFF)
|
option(SPDLOG_NO_THREAD_ID "prevent spdlog from querying the thread id on each log call if thread id is not needed" OFF)
|
||||||
@ -127,19 +133,20 @@ option(
|
|||||||
"prevent spdlog from using of std::atomic log levels (use only if your code never modifies log levels concurrently"
|
"prevent spdlog from using of std::atomic log levels (use only if your code never modifies log levels concurrently"
|
||||||
OFF)
|
OFF)
|
||||||
option(SPDLOG_DISABLE_DEFAULT_LOGGER "Disable default logger creation" OFF)
|
option(SPDLOG_DISABLE_DEFAULT_LOGGER "Disable default logger creation" OFF)
|
||||||
|
option(SPDLOG_FWRITE_UNLOCKED "Use the unlocked variant of fwrite. Leave this on unless your libc doesn't have it" ON)
|
||||||
|
|
||||||
# clang-tidy
|
# clang-tidy
|
||||||
option(SPDLOG_TIDY "run clang-tidy" OFF)
|
option(SPDLOG_TIDY "run clang-tidy" OFF)
|
||||||
|
|
||||||
if(SPDLOG_TIDY)
|
if (SPDLOG_TIDY)
|
||||||
set(CMAKE_CXX_CLANG_TIDY "clang-tidy")
|
set(CMAKE_CXX_CLANG_TIDY "clang-tidy")
|
||||||
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
|
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
|
||||||
message(STATUS "Enabled clang-tidy")
|
message(STATUS "Enabled clang-tidy")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_BUILD_PIC)
|
if (SPDLOG_BUILD_PIC)
|
||||||
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
find_package(Threads REQUIRED)
|
find_package(Threads REQUIRED)
|
||||||
message(STATUS "Build type: " ${CMAKE_BUILD_TYPE})
|
message(STATUS "Build type: " ${CMAKE_BUILD_TYPE})
|
||||||
@ -148,34 +155,34 @@ message(STATUS "Build type: " ${CMAKE_BUILD_TYPE})
|
|||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
set(SPDLOG_SRCS src/spdlog.cpp src/stdout_sinks.cpp src/color_sinks.cpp src/file_sinks.cpp src/async.cpp src/cfg.cpp)
|
set(SPDLOG_SRCS src/spdlog.cpp src/stdout_sinks.cpp src/color_sinks.cpp src/file_sinks.cpp src/async.cpp src/cfg.cpp)
|
||||||
|
|
||||||
if(NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
if (NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
||||||
list(APPEND SPDLOG_SRCS src/bundled_fmtlib_format.cpp)
|
list(APPEND SPDLOG_SRCS src/bundled_fmtlib_format.cpp)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_BUILD_SHARED OR BUILD_SHARED_LIBS)
|
if (SPDLOG_BUILD_SHARED OR BUILD_SHARED_LIBS)
|
||||||
if(WIN32)
|
if (WIN32)
|
||||||
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmake/version.rc.in ${CMAKE_CURRENT_BINARY_DIR}/version.rc @ONLY)
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmake/version.rc.in ${CMAKE_CURRENT_BINARY_DIR}/version.rc @ONLY)
|
||||||
list(APPEND SPDLOG_SRCS ${CMAKE_CURRENT_BINARY_DIR}/version.rc)
|
list(APPEND SPDLOG_SRCS ${CMAKE_CURRENT_BINARY_DIR}/version.rc)
|
||||||
endif()
|
endif ()
|
||||||
add_library(spdlog SHARED ${SPDLOG_SRCS} ${SPDLOG_ALL_HEADERS})
|
add_library(spdlog SHARED ${SPDLOG_SRCS} ${SPDLOG_ALL_HEADERS})
|
||||||
target_compile_definitions(spdlog PUBLIC SPDLOG_SHARED_LIB)
|
target_compile_definitions(spdlog PUBLIC SPDLOG_SHARED_LIB)
|
||||||
if(MSVC)
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "MSVC")
|
||||||
target_compile_options(spdlog PUBLIC $<$<AND:$<CXX_COMPILER_ID:MSVC>,$<NOT:$<COMPILE_LANGUAGE:CUDA>>>:/wd4251
|
target_compile_options(spdlog PUBLIC $<$<AND:$<CXX_COMPILER_ID:MSVC>,$<NOT:$<COMPILE_LANGUAGE:CUDA>>>:/wd4251
|
||||||
/wd4275>)
|
/wd4275>)
|
||||||
endif()
|
endif ()
|
||||||
if(NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
if (NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
||||||
target_compile_definitions(spdlog PRIVATE FMT_LIB_EXPORT PUBLIC FMT_SHARED)
|
target_compile_definitions(spdlog PRIVATE FMT_LIB_EXPORT PUBLIC FMT_SHARED)
|
||||||
endif()
|
endif ()
|
||||||
else()
|
else ()
|
||||||
add_library(spdlog STATIC ${SPDLOG_SRCS} ${SPDLOG_ALL_HEADERS})
|
add_library(spdlog STATIC ${SPDLOG_SRCS} ${SPDLOG_ALL_HEADERS})
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
add_library(spdlog::spdlog ALIAS spdlog)
|
add_library(spdlog::spdlog ALIAS spdlog)
|
||||||
|
|
||||||
set(SPDLOG_INCLUDES_LEVEL "")
|
set(SPDLOG_INCLUDES_LEVEL "")
|
||||||
if(SPDLOG_SYSTEM_INCLUDES)
|
if (SPDLOG_SYSTEM_INCLUDES)
|
||||||
set(SPDLOG_INCLUDES_LEVEL "SYSTEM")
|
set(SPDLOG_INCLUDES_LEVEL "SYSTEM")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
target_compile_definitions(spdlog PUBLIC SPDLOG_COMPILED_LIB)
|
target_compile_definitions(spdlog PUBLIC SPDLOG_COMPILED_LIB)
|
||||||
target_include_directories(spdlog ${SPDLOG_INCLUDES_LEVEL} PUBLIC "$<BUILD_INTERFACE:${CMAKE_CURRENT_LIST_DIR}/include>"
|
target_include_directories(spdlog ${SPDLOG_INCLUDES_LEVEL} PUBLIC "$<BUILD_INTERFACE:${CMAKE_CURRENT_LIST_DIR}/include>"
|
||||||
@ -187,10 +194,17 @@ set_target_properties(spdlog PROPERTIES VERSION ${SPDLOG_VERSION} SOVERSION
|
|||||||
${SPDLOG_VERSION_MAJOR}.${SPDLOG_VERSION_MINOR})
|
${SPDLOG_VERSION_MAJOR}.${SPDLOG_VERSION_MINOR})
|
||||||
set_target_properties(spdlog PROPERTIES DEBUG_POSTFIX d)
|
set_target_properties(spdlog PROPERTIES DEBUG_POSTFIX d)
|
||||||
|
|
||||||
if(COMMAND target_precompile_headers AND SPDLOG_ENABLE_PCH)
|
if (COMMAND target_precompile_headers AND SPDLOG_ENABLE_PCH)
|
||||||
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmake/pch.h.in ${PROJECT_BINARY_DIR}/spdlog_pch.h @ONLY)
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmake/pch.h.in ${PROJECT_BINARY_DIR}/spdlog_pch.h @ONLY)
|
||||||
target_precompile_headers(spdlog PRIVATE ${PROJECT_BINARY_DIR}/spdlog_pch.h)
|
target_precompile_headers(spdlog PRIVATE ${PROJECT_BINARY_DIR}/spdlog_pch.h)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
|
# sanitizer support
|
||||||
|
if (SPDLOG_SANITIZE_ADDRESS)
|
||||||
|
spdlog_enable_addr_sanitizer(spdlog)
|
||||||
|
elseif (SPDLOG_SANITIZE_THREAD)
|
||||||
|
spdlog_enable_thread_sanitizer(spdlog)
|
||||||
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Header only version
|
# Header only version
|
||||||
@ -206,40 +220,58 @@ target_link_libraries(spdlog_header_only INTERFACE Threads::Threads)
|
|||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Use fmt package if using external fmt
|
# Use fmt package if using external fmt
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(SPDLOG_FMT_EXTERNAL OR SPDLOG_FMT_EXTERNAL_HO)
|
if (SPDLOG_FMT_EXTERNAL OR SPDLOG_FMT_EXTERNAL_HO)
|
||||||
if(NOT TARGET fmt::fmt)
|
if (NOT TARGET fmt::fmt)
|
||||||
find_package(fmt CONFIG REQUIRED)
|
find_package(fmt CONFIG REQUIRED)
|
||||||
endif()
|
endif ()
|
||||||
target_compile_definitions(spdlog PUBLIC SPDLOG_FMT_EXTERNAL)
|
target_compile_definitions(spdlog PUBLIC SPDLOG_FMT_EXTERNAL)
|
||||||
target_compile_definitions(spdlog_header_only INTERFACE SPDLOG_FMT_EXTERNAL)
|
target_compile_definitions(spdlog_header_only INTERFACE SPDLOG_FMT_EXTERNAL)
|
||||||
|
|
||||||
# use external fmt-header-nly
|
# use external fmt-header-only
|
||||||
if(SPDLOG_FMT_EXTERNAL_HO)
|
if (SPDLOG_FMT_EXTERNAL_HO)
|
||||||
target_link_libraries(spdlog PUBLIC fmt::fmt-header-only)
|
target_link_libraries(spdlog PUBLIC fmt::fmt-header-only)
|
||||||
target_link_libraries(spdlog_header_only INTERFACE fmt::fmt-header-only)
|
target_link_libraries(spdlog_header_only INTERFACE fmt::fmt-header-only)
|
||||||
else() # use external compile fmt
|
else () # use external compile fmt
|
||||||
target_link_libraries(spdlog PUBLIC fmt::fmt)
|
target_link_libraries(spdlog PUBLIC fmt::fmt)
|
||||||
target_link_libraries(spdlog_header_only INTERFACE fmt::fmt)
|
target_link_libraries(spdlog_header_only INTERFACE fmt::fmt)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
set(PKG_CONFIG_REQUIRES fmt) # add dependency to pkg-config
|
set(PKG_CONFIG_REQUIRES fmt) # add dependency to pkg-config
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
|
# ---------------------------------------------------------------------------------------
|
||||||
|
# Check if fwrite_unlocked/_fwrite_nolock is available
|
||||||
|
# ---------------------------------------------------------------------------------------
|
||||||
|
if (SPDLOG_FWRITE_UNLOCKED)
|
||||||
|
include(CheckSymbolExists)
|
||||||
|
if (WIN32)
|
||||||
|
check_symbol_exists(_fwrite_nolock "stdio.h" HAVE_FWRITE_UNLOCKED)
|
||||||
|
else ()
|
||||||
|
check_symbol_exists(fwrite_unlocked "stdio.h" HAVE_FWRITE_UNLOCKED)
|
||||||
|
endif ()
|
||||||
|
if (HAVE_FWRITE_UNLOCKED)
|
||||||
|
target_compile_definitions(spdlog PRIVATE SPDLOG_FWRITE_UNLOCKED)
|
||||||
|
target_compile_definitions(spdlog_header_only INTERFACE SPDLOG_FWRITE_UNLOCKED)
|
||||||
|
endif ()
|
||||||
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Add required libraries for Android CMake build
|
# Add required libraries for Android CMake build
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(ANDROID)
|
if (ANDROID)
|
||||||
target_link_libraries(spdlog PUBLIC log)
|
target_link_libraries(spdlog PUBLIC log)
|
||||||
target_link_libraries(spdlog_header_only INTERFACE log)
|
target_link_libraries(spdlog_header_only INTERFACE log)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Misc definitions according to tweak options
|
# Misc definitions according to tweak options
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
set(SPDLOG_WCHAR_TO_UTF8_SUPPORT ${SPDLOG_WCHAR_SUPPORT})
|
set(SPDLOG_WCHAR_TO_UTF8_SUPPORT ${SPDLOG_WCHAR_SUPPORT})
|
||||||
foreach(
|
set(SPDLOG_UTF8_TO_WCHAR_CONSOLE ${SPDLOG_WCHAR_CONSOLE})
|
||||||
|
foreach (
|
||||||
SPDLOG_OPTION
|
SPDLOG_OPTION
|
||||||
SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
||||||
|
SPDLOG_UTF8_TO_WCHAR_CONSOLE
|
||||||
SPDLOG_WCHAR_FILENAMES
|
SPDLOG_WCHAR_FILENAMES
|
||||||
SPDLOG_NO_EXCEPTIONS
|
SPDLOG_NO_EXCEPTIONS
|
||||||
SPDLOG_CLOCK_COARSE
|
SPDLOG_CLOCK_COARSE
|
||||||
@ -249,52 +281,65 @@ foreach(
|
|||||||
SPDLOG_NO_ATOMIC_LEVELS
|
SPDLOG_NO_ATOMIC_LEVELS
|
||||||
SPDLOG_DISABLE_DEFAULT_LOGGER
|
SPDLOG_DISABLE_DEFAULT_LOGGER
|
||||||
SPDLOG_USE_STD_FORMAT)
|
SPDLOG_USE_STD_FORMAT)
|
||||||
if(${SPDLOG_OPTION})
|
if (${SPDLOG_OPTION})
|
||||||
target_compile_definitions(spdlog PUBLIC ${SPDLOG_OPTION})
|
target_compile_definitions(spdlog PUBLIC ${SPDLOG_OPTION})
|
||||||
target_compile_definitions(spdlog_header_only INTERFACE ${SPDLOG_OPTION})
|
target_compile_definitions(spdlog_header_only INTERFACE ${SPDLOG_OPTION})
|
||||||
endif()
|
endif ()
|
||||||
endforeach()
|
endforeach ()
|
||||||
|
|
||||||
|
if (MSVC)
|
||||||
|
target_compile_options(spdlog PRIVATE "/Zc:__cplusplus")
|
||||||
|
target_compile_options(spdlog_header_only INTERFACE "/Zc:__cplusplus")
|
||||||
|
if (SPDLOG_MSVC_UTF8)
|
||||||
|
# fmtlib requires the /utf-8 flag when building with msvc.
|
||||||
|
# see https://github.com/fmtlib/fmt/pull/4159 on the purpose of the additional
|
||||||
|
# "$<$<AND:$<COMPILE_LANGUAGE:CXX>,$<CXX_COMPILER_ID:MSVC>>"
|
||||||
|
target_compile_options(spdlog PUBLIC $<$<AND:$<COMPILE_LANGUAGE:CXX>,$<CXX_COMPILER_ID:MSVC>>:/utf-8>)
|
||||||
|
target_compile_options(spdlog_header_only INTERFACE $<$<AND:$<COMPILE_LANGUAGE:CXX>,$<CXX_COMPILER_ID:MSVC>>:/utf-8>)
|
||||||
|
endif ()
|
||||||
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# If exceptions are disabled, disable them in the bundled fmt as well
|
# If exceptions are disabled, disable them in the bundled fmt as well
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(SPDLOG_NO_EXCEPTIONS)
|
if (SPDLOG_NO_EXCEPTIONS)
|
||||||
if(NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
if (NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
||||||
target_compile_definitions(spdlog PUBLIC FMT_EXCEPTIONS=0)
|
target_compile_definitions(spdlog PUBLIC FMT_USE_EXCEPTIONS=0)
|
||||||
endif()
|
endif ()
|
||||||
if(NOT MSVC)
|
if (NOT MSVC)
|
||||||
target_compile_options(spdlog PRIVATE -fno-exceptions)
|
target_compile_options(spdlog PRIVATE -fno-exceptions)
|
||||||
else()
|
else ()
|
||||||
target_compile_options(spdlog PRIVATE /EHs-c-)
|
target_compile_options(spdlog PRIVATE /EHs-c-)
|
||||||
endif()
|
target_compile_definitions(spdlog PRIVATE _HAS_EXCEPTIONS=0)
|
||||||
endif()
|
endif ()
|
||||||
|
endif ()
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Build binaries
|
# Build binaries
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(SPDLOG_BUILD_EXAMPLE OR SPDLOG_BUILD_EXAMPLE_HO OR SPDLOG_BUILD_ALL)
|
if (SPDLOG_BUILD_EXAMPLE OR SPDLOG_BUILD_EXAMPLE_HO OR SPDLOG_BUILD_ALL)
|
||||||
message(STATUS "Generating example(s)")
|
message(STATUS "Generating example(s)")
|
||||||
add_subdirectory(example)
|
add_subdirectory(example)
|
||||||
spdlog_enable_warnings(example)
|
spdlog_enable_warnings(example)
|
||||||
if(SPDLOG_BUILD_EXAMPLE_HO)
|
if (SPDLOG_BUILD_EXAMPLE_HO)
|
||||||
spdlog_enable_warnings(example_header_only)
|
spdlog_enable_warnings(example_header_only)
|
||||||
endif()
|
endif ()
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_BUILD_TESTS OR SPDLOG_BUILD_TESTS_HO OR SPDLOG_BUILD_ALL)
|
if (SPDLOG_BUILD_TESTS OR SPDLOG_BUILD_TESTS_HO OR SPDLOG_BUILD_ALL)
|
||||||
message(STATUS "Generating tests")
|
message(STATUS "Generating tests")
|
||||||
enable_testing()
|
enable_testing()
|
||||||
add_subdirectory(tests)
|
add_subdirectory(tests)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
if(SPDLOG_BUILD_BENCH OR SPDLOG_BUILD_ALL)
|
if (SPDLOG_BUILD_BENCH OR SPDLOG_BUILD_ALL)
|
||||||
message(STATUS "Generating benchmarks")
|
message(STATUS "Generating benchmarks")
|
||||||
add_subdirectory(bench)
|
add_subdirectory(bench)
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Install
|
# Install
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(SPDLOG_INSTALL)
|
if (SPDLOG_INSTALL)
|
||||||
message(STATUS "Generating install")
|
message(STATUS "Generating install")
|
||||||
set(project_config_in "${CMAKE_CURRENT_LIST_DIR}/cmake/spdlogConfig.cmake.in")
|
set(project_config_in "${CMAKE_CURRENT_LIST_DIR}/cmake/spdlogConfig.cmake.in")
|
||||||
set(project_config_out "${CMAKE_CURRENT_BINARY_DIR}/spdlogConfig.cmake")
|
set(project_config_out "${CMAKE_CURRENT_BINARY_DIR}/spdlogConfig.cmake")
|
||||||
@ -315,24 +360,24 @@ if(SPDLOG_INSTALL)
|
|||||||
ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||||||
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR})
|
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR})
|
||||||
|
|
||||||
if(NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
if (NOT SPDLOG_USE_STD_FORMAT AND NOT SPDLOG_FMT_EXTERNAL AND NOT SPDLOG_FMT_EXTERNAL_HO)
|
||||||
install(DIRECTORY include/${PROJECT_NAME}/fmt/bundled/
|
install(DIRECTORY include/${PROJECT_NAME}/fmt/bundled/
|
||||||
DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/fmt/bundled/")
|
DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/fmt/bundled/")
|
||||||
endif()
|
endif ()
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
# Install pkg-config file
|
# Install pkg-config file
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
if(IS_ABSOLUTE "${CMAKE_INSTALL_INCLUDEDIR}")
|
if (IS_ABSOLUTE "${CMAKE_INSTALL_INCLUDEDIR}")
|
||||||
set(PKG_CONFIG_INCLUDEDIR "${CMAKE_INSTALL_INCLUDEDIR}")
|
set(PKG_CONFIG_INCLUDEDIR "${CMAKE_INSTALL_INCLUDEDIR}")
|
||||||
else()
|
else ()
|
||||||
set(PKG_CONFIG_INCLUDEDIR "\${prefix}/${CMAKE_INSTALL_INCLUDEDIR}")
|
set(PKG_CONFIG_INCLUDEDIR "\${prefix}/${CMAKE_INSTALL_INCLUDEDIR}")
|
||||||
endif()
|
endif ()
|
||||||
if(IS_ABSOLUTE "${CMAKE_INSTALL_LIBDIR}")
|
if (IS_ABSOLUTE "${CMAKE_INSTALL_LIBDIR}")
|
||||||
set(PKG_CONFIG_LIBDIR "${CMAKE_INSTALL_LIBDIR}")
|
set(PKG_CONFIG_LIBDIR "${CMAKE_INSTALL_LIBDIR}")
|
||||||
else()
|
else ()
|
||||||
set(PKG_CONFIG_LIBDIR "\${exec_prefix}/${CMAKE_INSTALL_LIBDIR}")
|
set(PKG_CONFIG_LIBDIR "\${exec_prefix}/${CMAKE_INSTALL_LIBDIR}")
|
||||||
endif()
|
endif ()
|
||||||
get_target_property(PKG_CONFIG_DEFINES spdlog INTERFACE_COMPILE_DEFINITIONS)
|
get_target_property(PKG_CONFIG_DEFINES spdlog INTERFACE_COMPILE_DEFINITIONS)
|
||||||
string(REPLACE ";" " -D" PKG_CONFIG_DEFINES "${PKG_CONFIG_DEFINES}")
|
string(REPLACE ";" " -D" PKG_CONFIG_DEFINES "${PKG_CONFIG_DEFINES}")
|
||||||
string(CONCAT PKG_CONFIG_DEFINES "-D" "${PKG_CONFIG_DEFINES}")
|
string(CONCAT PKG_CONFIG_DEFINES "-D" "${PKG_CONFIG_DEFINES}")
|
||||||
@ -356,4 +401,4 @@ if(SPDLOG_INSTALL)
|
|||||||
# Support creation of installable packages
|
# Support creation of installable packages
|
||||||
# ---------------------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------------------
|
||||||
include(cmake/spdlogCPack.cmake)
|
include(cmake/spdlogCPack.cmake)
|
||||||
endif()
|
endif ()
|
||||||
|
32
README.md
32
README.md
@ -1,18 +1,25 @@
|
|||||||
# spdlog
|
# spdlog
|
||||||
|
|
||||||
Very fast, header-only/compiled, C++ logging library. [](https://github.com/gabime/spdlog/actions/workflows/ci.yml) [](https://ci.appveyor.com/project/gabime/spdlog) [](https://github.com/gabime/spdlog/releases/latest)
|
|
||||||
|
[](https://github.com/gabime/spdlog/actions/workflows/linux.yml)
|
||||||
|
[](https://github.com/gabime/spdlog/actions/workflows/windows.yml)
|
||||||
|
[](https://github.com/gabime/spdlog/actions/workflows/macos.yml)
|
||||||
|
[](https://ci.appveyor.com/project/gabime/spdlog) [](https://github.com/gabime/spdlog/releases/latest)
|
||||||
|
|
||||||
|
Fast C++ logging library
|
||||||
|
|
||||||
|
|
||||||
## Install
|
## Install
|
||||||
#### Header-only version
|
#### Header-only version
|
||||||
Copy the include [folder](https://github.com/gabime/spdlog/tree/v1.x/include/spdlog) to your build tree and use a C++11 compiler.
|
Copy the include [folder](include/spdlog) to your build tree and use a C++11 compiler.
|
||||||
|
|
||||||
#### Compiled version (recommended - much faster compile times)
|
#### Compiled version (recommended - much faster compile times)
|
||||||
```console
|
```console
|
||||||
$ git clone https://github.com/gabime/spdlog.git
|
$ git clone https://github.com/gabime/spdlog.git
|
||||||
$ cd spdlog && mkdir build && cd build
|
$ cd spdlog && mkdir build && cd build
|
||||||
$ cmake .. && make -j
|
$ cmake .. && cmake --build .
|
||||||
```
|
```
|
||||||
see example [CMakeLists.txt](https://github.com/gabime/spdlog/blob/v1.x/example/CMakeLists.txt) on how to use.
|
see example [CMakeLists.txt](example/CMakeLists.txt) on how to use.
|
||||||
|
|
||||||
## Platforms
|
## Platforms
|
||||||
* Linux, FreeBSD, OpenBSD, Solaris, AIX
|
* Linux, FreeBSD, OpenBSD, Solaris, AIX
|
||||||
@ -29,8 +36,9 @@ see example [CMakeLists.txt](https://github.com/gabime/spdlog/blob/v1.x/example/
|
|||||||
* Gentoo: `emerge dev-libs/spdlog`
|
* Gentoo: `emerge dev-libs/spdlog`
|
||||||
* Arch Linux: `pacman -S spdlog`
|
* Arch Linux: `pacman -S spdlog`
|
||||||
* openSUSE: `sudo zypper in spdlog-devel`
|
* openSUSE: `sudo zypper in spdlog-devel`
|
||||||
|
* ALT Linux: `apt-get install libspdlog-devel`
|
||||||
* vcpkg: `vcpkg install spdlog`
|
* vcpkg: `vcpkg install spdlog`
|
||||||
* conan: `spdlog/[>=1.4.1]`
|
* conan: `conan install --requires=spdlog/[*]`
|
||||||
* conda: `conda install -c conda-forge spdlog`
|
* conda: `conda install -c conda-forge spdlog`
|
||||||
* build2: ```depends: spdlog ^1.8.2```
|
* build2: ```depends: spdlog ^1.8.2```
|
||||||
|
|
||||||
@ -40,7 +48,7 @@ see example [CMakeLists.txt](https://github.com/gabime/spdlog/blob/v1.x/example/
|
|||||||
* Headers only or compiled
|
* Headers only or compiled
|
||||||
* Feature-rich formatting, using the excellent [fmt](https://github.com/fmtlib/fmt) library.
|
* Feature-rich formatting, using the excellent [fmt](https://github.com/fmtlib/fmt) library.
|
||||||
* Asynchronous mode (optional)
|
* Asynchronous mode (optional)
|
||||||
* [Custom](https://github.com/gabime/spdlog/wiki/3.-Custom-formatting) formatting.
|
* [Custom](https://github.com/gabime/spdlog/wiki/Custom-formatting) formatting.
|
||||||
* Multi/Single threaded loggers.
|
* Multi/Single threaded loggers.
|
||||||
* Various log targets:
|
* Various log targets:
|
||||||
* Rotating log files.
|
* Rotating log files.
|
||||||
@ -50,7 +58,7 @@ see example [CMakeLists.txt](https://github.com/gabime/spdlog/blob/v1.x/example/
|
|||||||
* Windows event log.
|
* Windows event log.
|
||||||
* Windows debugger (```OutputDebugString(..)```).
|
* Windows debugger (```OutputDebugString(..)```).
|
||||||
* Log to Qt widgets ([example](#log-to-qt-with-nice-colors)).
|
* Log to Qt widgets ([example](#log-to-qt-with-nice-colors)).
|
||||||
* Easily [extendable](https://github.com/gabime/spdlog/wiki/4.-Sinks#implementing-your-own-sink) with custom log targets.
|
* Easily [extendable](https://github.com/gabime/spdlog/wiki/Sinks#implementing-your-own-sink) with custom log targets.
|
||||||
* Log filtering - log levels can be modified at runtime as well as compile time.
|
* Log filtering - log levels can be modified at runtime as well as compile time.
|
||||||
* Support for loading log levels from argv or environment var.
|
* Support for loading log levels from argv or environment var.
|
||||||
* [Backtrace](#backtrace-support) support - store debug messages in a ring buffer and display them later on demand.
|
* [Backtrace](#backtrace-support) support - store debug messages in a ring buffer and display them later on demand.
|
||||||
@ -296,7 +304,7 @@ struct fmt::formatter<my_type> : fmt::formatter<std::string>
|
|||||||
{
|
{
|
||||||
auto format(my_type my, format_context &ctx) const -> decltype(ctx.out())
|
auto format(my_type my, format_context &ctx) const -> decltype(ctx.out())
|
||||||
{
|
{
|
||||||
return format_to(ctx.out(), "[my_type i={}]", my.i);
|
return fmt::format_to(ctx.out(), "[my_type i={}]", my.i);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -380,6 +388,9 @@ void android_example()
|
|||||||
int main (int argc, char *argv[])
|
int main (int argc, char *argv[])
|
||||||
{
|
{
|
||||||
spdlog::cfg::load_env_levels();
|
spdlog::cfg::load_env_levels();
|
||||||
|
// or specify the env variable name:
|
||||||
|
// MYAPP_LEVEL=info,mylogger=trace && ./example
|
||||||
|
// spdlog::cfg::load_env_levels("MYAPP_LEVEL");
|
||||||
// or from the command line:
|
// or from the command line:
|
||||||
// ./example SPDLOG_LEVEL=info,mylogger=trace
|
// ./example SPDLOG_LEVEL=info,mylogger=trace
|
||||||
// #include "spdlog/cfg/argv.h" // for loading levels from argv
|
// #include "spdlog/cfg/argv.h" // for loading levels from argv
|
||||||
@ -456,7 +467,7 @@ void mdc_example()
|
|||||||
---
|
---
|
||||||
## Benchmarks
|
## Benchmarks
|
||||||
|
|
||||||
Below are some [benchmarks](https://github.com/gabime/spdlog/blob/v1.x/bench/bench.cpp) done in Ubuntu 64 bit, Intel i7-4770 CPU @ 3.40GHz
|
Below are some [benchmarks](bench/bench.cpp) done in Ubuntu 64 bit, Intel i7-4770 CPU @ 3.40GHz
|
||||||
|
|
||||||
#### Synchronous mode
|
#### Synchronous mode
|
||||||
```
|
```
|
||||||
@ -508,7 +519,8 @@ Below are some [benchmarks](https://github.com/gabime/spdlog/blob/v1.x/bench/ben
|
|||||||
```
|
```
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
Documentation can be found in the [wiki](https://github.com/gabime/spdlog/wiki/1.-QuickStart) pages.
|
|
||||||
|
Documentation can be found in the [wiki](https://github.com/gabime/spdlog/wiki) pages.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
@ -160,7 +160,7 @@ void bench_mt(int howmany, std::shared_ptr<spdlog::logger> logger, int thread_co
|
|||||||
|
|
||||||
for (auto &t : threads) {
|
for (auto &t : threads) {
|
||||||
t.join();
|
t.join();
|
||||||
};
|
}
|
||||||
|
|
||||||
auto delta = high_resolution_clock::now() - start;
|
auto delta = high_resolution_clock::now() - start;
|
||||||
auto delta_d = duration_cast<duration<double>>(delta).count();
|
auto delta_d = duration_cast<duration<double>>(delta).count();
|
||||||
|
@ -181,7 +181,7 @@ void bench_mt(int howmany, std::shared_ptr<spdlog::logger> log, size_t thread_co
|
|||||||
|
|
||||||
for (auto &t : threads) {
|
for (auto &t : threads) {
|
||||||
t.join();
|
t.join();
|
||||||
};
|
}
|
||||||
|
|
||||||
auto delta = high_resolution_clock::now() - start;
|
auto delta = high_resolution_clock::now() - start;
|
||||||
auto delta_d = duration_cast<duration<double>>(delta).count();
|
auto delta_d = duration_cast<duration<double>>(delta).count();
|
||||||
|
@ -49,7 +49,7 @@ function(spdlog_enable_warnings target_name)
|
|||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
# Enable address sanitizer (gcc/clang only)
|
# Enable address sanitizer (gcc/clang only)
|
||||||
function(spdlog_enable_sanitizer target_name)
|
function(spdlog_enable_addr_sanitizer target_name)
|
||||||
if(NOT CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
if(NOT CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
||||||
message(FATAL_ERROR "Sanitizer supported only for gcc/clang")
|
message(FATAL_ERROR "Sanitizer supported only for gcc/clang")
|
||||||
endif()
|
endif()
|
||||||
@ -58,5 +58,16 @@ function(spdlog_enable_sanitizer target_name)
|
|||||||
target_compile_options(${target_name} PRIVATE -fno-sanitize=signed-integer-overflow)
|
target_compile_options(${target_name} PRIVATE -fno-sanitize=signed-integer-overflow)
|
||||||
target_compile_options(${target_name} PRIVATE -fno-sanitize-recover=all)
|
target_compile_options(${target_name} PRIVATE -fno-sanitize-recover=all)
|
||||||
target_compile_options(${target_name} PRIVATE -fno-omit-frame-pointer)
|
target_compile_options(${target_name} PRIVATE -fno-omit-frame-pointer)
|
||||||
target_link_libraries(${target_name} PRIVATE -fsanitize=address,undefined -fuse-ld=gold)
|
target_link_libraries(${target_name} PRIVATE -fsanitize=address,undefined)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
# Enable thread sanitizer (gcc/clang only)
|
||||||
|
function(spdlog_enable_thread_sanitizer target_name)
|
||||||
|
if(NOT CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
||||||
|
message(FATAL_ERROR "Sanitizer supported only for gcc/clang")
|
||||||
|
endif()
|
||||||
|
message(STATUS "Thread sanitizer enabled")
|
||||||
|
target_compile_options(${target_name} PRIVATE -fsanitize=thread)
|
||||||
|
target_compile_options(${target_name} PRIVATE -fno-omit-frame-pointer)
|
||||||
|
target_link_libraries(${target_name} PRIVATE -fsanitize=thread)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
@ -148,6 +148,9 @@ void load_levels_example() {
|
|||||||
// Set the log level to "info" and mylogger to "trace":
|
// Set the log level to "info" and mylogger to "trace":
|
||||||
// SPDLOG_LEVEL=info,mylogger=trace && ./example
|
// SPDLOG_LEVEL=info,mylogger=trace && ./example
|
||||||
spdlog::cfg::load_env_levels();
|
spdlog::cfg::load_env_levels();
|
||||||
|
// or specify the env variable name:
|
||||||
|
// MYAPP_LEVEL=info,mylogger=trace && ./example
|
||||||
|
// spdlog::cfg::load_env_levels("MYAPP_LEVEL");
|
||||||
// or from command line:
|
// or from command line:
|
||||||
// ./example SPDLOG_LEVEL=info,mylogger=trace
|
// ./example SPDLOG_LEVEL=info,mylogger=trace
|
||||||
// #include "spdlog/cfg/argv.h" // for loading levels from argv
|
// #include "spdlog/cfg/argv.h" // for loading levels from argv
|
||||||
@ -266,13 +269,13 @@ void multi_sink_example() {
|
|||||||
struct my_type {
|
struct my_type {
|
||||||
int i = 0;
|
int i = 0;
|
||||||
explicit my_type(int i)
|
explicit my_type(int i)
|
||||||
: i(i){};
|
: i(i){}
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef SPDLOG_USE_STD_FORMAT // when using fmtlib
|
#ifndef SPDLOG_USE_STD_FORMAT // when using fmtlib
|
||||||
template <>
|
template <>
|
||||||
struct fmt::formatter<my_type> : fmt::formatter<std::string> {
|
struct fmt::formatter<my_type> : fmt::formatter<std::string> {
|
||||||
auto format(my_type my, format_context &ctx) -> decltype(ctx.out()) {
|
auto format(my_type my, format_context &ctx) const -> decltype(ctx.out()) {
|
||||||
return fmt::format_to(ctx.out(), "[my_type i={}]", my.i);
|
return fmt::format_to(ctx.out(), "[my_type i={}]", my.i);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -281,7 +284,7 @@ struct fmt::formatter<my_type> : fmt::formatter<std::string> {
|
|||||||
template <>
|
template <>
|
||||||
struct std::formatter<my_type> : std::formatter<std::string> {
|
struct std::formatter<my_type> : std::formatter<std::string> {
|
||||||
auto format(my_type my, format_context &ctx) const -> decltype(ctx.out()) {
|
auto format(my_type my, format_context &ctx) const -> decltype(ctx.out()) {
|
||||||
return format_to(ctx.out(), "[my_type i={}]", my.i);
|
return std::format_to(ctx.out(), "[my_type i={}]", my.i);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
@ -382,7 +385,9 @@ void replace_default_logger_example() {
|
|||||||
// Mapped Diagnostic Context (MDC) is a map that stores key-value pairs (string values) in thread local storage.
|
// Mapped Diagnostic Context (MDC) is a map that stores key-value pairs (string values) in thread local storage.
|
||||||
// Each thread maintains its own MDC, which loggers use to append diagnostic information to log outputs.
|
// Each thread maintains its own MDC, which loggers use to append diagnostic information to log outputs.
|
||||||
// Note: it is not supported in asynchronous mode due to its reliance on thread-local storage.
|
// Note: it is not supported in asynchronous mode due to its reliance on thread-local storage.
|
||||||
#include "spdlog/mdc.h"
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
|
#include "spdlog/mdc.h"
|
||||||
void mdc_example()
|
void mdc_example()
|
||||||
{
|
{
|
||||||
spdlog::mdc::put("key1", "value1");
|
spdlog::mdc::put("key1", "value1");
|
||||||
@ -391,3 +396,8 @@ void mdc_example()
|
|||||||
spdlog::set_pattern("[%H:%M:%S %z] [%^%L%$] [%&] %v");
|
spdlog::set_pattern("[%H:%M:%S %z] [%^%L%$] [%&] %v");
|
||||||
spdlog::info("Some log message with context");
|
spdlog::info("Some log message with context");
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
void mdc_example() {
|
||||||
|
// if TLS feature is disabled
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
@ -43,15 +43,13 @@ SPDLOG_LOGGER_CATCH(msg.source)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// send flush request to the thread pool
|
// send flush request to the thread pool
|
||||||
SPDLOG_INLINE void spdlog::async_logger::flush_(){SPDLOG_TRY{auto pool_ptr = thread_pool_.lock();
|
SPDLOG_INLINE void spdlog::async_logger::flush_(){
|
||||||
if (!pool_ptr) {
|
SPDLOG_TRY{if (auto pool_ptr = thread_pool_.lock()){
|
||||||
|
pool_ptr->post_flush(shared_from_this(), overflow_policy_);
|
||||||
|
}
|
||||||
|
else {
|
||||||
throw_spdlog_ex("async flush: thread pool doesn't exist anymore");
|
throw_spdlog_ex("async flush: thread pool doesn't exist anymore");
|
||||||
}
|
}
|
||||||
|
|
||||||
std::future<void> future = pool_ptr->post_flush(shared_from_this(), overflow_policy_);
|
|
||||||
// Wait for the flush operation to complete.
|
|
||||||
// This might throw exception if the flush message get dropped because of overflow.
|
|
||||||
future.get();
|
|
||||||
}
|
}
|
||||||
SPDLOG_LOGGER_CATCH(source_loc())
|
SPDLOG_LOGGER_CATCH(source_loc())
|
||||||
}
|
}
|
||||||
|
@ -25,8 +25,8 @@
|
|||||||
|
|
||||||
namespace spdlog {
|
namespace spdlog {
|
||||||
namespace cfg {
|
namespace cfg {
|
||||||
inline void load_env_levels() {
|
inline void load_env_levels(const char* var = "SPDLOG_LEVEL") {
|
||||||
auto env_val = details::os::getenv("SPDLOG_LEVEL");
|
auto env_val = details::os::getenv(var);
|
||||||
if (!env_val.empty()) {
|
if (!env_val.empty()) {
|
||||||
helpers::load_levels(env_val);
|
helpers::load_levels(env_val);
|
||||||
}
|
}
|
||||||
|
@ -364,12 +364,7 @@ SPDLOG_CONSTEXPR_FUNC spdlog::wstring_view_t to_string_view(spdlog::wstring_view
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef SPDLOG_USE_STD_FORMAT
|
#if defined(SPDLOG_USE_STD_FORMAT) && __cpp_lib_format >= 202207L
|
||||||
template <typename T, typename... Args>
|
|
||||||
inline fmt::basic_string_view<T> to_string_view(fmt::basic_format_string<T, Args...> fmt) {
|
|
||||||
return fmt;
|
|
||||||
}
|
|
||||||
#elif __cpp_lib_format >= 202207L
|
|
||||||
template <typename T, typename... Args>
|
template <typename T, typename... Args>
|
||||||
SPDLOG_CONSTEXPR_FUNC std::basic_string_view<T> to_string_view(
|
SPDLOG_CONSTEXPR_FUNC std::basic_string_view<T> to_string_view(
|
||||||
std::basic_format_string<T, Args...> fmt) SPDLOG_NOEXCEPT {
|
std::basic_format_string<T, Args...> fmt) SPDLOG_NOEXCEPT {
|
||||||
|
@ -101,7 +101,8 @@ SPDLOG_INLINE void file_helper::write(const memory_buf_t &buf) {
|
|||||||
if (fd_ == nullptr) return;
|
if (fd_ == nullptr) return;
|
||||||
size_t msg_size = buf.size();
|
size_t msg_size = buf.size();
|
||||||
auto data = buf.data();
|
auto data = buf.data();
|
||||||
if (std::fwrite(data, 1, msg_size, fd_) != msg_size) {
|
|
||||||
|
if (!details::os::fwrite_bytes(data, msg_size, fd_)) {
|
||||||
throw_spdlog_ex("Failed writing to file " + os::filename_to_str(filename_), errno);
|
throw_spdlog_ex("Failed writing to file " + os::filename_to_str(filename_), errno);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -148,19 +148,19 @@ public:
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
size_t overrun_counter() {
|
size_t overrun_counter() {
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
std::lock_guard<std::mutex> lock(queue_mutex_);
|
||||||
return q_.overrun_counter();
|
return q_.overrun_counter();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t discard_counter() { return discard_counter_.load(std::memory_order_relaxed); }
|
size_t discard_counter() { return discard_counter_.load(std::memory_order_relaxed); }
|
||||||
|
|
||||||
size_t size() {
|
size_t size() {
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
std::lock_guard<std::mutex> lock(queue_mutex_);
|
||||||
return q_.size();
|
return q_.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
void reset_overrun_counter() {
|
void reset_overrun_counter() {
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
std::lock_guard<std::mutex> lock(queue_mutex_);
|
||||||
q_.reset_overrun_counter();
|
q_.reset_overrun_counter();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -267,7 +267,8 @@ SPDLOG_INLINE int utc_minutes_offset(const std::tm &tm) {
|
|||||||
|
|
||||||
#if defined(sun) || defined(__sun) || defined(_AIX) || \
|
#if defined(sun) || defined(__sun) || defined(_AIX) || \
|
||||||
(defined(__NEWLIB__) && !defined(__TM_GMTOFF)) || \
|
(defined(__NEWLIB__) && !defined(__TM_GMTOFF)) || \
|
||||||
(!defined(_BSD_SOURCE) && !defined(_GNU_SOURCE))
|
(!defined(__APPLE__) && !defined(_BSD_SOURCE) && !defined(_GNU_SOURCE) && \
|
||||||
|
(!defined(_POSIX_VERSION) || (_POSIX_VERSION < 202405L)))
|
||||||
// 'tm_gmtoff' field is BSD extension and it's missing on SunOS/Solaris
|
// 'tm_gmtoff' field is BSD extension and it's missing on SunOS/Solaris
|
||||||
struct helper {
|
struct helper {
|
||||||
static long int calculate_gmt_offset(const std::tm &localtm = details::os::localtime(),
|
static long int calculate_gmt_offset(const std::tm &localtm = details::os::localtime(),
|
||||||
@ -483,12 +484,12 @@ SPDLOG_INLINE void utf8_to_wstrbuf(string_view_t str, wmemory_buf_t &target) {
|
|||||||
|
|
||||||
// find the size to allocate for the result buffer
|
// find the size to allocate for the result buffer
|
||||||
int result_size =
|
int result_size =
|
||||||
::MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, str.data(), str_size, NULL, 0);
|
::MultiByteToWideChar(CP_UTF8, 0, str.data(), str_size, NULL, 0);
|
||||||
|
|
||||||
if (result_size > 0) {
|
if (result_size > 0) {
|
||||||
target.resize(result_size);
|
target.resize(result_size);
|
||||||
result_size = ::MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, str.data(), str_size,
|
result_size = ::MultiByteToWideChar(CP_UTF8, 0, str.data(), str_size, target.data(),
|
||||||
target.data(), result_size);
|
result_size);
|
||||||
if (result_size > 0) {
|
if (result_size > 0) {
|
||||||
assert(result_size == target.size());
|
assert(result_size == target.size());
|
||||||
return;
|
return;
|
||||||
@ -589,6 +590,18 @@ SPDLOG_INLINE bool fsync(FILE *fp) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Do non-locking fwrite if possible by the os or use the regular locking fwrite
|
||||||
|
// Return true on success.
|
||||||
|
SPDLOG_INLINE bool fwrite_bytes(const void *ptr, const size_t n_bytes, FILE *fp) {
|
||||||
|
#if defined(_WIN32) && defined(SPDLOG_FWRITE_UNLOCKED)
|
||||||
|
return _fwrite_nolock(ptr, 1, n_bytes, fp) == n_bytes;
|
||||||
|
#elif defined(SPDLOG_FWRITE_UNLOCKED)
|
||||||
|
return ::fwrite_unlocked(ptr, 1, n_bytes, fp) == n_bytes;
|
||||||
|
#else
|
||||||
|
return std::fwrite(ptr, 1, n_bytes, fp) == n_bytes;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace os
|
} // namespace os
|
||||||
} // namespace details
|
} // namespace details
|
||||||
} // namespace spdlog
|
} // namespace spdlog
|
||||||
|
@ -114,6 +114,10 @@ SPDLOG_API std::string getenv(const char *field);
|
|||||||
// Return true on success.
|
// Return true on success.
|
||||||
SPDLOG_API bool fsync(FILE *fp);
|
SPDLOG_API bool fsync(FILE *fp);
|
||||||
|
|
||||||
|
// Do non-locking fwrite if possible by the os or use the regular locking fwrite
|
||||||
|
// Return true on success.
|
||||||
|
SPDLOG_API bool fwrite_bytes(const void *ptr, const size_t n_bytes, FILE *fp);
|
||||||
|
|
||||||
} // namespace os
|
} // namespace os
|
||||||
} // namespace details
|
} // namespace details
|
||||||
} // namespace spdlog
|
} // namespace spdlog
|
||||||
|
@ -62,13 +62,9 @@ void SPDLOG_INLINE thread_pool::post_log(async_logger_ptr &&worker_ptr,
|
|||||||
post_async_msg_(std::move(async_m), overflow_policy);
|
post_async_msg_(std::move(async_m), overflow_policy);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::future<void> SPDLOG_INLINE thread_pool::post_flush(async_logger_ptr &&worker_ptr,
|
void SPDLOG_INLINE thread_pool::post_flush(async_logger_ptr &&worker_ptr,
|
||||||
async_overflow_policy overflow_policy) {
|
async_overflow_policy overflow_policy) {
|
||||||
std::promise<void> promise;
|
post_async_msg_(async_msg(std::move(worker_ptr), async_msg_type::flush), overflow_policy);
|
||||||
std::future<void> future = promise.get_future();
|
|
||||||
post_async_msg_(async_msg(std::move(worker_ptr), async_msg_type::flush, std::move(promise)),
|
|
||||||
overflow_policy);
|
|
||||||
return future;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t SPDLOG_INLINE thread_pool::overrun_counter() { return q_.overrun_counter(); }
|
size_t SPDLOG_INLINE thread_pool::overrun_counter() { return q_.overrun_counter(); }
|
||||||
@ -112,7 +108,6 @@ bool SPDLOG_INLINE thread_pool::process_next_msg_() {
|
|||||||
}
|
}
|
||||||
case async_msg_type::flush: {
|
case async_msg_type::flush: {
|
||||||
incoming_async_msg.worker_ptr->backend_flush_();
|
incoming_async_msg.worker_ptr->backend_flush_();
|
||||||
incoming_async_msg.flush_promise.set_value();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,7 +9,6 @@
|
|||||||
|
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <future>
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
@ -28,7 +27,6 @@ enum class async_msg_type { log, flush, terminate };
|
|||||||
struct async_msg : log_msg_buffer {
|
struct async_msg : log_msg_buffer {
|
||||||
async_msg_type msg_type{async_msg_type::log};
|
async_msg_type msg_type{async_msg_type::log};
|
||||||
async_logger_ptr worker_ptr;
|
async_logger_ptr worker_ptr;
|
||||||
std::promise<void> flush_promise;
|
|
||||||
|
|
||||||
async_msg() = default;
|
async_msg() = default;
|
||||||
~async_msg() = default;
|
~async_msg() = default;
|
||||||
@ -58,20 +56,12 @@ struct async_msg : log_msg_buffer {
|
|||||||
async_msg(async_logger_ptr &&worker, async_msg_type the_type, const details::log_msg &m)
|
async_msg(async_logger_ptr &&worker, async_msg_type the_type, const details::log_msg &m)
|
||||||
: log_msg_buffer{m},
|
: log_msg_buffer{m},
|
||||||
msg_type{the_type},
|
msg_type{the_type},
|
||||||
worker_ptr{std::move(worker)},
|
worker_ptr{std::move(worker)} {}
|
||||||
flush_promise{} {}
|
|
||||||
|
|
||||||
async_msg(async_logger_ptr &&worker, async_msg_type the_type)
|
async_msg(async_logger_ptr &&worker, async_msg_type the_type)
|
||||||
: log_msg_buffer{},
|
: log_msg_buffer{},
|
||||||
msg_type{the_type},
|
msg_type{the_type},
|
||||||
worker_ptr{std::move(worker)},
|
worker_ptr{std::move(worker)} {}
|
||||||
flush_promise{} {}
|
|
||||||
|
|
||||||
async_msg(async_logger_ptr &&worker, async_msg_type the_type, std::promise<void> &&promise)
|
|
||||||
: log_msg_buffer{},
|
|
||||||
msg_type{the_type},
|
|
||||||
worker_ptr{std::move(worker)},
|
|
||||||
flush_promise{std::move(promise)} {}
|
|
||||||
|
|
||||||
explicit async_msg(async_msg_type the_type)
|
explicit async_msg(async_msg_type the_type)
|
||||||
: async_msg{nullptr, the_type} {}
|
: async_msg{nullptr, the_type} {}
|
||||||
@ -98,8 +88,7 @@ public:
|
|||||||
void post_log(async_logger_ptr &&worker_ptr,
|
void post_log(async_logger_ptr &&worker_ptr,
|
||||||
const details::log_msg &msg,
|
const details::log_msg &msg,
|
||||||
async_overflow_policy overflow_policy);
|
async_overflow_policy overflow_policy);
|
||||||
std::future<void> post_flush(async_logger_ptr &&worker_ptr,
|
void post_flush(async_logger_ptr &&worker_ptr, async_overflow_policy overflow_policy);
|
||||||
async_overflow_policy overflow_policy);
|
|
||||||
size_t overrun_counter();
|
size_t overrun_counter();
|
||||||
void reset_overrun_counter();
|
void reset_overrun_counter();
|
||||||
size_t discard_counter();
|
size_t discard_counter();
|
||||||
|
@ -102,7 +102,7 @@ namespace
|
|||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
struct formatter<spdlog::details::dump_info<T>, char> {
|
struct formatter<spdlog::details::dump_info<T>, char> {
|
||||||
const char delimiter = ' ';
|
char delimiter = ' ';
|
||||||
bool put_newlines = true;
|
bool put_newlines = true;
|
||||||
bool put_delimiters = true;
|
bool put_delimiters = true;
|
||||||
bool use_uppercase = false;
|
bool use_uppercase = false;
|
||||||
|
@ -8,14 +8,15 @@
|
|||||||
#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 {};
|
||||||
@ -28,15 +29,18 @@ 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
|
||||||
|
// unit for placing vtable. So node is made a fake template.
|
||||||
|
template <typename = void> struct node {
|
||||||
virtual ~node() = default;
|
virtual ~node() = default;
|
||||||
std::unique_ptr<node<>> next;
|
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;
|
||||||
|
|
||||||
@ -62,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>
|
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 ||
|
||||||
@ -96,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>;
|
||||||
@ -111,79 +105,71 @@ class dynamic_format_arg_store
|
|||||||
|
|
||||||
friend class basic_format_args<Context>;
|
friend class basic_format_args<Context>;
|
||||||
|
|
||||||
auto get_types() const -> unsigned long long {
|
|
||||||
return detail::is_unpacked_bit | data_.size() |
|
|
||||||
(named_info_.empty()
|
|
||||||
? 0ULL
|
|
||||||
: static_cast<unsigned long long>(detail::has_named_args_bit));
|
|
||||||
}
|
|
||||||
|
|
||||||
auto data() const -> const basic_format_arg<Context>* {
|
auto data() const -> const basic_format_arg<Context>* {
|
||||||
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(
|
||||||
@ -193,9 +179,9 @@ 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) {
|
||||||
@ -203,31 +189,30 @@ class dynamic_format_arg_store
|
|||||||
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.
|
||||||
|
size_t size() const noexcept { return data_.size(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
2962
include/spdlog/fmt/bundled/base.h
Normal file
2962
include/spdlog/fmt/bundled/base.h
Normal file
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -227,7 +227,7 @@ struct color_type {
|
|||||||
};
|
};
|
||||||
} // 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 {
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR text_style(emphasis em = emphasis()) noexcept
|
FMT_CONSTEXPR text_style(emphasis em = emphasis()) noexcept
|
||||||
@ -239,7 +239,7 @@ class text_style {
|
|||||||
foreground_color = rhs.foreground_color;
|
foreground_color = rhs.foreground_color;
|
||||||
} else if (rhs.set_foreground_color) {
|
} else if (rhs.set_foreground_color) {
|
||||||
if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb)
|
if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb)
|
||||||
FMT_THROW(format_error("can't OR a terminal color"));
|
report_error("can't OR a terminal color");
|
||||||
foreground_color.value.rgb_color |= rhs.foreground_color.value.rgb_color;
|
foreground_color.value.rgb_color |= rhs.foreground_color.value.rgb_color;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -248,7 +248,7 @@ class text_style {
|
|||||||
background_color = rhs.background_color;
|
background_color = rhs.background_color;
|
||||||
} else if (rhs.set_background_color) {
|
} else if (rhs.set_background_color) {
|
||||||
if (!background_color.is_rgb || !rhs.background_color.is_rgb)
|
if (!background_color.is_rgb || !rhs.background_color.is_rgb)
|
||||||
FMT_THROW(format_error("can't OR a terminal color"));
|
report_error("can't OR a terminal color");
|
||||||
background_color.value.rgb_color |= rhs.background_color.value.rgb_color;
|
background_color.value.rgb_color |= rhs.background_color.value.rgb_color;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -310,13 +310,13 @@ class text_style {
|
|||||||
emphasis ems;
|
emphasis ems;
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Creates a text style from the foreground (text) color. */
|
/// Creates a text style from the foreground (text) color.
|
||||||
FMT_CONSTEXPR inline auto fg(detail::color_type foreground) noexcept
|
FMT_CONSTEXPR inline auto fg(detail::color_type foreground) noexcept
|
||||||
-> text_style {
|
-> text_style {
|
||||||
return text_style(true, foreground);
|
return text_style(true, foreground);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Creates a text style from the background color. */
|
/// Creates a text style from the background color.
|
||||||
FMT_CONSTEXPR inline auto bg(detail::color_type background) noexcept
|
FMT_CONSTEXPR inline auto bg(detail::color_type background) noexcept
|
||||||
-> text_style {
|
-> text_style {
|
||||||
return text_style(false, background);
|
return text_style(false, background);
|
||||||
@ -330,7 +330,7 @@ FMT_CONSTEXPR inline auto operator|(emphasis lhs, emphasis rhs) noexcept
|
|||||||
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.
|
||||||
@ -390,8 +390,8 @@ template <typename Char> struct ansi_color_escape {
|
|||||||
FMT_CONSTEXPR operator const Char*() const noexcept { return buffer; }
|
FMT_CONSTEXPR operator const Char*() const noexcept { return buffer; }
|
||||||
|
|
||||||
FMT_CONSTEXPR auto begin() const noexcept -> const Char* { return buffer; }
|
FMT_CONSTEXPR auto begin() const noexcept -> const Char* { return buffer; }
|
||||||
FMT_CONSTEXPR_CHAR_TRAITS auto end() const noexcept -> const Char* {
|
FMT_CONSTEXPR20 auto end() const noexcept -> const Char* {
|
||||||
return buffer + std::char_traits<Char>::length(buffer);
|
return buffer + basic_string_view<Char>(buffer).size();
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -412,13 +412,13 @@ template <typename Char> struct ansi_color_escape {
|
|||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_CONSTEXPR auto make_foreground_color(detail::color_type foreground) noexcept
|
FMT_CONSTEXPR auto make_foreground_color(color_type foreground) noexcept
|
||||||
-> ansi_color_escape<Char> {
|
-> 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 auto make_background_color(detail::color_type background) noexcept
|
FMT_CONSTEXPR auto make_background_color(color_type background) noexcept
|
||||||
-> ansi_color_escape<Char> {
|
-> ansi_color_escape<Char> {
|
||||||
return ansi_color_escape<Char>(background, "\x1b[48;2;");
|
return ansi_color_escape<Char>(background, "\x1b[48;2;");
|
||||||
}
|
}
|
||||||
@ -434,7 +434,7 @@ 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 : detail::view {
|
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) {}
|
styled_arg(const T& v, text_style s) : value(v), style(s) {}
|
||||||
@ -442,145 +442,115 @@ template <typename T> struct styled_arg : detail::view {
|
|||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
void vformat_to(buffer<Char>& buf, const text_style& ts,
|
void vformat_to(buffer<Char>& buf, const text_style& ts,
|
||||||
basic_string_view<Char> format_str,
|
basic_string_view<Char> fmt,
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) {
|
basic_format_args<buffered_context<Char>> args) {
|
||||||
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 = 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;
|
has_style = true;
|
||||||
auto foreground = detail::make_foreground_color<Char>(ts.get_foreground());
|
auto foreground = 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;
|
has_style = true;
|
||||||
auto background = detail::make_background_color<Char>(ts.get_background());
|
auto background = 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 (has_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, const 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, const text_style& ts, format_string<T...> fmt,
|
||||||
void print(std::FILE* f, const text_style& ts, const S& format_str,
|
T&&... args) {
|
||||||
const Args&... args) {
|
vprint(f, ts, fmt.str, vargs<T...>{{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(const 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(const text_style& ts, string_view fmt, format_args args)
|
||||||
inline auto 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);
|
||||||
-> std::basic_string<Char> {
|
|
||||||
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 auto format(const text_style& ts, const S& format_str,
|
inline auto format(const text_style& ts, format_string<T...> fmt, T&&... args)
|
||||||
const Args&... args) -> std::basic_string<Char> {
|
-> 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, const text_style& ts, string_view fmt,
|
||||||
FMT_ENABLE_IF(detail::is_output_iterator<OutputIt, Char>::value)>
|
format_args args) -> OutputIt {
|
||||||
auto vformat_to(OutputIt out, const text_style& ts,
|
auto&& buf = detail::get_buffer<char>(out);
|
||||||
basic_string_view<Char> format_str,
|
detail::vformat_to(buf, ts, fmt, args);
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args)
|
|
||||||
-> OutputIt {
|
|
||||||
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 <
|
inline auto format_to(OutputIt out, const text_style& ts,
|
||||||
typename OutputIt, typename S, typename... Args,
|
format_string<T...> fmt, T&&... args) -> OutputIt {
|
||||||
bool enable = detail::is_output_iterator<OutputIt, char_t<S>>::value &&
|
return vformat_to(out, ts, fmt.str, vargs<T...>{{args...}});
|
||||||
detail::is_string<S>::value>
|
|
||||||
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>
|
||||||
@ -589,47 +559,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)
|
||||||
|
@ -8,54 +8,39 @@
|
|||||||
#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 {
|
|
||||||
|
|
||||||
template <typename Char, typename InputIt>
|
|
||||||
FMT_CONSTEXPR inline auto copy_str(InputIt begin, InputIt end,
|
|
||||||
counting_iterator it) -> counting_iterator {
|
|
||||||
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 {};
|
FMT_EXPORT class compiled_string {};
|
||||||
|
|
||||||
template <typename S>
|
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> {};
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\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
|
||||||
|
|
||||||
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
|
||||||
template <typename Char, size_t N,
|
|
||||||
fmt::detail_exported::fixed_string<Char, N> Str>
|
|
||||||
struct udl_compiled_string : compiled_string {
|
|
||||||
using char_type = Char;
|
|
||||||
explicit constexpr operator basic_string_view<char_type>() const {
|
|
||||||
return {Str.data, N - 1};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
template <typename T, typename... Tail>
|
template <typename T, typename... Tail>
|
||||||
auto first(const T& value, const Tail&...) -> const T& {
|
auto first(const T& value, const Tail&...) -> const T& {
|
||||||
return value;
|
return value;
|
||||||
@ -75,6 +60,29 @@ 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 int get_arg_index_by_name(basic_string_view<Char> name,
|
||||||
type_list<Args...>) {
|
type_list<Args...>) {
|
||||||
@ -144,12 +152,13 @@ template <typename Char, typename T, int N> struct field {
|
|||||||
template <typename OutputIt, typename... Args>
|
template <typename OutputIt, typename... Args>
|
||||||
constexpr OutputIt format(OutputIt out, const Args&... args) const {
|
constexpr OutputIt format(OutputIt out, const Args&... args) const {
|
||||||
const T& arg = get_arg_checked<T, N>(args...);
|
const T& arg = get_arg_checked<T, N>(args...);
|
||||||
if constexpr (std::is_convertible_v<T, basic_string_view<Char>>) {
|
if constexpr (std::is_convertible<T, 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);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char, typename T, int N>
|
template <typename Char, typename T, int N>
|
||||||
@ -236,13 +245,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;
|
||||||
@ -274,6 +282,7 @@ 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 int on_auto() {
|
||||||
@ -281,25 +290,28 @@ template <typename Char> struct arg_id_handler {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
constexpr int on_index(int id) {
|
constexpr int on_index(int id) {
|
||||||
|
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 int on_name(basic_string_view<Char> id) {
|
||||||
|
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 +325,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 +344,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 +352,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 +374,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 +403,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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -445,7 +451,7 @@ constexpr FMT_INLINE OutputIt format_to(OutputIt out, const CompiledFormat& cf,
|
|||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... Args,
|
||||||
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 std::basic_string<typename S::char_type> format(const S&,
|
||||||
Args&&... args) {
|
Args&&... args) {
|
||||||
if constexpr (std::is_same<typename S::char_type, char>::value) {
|
if constexpr (std::is_same<typename S::char_type, char>::value) {
|
||||||
@ -472,7 +478,7 @@ FMT_INLINE std::basic_string<typename S::char_type> format(const S&,
|
|||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... Args,
|
template <typename OutputIt, typename S, typename... Args,
|
||||||
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 OutputIt format_to(OutputIt out, const S&, Args&&... args) {
|
||||||
constexpr auto compiled = detail::compile<Args...>(S());
|
constexpr auto compiled = detail::compile<Args...>(S());
|
||||||
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
if constexpr (std::is_same<remove_cvref_t<decltype(compiled)>,
|
||||||
@ -487,44 +493,42 @@ FMT_CONSTEXPR OutputIt format_to(OutputIt out, const S&, Args&&... args) {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... Args,
|
template <typename OutputIt, typename S, typename... Args,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
auto format_to_n(OutputIt out, size_t n, const S& format_str, Args&&... args)
|
auto format_to_n(OutputIt out, size_t n, const S& fmt, Args&&... 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);
|
||||||
fmt::format_to(std::back_inserter(buf), format_str,
|
fmt::format_to(std::back_inserter(buf), fmt, std::forward<Args>(args)...);
|
||||||
std::forward<Args>(args)...);
|
|
||||||
return {buf.out(), buf.count()};
|
return {buf.out(), buf.count()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... Args,
|
||||||
FMT_ENABLE_IF(detail::is_compiled_string<S>::value)>
|
FMT_ENABLE_IF(is_compiled_string<S>::value)>
|
||||||
FMT_CONSTEXPR20 auto formatted_size(const S& format_str, const Args&... args)
|
FMT_CONSTEXPR20 auto formatted_size(const S& fmt, const Args&... args)
|
||||||
-> size_t {
|
-> size_t {
|
||||||
return fmt::format_to(detail::counting_iterator(), format_str, args...)
|
auto buf = detail::counting_buffer<>();
|
||||||
.count();
|
fmt::format_to(appender(buf), fmt, args...);
|
||||||
|
return buf.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... Args,
|
||||||
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, const Args&... args) {
|
||||||
memory_buffer buffer;
|
auto buf = memory_buffer();
|
||||||
fmt::format_to(std::back_inserter(buffer), format_str, args...);
|
fmt::format_to(appender(buf), fmt, args...);
|
||||||
detail::print(f, {buffer.data(), buffer.size()});
|
detail::print(f, {buf.data(), buf.size()});
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename S, typename... Args,
|
template <typename S, typename... Args,
|
||||||
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, const Args&... args) {
|
||||||
print(stdout, format_str, args...);
|
print(stdout, fmt, args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
#if FMT_USE_NONTYPE_TEMPLATE_ARGS
|
||||||
inline namespace literals {
|
inline namespace literals {
|
||||||
template <detail_exported::fixed_string Str> constexpr auto operator""_cf() {
|
template <detail::fixed_string Str> constexpr auto operator""_cf() {
|
||||||
using char_t = remove_cvref_t<decltype(Str.data[0])>;
|
return FMT_COMPILE(Str.data);
|
||||||
return detail::udl_compiled_string<char_t, sizeof(Str.data) / sizeof(char_t),
|
|
||||||
Str>();
|
|
||||||
}
|
}
|
||||||
} // namespace literals
|
} // namespace literals
|
||||||
#endif
|
#endif
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -8,36 +8,36 @@
|
|||||||
#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
|
||||||
|
|
||||||
#if defined(_WIN32) && !defined(FMT_WINDOWS_NO_WCHAR)
|
#if defined(_WIN32) && !defined(FMT_USE_WRITE_CONSOLE)
|
||||||
# include <io.h> // _isatty
|
# include <io.h> // _isatty
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
|
#if FMT_USE_LOCALE
|
||||||
|
# include <locale>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef FMT_FUNC
|
||||||
|
# define FMT_FUNC
|
||||||
|
#endif
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
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);
|
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();
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void throw_format_error(const char* message) {
|
|
||||||
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,
|
||||||
@ -56,89 +56,105 @@ 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)
|
||||||
fmt::format_to(it, FMT_STRING("{}{}"), message, SEP);
|
fmt::format_to(it, FMT_STRING("{}{}"), message, SEP);
|
||||||
fmt::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 count, FILE* stream) {
|
inline void fwrite_all(const void* ptr, size_t count, FILE* stream) {
|
||||||
size_t written = std::fwrite(ptr, 1, count, stream);
|
size_t written = std::fwrite(ptr, 1, 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
|
#if FMT_USE_LOCALE
|
||||||
|
using std::locale;
|
||||||
|
using std::numpunct;
|
||||||
|
using std::use_facet;
|
||||||
|
|
||||||
template <typename Locale>
|
template <typename Locale>
|
||||||
locale_ref::locale_ref(const Locale& loc) : locale_(&loc) {
|
locale_ref::locale_ref(const Locale& loc) : locale_(&loc) {
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
static_assert(std::is_same<Locale, locale>::value, "");
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
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 {}; }
|
||||||
|
#endif // FMT_USE_LOCALE
|
||||||
|
|
||||||
template <typename Locale> auto locale_ref::get() const -> Locale {
|
template <typename Locale> auto locale_ref::get() const -> Locale {
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
static_assert(std::is_same<Locale, locale>::value, "");
|
||||||
return locale_ ? *static_cast<const std::locale*>(locale_) : std::locale();
|
#if FMT_USE_LOCALE
|
||||||
|
if (locale_) return *static_cast<const locale*>(locale_);
|
||||||
|
#endif
|
||||||
|
return 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>
|
template <typename Char>
|
||||||
FMT_FUNC auto decimal_point_impl(locale_ref loc) -> Char {
|
FMT_FUNC auto decimal_point_impl(locale_ref loc) -> Char {
|
||||||
return std::use_facet<std::numpunct<Char>>(loc.get<std::locale>())
|
return use_facet<numpunct<Char>>(loc.get<locale>()).decimal_point();
|
||||||
.decimal_point();
|
|
||||||
}
|
}
|
||||||
#else
|
|
||||||
template <typename Char>
|
|
||||||
FMT_FUNC auto thousands_sep_impl(locale_ref) -> thousands_sep_result<Char> {
|
|
||||||
return {"\03", FMT_STATIC_THOUSANDS_SEPARATOR};
|
|
||||||
}
|
|
||||||
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_USE_EXCEPTIONS
|
||||||
|
// Use FMT_THROW instead of throw to avoid bogus unreachable code warnings
|
||||||
|
// from MSVC.
|
||||||
|
FMT_THROW(format_error(message));
|
||||||
|
#else
|
||||||
|
fputs(message, stderr);
|
||||||
|
abort();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
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_});
|
||||||
}
|
}
|
||||||
@ -1411,7 +1427,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(...) {}
|
||||||
@ -1420,7 +1436,7 @@ 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 auto vformat(string_view fmt, format_args args) -> std::string {
|
FMT_FUNC auto vformat(string_view fmt, format_args args) -> std::string {
|
||||||
@ -1432,9 +1448,252 @@ FMT_FUNC auto vformat(string_view fmt, format_args args) -> std::string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
#if !defined(_WIN32) || defined(FMT_WINDOWS_NO_WCHAR)
|
|
||||||
|
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<char>{parse_context<char>(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) return;
|
||||||
|
// Force buffer initialization by placing and removing a char in a buffer.
|
||||||
|
assume(this->file_->_IO_write_ptr >= this->file_->_IO_write_end);
|
||||||
|
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; }
|
||||||
|
|
||||||
|
bool needs_flush() const {
|
||||||
|
if ((this->file_->_flags & line_buffered) == 0) return false;
|
||||||
|
char* end = this->file_->_IO_write_end;
|
||||||
|
return memchr(end, '\n', to_unsigned(this->file_->_IO_write_ptr - end));
|
||||||
|
}
|
||||||
|
|
||||||
|
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.
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool needs_flush() const {
|
||||||
|
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; }
|
FMT_FUNC auto write_console(int, string_view) -> bool { return false; }
|
||||||
FMT_FUNC auto write_console(std::FILE*, 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( //
|
||||||
@ -1445,39 +1704,51 @@ FMT_FUNC bool write_console(int fd, string_view text) {
|
|||||||
return WriteConsoleW(reinterpret_cast<void*>(_get_osfhandle(fd)), u16.c_str(),
|
return WriteConsoleW(reinterpret_cast<void*>(_get_osfhandle(fd)), u16.c_str(),
|
||||||
static_cast<dword>(u16.size()), nullptr, nullptr) != 0;
|
static_cast<dword>(u16.size()), nullptr, nullptr) != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_FUNC auto write_console(std::FILE* f, string_view text) -> bool {
|
|
||||||
return write_console(_fileno(f), text);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef _WIN32
|
#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, args);
|
detail::vformat_to(buffer, fmt, args);
|
||||||
fwrite_fully(buffer.data(), buffer.size(), f);
|
if (newline) buffer.push_back('\n');
|
||||||
|
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) {
|
||||||
#ifdef _WIN32
|
#if defined(_WIN32) && !defined(FMT_USE_WRITE_CONSOLE)
|
||||||
int fd = _fileno(f);
|
int fd = _fileno(f);
|
||||||
if (_isatty(fd)) {
|
if (_isatty(fd)) {
|
||||||
std::fflush(f);
|
std::fflush(f);
|
||||||
if (write_console(fd, text)) return;
|
if (write_console(fd, text)) return;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
fwrite_fully(text.data(), text.size(), f);
|
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);
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,2 +0,0 @@
|
|||||||
#include "xchar.h"
|
|
||||||
#warning fmt/locale.h is deprecated, include fmt/format.h or fmt/xchar.h instead
|
|
@ -8,18 +8,18 @@
|
|||||||
#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
|
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
#if defined __APPLE__ || defined(__FreeBSD__)
|
#ifndef FMT_MODULE
|
||||||
|
# include <cerrno>
|
||||||
|
# include <cstddef>
|
||||||
|
# include <cstdio>
|
||||||
|
# include <system_error> // std::system_error
|
||||||
|
|
||||||
# if FMT_HAS_INCLUDE(<xlocale.h>)
|
# if FMT_HAS_INCLUDE(<xlocale.h>)
|
||||||
# include <xlocale.h> // for LC_NUMERIC_MASK on OS X
|
# include <xlocale.h> // LC_NUMERIC_MASK on macOS
|
||||||
# endif
|
# endif
|
||||||
#endif
|
#endif // FMT_MODULE
|
||||||
|
|
||||||
#ifndef FMT_USE_FCNTL
|
#ifndef FMT_USE_FCNTL
|
||||||
// UWP doesn't provide _pipe.
|
// UWP doesn't provide _pipe.
|
||||||
@ -77,46 +77,33 @@ 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.
|
||||||
auto c_str() const -> const Char* { return data_; }
|
auto c_str() const -> const Char* { return data_; }
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -131,41 +118,38 @@ 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 'madeup': The system cannot find the file
|
||||||
|
* specified.
|
||||||
// This throws a system_error with the description
|
* // or similar (system message may vary).
|
||||||
// cannot open file 'madeup': The system cannot find the file specified.
|
* const char *filename = "madeup";
|
||||||
// or similar (system message may vary).
|
* LPOFSTRUCT of = LPOFSTRUCT();
|
||||||
const char *filename = "madeup";
|
* HFILE file = OpenFile(filename, &of, OF_READ);
|
||||||
LPOFSTRUCT of = LPOFSTRUCT();
|
* if (file == HFILE_ERROR) {
|
||||||
HFILE file = OpenFile(filename, &of, OF_READ);
|
* throw fmt::windows_error(GetLastError(),
|
||||||
if (file == HFILE_ERROR) {
|
* "cannot open file '{}'", filename);
|
||||||
throw fmt::windows_error(GetLastError(),
|
* }
|
||||||
"cannot open file '{}'", filename);
|
*/
|
||||||
}
|
template <typename... T>
|
||||||
\endrst
|
auto windows_error(int error_code, string_view message, const T&... args)
|
||||||
*/
|
-> std::system_error {
|
||||||
template <typename... Args>
|
return vwindows_error(error_code, message, vargs<T...>{{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.
|
||||||
@ -180,8 +164,8 @@ inline auto system_category() noexcept -> const std::error_category& {
|
|||||||
// 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
|
||||||
|
|
||||||
@ -192,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto operator=(buffered_file&& other) -> buffered_file& {
|
inline auto operator=(buffered_file&& other) -> buffered_file& {
|
||||||
close();
|
close();
|
||||||
file_ = other.file_;
|
file_ = other.file_;
|
||||||
other.file_ = nullptr;
|
other.file_ = nullptr;
|
||||||
@ -223,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.
|
||||||
auto get() const noexcept -> FILE* { return file_; }
|
inline auto get() const noexcept -> FILE* { return file_; }
|
||||||
|
|
||||||
FMT_API auto descriptor() const -> int;
|
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
|
||||||
@ -251,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 {
|
||||||
@ -263,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);
|
||||||
@ -272,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.
|
||||||
auto operator=(file&& other) -> file& {
|
inline auto operator=(file&& other) -> file& {
|
||||||
close();
|
close();
|
||||||
fd_ = other.fd_;
|
fd_ = other.fd_;
|
||||||
other.fd_ = -1;
|
other.fd_ = -1;
|
||||||
@ -286,7 +271,7 @@ class FMT_API file {
|
|||||||
~file() noexcept;
|
~file() noexcept;
|
||||||
|
|
||||||
// Returns the file descriptor.
|
// Returns the file descriptor.
|
||||||
auto descriptor() const noexcept -> int { return fd_; }
|
inline auto descriptor() const noexcept -> int { return fd_; }
|
||||||
|
|
||||||
// Closes the file.
|
// Closes the file.
|
||||||
void close();
|
void close();
|
||||||
@ -313,11 +298,6 @@ 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.
|
|
||||||
// DEPRECATED! Taking files as out parameters is deprecated.
|
|
||||||
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.
|
||||||
auto fdopen(const char* mode) -> buffered_file;
|
auto fdopen(const char* mode) -> buffered_file;
|
||||||
@ -329,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.
|
||||||
auto getpagesize() -> long;
|
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;
|
||||||
auto operator=(size_t val) const -> buffer_size {
|
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;
|
||||||
@ -348,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...) {
|
||||||
@ -369,79 +358,62 @@ 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 FMT_API ostream : private detail::buffer<char> {
|
||||||
|
private:
|
||||||
file file_;
|
file file_;
|
||||||
|
|
||||||
FMT_API void grow(size_t) override;
|
ostream(cstring_view path, const detail::ostream_params& params);
|
||||||
|
|
||||||
|
static void grow(buffer<char>& buf, size_t);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_API file_buffer(cstring_view path, const ostream_params& params);
|
ostream(ostream&& other) noexcept;
|
||||||
FMT_API file_buffer(file_buffer&& other);
|
~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() {
|
|
||||||
flush();
|
|
||||||
file_.close();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
// 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>
|
template <typename... T>
|
||||||
friend auto output_file(cstring_view path, T... params) -> ostream;
|
friend auto output_file(cstring_view path, T... params) -> ostream;
|
||||||
|
|
||||||
void close() { buffer_.close(); }
|
inline void close() {
|
||||||
|
flush();
|
||||||
|
file_.close();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/// Formats `args` according to specifications in `fmt` and writes the
|
||||||
Formats ``args`` according to specifications in ``fmt`` and writes the
|
/// output to the file.
|
||||||
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(std::back_inserter(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 auto output_file(cstring_view path, T... params) -> ostream {
|
inline auto output_file(cstring_view path, T... params) -> ostream {
|
||||||
|
@ -8,7 +8,9 @@
|
|||||||
#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
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
# ifdef __GLIBCXX__
|
# ifdef __GLIBCXX__
|
||||||
@ -18,42 +20,19 @@
|
|||||||
# include <io.h>
|
# include <io.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#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 {
|
||||||
|
|
||||||
template <typename Streambuf> class formatbuf : public Streambuf {
|
|
||||||
private:
|
|
||||||
using char_type = typename Streambuf::char_type;
|
|
||||||
using streamsize = decltype(std::declval<Streambuf>().sputn(nullptr, 0));
|
|
||||||
using int_type = typename Streambuf::int_type;
|
|
||||||
using traits_type = typename Streambuf::traits_type;
|
|
||||||
|
|
||||||
buffer<char_type>& buffer_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit formatbuf(buffer<char_type>& buf) : buffer_(buf) {}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
// The put area is always empty. This makes the implementation simpler and has
|
|
||||||
// the advantage that the streambuf and the buffer are always in sync and
|
|
||||||
// sputc never writes into uninitialized memory. A disadvantage is that each
|
|
||||||
// call to sputc always results in a (virtual) call to overflow. There is no
|
|
||||||
// disadvantage here for sputn since this always results in a call to xsputn.
|
|
||||||
|
|
||||||
auto overflow(int_type ch) -> int_type override {
|
|
||||||
if (!traits_type::eq_int_type(ch, traits_type::eof()))
|
|
||||||
buffer_.push_back(static_cast<char_type>(ch));
|
|
||||||
return ch;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto xsputn(const char_type* s, streamsize count) -> streamsize override {
|
|
||||||
buffer_.append(s, s + count);
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Generate a unique explicit instantion in every translation unit using a tag
|
// Generate a unique explicit instantion in every translation unit using a tag
|
||||||
// type in an anonymous namespace.
|
// type in an anonymous namespace.
|
||||||
namespace {
|
namespace {
|
||||||
@ -64,53 +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*;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
inline auto write_ostream_unicode(std::ostream& os, fmt::string_view data)
|
|
||||||
-> bool {
|
|
||||||
FILE* f = nullptr;
|
|
||||||
#if FMT_MSC_VERSION
|
|
||||||
if (auto* buf = dynamic_cast<std::filebuf*>(os.rdbuf()))
|
|
||||||
f = get_file(*buf);
|
|
||||||
else
|
|
||||||
return false;
|
|
||||||
#elif defined(_WIN32) && defined(__GLIBCXX__)
|
|
||||||
auto* rdbuf = os.rdbuf();
|
|
||||||
if (auto* sfbuf = dynamic_cast<__gnu_cxx::stdio_sync_filebuf<char>*>(rdbuf))
|
|
||||||
f = sfbuf->file();
|
|
||||||
else if (auto* fbuf = dynamic_cast<__gnu_cxx::stdio_filebuf<char>*>(rdbuf))
|
|
||||||
f = fbuf->file();
|
|
||||||
else
|
|
||||||
return false;
|
|
||||||
#else
|
|
||||||
ignore_unused(os, data, f);
|
|
||||||
#endif
|
|
||||||
#ifdef _WIN32
|
|
||||||
if (f) {
|
|
||||||
int fd = _fileno(f);
|
|
||||||
if (_isatty(fd)) {
|
|
||||||
os.flush();
|
|
||||||
return write_console(fd, data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
inline auto write_ostream_unicode(std::wostream&,
|
|
||||||
fmt::basic_string_view<wchar_t>) -> bool {
|
|
||||||
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 {
|
||||||
@ -121,21 +65,9 @@ void write_buffer(std::basic_ostream<Char>& os, buffer<Char>& buf) {
|
|||||||
} while (size != 0);
|
} while (size != 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char, typename T>
|
|
||||||
void format_value(buffer<Char>& buf, const T& value) {
|
|
||||||
auto&& format_buf = formatbuf<std::basic_streambuf<Char>>(buf);
|
|
||||||
auto&& output = std::basic_ostream<Char>(&format_buf);
|
|
||||||
#if !defined(FMT_STATIC_THOUSANDS_SEPARATOR)
|
|
||||||
output.imbue(std::locale::classic()); // The default is always unlocalized.
|
|
||||||
#endif
|
|
||||||
output << value;
|
|
||||||
output.exceptions(std::ios_base::failbit | std::ios_base::badbit);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T> struct streamed_view {
|
template <typename T> struct streamed_view {
|
||||||
const T& value;
|
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<<.
|
||||||
@ -143,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);
|
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);
|
||||||
}
|
}
|
||||||
@ -158,73 +93,67 @@ 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>
|
||||||
constexpr 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>
|
||||||
@ -232,14 +161,6 @@ 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, "{}\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
|
||||||
|
|
||||||
#endif // FMT_OSTREAM_H_
|
#endif // FMT_OSTREAM_H_
|
||||||
|
@ -8,8 +8,10 @@
|
|||||||
#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::max
|
||||||
|
# include <limits> // std::numeric_limits
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
@ -22,7 +24,7 @@ template <typename T> struct printf_formatter {
|
|||||||
|
|
||||||
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 ||
|
static_assert(std::is_same<Char, char>::value ||
|
||||||
@ -31,43 +33,53 @@ template <typename Char> class basic_printf_context {
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
using char_type = Char;
|
using char_type = Char;
|
||||||
using parse_context_type = basic_format_parse_context<Char>;
|
using parse_context_type = parse_context<Char>;
|
||||||
template <typename T> using formatter_type = printf_formatter<T>;
|
template <typename T> using formatter_type = printf_formatter<T>;
|
||||||
|
enum { builtin_types = 1 };
|
||||||
|
|
||||||
/**
|
/// 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() -> detail::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 IsSigned> 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>();
|
unsigned max = to_unsigned(max_value<int>());
|
||||||
return value <= max;
|
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> {
|
||||||
@ -75,20 +87,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 (std::max)(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;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -133,25 +145,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) {
|
|
||||||
auto n = static_cast<int>(static_cast<target_type>(value));
|
|
||||||
arg_ = detail::make_arg<Context>(n);
|
|
||||||
} else {
|
|
||||||
using unsigned_type = typename make_unsigned_or_bool<target_type>::type;
|
using unsigned_type = typename make_unsigned_or_bool<target_type>::type;
|
||||||
auto n = static_cast<unsigned>(static_cast<unsigned_type>(value));
|
if (is_signed)
|
||||||
arg_ = detail::make_arg<Context>(n);
|
arg_ = static_cast<int>(static_cast<target_type>(value));
|
||||||
}
|
else
|
||||||
|
arg_ = static_cast<unsigned>(static_cast<unsigned_type>(value));
|
||||||
} 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.
|
||||||
auto n = static_cast<long long>(value);
|
if (is_signed)
|
||||||
arg_ = detail::make_arg<Context>(n);
|
arg_ = static_cast<long long>(value);
|
||||||
} else {
|
else
|
||||||
auto n = static_cast<typename make_unsigned_or_bool<U>::type>(value);
|
arg_ = static_cast<typename make_unsigned_or_bool<U>::type>(value);
|
||||||
arg_ = detail::make_arg<Context>(n);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -165,7 +171,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.
|
||||||
@ -178,8 +184,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)>
|
||||||
@ -195,28 +200,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;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -224,12 +229,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:
|
||||||
@ -240,105 +245,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;
|
||||||
@ -350,11 +346,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;
|
||||||
}
|
}
|
||||||
@ -365,63 +361,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.
|
||||||
@ -449,12 +429,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 == '.') {
|
||||||
@ -464,8 +444,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;
|
||||||
}
|
}
|
||||||
@ -474,25 +454,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() && arg.visit(is_zero_int())) specs.clear_alt();
|
||||||
|
if (specs.fill_unit<Char>() == '0') {
|
||||||
|
if (is_arithmetic_type(arg.type()) && specs.align() != align::left) {
|
||||||
|
specs.set_align(align::numeric);
|
||||||
|
} else {
|
||||||
|
// Ignore '0' flag for non-numeric types or if '-' flag is also present.
|
||||||
|
specs.set_fill(' ');
|
||||||
}
|
}
|
||||||
if (specs.alt && visit_format_arg(is_zero_int(), arg)) specs.alt = false;
|
|
||||||
if (specs.fill[0] == '0') {
|
|
||||||
if (arg.is_arithmetic() && specs.align != align::left)
|
|
||||||
specs.align = align::numeric;
|
|
||||||
else
|
|
||||||
specs.fill[0] = ' '; // Ignore '0' flag for non-numeric types or if '-'
|
|
||||||
// flag is also present.
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse length and convert the argument to the required type.
|
// Parse length and convert the argument to the required type.
|
||||||
@ -517,47 +498,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)));
|
||||||
}
|
}
|
||||||
@ -569,56 +542,44 @@ 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 S, typename... T, typename Char = detail::char_t<S>>
|
||||||
template <typename S, typename... T,
|
|
||||||
typename Char = enable_if_t<detail::is_string<S>::value, char_t<S>>>
|
|
||||||
inline auto sprintf(const S& fmt, const T&... args) -> std::basic_string<Char> {
|
inline auto sprintf(const S& fmt, const T&... args) -> std::basic_string<Char> {
|
||||||
return vsprintf(detail::to_string_view(fmt),
|
return vsprintf(detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<basic_printf_context<Char>>(args...));
|
fmt::make_format_args<basic_printf_context<Char>>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
inline auto vfprintf(
|
inline 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();
|
||||||
@ -628,36 +589,33 @@ 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 S, typename... T, typename Char = detail::char_t<S>>
|
||||||
inline auto fprintf(std::FILE* f, const S& fmt, const T&... args) -> int {
|
inline auto fprintf(std::FILE* f, const S& fmt, const T&... args) -> int {
|
||||||
return vfprintf(f, detail::to_string_view(fmt),
|
return vfprintf(f, detail::to_string_view(fmt),
|
||||||
fmt::make_format_args<basic_printf_context<Char>>(args...));
|
make_printf_args<Char>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Char>
|
template <typename Char>
|
||||||
FMT_DEPRECATED inline auto vprintf(
|
FMT_DEPRECATED inline auto vprintf(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)
|
|
||||||
-> int {
|
-> int {
|
||||||
return vfprintf(stdout, fmt, args);
|
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 {
|
||||||
@ -666,7 +624,7 @@ inline auto printf(string_view fmt, const T&... args) -> int {
|
|||||||
template <typename... T>
|
template <typename... T>
|
||||||
FMT_DEPRECATED inline auto printf(basic_string_view<wchar_t> fmt,
|
FMT_DEPRECATED inline auto printf(basic_string_view<wchar_t> fmt,
|
||||||
const T&... args) -> int {
|
const T&... args) -> int {
|
||||||
return vfprintf(stdout, fmt, make_wprintf_args(args...));
|
return vfprintf(stdout, fmt, make_printf_args<wchar_t>(args...));
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_END_EXPORT
|
FMT_END_EXPORT
|
||||||
|
@ -8,67 +8,31 @@
|
|||||||
#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 <string>
|
||||||
|
# include <tuple>
|
||||||
|
# include <type_traits>
|
||||||
|
# include <utility>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "format.h"
|
#include "format.h"
|
||||||
|
|
||||||
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 const bool value = false;
|
|
||||||
#else
|
|
||||||
static constexpr const bool value =
|
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 {
|
||||||
@ -76,26 +40,10 @@ 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 const bool value = false;
|
|
||||||
#else
|
|
||||||
static constexpr const bool value =
|
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, std::size_t N>
|
||||||
auto range_begin(const T (&arr)[N]) -> const T* {
|
auto range_begin(const T (&arr)[N]) -> const T* {
|
||||||
@ -110,17 +58,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)
|
||||||
@ -141,31 +93,30 @@ 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:
|
||||||
@ -206,12 +157,13 @@ class is_tuple_formattable_ {
|
|||||||
static constexpr const bool value = false;
|
static constexpr const 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 auto check2(index_sequence<Is...>,
|
static auto all_true(index_sequence<Is...>,
|
||||||
integer_sequence<bool, (Is == Is)...>) -> std::true_type;
|
integer_sequence<bool, (Is >= 0)...>) -> std::true_type;
|
||||||
static auto check2(...) -> std::false_type;
|
static auto all_true(...) -> std::false_type;
|
||||||
template <std::size_t... Is>
|
|
||||||
static auto check(index_sequence<Is...>) -> decltype(check2(
|
template <size_t... Is>
|
||||||
|
static auto check(index_sequence<Is...>) -> decltype(all_true(
|
||||||
index_sequence<Is...>{},
|
index_sequence<Is...>{},
|
||||||
integer_sequence<bool,
|
integer_sequence<bool,
|
||||||
(is_formattable<typename std::tuple_element<Is, T>::type,
|
(is_formattable<typename std::tuple_element<Is, T>::type,
|
||||||
@ -292,21 +244,32 @@ FMT_CONSTEXPR auto maybe_set_debug_format(Formatter& f, bool set)
|
|||||||
template <typename Formatter>
|
template <typename Formatter>
|
||||||
FMT_CONSTEXPR void maybe_set_debug_format(Formatter&, ...) {}
|
FMT_CONSTEXPR void maybe_set_debug_format(Formatter&, ...) {}
|
||||||
|
|
||||||
|
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>
|
||||||
|
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;
|
||||||
}
|
}
|
||||||
@ -355,66 +318,48 @@ 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());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T, typename Char> struct is_range {
|
template <typename T, typename Char> struct is_range {
|
||||||
static constexpr const bool value =
|
static constexpr const 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...> struct conjunction : std::true_type {};
|
||||||
@ -438,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() {}
|
||||||
@ -456,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);
|
||||||
@ -479,80 +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);
|
||||||
auto&& item = *it;
|
auto&& item = *it; // Need an lvalue
|
||||||
out = underlying_.format(mapper.map(item), ctx);
|
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<
|
||||||
@ -562,23 +490,191 @@ 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;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename Char, typename Tuple> 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} {}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -589,66 +685,65 @@ 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 Char, typename Tuple>
|
||||||
struct formatter<tuple_join_view<Char, T...>, Char> {
|
struct formatter<tuple_join_view<Char, Tuple>, 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<Char, Tuple>& 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<Char, Tuple>&, 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<Char, Tuple>& 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);
|
||||||
|
if (N <= 1) return out;
|
||||||
|
out = detail::copy<Char>(value.sep, out);
|
||||||
ctx.advance_to(out);
|
ctx.advance_to(out);
|
||||||
return do_format(value, ctx, std::integral_constant<size_t, N - 1>());
|
return do_format(value, ctx, std::integral_constant<size_t, N - 1>());
|
||||||
}
|
}
|
||||||
return out;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace detail {
|
namespace detail {
|
||||||
@ -691,40 +786,57 @@ struct formatter<
|
|||||||
|
|
||||||
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, string_view sep)
|
||||||
|
-> tuple_join_view<char, Tuple> {
|
||||||
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)
|
||||||
|
@ -8,26 +8,27 @@
|
|||||||
#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"
|
||||||
|
|
||||||
#if FMT_HAS_INCLUDE(<version>)
|
#ifndef FMT_MODULE
|
||||||
# include <version>
|
# include <atomic>
|
||||||
#endif
|
# include <bitset>
|
||||||
// Checking FMT_CPLUSPLUS for warning suppression in MSVC.
|
# include <complex>
|
||||||
#if FMT_CPLUSPLUS >= 201703L
|
# include <cstdlib>
|
||||||
# if FMT_HAS_INCLUDE(<filesystem>)
|
# include <exception>
|
||||||
|
# include <functional>
|
||||||
|
# include <memory>
|
||||||
|
# include <thread>
|
||||||
|
# include <type_traits>
|
||||||
|
# include <typeinfo>
|
||||||
|
# include <utility>
|
||||||
|
# include <vector>
|
||||||
|
|
||||||
|
// 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>
|
# include <filesystem>
|
||||||
# endif
|
# endif
|
||||||
# if FMT_HAS_INCLUDE(<variant>)
|
# if FMT_HAS_INCLUDE(<variant>)
|
||||||
@ -36,10 +37,19 @@
|
|||||||
# if FMT_HAS_INCLUDE(<optional>)
|
# if FMT_HAS_INCLUDE(<optional>)
|
||||||
# include <optional>
|
# include <optional>
|
||||||
# endif
|
# endif
|
||||||
#endif
|
# endif
|
||||||
|
// Use > instead of >= in the version check because <source_location> may be
|
||||||
#if FMT_CPLUSPLUS > 201703L && FMT_HAS_INCLUDE(<source_location>)
|
// available after C++17 but before C++20 is marked as implemented.
|
||||||
|
# if FMT_CPLUSPLUS > 201703L && FMT_HAS_INCLUDE(<source_location>)
|
||||||
# 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>)
|
||||||
|
# include <version>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// GCC 4 does not support FMT_HAS_INCLUDE.
|
// GCC 4 does not support FMT_HAS_INCLUDE.
|
||||||
@ -52,17 +62,6 @@
|
|||||||
# endif
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Check if typeid is available.
|
|
||||||
#ifndef FMT_USE_TYPEID
|
|
||||||
// __RTTI is for EDG compilers. In MSVC typeid is available without RTTI.
|
|
||||||
# if defined(__GXX_RTTI) || FMT_HAS_FEATURE(cxx_rtti) || FMT_MSC_VERSION || \
|
|
||||||
defined(__INTEL_RTTI__) || defined(__RTTI)
|
|
||||||
# define FMT_USE_TYPEID 1
|
|
||||||
# else
|
|
||||||
# define FMT_USE_TYPEID 0
|
|
||||||
# endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// For older Xcode versions, __cpp_lib_xxx flags are inaccurately defined.
|
// For older Xcode versions, __cpp_lib_xxx flags are inaccurately defined.
|
||||||
#ifndef FMT_CPP_LIB_FILESYSTEM
|
#ifndef FMT_CPP_LIB_FILESYSTEM
|
||||||
# ifdef __cpp_lib_filesystem
|
# ifdef __cpp_lib_filesystem
|
||||||
@ -117,7 +116,7 @@ void write_escaped_path(basic_memory_buffer<Char>& quoted,
|
|||||||
FMT_EXPORT
|
FMT_EXPORT
|
||||||
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;
|
char path_type_ = 0;
|
||||||
@ -125,32 +124,32 @@ template <typename Char> struct formatter<std::filesystem::path, Char> {
|
|||||||
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_ = *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_;
|
||||||
# ifdef _WIN32
|
auto path_string =
|
||||||
auto path_string = !path_type_ ? p.native() : p.generic_wstring();
|
!path_type_ ? p.native()
|
||||||
# else
|
: p.generic_string<std::filesystem::path::value_type>();
|
||||||
auto path_string = !path_type_ ? p.native() : p.generic_string();
|
|
||||||
# endif
|
|
||||||
|
|
||||||
detail::handle_dynamic_spec<detail::width_checker>(specs.width, width_ref_,
|
detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
|
||||||
ctx);
|
ctx);
|
||||||
if (!debug_) {
|
if (!debug_) {
|
||||||
auto s = detail::get_path_string<Char>(p, path_string);
|
auto s = detail::get_path_string<Char>(p, path_string);
|
||||||
@ -163,13 +162,30 @@ template <typename Char> struct formatter<std::filesystem::path, Char> {
|
|||||||
specs);
|
specs);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class path : public std::filesystem::path {
|
||||||
|
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(); }
|
||||||
|
};
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
#endif // FMT_CPP_LIB_FILESYSTEM
|
#endif // FMT_CPP_LIB_FILESYSTEM
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
FMT_EXPORT
|
FMT_EXPORT
|
||||||
template <std::size_t N, typename Char>
|
template <std::size_t N, typename Char>
|
||||||
struct formatter<std::bitset<N>, Char> : nested_formatter<string_view> {
|
struct formatter<std::bitset<N>, Char>
|
||||||
|
: nested_formatter<basic_string_view<Char>, Char> {
|
||||||
private:
|
private:
|
||||||
// Functor because C++11 doesn't support generic lambdas.
|
// Functor because C++11 doesn't support generic lambdas.
|
||||||
struct writer {
|
struct writer {
|
||||||
@ -189,7 +205,7 @@ struct formatter<std::bitset<N>, Char> : nested_formatter<string_view> {
|
|||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const std::bitset<N>& bs, FormatContext& ctx) const
|
auto format(const std::bitset<N>& bs, FormatContext& ctx) const
|
||||||
-> decltype(ctx.out()) {
|
-> decltype(ctx.out()) {
|
||||||
return write_padded(ctx, writer{bs});
|
return this->write_padded(ctx, writer{bs});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -222,7 +238,7 @@ struct formatter<std::optional<T>, Char,
|
|||||||
FMT_CONSTEXPR static void maybe_set_debug_format(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);
|
maybe_set_debug_format(underlying_, true);
|
||||||
return underlying_.parse(ctx);
|
return underlying_.parse(ctx);
|
||||||
}
|
}
|
||||||
@ -242,13 +258,62 @@ struct formatter<std::optional<T>, Char,
|
|||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
#endif // __cpp_lib_optional
|
#endif // __cpp_lib_optional
|
||||||
|
|
||||||
|
#if defined(__cpp_lib_expected) || FMT_CPP_LIB_VARIANT
|
||||||
|
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
template <typename Char, typename OutputIt, typename T>
|
||||||
|
auto write_escaped_alternative(OutputIt out, const T& v) -> 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);
|
||||||
|
return write<Char>(out, v);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
|
FMT_END_NAMESPACE
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __cpp_lib_expected
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
template <typename T, typename E, typename Char>
|
||||||
|
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 FormatContext>
|
||||||
|
auto format(const std::expected<T, E>& value, FormatContext& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
|
||||||
|
if (value.has_value()) {
|
||||||
|
out = detail::write<Char>(out, "expected(");
|
||||||
|
if constexpr (!std::is_void<T>::value)
|
||||||
|
out = detail::write_escaped_alternative<Char>(out, *value);
|
||||||
|
} else {
|
||||||
|
out = detail::write<Char>(out, "unexpected(");
|
||||||
|
out = detail::write_escaped_alternative<Char>(out, value.error());
|
||||||
|
}
|
||||||
|
*out++ = ')';
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
FMT_END_NAMESPACE
|
||||||
|
#endif // __cpp_lib_expected
|
||||||
|
|
||||||
#ifdef __cpp_lib_source_location
|
#ifdef __cpp_lib_source_location
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
FMT_EXPORT
|
FMT_EXPORT
|
||||||
template <> struct formatter<std::source_location> {
|
template <> struct formatter<std::source_location> {
|
||||||
template <typename ParseContext> FMT_CONSTEXPR auto parse(ParseContext& ctx) {
|
FMT_CONSTEXPR auto parse(parse_context<>& ctx) { return ctx.begin(); }
|
||||||
return ctx.begin();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
template <typename FormatContext>
|
||||||
auto format(const std::source_location& loc, FormatContext& ctx) const
|
auto format(const std::source_location& loc, FormatContext& ctx) const
|
||||||
@ -291,16 +356,6 @@ template <typename T, typename C> class is_variant_formattable_ {
|
|||||||
decltype(check(variant_index_sequence<T>{}))::value;
|
decltype(check(variant_index_sequence<T>{}))::value;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename Char, typename OutputIt, typename T>
|
|
||||||
auto write_variant_alternative(OutputIt out, const T& v) -> OutputIt {
|
|
||||||
if constexpr (is_string<T>::value)
|
|
||||||
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
|
} // namespace detail
|
||||||
|
|
||||||
template <typename T> struct is_variant_like {
|
template <typename T> struct is_variant_like {
|
||||||
@ -314,8 +369,7 @@ template <typename T, typename C> struct is_variant_formattable {
|
|||||||
|
|
||||||
FMT_EXPORT
|
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();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -332,8 +386,7 @@ 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_formattable<Variant, Char>>>> {
|
is_variant_like<Variant>, 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();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -346,7 +399,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);
|
||||||
},
|
},
|
||||||
value);
|
value);
|
||||||
}
|
}
|
||||||
@ -362,54 +415,45 @@ FMT_END_NAMESPACE
|
|||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
FMT_EXPORT
|
FMT_EXPORT
|
||||||
template <typename Char> struct formatter<std::error_code, Char> {
|
template <> struct formatter<std::error_code> {
|
||||||
template <typename ParseContext>
|
|
||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
FMT_CONSTEXPR auto format(const std::error_code& ec, FormatContext& ctx) const
|
|
||||||
-> decltype(ctx.out()) {
|
|
||||||
auto out = ctx.out();
|
|
||||||
out = detail::write_bytes(out, ec.category().name(), format_specs<Char>());
|
|
||||||
out = detail::write<Char>(out, Char(':'));
|
|
||||||
out = detail::write<Char>(out, ec.value());
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_EXPORT
|
|
||||||
template <typename T, typename Char>
|
|
||||||
struct formatter<
|
|
||||||
T, Char, // DEPRECATED! Mixing code unit types.
|
|
||||||
typename std::enable_if<std::is_base_of<std::exception, T>::value>::type> {
|
|
||||||
private:
|
private:
|
||||||
bool with_typename_ = false;
|
format_specs specs_;
|
||||||
|
detail::arg_ref<char> width_ref_;
|
||||||
|
|
||||||
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(), end = ctx.end();
|
||||||
auto it = ctx.begin();
|
if (it == end) return it;
|
||||||
auto end = ctx.end();
|
|
||||||
if (it == end || *it == '}') return it;
|
it = detail::parse_align(it, end, specs_);
|
||||||
if (*it == 't') {
|
if (it == end) return it;
|
||||||
++it;
|
|
||||||
with_typename_ = FMT_USE_TYPEID != 0;
|
char c = *it;
|
||||||
}
|
if ((c >= '0' && c <= '9') || c == '{')
|
||||||
|
it = detail::parse_width(it, end, specs_, width_ref_, ctx);
|
||||||
return it;
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt>
|
template <typename FormatContext>
|
||||||
auto format(const std::exception& ex,
|
FMT_CONSTEXPR20 auto format(const std::error_code& ec,
|
||||||
basic_format_context<OutputIt, Char>& ctx) const -> OutputIt {
|
FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
format_specs<Char> spec;
|
auto specs = specs_;
|
||||||
auto out = ctx.out();
|
detail::handle_dynamic_spec(specs.dynamic_width(), specs.width, width_ref_,
|
||||||
if (!with_typename_)
|
ctx);
|
||||||
return detail::write_bytes(out, string_view(ex.what()), spec);
|
memory_buffer buf;
|
||||||
|
buf.append(string_view(ec.category().name()));
|
||||||
|
buf.push_back(':');
|
||||||
|
detail::write<char>(appender(buf), ec.value());
|
||||||
|
return detail::write<char>(ctx.out(), string_view(buf.data(), buf.size()),
|
||||||
|
specs);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
#if FMT_USE_TYPEID
|
#if FMT_USE_RTTI
|
||||||
const std::type_info& ti = typeid(ex);
|
namespace detail {
|
||||||
|
|
||||||
|
template <typename Char, typename OutputIt>
|
||||||
|
auto write_demangled_name(OutputIt out, const std::type_info& ti) -> OutputIt {
|
||||||
# ifdef FMT_HAS_ABI_CXA_DEMANGLE
|
# ifdef FMT_HAS_ABI_CXA_DEMANGLE
|
||||||
int status = 0;
|
int status = 0;
|
||||||
std::size_t size = 0;
|
std::size_t size = 0;
|
||||||
@ -448,21 +492,83 @@ struct formatter<
|
|||||||
} else {
|
} else {
|
||||||
demangled_name_view = string_view(ti.name());
|
demangled_name_view = string_view(ti.name());
|
||||||
}
|
}
|
||||||
out = detail::write_bytes(out, demangled_name_view, spec);
|
return detail::write_bytes<Char>(out, demangled_name_view);
|
||||||
# elif FMT_MSC_VERSION
|
# elif FMT_MSC_VERSION
|
||||||
string_view demangled_name_view(ti.name());
|
const string_view demangled_name(ti.name());
|
||||||
if (demangled_name_view.starts_with("class "))
|
for (std::size_t i = 0; i < demangled_name.size(); ++i) {
|
||||||
demangled_name_view.remove_prefix(6);
|
auto sub = demangled_name;
|
||||||
else if (demangled_name_view.starts_with("struct "))
|
sub.remove_prefix(i);
|
||||||
demangled_name_view.remove_prefix(7);
|
if (sub.starts_with("enum ")) {
|
||||||
out = detail::write_bytes(out, demangled_name_view, spec);
|
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;
|
||||||
# else
|
# else
|
||||||
out = detail::write_bytes(out, string_view(ti.name()), spec);
|
return detail::write_bytes<Char>(out, string_view(ti.name()));
|
||||||
# endif
|
# endif
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
template <typename Char>
|
||||||
|
struct formatter<std::type_info, Char // DEPRECATED! Mixing code unit types.
|
||||||
|
> {
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& 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<Char>(ctx.out(), ti);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
template <typename T, typename Char>
|
||||||
|
struct formatter<
|
||||||
|
T, Char, // DEPRECATED! Mixing code unit types.
|
||||||
|
typename std::enable_if<std::is_base_of<std::exception, T>::value>::type> {
|
||||||
|
private:
|
||||||
|
bool with_typename_ = false;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FMT_CONSTEXPR auto parse(parse_context<Char>& ctx) -> const Char* {
|
||||||
|
auto it = ctx.begin();
|
||||||
|
auto end = ctx.end();
|
||||||
|
if (it == end || *it == '}') return it;
|
||||||
|
if (*it == 't') {
|
||||||
|
++it;
|
||||||
|
with_typename_ = FMT_USE_RTTI != 0;
|
||||||
|
}
|
||||||
|
return it;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Context>
|
||||||
|
auto format(const std::exception& ex, Context& ctx) const
|
||||||
|
-> decltype(ctx.out()) {
|
||||||
|
auto out = ctx.out();
|
||||||
|
#if FMT_USE_RTTI
|
||||||
|
if (with_typename_) {
|
||||||
|
out = detail::write_demangled_name<Char>(out, typeid(ex));
|
||||||
*out++ = ':';
|
*out++ = ':';
|
||||||
*out++ = ' ';
|
*out++ = ' ';
|
||||||
return detail::write_bytes(out, string_view(ex.what()), spec);
|
}
|
||||||
#endif
|
#endif
|
||||||
|
return detail::write_bytes<Char>(out, string_view(ex.what()));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -509,6 +615,14 @@ struct formatter<BitRef, Char,
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
FMT_EXPORT
|
FMT_EXPORT
|
||||||
template <typename T, typename Char>
|
template <typename T, typename Char>
|
||||||
struct formatter<std::atomic<T>, Char,
|
struct formatter<std::atomic<T>, Char,
|
||||||
@ -533,5 +647,80 @@ struct formatter<std::atomic_flag, Char> : formatter<bool, Char> {
|
|||||||
};
|
};
|
||||||
#endif // __cpp_lib_atomic_flag_test
|
#endif // __cpp_lib_atomic_flag_test
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
FMT_EXPORT
|
||||||
|
template <typename T, typename Char>
|
||||||
|
struct formatter<std::reference_wrapper<T>, Char,
|
||||||
|
enable_if_t<is_formattable<remove_cvref_t<T>, Char>::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_
|
||||||
|
@ -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 <cwchar>
|
||||||
|
# if FMT_USE_LOCALE
|
||||||
# include <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();
|
||||||
@ -40,42 +60,79 @@ inline auto write_loc(std::back_insert_iterator<detail::buffer<wchar_t>> out,
|
|||||||
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<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<char16_t> : std::true_type {};
|
||||||
template <> struct is_char<char32_t> : std::true_type {};
|
template <> struct is_char<char32_t> : std::true_type {};
|
||||||
|
|
||||||
|
#ifdef __cpp_char8_t
|
||||||
|
template <> struct is_char<char8_t> : bool_constant<detail::is_utf8_enabled> {};
|
||||||
|
#endif
|
||||||
|
|
||||||
template <typename... T>
|
template <typename... T>
|
||||||
constexpr auto make_wformat_args(const T&... args)
|
constexpr auto make_wformat_args(T&... args)
|
||||||
-> format_arg_store<wformat_context, T...> {
|
-> 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 auto operator""_a(const wchar_t* s, size_t)
|
|
||||||
-> detail::udl_arg<wchar_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)
|
||||||
@ -83,9 +140,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);
|
||||||
}
|
}
|
||||||
@ -96,13 +153,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<wchar_t, Tuple> {
|
||||||
|
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)
|
typename detail::vformat_args<Char>::type 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>
|
||||||
@ -110,110 +173,122 @@ 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 Locale, typename S,
|
||||||
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
inline auto vformat(
|
inline auto vformat(const Locale& loc, const S& fmt,
|
||||||
const Locale& loc, const S& format_str,
|
typename detail::vformat_args<Char>::type args)
|
||||||
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,
|
||||||
|
detail::locale_ref(loc));
|
||||||
|
return {buf.data(), buf.size()};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Locale, typename S, typename... T, typename Char = char_t<S>,
|
template <typename Locale, typename S, typename... T,
|
||||||
|
typename Char = detail::format_string_char_t<S>,
|
||||||
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
FMT_ENABLE_IF(detail::is_locale<Locale>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
inline auto format(const Locale& loc, const S& format_str, T&&... args)
|
inline auto format(const Locale& 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)
|
typename detail::vformat_args<Char>::type 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 Locale, 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_locale<Locale>::value&&
|
||||||
detail::is_exotic_char<Char>::value)>
|
detail::is_exotic_char<Char>::value)>
|
||||||
inline auto vformat_to(
|
inline auto vformat_to(OutputIt out, const Locale& loc, const S& fmt,
|
||||||
OutputIt out, const Locale& loc, const S& format_str,
|
typename detail::vformat_args<Char>::type args)
|
||||||
basic_format_args<buffer_context<type_identity_t<Char>>> args) -> OutputIt {
|
-> 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, detail::locale_ref(loc));
|
||||||
detail::locale_ref(loc));
|
|
||||||
return detail::get_iterator(buf, out);
|
return detail::get_iterator(buf, out);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename OutputIt, typename Locale, typename S, typename... T,
|
template <typename Locale, typename OutputIt, typename S, typename... T,
|
||||||
typename Char = char_t<S>,
|
typename Char = detail::format_string_char_t<S>,
|
||||||
bool enable = detail::is_output_iterator<OutputIt, Char>::value &&
|
bool enable = detail::is_output_iterator<OutputIt, Char>::value &&
|
||||||
detail::is_locale<Locale>::value &&
|
detail::is_locale<Locale>::value &&
|
||||||
detail::is_exotic_char<Char>::value>
|
detail::is_exotic_char<Char>::value>
|
||||||
inline auto format_to(OutputIt out, const Locale& loc, const S& format_str,
|
inline auto format_to(OutputIt out, const Locale& loc, const S& fmt,
|
||||||
T&&... args) ->
|
T&&... args) ->
|
||||||
typename std::enable_if<enable, OutputIt>::type {
|
typename std::enable_if<enable, OutputIt>::type {
|
||||||
return vformat_to(out, loc, detail::to_string_view(format_str),
|
return vformat_to(out, 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 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,
|
typename detail::vformat_args<Char>::type 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();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -247,9 +322,48 @@ 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(const 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(const text_style& ts, wformat_string<T...> fmt, T&&... args)
|
||||||
|
-> std::wstring {
|
||||||
|
return fmt::vformat(ts, fmt, fmt::make_wformat_args(args...));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
FMT_DEPRECATED void print(std::FILE* f, const text_style& ts,
|
||||||
|
wformat_string<T...> fmt, const T&... args) {
|
||||||
|
vprint(f, ts, fmt, fmt::make_wformat_args(args...));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename... T>
|
||||||
|
FMT_DEPRECATED void print(const text_style& ts, wformat_string<T...> fmt,
|
||||||
|
const T&... args) {
|
||||||
|
return print(stdout, ts, fmt, 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);
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,10 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#if defined(SPDLOG_NO_TLS)
|
||||||
|
#error "This header requires thread local storage support, but SPDLOG_NO_TLS is defined."
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
@ -10,7 +10,11 @@
|
|||||||
#include <spdlog/details/fmt_helper.h>
|
#include <spdlog/details/fmt_helper.h>
|
||||||
#include <spdlog/details/log_msg.h>
|
#include <spdlog/details/log_msg.h>
|
||||||
#include <spdlog/details/os.h>
|
#include <spdlog/details/os.h>
|
||||||
#include <spdlog/mdc.h>
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
|
#include <spdlog/mdc.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <spdlog/fmt/fmt.h>
|
#include <spdlog/fmt/fmt.h>
|
||||||
#include <spdlog/formatter.h>
|
#include <spdlog/formatter.h>
|
||||||
|
|
||||||
@ -66,6 +70,9 @@ public:
|
|||||||
pad_it(remaining_pad_);
|
pad_it(remaining_pad_);
|
||||||
} else if (padinfo_.truncate_) {
|
} else if (padinfo_.truncate_) {
|
||||||
long new_size = static_cast<long>(dest_.size()) + remaining_pad_;
|
long new_size = static_cast<long>(dest_.size()) + remaining_pad_;
|
||||||
|
if (new_size < 0) {
|
||||||
|
new_size = 0;
|
||||||
|
}
|
||||||
dest_.resize(static_cast<size_t>(new_size));
|
dest_.resize(static_cast<size_t>(new_size));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -176,7 +183,7 @@ public:
|
|||||||
|
|
||||||
// Abbreviated month
|
// Abbreviated month
|
||||||
static const std::array<const char *, 12> months{
|
static const std::array<const char *, 12> months{
|
||||||
{"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sept", "Oct", "Nov", "Dec"}};
|
{"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}};
|
||||||
|
|
||||||
template <typename ScopedPadder>
|
template <typename ScopedPadder>
|
||||||
class b_formatter final : public flag_formatter {
|
class b_formatter final : public flag_formatter {
|
||||||
@ -260,7 +267,7 @@ public:
|
|||||||
: flag_formatter(padinfo) {}
|
: flag_formatter(padinfo) {}
|
||||||
|
|
||||||
void format(const details::log_msg &, const std::tm &tm_time, memory_buf_t &dest) override {
|
void format(const details::log_msg &, const std::tm &tm_time, memory_buf_t &dest) override {
|
||||||
const size_t field_size = 10;
|
const size_t field_size = 8;
|
||||||
ScopedPadder p(field_size, padinfo_, dest);
|
ScopedPadder p(field_size, padinfo_, dest);
|
||||||
|
|
||||||
fmt_helper::pad2(tm_time.tm_mon + 1, dest);
|
fmt_helper::pad2(tm_time.tm_mon + 1, dest);
|
||||||
@ -786,6 +793,7 @@ private:
|
|||||||
|
|
||||||
// Class for formatting Mapped Diagnostic Context (MDC) in log messages.
|
// Class for formatting Mapped Diagnostic Context (MDC) in log messages.
|
||||||
// Example: [logger-name] [info] [mdc_key_1:mdc_value_1 mdc_key_2:mdc_value_2] some message
|
// Example: [logger-name] [info] [mdc_key_1:mdc_value_1 mdc_key_2:mdc_value_2] some message
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
template <typename ScopedPadder>
|
template <typename ScopedPadder>
|
||||||
class mdc_formatter : public flag_formatter {
|
class mdc_formatter : public flag_formatter {
|
||||||
public:
|
public:
|
||||||
@ -824,6 +832,7 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
// Full info formatter
|
// Full info formatter
|
||||||
// pattern: [%Y-%m-%d %H:%M:%S.%e] [%n] [%l] [%s:%#] %v
|
// pattern: [%Y-%m-%d %H:%M:%S.%e] [%n] [%l] [%s:%#] %v
|
||||||
@ -901,6 +910,7 @@ public:
|
|||||||
dest.push_back(' ');
|
dest.push_back(' ');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
// add mdc if present
|
// add mdc if present
|
||||||
auto &mdc_map = mdc::get_context();
|
auto &mdc_map = mdc::get_context();
|
||||||
if (!mdc_map.empty()) {
|
if (!mdc_map.empty()) {
|
||||||
@ -909,6 +919,7 @@ public:
|
|||||||
dest.push_back(']');
|
dest.push_back(']');
|
||||||
dest.push_back(' ');
|
dest.push_back(' ');
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
// fmt_helper::append_string_view(msg.msg(), dest);
|
// fmt_helper::append_string_view(msg.msg(), dest);
|
||||||
fmt_helper::append_string_view(msg.payload, dest);
|
fmt_helper::append_string_view(msg.payload, dest);
|
||||||
}
|
}
|
||||||
@ -916,7 +927,10 @@ public:
|
|||||||
private:
|
private:
|
||||||
std::chrono::seconds cache_timestamp_{0};
|
std::chrono::seconds cache_timestamp_{0};
|
||||||
memory_buf_t cached_datetime_;
|
memory_buf_t cached_datetime_;
|
||||||
mdc_formatter<null_scoped_padder> mdc_formatter_{padding_info{}};
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
|
mdc_formatter<null_scoped_padder> mdc_formatter_{padding_info {}};
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace details
|
} // namespace details
|
||||||
@ -1211,9 +1225,11 @@ SPDLOG_INLINE void pattern_formatter::handle_flag_(char flag, details::padding_i
|
|||||||
padding));
|
padding));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
#ifndef SPDLOG_NO_TLS // mdc formatter requires TLS support
|
||||||
case ('&'):
|
case ('&'):
|
||||||
formatters_.push_back(details::make_unique<details::mdc_formatter<Padder>>(padding));
|
formatters_.push_back(details::make_unique<details::mdc_formatter<Padder>>(padding));
|
||||||
break;
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
default: // Unknown flag appears as is
|
default: // Unknown flag appears as is
|
||||||
auto unknown_flag = details::make_unique<details::aggregate_formatter>();
|
auto unknown_flag = details::make_unique<details::aggregate_formatter>();
|
||||||
|
@ -20,7 +20,7 @@ SPDLOG_INLINE ansicolor_sink<ConsoleMutex>::ansicolor_sink(FILE *target_file, co
|
|||||||
formatter_(details::make_unique<spdlog::pattern_formatter>())
|
formatter_(details::make_unique<spdlog::pattern_formatter>())
|
||||||
|
|
||||||
{
|
{
|
||||||
set_color_mode(mode);
|
set_color_mode_(mode);
|
||||||
colors_.at(level::trace) = to_string_(white);
|
colors_.at(level::trace) = to_string_(white);
|
||||||
colors_.at(level::debug) = to_string_(cyan);
|
colors_.at(level::debug) = to_string_(cyan);
|
||||||
colors_.at(level::info) = to_string_(green);
|
colors_.at(level::info) = to_string_(green);
|
||||||
@ -82,12 +82,18 @@ SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::set_formatter(
|
|||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
SPDLOG_INLINE bool ansicolor_sink<ConsoleMutex>::should_color() {
|
SPDLOG_INLINE bool ansicolor_sink<ConsoleMutex>::should_color() const {
|
||||||
return should_do_colors_;
|
return should_do_colors_;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::set_color_mode(color_mode mode) {
|
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::set_color_mode(color_mode mode) {
|
||||||
|
std::lock_guard<mutex_t> lock(mutex_);
|
||||||
|
set_color_mode_(mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename ConsoleMutex>
|
||||||
|
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::set_color_mode_(color_mode mode) {
|
||||||
switch (mode) {
|
switch (mode) {
|
||||||
case color_mode::always:
|
case color_mode::always:
|
||||||
should_do_colors_ = true;
|
should_do_colors_ = true;
|
||||||
@ -105,15 +111,15 @@ SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::set_color_mode(color_mode mode)
|
|||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::print_ccode_(const string_view_t &color_code) {
|
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::print_ccode_(const string_view_t &color_code) const {
|
||||||
fwrite(color_code.data(), sizeof(char), color_code.size(), target_file_);
|
details::os::fwrite_bytes(color_code.data(), color_code.size(), target_file_);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::print_range_(const memory_buf_t &formatted,
|
SPDLOG_INLINE void ansicolor_sink<ConsoleMutex>::print_range_(const memory_buf_t &formatted,
|
||||||
size_t start,
|
size_t start,
|
||||||
size_t end) {
|
size_t end) const {
|
||||||
fwrite(formatted.data() + start, sizeof(char), end - start, target_file_);
|
details::os::fwrite_bytes(formatted.data() + start, end - start, target_file_);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
|
@ -36,11 +36,11 @@ public:
|
|||||||
|
|
||||||
void set_color(level::level_enum color_level, string_view_t color);
|
void set_color(level::level_enum color_level, string_view_t color);
|
||||||
void set_color_mode(color_mode mode);
|
void set_color_mode(color_mode mode);
|
||||||
bool should_color();
|
bool should_color() const;
|
||||||
|
|
||||||
void log(const details::log_msg &msg) override;
|
void log(const details::log_msg &msg) override;
|
||||||
void flush() override;
|
void flush() override;
|
||||||
void set_pattern(const std::string &pattern) final;
|
void set_pattern(const std::string &pattern) final override;
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override;
|
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override;
|
||||||
|
|
||||||
// Formatting codes
|
// Formatting codes
|
||||||
@ -84,8 +84,9 @@ private:
|
|||||||
bool should_do_colors_;
|
bool should_do_colors_;
|
||||||
std::unique_ptr<spdlog::formatter> formatter_;
|
std::unique_ptr<spdlog::formatter> formatter_;
|
||||||
std::array<std::string, level::n_levels> colors_;
|
std::array<std::string, level::n_levels> colors_;
|
||||||
void print_ccode_(const string_view_t &color_code);
|
void set_color_mode_(color_mode mode);
|
||||||
void print_range_(const memory_buf_t &formatted, size_t start, size_t end);
|
void print_ccode_(const string_view_t &color_code) const;
|
||||||
|
void print_range_(const memory_buf_t &formatted, size_t start, size_t end) const;
|
||||||
static std::string to_string_(const string_view_t &sv);
|
static std::string to_string_(const string_view_t &sv);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -28,10 +28,10 @@ public:
|
|||||||
base_sink &operator=(const base_sink &) = delete;
|
base_sink &operator=(const base_sink &) = delete;
|
||||||
base_sink &operator=(base_sink &&) = delete;
|
base_sink &operator=(base_sink &&) = delete;
|
||||||
|
|
||||||
void log(const details::log_msg &msg) final;
|
void log(const details::log_msg &msg) final override;
|
||||||
void flush() final;
|
void flush() final override;
|
||||||
void set_pattern(const std::string &pattern) final;
|
void set_pattern(const std::string &pattern) final override;
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) final;
|
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) final override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// sink formatter
|
// sink formatter
|
||||||
|
@ -26,6 +26,12 @@ SPDLOG_INLINE const filename_t &basic_file_sink<Mutex>::filename() const {
|
|||||||
return file_helper_.filename();
|
return file_helper_.filename();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Mutex>
|
||||||
|
SPDLOG_INLINE void basic_file_sink<Mutex>::truncate() {
|
||||||
|
std::lock_guard<Mutex> lock(base_sink<Mutex>::mutex_);
|
||||||
|
file_helper_.reopen(true);
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Mutex>
|
template <typename Mutex>
|
||||||
SPDLOG_INLINE void basic_file_sink<Mutex>::sink_it_(const details::log_msg &msg) {
|
SPDLOG_INLINE void basic_file_sink<Mutex>::sink_it_(const details::log_msg &msg) {
|
||||||
memory_buf_t formatted;
|
memory_buf_t formatted;
|
||||||
|
@ -23,6 +23,7 @@ public:
|
|||||||
bool truncate = false,
|
bool truncate = false,
|
||||||
const file_event_handlers &event_handlers = {});
|
const file_event_handlers &event_handlers = {});
|
||||||
const filename_t &filename() const;
|
const filename_t &filename() const;
|
||||||
|
void truncate();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void sink_it_(const details::log_msg &msg) override;
|
void sink_it_(const details::log_msg &msg) override;
|
||||||
|
@ -27,7 +27,7 @@ public:
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
void sink_it_(const details::log_msg &msg) override { callback_(msg); }
|
void sink_it_(const details::log_msg &msg) override { callback_(msg); }
|
||||||
void flush_() override{};
|
void flush_() override{}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
custom_log_callback callback_;
|
custom_log_callback callback_;
|
||||||
|
@ -62,6 +62,8 @@ struct daily_filename_format_calculator {
|
|||||||
* Rotating file sink based on date.
|
* Rotating file sink based on date.
|
||||||
* If truncate != false , the created file will be truncated.
|
* If truncate != false , the created file will be truncated.
|
||||||
* If max_files > 0, retain only the last max_files and delete previous.
|
* If max_files > 0, retain only the last max_files and delete previous.
|
||||||
|
* Note that old log files from previous executions will not be deleted by this class,
|
||||||
|
* rotation and deletion is only applied while the program is running.
|
||||||
*/
|
*/
|
||||||
template <typename Mutex, typename FileNameCalc = daily_filename_calculator>
|
template <typename Mutex, typename FileNameCalc = daily_filename_calculator>
|
||||||
class daily_file_sink final : public base_sink<Mutex> {
|
class daily_file_sink final : public base_sink<Mutex> {
|
||||||
|
@ -40,22 +40,21 @@ template <typename Mutex>
|
|||||||
class dup_filter_sink : public dist_sink<Mutex> {
|
class dup_filter_sink : public dist_sink<Mutex> {
|
||||||
public:
|
public:
|
||||||
template <class Rep, class Period>
|
template <class Rep, class Period>
|
||||||
explicit dup_filter_sink(std::chrono::duration<Rep, Period> max_skip_duration,
|
explicit dup_filter_sink(std::chrono::duration<Rep, Period> max_skip_duration)
|
||||||
level::level_enum notification_level = level::info)
|
: max_skip_duration_{max_skip_duration} {}
|
||||||
: max_skip_duration_{max_skip_duration},
|
|
||||||
log_level_{notification_level} {}
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::chrono::microseconds max_skip_duration_;
|
std::chrono::microseconds max_skip_duration_;
|
||||||
log_clock::time_point last_msg_time_;
|
log_clock::time_point last_msg_time_;
|
||||||
std::string last_msg_payload_;
|
std::string last_msg_payload_;
|
||||||
size_t skip_counter_ = 0;
|
size_t skip_counter_ = 0;
|
||||||
level::level_enum log_level_;
|
level::level_enum skipped_msg_log_level_ = spdlog::level::level_enum::off;
|
||||||
|
|
||||||
void sink_it_(const details::log_msg &msg) override {
|
void sink_it_(const details::log_msg &msg) override {
|
||||||
bool filtered = filter_(msg);
|
bool filtered = filter_(msg);
|
||||||
if (!filtered) {
|
if (!filtered) {
|
||||||
skip_counter_ += 1;
|
skip_counter_ += 1;
|
||||||
|
skipped_msg_log_level_ = msg.level;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,7 +64,7 @@ protected:
|
|||||||
auto msg_size = ::snprintf(buf, sizeof(buf), "Skipped %u duplicate messages..",
|
auto msg_size = ::snprintf(buf, sizeof(buf), "Skipped %u duplicate messages..",
|
||||||
static_cast<unsigned>(skip_counter_));
|
static_cast<unsigned>(skip_counter_));
|
||||||
if (msg_size > 0 && static_cast<size_t>(msg_size) < sizeof(buf)) {
|
if (msg_size > 0 && static_cast<size_t>(msg_size) < sizeof(buf)) {
|
||||||
details::log_msg skipped_msg{msg.source, msg.logger_name, log_level_,
|
details::log_msg skipped_msg{msg.source, msg.logger_name, skipped_msg_log_level_,
|
||||||
string_view_t{buf, static_cast<size_t>(msg_size)}};
|
string_view_t{buf, static_cast<size_t>(msg_size)}};
|
||||||
dist_sink<Mutex>::sink_it_(skipped_msg);
|
dist_sink<Mutex>::sink_it_(skipped_msg);
|
||||||
}
|
}
|
||||||
|
@ -39,6 +39,8 @@ struct hourly_filename_calculator {
|
|||||||
* Rotating file sink based on time.
|
* Rotating file sink based on time.
|
||||||
* If truncate != false , the created file will be truncated.
|
* If truncate != false , the created file will be truncated.
|
||||||
* If max_files > 0, retain only the last max_files and delete previous.
|
* If max_files > 0, retain only the last max_files and delete previous.
|
||||||
|
* Note that old log files from previous executions will not be deleted by this class,
|
||||||
|
* rotation and deletion is only applied while the program is running.
|
||||||
*/
|
*/
|
||||||
template <typename Mutex, typename FileNameCalc = hourly_filename_calculator>
|
template <typename Mutex, typename FileNameCalc = hourly_filename_calculator>
|
||||||
class hourly_file_sink final : public base_sink<Mutex> {
|
class hourly_file_sink final : public base_sink<Mutex> {
|
||||||
|
@ -32,7 +32,7 @@ class msvc_sink : public base_sink<Mutex> {
|
|||||||
public:
|
public:
|
||||||
msvc_sink() = default;
|
msvc_sink() = default;
|
||||||
msvc_sink(bool check_debugger_present)
|
msvc_sink(bool check_debugger_present)
|
||||||
: check_debugger_present_{check_debugger_present} {};
|
: check_debugger_present_{check_debugger_present} {}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void sink_it_(const details::log_msg &msg) override {
|
void sink_it_(const details::log_msg &msg) override {
|
||||||
|
@ -13,7 +13,7 @@ namespace spdlog {
|
|||||||
namespace sinks {
|
namespace sinks {
|
||||||
|
|
||||||
template <typename Mutex>
|
template <typename Mutex>
|
||||||
class null_sink : public base_sink<Mutex> {
|
class null_sink final : public base_sink<Mutex> {
|
||||||
protected:
|
protected:
|
||||||
void sink_it_(const details::log_msg &) override {}
|
void sink_it_(const details::log_msg &) override {}
|
||||||
void flush_() override {}
|
void flush_() override {}
|
||||||
|
@ -60,7 +60,7 @@ SPDLOG_INLINE filename_t rotating_file_sink<Mutex>::calc_filename(const filename
|
|||||||
|
|
||||||
filename_t basename, ext;
|
filename_t basename, ext;
|
||||||
std::tie(basename, ext) = details::file_helper::split_by_extension(filename);
|
std::tie(basename, ext) = details::file_helper::split_by_extension(filename);
|
||||||
return fmt_lib::format(SPDLOG_FILENAME_T("{}.{}{}"), basename, index, ext);
|
return fmt_lib::format(SPDLOG_FMT_STRING(SPDLOG_FILENAME_T("{}.{}{}")), basename, index, ext);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Mutex>
|
template <typename Mutex>
|
||||||
@ -69,6 +69,12 @@ SPDLOG_INLINE filename_t rotating_file_sink<Mutex>::filename() {
|
|||||||
return file_helper_.filename();
|
return file_helper_.filename();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Mutex>
|
||||||
|
SPDLOG_INLINE void rotating_file_sink<Mutex>::rotate_now() {
|
||||||
|
std::lock_guard<Mutex> lock(base_sink<Mutex>::mutex_);
|
||||||
|
rotate_();
|
||||||
|
}
|
||||||
|
|
||||||
template <typename Mutex>
|
template <typename Mutex>
|
||||||
SPDLOG_INLINE void rotating_file_sink<Mutex>::sink_it_(const details::log_msg &msg) {
|
SPDLOG_INLINE void rotating_file_sink<Mutex>::sink_it_(const details::log_msg &msg) {
|
||||||
memory_buf_t formatted;
|
memory_buf_t formatted;
|
||||||
|
@ -28,6 +28,7 @@ public:
|
|||||||
const file_event_handlers &event_handlers = {});
|
const file_event_handlers &event_handlers = {});
|
||||||
static filename_t calc_filename(const filename_t &filename, std::size_t index);
|
static filename_t calc_filename(const filename_t &filename, std::size_t index);
|
||||||
filename_t filename();
|
filename_t filename();
|
||||||
|
void rotate_now();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void sink_it_(const details::log_msg &msg) override;
|
void sink_it_(const details::log_msg &msg) override;
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <spdlog/details/console_globals.h>
|
#include <spdlog/details/console_globals.h>
|
||||||
#include <spdlog/pattern_formatter.h>
|
#include <spdlog/pattern_formatter.h>
|
||||||
|
#include <spdlog/details/os.h>
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
// under windows using fwrite to non-binary stream results in \r\r\n (see issue #1675)
|
// under windows using fwrite to non-binary stream results in \r\r\n (see issue #1675)
|
||||||
@ -22,7 +23,7 @@
|
|||||||
|
|
||||||
#include <io.h> // _get_osfhandle(..)
|
#include <io.h> // _get_osfhandle(..)
|
||||||
#include <stdio.h> // _fileno(..)
|
#include <stdio.h> // _fileno(..)
|
||||||
#endif // WIN32
|
#endif // _WIN32
|
||||||
|
|
||||||
namespace spdlog {
|
namespace spdlog {
|
||||||
|
|
||||||
@ -44,7 +45,7 @@ SPDLOG_INLINE stdout_sink_base<ConsoleMutex>::stdout_sink_base(FILE *file)
|
|||||||
if (handle_ == INVALID_HANDLE_VALUE && file != stdout && file != stderr) {
|
if (handle_ == INVALID_HANDLE_VALUE && file != stdout && file != stderr) {
|
||||||
throw_spdlog_ex("spdlog::stdout_sink_base: _get_osfhandle() failed", errno);
|
throw_spdlog_ex("spdlog::stdout_sink_base: _get_osfhandle() failed", errno);
|
||||||
}
|
}
|
||||||
#endif // WIN32
|
#endif // _WIN32
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ConsoleMutex>
|
template <typename ConsoleMutex>
|
||||||
@ -67,8 +68,8 @@ SPDLOG_INLINE void stdout_sink_base<ConsoleMutex>::log(const details::log_msg &m
|
|||||||
std::lock_guard<mutex_t> lock(mutex_);
|
std::lock_guard<mutex_t> lock(mutex_);
|
||||||
memory_buf_t formatted;
|
memory_buf_t formatted;
|
||||||
formatter_->format(msg, formatted);
|
formatter_->format(msg, formatted);
|
||||||
::fwrite(formatted.data(), sizeof(char), formatted.size(), file_);
|
details::os::fwrite_bytes(formatted.data(), formatted.size(), file_);
|
||||||
#endif // WIN32
|
#endif // _WIN32
|
||||||
::fflush(file_); // flush every line to terminal
|
::fflush(file_); // flush every line to terminal
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -64,13 +64,14 @@ protected:
|
|||||||
//
|
//
|
||||||
// Simply maps spdlog's log level to syslog priority level.
|
// Simply maps spdlog's log level to syslog priority level.
|
||||||
//
|
//
|
||||||
int syslog_prio_from_level(const details::log_msg &msg) const {
|
virtual int syslog_prio_from_level(const details::log_msg &msg) const {
|
||||||
return syslog_levels_.at(static_cast<levels_array::size_type>(msg.level));
|
return syslog_levels_.at(static_cast<levels_array::size_type>(msg.level));
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
|
||||||
using levels_array = std::array<int, 7>;
|
using levels_array = std::array<int, 7>;
|
||||||
levels_array syslog_levels_;
|
levels_array syslog_levels_;
|
||||||
|
|
||||||
|
private:
|
||||||
// must store the ident because the man says openlog might use the pointer as
|
// must store the ident because the man says openlog might use the pointer as
|
||||||
// is and not a string copy
|
// is and not a string copy
|
||||||
const std::string ident_;
|
const std::string ident_;
|
||||||
|
@ -134,9 +134,18 @@ void SPDLOG_INLINE wincolor_sink<ConsoleMutex>::print_range_(const memory_buf_t
|
|||||||
size_t start,
|
size_t start,
|
||||||
size_t end) {
|
size_t end) {
|
||||||
if (end > start) {
|
if (end > start) {
|
||||||
|
#if defined(SPDLOG_UTF8_TO_WCHAR_CONSOLE)
|
||||||
|
wmemory_buf_t wformatted;
|
||||||
|
details::os::utf8_to_wstrbuf(string_view_t(formatted.data() + start, end - start),
|
||||||
|
wformatted);
|
||||||
|
auto size = static_cast<DWORD>(wformatted.size());
|
||||||
|
auto ignored = ::WriteConsoleW(static_cast<HANDLE>(out_handle_), wformatted.data(), size,
|
||||||
|
nullptr, nullptr);
|
||||||
|
#else
|
||||||
auto size = static_cast<DWORD>(end - start);
|
auto size = static_cast<DWORD>(end - start);
|
||||||
auto ignored = ::WriteConsoleA(static_cast<HANDLE>(out_handle_), formatted.data() + start,
|
auto ignored = ::WriteConsoleA(static_cast<HANDLE>(out_handle_), formatted.data() + start,
|
||||||
size, nullptr, nullptr);
|
size, nullptr, nullptr);
|
||||||
|
#endif
|
||||||
(void)(ignored);
|
(void)(ignored);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -104,6 +104,13 @@
|
|||||||
//
|
//
|
||||||
// #define SPDLOG_LEVEL_NAMES { "MY TRACE", "MY DEBUG", "MY INFO", "MY WARNING", "MY ERROR", "MY
|
// #define SPDLOG_LEVEL_NAMES { "MY TRACE", "MY DEBUG", "MY INFO", "MY WARNING", "MY ERROR", "MY
|
||||||
// CRITICAL", "OFF" }
|
// CRITICAL", "OFF" }
|
||||||
|
//
|
||||||
|
// For C++17 use string_view_literals:
|
||||||
|
//
|
||||||
|
// #include <string_view>
|
||||||
|
// using namespace std::string_view_literals;
|
||||||
|
// #define SPDLOG_LEVEL_NAMES { "MY TRACE"sv, "MY DEBUG"sv, "MY INFO"sv, "MY WARNING"sv, "MY ERROR"sv, "MY
|
||||||
|
// CRITICAL"sv, "OFF"sv }
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -4,8 +4,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#define SPDLOG_VER_MAJOR 1
|
#define SPDLOG_VER_MAJOR 1
|
||||||
#define SPDLOG_VER_MINOR 14
|
#define SPDLOG_VER_MINOR 15
|
||||||
#define SPDLOG_VER_PATCH 1
|
#define SPDLOG_VER_PATCH 2
|
||||||
|
|
||||||
#define SPDLOG_TO_VERSION(major, minor, patch) (major * 10000 + minor * 100 + patch)
|
#define SPDLOG_TO_VERSION(major, minor, patch) (major * 10000 + minor * 100 + patch)
|
||||||
#define SPDLOG_VERSION SPDLOG_TO_VERSION(SPDLOG_VER_MAJOR, SPDLOG_VER_MINOR, SPDLOG_VER_PATCH)
|
#define SPDLOG_VERSION SPDLOG_TO_VERSION(SPDLOG_VER_MAJOR, SPDLOG_VER_MINOR, SPDLOG_VER_PATCH)
|
||||||
|
@ -13,29 +13,34 @@
|
|||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
template FMT_API auto dragonbox::to_decimal(float x) noexcept -> dragonbox::decimal_fp<float>;
|
template FMT_API auto dragonbox::to_decimal(float x) noexcept
|
||||||
template FMT_API auto dragonbox::to_decimal(double x) noexcept -> dragonbox::decimal_fp<double>;
|
-> dragonbox::decimal_fp<float>;
|
||||||
|
template FMT_API auto dragonbox::to_decimal(double x) noexcept
|
||||||
|
-> dragonbox::decimal_fp<double>;
|
||||||
|
|
||||||
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
|
#if FMT_USE_LOCALE
|
||||||
|
// DEPRECATED! locale_ref in the detail namespace
|
||||||
template FMT_API locale_ref::locale_ref(const std::locale& loc);
|
template FMT_API locale_ref::locale_ref(const std::locale& loc);
|
||||||
template FMT_API auto locale_ref::get<std::locale>() const -> std::locale;
|
template FMT_API auto locale_ref::get<std::locale>() const -> std::locale;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Explicit instantiations for char.
|
// Explicit instantiations for char.
|
||||||
|
|
||||||
template FMT_API auto thousands_sep_impl(locale_ref) -> thousands_sep_result<char>;
|
template FMT_API auto thousands_sep_impl(locale_ref)
|
||||||
|
-> 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>&,
|
// DEPRECATED!
|
||||||
string_view,
|
template FMT_API void vformat_to(buffer<char>&, string_view,
|
||||||
typename vformat_args<>::type,
|
typename vformat_args<>::type, locale_ref);
|
||||||
locale_ref);
|
|
||||||
|
|
||||||
// Explicit instantiations for wchar_t.
|
// Explicit instantiations for wchar_t.
|
||||||
|
|
||||||
template FMT_API auto thousands_sep_impl(locale_ref) -> thousands_sep_result<wchar_t>;
|
template FMT_API auto thousands_sep_impl(locale_ref)
|
||||||
|
-> 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;
|
||||||
|
|
||||||
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*);
|
||||||
@ -43,4 +48,5 @@ template FMT_API void buffer<wchar_t>::append(const wchar_t*, const wchar_t*);
|
|||||||
} // namespace detail
|
} // namespace detail
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
|
||||||
|
|
||||||
#endif // !SPDLOG_FMT_EXTERNAL
|
#endif // !SPDLOG_FMT_EXTERNAL
|
||||||
|
@ -48,7 +48,8 @@ set(SPDLOG_UTESTS_SOURCES
|
|||||||
test_cfg.cpp
|
test_cfg.cpp
|
||||||
test_time_point.cpp
|
test_time_point.cpp
|
||||||
test_stopwatch.cpp
|
test_stopwatch.cpp
|
||||||
test_circular_q.cpp)
|
test_circular_q.cpp
|
||||||
|
test_bin_to_hex.cpp)
|
||||||
|
|
||||||
if(NOT SPDLOG_NO_EXCEPTIONS)
|
if(NOT SPDLOG_NO_EXCEPTIONS)
|
||||||
list(APPEND SPDLOG_UTESTS_SOURCES test_errors.cpp)
|
list(APPEND SPDLOG_UTESTS_SOURCES test_errors.cpp)
|
||||||
@ -58,10 +59,6 @@ if(systemd_FOUND)
|
|||||||
list(APPEND SPDLOG_UTESTS_SOURCES test_systemd.cpp)
|
list(APPEND SPDLOG_UTESTS_SOURCES test_systemd.cpp)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(NOT SPDLOG_USE_STD_FORMAT)
|
|
||||||
list(APPEND SPDLOG_UTESTS_SOURCES test_bin_to_hex.cpp)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
enable_testing()
|
enable_testing()
|
||||||
|
|
||||||
function(spdlog_prepare_test test_target spdlog_lib)
|
function(spdlog_prepare_test test_target spdlog_lib)
|
||||||
@ -73,8 +70,10 @@ function(spdlog_prepare_test test_target spdlog_lib)
|
|||||||
endif()
|
endif()
|
||||||
target_link_libraries(${test_target} PRIVATE Catch2::Catch2WithMain)
|
target_link_libraries(${test_target} PRIVATE Catch2::Catch2WithMain)
|
||||||
if(SPDLOG_SANITIZE_ADDRESS)
|
if(SPDLOG_SANITIZE_ADDRESS)
|
||||||
spdlog_enable_sanitizer(${test_target})
|
spdlog_enable_addr_sanitizer(${test_target})
|
||||||
endif()
|
elseif (SPDLOG_SANITIZE_THREAD)
|
||||||
|
spdlog_enable_thread_sanitizer(${test_target})
|
||||||
|
endif ()
|
||||||
add_test(NAME ${test_target} COMMAND ${test_target})
|
add_test(NAME ${test_target} COMMAND ${test_target})
|
||||||
set_tests_properties(${test_target} PROPERTIES RUN_SERIAL ON)
|
set_tests_properties(${test_target} PROPERTIES RUN_SERIAL ON)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
@ -26,7 +26,12 @@
|
|||||||
#include "spdlog/spdlog.h"
|
#include "spdlog/spdlog.h"
|
||||||
#include "spdlog/async.h"
|
#include "spdlog/async.h"
|
||||||
#include "spdlog/details/fmt_helper.h"
|
#include "spdlog/details/fmt_helper.h"
|
||||||
#include "spdlog/mdc.h"
|
#include "spdlog/details/os.h"
|
||||||
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
|
#include "spdlog/mdc.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "spdlog/sinks/basic_file_sink.h"
|
#include "spdlog/sinks/basic_file_sink.h"
|
||||||
#include "spdlog/sinks/daily_file_sink.h"
|
#include "spdlog/sinks/daily_file_sink.h"
|
||||||
#include "spdlog/sinks/null_sink.h"
|
#include "spdlog/sinks/null_sink.h"
|
||||||
|
@ -93,50 +93,6 @@ TEST_CASE("flush", "[async]") {
|
|||||||
REQUIRE(test_sink->flush_counter() == 1);
|
REQUIRE(test_sink->flush_counter() == 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("multithread flush", "[async]") {
|
|
||||||
auto test_sink = std::make_shared<spdlog::sinks::test_sink_mt>();
|
|
||||||
size_t queue_size = 2;
|
|
||||||
size_t messages = 10;
|
|
||||||
size_t n_threads = 10;
|
|
||||||
size_t flush_count = 1024;
|
|
||||||
std::mutex mtx;
|
|
||||||
std::vector<std::string> errmsgs;
|
|
||||||
{
|
|
||||||
auto tp = std::make_shared<spdlog::details::thread_pool>(queue_size, 1);
|
|
||||||
auto logger = std::make_shared<spdlog::async_logger>(
|
|
||||||
"as", test_sink, tp, spdlog::async_overflow_policy::discard_new);
|
|
||||||
|
|
||||||
logger->set_error_handler([&](const std::string &) {
|
|
||||||
std::unique_lock<std::mutex> lock(mtx);
|
|
||||||
errmsgs.push_back("Broken promise");
|
|
||||||
});
|
|
||||||
|
|
||||||
for (size_t i = 0; i < messages; i++) {
|
|
||||||
logger->info("Hello message #{}", i);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<std::thread> threads;
|
|
||||||
for (size_t i = 0; i < n_threads; i++) {
|
|
||||||
threads.emplace_back([logger, flush_count] {
|
|
||||||
for (size_t j = 0; j < flush_count; j++) {
|
|
||||||
// flush does not throw exception even if failed.
|
|
||||||
// Instead, the error handler is invoked.
|
|
||||||
logger->flush();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &t : threads) {
|
|
||||||
t.join();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
REQUIRE(test_sink->flush_counter() >= 1);
|
|
||||||
REQUIRE(test_sink->flush_counter() + errmsgs.size() == n_threads * flush_count);
|
|
||||||
if (errmsgs.size() > 0) {
|
|
||||||
REQUIRE(errmsgs[0] == "Broken promise");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST_CASE("async periodic flush", "[async]") {
|
TEST_CASE("async periodic flush", "[async]") {
|
||||||
auto logger = spdlog::create_async<spdlog::sinks::test_sink_mt>("as");
|
auto logger = spdlog::create_async<spdlog::sinks::test_sink_mt>("as");
|
||||||
auto test_sink = std::static_pointer_cast<spdlog::sinks::test_sink_mt>(logger->sinks()[0]);
|
auto test_sink = std::static_pointer_cast<spdlog::sinks::test_sink_mt>(logger->sinks()[0]);
|
||||||
|
@ -19,6 +19,15 @@ TEST_CASE("env", "[cfg]") {
|
|||||||
#endif
|
#endif
|
||||||
load_env_levels();
|
load_env_levels();
|
||||||
REQUIRE(l1->level() == spdlog::level::warn);
|
REQUIRE(l1->level() == spdlog::level::warn);
|
||||||
|
|
||||||
|
#ifdef CATCH_PLATFORM_WINDOWS
|
||||||
|
_putenv_s("MYAPP_LEVEL", "l1=trace");
|
||||||
|
#else
|
||||||
|
setenv("MYAPP_LEVEL", "l1=trace", 1);
|
||||||
|
#endif
|
||||||
|
load_env_levels("MYAPP_LEVEL");
|
||||||
|
REQUIRE(l1->level() == spdlog::level::trace);
|
||||||
|
|
||||||
spdlog::set_default_logger(spdlog::create<test_sink_st>("cfg-default"));
|
spdlog::set_default_logger(spdlog::create<test_sink_st>("cfg-default"));
|
||||||
REQUIRE(spdlog::default_logger()->level() == spdlog::level::info);
|
REQUIRE(spdlog::default_logger()->level() == spdlog::level::info);
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,8 @@ TEST_CASE("custom_callback_logger", "[custom_callback_logger]") {
|
|||||||
spdlog::memory_buf_t formatted;
|
spdlog::memory_buf_t formatted;
|
||||||
formatter.format(msg, formatted);
|
formatter.format(msg, formatted);
|
||||||
auto eol_len = strlen(spdlog::details::os::default_eol);
|
auto eol_len = strlen(spdlog::details::os::default_eol);
|
||||||
lines.emplace_back(formatted.begin(), formatted.end() - eol_len);
|
using diff_t = typename std::iterator_traits<decltype(formatted.end())>::difference_type;
|
||||||
|
lines.emplace_back(formatted.begin(), formatted.end() - static_cast<diff_t>(eol_len));
|
||||||
});
|
});
|
||||||
std::shared_ptr<spdlog::sinks::test_sink_st> test_sink(new spdlog::sinks::test_sink_st);
|
std::shared_ptr<spdlog::sinks::test_sink_st> test_sink(new spdlog::sinks::test_sink_st);
|
||||||
|
|
||||||
|
@ -46,12 +46,10 @@ TEST_CASE("daily_logger with dateonly calculator", "[daily_logger]") {
|
|||||||
|
|
||||||
struct custom_daily_file_name_calculator {
|
struct custom_daily_file_name_calculator {
|
||||||
static spdlog::filename_t calc_filename(const spdlog::filename_t &basename, const tm &now_tm) {
|
static spdlog::filename_t calc_filename(const spdlog::filename_t &basename, const tm &now_tm) {
|
||||||
filename_memory_buf_t w;
|
|
||||||
spdlog::fmt_lib::format_to(std::back_inserter(w), SPDLOG_FILENAME_T("{}{:04d}{:02d}{:02d}"),
|
return spdlog::fmt_lib::format(SPDLOG_FILENAME_T("{}{:04d}{:02d}{:02d}"),
|
||||||
basename, now_tm.tm_year + 1900, now_tm.tm_mon + 1,
|
basename, now_tm.tm_year + 1900, now_tm.tm_mon + 1,
|
||||||
now_tm.tm_mday);
|
now_tm.tm_mday);
|
||||||
|
|
||||||
return SPDLOG_BUF_TO_STRING(w);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -45,6 +45,26 @@ TEST_CASE("flush_on", "[flush_on]") {
|
|||||||
default_eol, default_eol, default_eol));
|
default_eol, default_eol, default_eol));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("simple_file_logger", "[truncate]") {
|
||||||
|
prepare_logdir();
|
||||||
|
const spdlog::filename_t filename = SPDLOG_FILENAME_T(SIMPLE_LOG);
|
||||||
|
const bool truncate = true;
|
||||||
|
const auto sink = std::make_shared<spdlog::sinks::basic_file_sink_mt>(filename, truncate);
|
||||||
|
const auto logger = std::make_shared<spdlog::logger>("simple_file_logger", sink);
|
||||||
|
|
||||||
|
logger->info("Test message {}", 3.14);
|
||||||
|
logger->info("Test message {}", 2.71);
|
||||||
|
logger->flush();
|
||||||
|
REQUIRE(count_lines(SIMPLE_LOG) == 2);
|
||||||
|
|
||||||
|
sink->truncate();
|
||||||
|
REQUIRE(count_lines(SIMPLE_LOG) == 0);
|
||||||
|
|
||||||
|
logger->info("Test message {}", 6.28);
|
||||||
|
logger->flush();
|
||||||
|
REQUIRE(count_lines(SIMPLE_LOG) == 1);
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE("rotating_file_logger1", "[rotating_logger]") {
|
TEST_CASE("rotating_file_logger1", "[rotating_logger]") {
|
||||||
prepare_logdir();
|
prepare_logdir();
|
||||||
size_t max_size = 1024 * 10;
|
size_t max_size = 1024 * 10;
|
||||||
@ -101,3 +121,23 @@ TEST_CASE("rotating_file_logger3", "[rotating_logger]") {
|
|||||||
REQUIRE_THROWS_AS(spdlog::rotating_logger_mt("logger", basename, max_size, 0),
|
REQUIRE_THROWS_AS(spdlog::rotating_logger_mt("logger", basename, max_size, 0),
|
||||||
spdlog::spdlog_ex);
|
spdlog::spdlog_ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// test on-demand rotation of logs
|
||||||
|
TEST_CASE("rotating_file_logger4", "[rotating_logger]") {
|
||||||
|
prepare_logdir();
|
||||||
|
size_t max_size = 1024 * 10;
|
||||||
|
spdlog::filename_t basename = SPDLOG_FILENAME_T(ROTATING_LOG);
|
||||||
|
auto sink = std::make_shared<spdlog::sinks::rotating_file_sink_st>(basename, max_size, 2);
|
||||||
|
auto logger = std::make_shared<spdlog::logger>("rotating_sink_logger", sink);
|
||||||
|
|
||||||
|
logger->info("Test message - pre-rotation");
|
||||||
|
logger->flush();
|
||||||
|
|
||||||
|
sink->rotate_now();
|
||||||
|
|
||||||
|
logger->info("Test message - post-rotation");
|
||||||
|
logger->flush();
|
||||||
|
|
||||||
|
REQUIRE(get_filesize(ROTATING_LOG) > 0);
|
||||||
|
REQUIRE(get_filesize(ROTATING_LOG ".1") > 0);
|
||||||
|
}
|
||||||
|
@ -1,3 +1,7 @@
|
|||||||
|
#ifdef _WIN32 // to prevent fopen warning on windows
|
||||||
|
#define _CRT_SECURE_NO_WARNINGS
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
#include "test_sink.h"
|
#include "test_sink.h"
|
||||||
|
|
||||||
@ -105,9 +109,9 @@ TEST_CASE("clone-logger", "[clone]") {
|
|||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("clone async", "[clone]") {
|
TEST_CASE("clone async", "[clone]") {
|
||||||
using spdlog::sinks::test_sink_st;
|
using spdlog::sinks::test_sink_mt;
|
||||||
spdlog::init_thread_pool(4, 1);
|
spdlog::init_thread_pool(4, 1);
|
||||||
auto test_sink = std::make_shared<test_sink_st>();
|
auto test_sink = std::make_shared<test_sink_mt>();
|
||||||
auto logger = std::make_shared<spdlog::async_logger>("orig", test_sink, spdlog::thread_pool());
|
auto logger = std::make_shared<spdlog::async_logger>("orig", test_sink, spdlog::thread_pool());
|
||||||
logger->set_pattern("%v");
|
logger->set_pattern("%v");
|
||||||
auto cloned = logger->clone("clone");
|
auto cloned = logger->clone("clone");
|
||||||
@ -167,3 +171,52 @@ TEST_CASE("default logger API", "[default logger]") {
|
|||||||
spdlog::drop_all();
|
spdlog::drop_all();
|
||||||
spdlog::set_pattern("%v");
|
spdlog::set_pattern("%v");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (defined(SPDLOG_WCHAR_TO_UTF8_SUPPORT) || defined(SPDLOG_WCHAR_FILENAMES)) && defined(_WIN32)
|
||||||
|
TEST_CASE("utf8 to utf16 conversion using windows api", "[windows utf]") {
|
||||||
|
spdlog::wmemory_buf_t buffer;
|
||||||
|
|
||||||
|
spdlog::details::os::utf8_to_wstrbuf("", buffer);
|
||||||
|
REQUIRE(std::wstring(buffer.data(), buffer.size()) == std::wstring(L""));
|
||||||
|
|
||||||
|
spdlog::details::os::utf8_to_wstrbuf("abc", buffer);
|
||||||
|
REQUIRE(std::wstring(buffer.data(), buffer.size()) == std::wstring(L"abc"));
|
||||||
|
|
||||||
|
spdlog::details::os::utf8_to_wstrbuf("\xc3\x28", buffer); // Invalid UTF-8 sequence.
|
||||||
|
REQUIRE(std::wstring(buffer.data(), buffer.size()) == std::wstring(L"\xfffd("));
|
||||||
|
|
||||||
|
spdlog::details::os::utf8_to_wstrbuf("\xe3\x81\xad\xe3\x81\x93", buffer); // "Neko" in hiragana.
|
||||||
|
REQUIRE(std::wstring(buffer.data(), buffer.size()) == std::wstring(L"\x306d\x3053"));
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct auto_closer {
|
||||||
|
FILE* fp = nullptr;
|
||||||
|
explicit auto_closer(FILE* f) : fp(f) {}
|
||||||
|
auto_closer(const auto_closer&) = delete;
|
||||||
|
auto_closer& operator=(const auto_closer&) = delete;
|
||||||
|
~auto_closer() {
|
||||||
|
if (fp != nullptr) (void)std::fclose(fp);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
TEST_CASE("os::fwrite_bytes", "[os]") {
|
||||||
|
using spdlog::details::os::fwrite_bytes;
|
||||||
|
using spdlog::details::os::create_dir;
|
||||||
|
const char* filename = "log_tests/test_fwrite_bytes.txt";
|
||||||
|
const char *msg = "hello";
|
||||||
|
prepare_logdir();
|
||||||
|
REQUIRE(create_dir(SPDLOG_FILENAME_T("log_tests")) == true);
|
||||||
|
{
|
||||||
|
auto_closer closer(std::fopen(filename, "wb"));
|
||||||
|
REQUIRE(closer.fp != nullptr);
|
||||||
|
REQUIRE(fwrite_bytes(msg, std::strlen(msg), closer.fp) == true);
|
||||||
|
REQUIRE(fwrite_bytes(msg, 0, closer.fp) == true);
|
||||||
|
std::fflush(closer.fp);
|
||||||
|
REQUIRE(spdlog::details::os::filesize(closer.fp) == 5);
|
||||||
|
}
|
||||||
|
// fwrite_bytes should return false on write failure
|
||||||
|
auto_closer closer(std::fopen(filename, "r"));
|
||||||
|
REQUIRE(closer.fp != nullptr);
|
||||||
|
REQUIRE_FALSE(fwrite_bytes("Hello", 5, closer.fp));
|
||||||
|
}
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
#include "includes.h"
|
#include "includes.h"
|
||||||
#include "test_sink.h"
|
#include "test_sink.h"
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
using spdlog::memory_buf_t;
|
using spdlog::memory_buf_t;
|
||||||
using spdlog::details::to_string_view;
|
using spdlog::details::to_string_view;
|
||||||
|
|
||||||
@ -19,6 +21,21 @@ static std::string log_to_str(const std::string &msg, const Args &...args) {
|
|||||||
return oss.str();
|
return oss.str();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// log to str and return it with time
|
||||||
|
template <typename... Args>
|
||||||
|
static std::string log_to_str_with_time(spdlog::log_clock::time_point log_time, const std::string &msg, const Args &...args) {
|
||||||
|
std::ostringstream oss;
|
||||||
|
auto oss_sink = std::make_shared<spdlog::sinks::ostream_sink_mt>(oss);
|
||||||
|
spdlog::logger oss_logger("pattern_tester", oss_sink);
|
||||||
|
oss_logger.set_level(spdlog::level::info);
|
||||||
|
|
||||||
|
oss_logger.set_formatter(
|
||||||
|
std::unique_ptr<spdlog::formatter>(new spdlog::pattern_formatter(args...)));
|
||||||
|
|
||||||
|
oss_logger.log(log_time, {}, spdlog::level::info, msg);
|
||||||
|
return oss.str();
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE("custom eol", "[pattern_formatter]") {
|
TEST_CASE("custom eol", "[pattern_formatter]") {
|
||||||
std::string msg = "Hello custom eol test";
|
std::string msg = "Hello custom eol test";
|
||||||
std::string eol = ";)";
|
std::string eol = ";)";
|
||||||
@ -58,6 +75,15 @@ TEST_CASE("date MM/DD/YY ", "[pattern_formatter]") {
|
|||||||
oss.str());
|
oss.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("GMT offset ", "[pattern_formatter]") {
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
const auto now = std::chrono::system_clock::now();
|
||||||
|
const auto yesterday = now - 24h;
|
||||||
|
|
||||||
|
REQUIRE(log_to_str_with_time(yesterday, "Some message", "%z", spdlog::pattern_time_type::utc, "\n") ==
|
||||||
|
"+00:00\n");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE("color range test1", "[pattern_formatter]") {
|
TEST_CASE("color range test1", "[pattern_formatter]") {
|
||||||
auto formatter = std::make_shared<spdlog::pattern_formatter>(
|
auto formatter = std::make_shared<spdlog::pattern_formatter>(
|
||||||
"%^%v%$", spdlog::pattern_time_type::local, "\n");
|
"%^%v%$", spdlog::pattern_time_type::local, "\n");
|
||||||
@ -501,6 +527,7 @@ TEST_CASE("override need_localtime", "[pattern_formatter]") {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef SPDLOG_NO_TLS
|
||||||
TEST_CASE("mdc formatter test-1", "[pattern_formatter]") {
|
TEST_CASE("mdc formatter test-1", "[pattern_formatter]") {
|
||||||
spdlog::mdc::put("mdc_key_1", "mdc_value_1");
|
spdlog::mdc::put("mdc_key_1", "mdc_value_1");
|
||||||
spdlog::mdc::put("mdc_key_2", "mdc_value_2");
|
spdlog::mdc::put("mdc_key_2", "mdc_value_2");
|
||||||
@ -628,3 +655,4 @@ TEST_CASE("mdc empty", "[pattern_formatter]") {
|
|||||||
|
|
||||||
SECTION("Tear down") { spdlog::mdc::clear(); }
|
SECTION("Tear down") { spdlog::mdc::clear(); }
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
@ -47,8 +47,9 @@ protected:
|
|||||||
base_sink<Mutex>::formatter_->format(msg, formatted);
|
base_sink<Mutex>::formatter_->format(msg, formatted);
|
||||||
// save the line without the eol
|
// save the line without the eol
|
||||||
auto eol_len = strlen(details::os::default_eol);
|
auto eol_len = strlen(details::os::default_eol);
|
||||||
|
using diff_t = typename std::iterator_traits<decltype(formatted.end())>::difference_type;
|
||||||
if (lines_.size() < lines_to_save) {
|
if (lines_.size() < lines_to_save) {
|
||||||
lines_.emplace_back(formatted.begin(), formatted.end() - eol_len);
|
lines_.emplace_back(formatted.begin(), formatted.end() - static_cast<diff_t>(eol_len));
|
||||||
}
|
}
|
||||||
msg_counter_++;
|
msg_counter_++;
|
||||||
std::this_thread::sleep_for(delay_);
|
std::this_thread::sleep_for(delay_);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user