Compare commits
39 commits
Author | SHA1 | Date | |
---|---|---|---|
ad4ba073a3 | |||
![]() |
6c8e87da57 | ||
![]() |
3f1664327a | ||
![]() |
174b23d76e | ||
![]() |
3d4c9820fc | ||
![]() |
72a9659576 | ||
![]() |
72ce6c9ae0 | ||
![]() |
4d097b9da0 | ||
![]() |
e86b94cb8d | ||
![]() |
c9ae9a57c3 | ||
![]() |
5443409085 | ||
![]() |
8aeb677413 | ||
![]() |
f2522c2cc0 | ||
![]() |
8a6813f242 | ||
![]() |
31ac2b3273 | ||
![]() |
8f416c881e | ||
![]() |
6708611448 | ||
![]() |
1f071407dc | ||
![]() |
9494cbd4aa | ||
![]() |
84ce6d87fb | ||
![]() |
173271ef06 | ||
![]() |
a83b66aac9 | ||
![]() |
34d8c2b74e | ||
![]() |
1cc3feb884 | ||
![]() |
b877a1a556 | ||
![]() |
bc183cd334 | ||
![]() |
70ca094db8 | ||
![]() |
f7b2f34053 | ||
![]() |
84c2d8bb19 | ||
![]() |
6a1efd31c1 | ||
![]() |
ea3871026a | ||
![]() |
54a6a795a9 | ||
![]() |
f14c1b5065 | ||
![]() |
00a333bb8e | ||
![]() |
09f879a995 | ||
![]() |
3237b9a367 | ||
![]() |
db94e8b345 | ||
![]() |
4d2d89eac7 | ||
![]() |
c44b65d57d |
23 changed files with 1708 additions and 638 deletions
2
.github/FUNDING.yml
vendored
2
.github/FUNDING.yml
vendored
|
@ -1,2 +0,0 @@
|
|||
custom: ["paypal.me/Neargye", "btc.com/btc/address/bc1qzevldln8tqz5xf4lyufu9msgl7t97xstth9zq8", "etherscan.io/address/0xbb42fdef9204fa6f3d535d202b088dee35fcbd31"]
|
||||
liberapay: neargye
|
36
.github/workflows/macos.yml
vendored
36
.github/workflows/macos.yml
vendored
|
@ -2,26 +2,36 @@ name: macos
|
|||
|
||||
on: [push, pull_request]
|
||||
|
||||
permissions: read-all
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ${{matrix.config.os}}
|
||||
runs-on: ${{ matrix.config.os }}
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
config:
|
||||
- { os: macos-11 } # https://github.com/actions/virtual-environments/blob/main/images/macos/macos-11-Readme.md#xcode
|
||||
- { os: macos-12 } # https://github.com/actions/virtual-environments/blob/main/images/macos/macos-12-Readme.md#xcode
|
||||
build: [Debug, Release]
|
||||
- { os: macos-13 } # https://github.com/actions/virtual-environments/blob/main/images/macos/macos-13-Readme.md#xcode
|
||||
- { os: macos-14 } # https://github.com/actions/virtual-environments/blob/main/images/macos/macos-14-Readme.md#xcode
|
||||
|
||||
name: "${{matrix.config.os}}:${{matrix.build}}"
|
||||
name: "${{ matrix.config.os }}"
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure CMake
|
||||
run: cmake -DCMAKE_BUILD_TYPE=${{matrix.build}}
|
||||
- name: Build Release
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -DCMAKE_BUILD_TYPE=Release
|
||||
cmake --build . -j 4 --config Release
|
||||
ctest --output-on-failure -C Release
|
||||
|
||||
- name: Build
|
||||
run: cmake --build . --config ${{matrix.build}}
|
||||
|
||||
- name: Tests
|
||||
run: ctest --output-on-failure -C ${{matrix.build}}
|
||||
- name: Build Debug
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -DCMAKE_BUILD_TYPE=Debug
|
||||
cmake --build . -j 4 --config Debug
|
||||
ctest --output-on-failure -C Debug
|
||||
|
|
76
.github/workflows/ubuntu.yml
vendored
76
.github/workflows/ubuntu.yml
vendored
|
@ -2,29 +2,75 @@ name: ubuntu
|
|||
|
||||
on: [push, pull_request]
|
||||
|
||||
permissions: read-all
|
||||
|
||||
jobs:
|
||||
ubuntu:
|
||||
runs-on: ubuntu-20.04
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
build: [Debug, Release]
|
||||
compiler:
|
||||
- { cc: "gcc-9", cxx: "g++-9" }
|
||||
- { cc: "gcc-10", cxx: "g++-10" }
|
||||
- { cc: "clang-10", cxx: "clang++-10" }
|
||||
- { cc: "clang-11", cxx: "clang++-11" }
|
||||
- { cc: "clang-12", cxx: "clang++-12" }
|
||||
- { cc: "gcc-9", cxx: "g++-9", os: "ubuntu-20.04" }
|
||||
- { cc: "gcc-10", cxx: "g++-10", os: "ubuntu-20.04" }
|
||||
- { cc: "gcc-10", cxx: "g++-10", os: "ubuntu-20.04" }
|
||||
- { cc: "gcc-11", cxx: "g++-11", os: "ubuntu-20.04" }
|
||||
- { cc: "gcc-11", cxx: "g++-11", os: "ubuntu-20.04" }
|
||||
- { cc: "gcc-12", cxx: "g++-12", os: "ubuntu-22.04" }
|
||||
- { cc: "gcc-13", cxx: "g++-12", os: "ubuntu-22.04" }
|
||||
- { cc: "gcc-14", cxx: "g++-12", os: "ubuntu-22.04" }
|
||||
- { cc: "clang-9", cxx: "clang++-9", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-10", cxx: "clang++-10", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-11", cxx: "clang++-11", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-12", cxx: "clang++-12", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-13", cxx: "clang++-13", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-14", cxx: "clang++-14", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-15", cxx: "clang++-15", os: "ubuntu-20.04" }
|
||||
- { cc: "clang-16", cxx: "clang++-16", os: "ubuntu-20.04" }
|
||||
|
||||
name: "${{matrix.compiler.cxx}}:${{matrix.build}}"
|
||||
name: "${{ matrix.compiler.cc }}"
|
||||
runs-on: ${{ matrix.compiler.os }}
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure CMake
|
||||
run: cmake -DCMAKE_BUILD_TYPE=${{matrix.build}} -DCMAKE_C_COMPILER=${{matrix.compiler.cc}} -DCMAKE_CXX_COMPILER=${{matrix.compiler.cxx}}
|
||||
- name: Configure clang
|
||||
run: |
|
||||
if [[ "${{ matrix.compiler.cc }}" == "clang"* ]]; then
|
||||
wget -O - https://apt.llvm.org/llvm-snapshot.gpg.key|sudo apt-key add -
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-9 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-10 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-11 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-12 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-13 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-14 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-15 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal-16 main"
|
||||
sudo apt-add-repository "deb http://apt.llvm.org/focal/ llvm-toolchain-focal main"
|
||||
sudo apt update
|
||||
sudo apt install ${{ matrix.compiler.cc }} -y
|
||||
fi
|
||||
|
||||
- name: Build
|
||||
run: cmake --build . --config ${{matrix.build}}
|
||||
- name: Configure gcc
|
||||
run: |
|
||||
if [[ "${{ matrix.compiler.cc }}" == "gcc"* ]]; then
|
||||
sudo add-apt-repository ppa:ubuntu-toolchain-r/test -y
|
||||
sudo apt update
|
||||
sudo apt install ${{ matrix.compiler.cxx }} -y
|
||||
fi
|
||||
|
||||
- name: Tests
|
||||
run: ctest --output-on-failure -C ${{matrix.build}}
|
||||
- name: Build Release
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -DCMAKE_BUILD_TYPE=Release -DCMAKE_CXX_COMPILER=${{ matrix.compiler.cxx }}
|
||||
cmake --build . -j 4 --config Release
|
||||
ctest --output-on-failure -C Release
|
||||
|
||||
- name: Build Debug
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -DCMAKE_BUILD_TYPE=Debug -DCMAKE_CXX_COMPILER=${{ matrix.compiler.cxx }}
|
||||
cmake --build . -j 4 --config Debug
|
||||
ctest --output-on-failure -C Debug
|
||||
|
|
39
.github/workflows/windows.yml
vendored
39
.github/workflows/windows.yml
vendored
|
@ -2,27 +2,42 @@ name: windows
|
|||
|
||||
on: [push, pull_request]
|
||||
|
||||
permissions: read-all
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ${{matrix.config.os}}
|
||||
runs-on: ${{ matrix.config.os }}
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
config:
|
||||
- { os: windows-2019, vs: "Visual Studio 2019" } # https://github.com/actions/virtual-environments/blob/main/images/win/Windows2019-Readme.md#visual-studio-enterprise-2019
|
||||
- { os: windows-2022, vs: "Visual Studio 2022" } # https://github.com/actions/virtual-environments/blob/main/images/win/Windows2022-Readme.md#visual-studio-enterprise-2022
|
||||
build: [Debug, Release]
|
||||
platform: [Win32, x64]
|
||||
|
||||
name: "${{matrix.config.vs}}:${{matrix.platform}}:${{matrix.build}}"
|
||||
name: "${{ matrix.config.vs }}"
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Configure CMake
|
||||
run: cmake -A ${{matrix.platform}}
|
||||
- name: Build Win32
|
||||
shell: bash
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -A Win32
|
||||
cmake --build . -j 4 --config Release
|
||||
ctest --output-on-failure -C Release
|
||||
cmake --build . -j 4 --config Debug
|
||||
ctest --output-on-failure -C Debug
|
||||
|
||||
- name: Build
|
||||
run: cmake --build . --config ${{matrix.build}}
|
||||
|
||||
- name: Tests
|
||||
run: ctest --output-on-failure -C ${{matrix.build}}
|
||||
- name: Build x64
|
||||
shell: bash
|
||||
run: |
|
||||
rm -rf build
|
||||
mkdir build
|
||||
cd build
|
||||
cmake .. -A x64
|
||||
cmake --build . -j 4 --config Release
|
||||
ctest --output-on-failure -C Release
|
||||
cmake --build . -j 4 --config Debug
|
||||
ctest --output-on-failure -C Debug
|
||||
|
|
156
CMakeLists.txt
156
CMakeLists.txt
|
@ -1,6 +1,17 @@
|
|||
cmake_minimum_required(VERSION 3.14)
|
||||
cmake_minimum_required(VERSION 3.30)
|
||||
|
||||
project(nameof VERSION "0.10.2" LANGUAGES CXX)
|
||||
include(GNUInstallDirs)
|
||||
|
||||
set(ADDITIONAL_MODULES_DIR "${CMAKE_CURRENT_LIST_DIR}/cmake")
|
||||
list(APPEND CMAKE_MODULE_PATH "${ADDITIONAL_MODULES_DIR}")
|
||||
|
||||
project(nameof
|
||||
VERSION "0.10.4"
|
||||
HOMEPAGE_URL "https://github.com/Neargye/nameof"
|
||||
DESCRIPTION "A library that provides nameof macros and functions to simply obtain the name of a variable, type, function, macro, and enum."
|
||||
LANGUAGES CXX
|
||||
)
|
||||
set(CPACK_PACKAGE_VENDOR "Daniil Goncharov")
|
||||
|
||||
if(CMAKE_PROJECT_NAME STREQUAL PROJECT_NAME)
|
||||
set(IS_TOPLEVEL_PROJECT TRUE)
|
||||
|
@ -11,6 +22,7 @@ endif()
|
|||
option(NAMEOF_OPT_BUILD_EXAMPLES "Build nameof examples" ${IS_TOPLEVEL_PROJECT})
|
||||
option(NAMEOF_OPT_BUILD_TESTS "Build and perform nameof tests" ${IS_TOPLEVEL_PROJECT})
|
||||
option(NAMEOF_OPT_INSTALL "Generate and install nameof target" ${IS_TOPLEVEL_PROJECT})
|
||||
option(NAMEOF_MODULE "Build nameof module" OFF)
|
||||
|
||||
if(NAMEOF_OPT_BUILD_EXAMPLES)
|
||||
add_subdirectory(example)
|
||||
|
@ -23,33 +35,119 @@ endif()
|
|||
|
||||
include(CMakePackageConfigHelpers)
|
||||
|
||||
add_library(${PROJECT_NAME} INTERFACE)
|
||||
add_library(${PROJECT_NAME}::${PROJECT_NAME} ALIAS ${PROJECT_NAME})
|
||||
target_include_directories(${PROJECT_NAME}
|
||||
set(EXPORT_NAMESPACE "${PROJECT_NAME}::")
|
||||
|
||||
|
||||
set(INCLUDES "${CMAKE_CURRENT_SOURCE_DIR}/include")
|
||||
|
||||
if(NAMEOF_MODULE)
|
||||
set(CMAKE_CXX_EXTENSIONS OFF)
|
||||
add_library(${PROJECT_NAME})
|
||||
|
||||
target_include_directories(${PROJECT_NAME}
|
||||
PUBLIC
|
||||
$<BUILD_INTERFACE:${INCLUDES}>
|
||||
$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>)
|
||||
target_compile_definitions(${PROJECT_NAME} PRIVATE NAMEOF_MODULE)
|
||||
target_compile_features(${PROJECT_NAME} PUBLIC cxx_std_20)
|
||||
|
||||
target_sources(${PROJECT_NAME} PUBLIC FILE_SET nameofModules TYPE CXX_MODULES
|
||||
FILES src/nameof.cpp)
|
||||
else()
|
||||
add_library("${PROJECT_NAME}" INTERFACE)
|
||||
target_include_directories(${PROJECT_NAME}
|
||||
INTERFACE
|
||||
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>
|
||||
$<INSTALL_INTERFACE:include>)
|
||||
|
||||
write_basic_package_version_file(${PROJECT_NAME}ConfigVersion.cmake
|
||||
VERSION ${PROJECT_VERSION}
|
||||
COMPATIBILITY AnyNewerVersion
|
||||
ARCH_INDEPENDENT)
|
||||
|
||||
if(NAMEOF_OPT_INSTALL)
|
||||
install(TARGETS ${PROJECT_NAME}
|
||||
EXPORT ${PROJECT_NAME}Config)
|
||||
|
||||
install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_NAME}ConfigVersion.cmake
|
||||
DESTINATION lib/cmake/${PROJECT_NAME})
|
||||
|
||||
install(EXPORT ${PROJECT_NAME}Config
|
||||
NAMESPACE ${PROJECT_NAME}::
|
||||
DESTINATION lib/cmake/${PROJECT_NAME})
|
||||
|
||||
install(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/include
|
||||
DESTINATION .)
|
||||
|
||||
export(EXPORT ${PROJECT_NAME}Config
|
||||
NAMESPACE ${PROJECT_NAME}::)
|
||||
$<BUILD_INTERFACE:${INCLUDES}>
|
||||
$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>)
|
||||
endif()
|
||||
|
||||
add_library("${EXPORT_NAMESPACE}${PROJECT_NAME}" ALIAS "${PROJECT_NAME}")
|
||||
|
||||
|
||||
|
||||
|
||||
if(NAMEOF_OPT_INSTALL)
|
||||
list(APPEND CMAKE_MODULE_PATH "${ADDITIONAL_MODULES_DIR}/GenPkgConfig")
|
||||
include(GenPkgConfig)
|
||||
include(CPackComponent)
|
||||
include(CMakePackageConfigHelpers)
|
||||
|
||||
string(REPLACE "/${CMAKE_LIBRARY_ARCHITECTURE}" "" CMAKE_INSTALL_LIBDIR_ARCHIND "${CMAKE_INSTALL_LIBDIR}")
|
||||
install(TARGETS "${PROJECT_NAME}"
|
||||
EXPORT ${PROJECT_NAME}
|
||||
ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||||
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||||
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
|
||||
FILE_SET nameofModules DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
||||
INCLUDES
|
||||
DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}"
|
||||
# COMPONENT "${SDK_COMPONENT_NAME}" # component is not allowed for includes! Headers are installed separately! Includes only marks the headers for export
|
||||
)
|
||||
|
||||
file(GLOB_RECURSE HEADERS "${INCLUDES}/*.h" "${INCLUDES}/*.hxx" "${INCLUDES}/*.hpp")
|
||||
foreach(headerFile ${HEADERS})
|
||||
get_filename_component(headerFileParentDir "${headerFile}" DIRECTORY)
|
||||
file(RELATIVE_PATH headerFileRelParentDir "${INCLUDES}" "${headerFileParentDir}")
|
||||
|
||||
install(FILES "${headerFile}"
|
||||
DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}/${headerFileRelParentDir}"
|
||||
)
|
||||
endforeach()
|
||||
|
||||
set(CPACK_PACKAGE_NAME "${PROJECT_NAME}")
|
||||
set(CPACK_PACKAGE_DESCRIPTION "${PROJECT_DESCRIPTION}")
|
||||
set(CPACK_DEBIAN_PACKAGE_ARCHITECTURE "all")
|
||||
set(CPACK_DEBIAN_PACKAGE_NAME "lib${CPACK_PACKAGE_NAME}-dev")
|
||||
set(CPACK_RPM_PACKAGE_NAME "lib${CPACK_PACKAGE_NAME}-devel")
|
||||
set(CPACK_PACKAGE_HOMEPAGE_URL "${PROJECT_HOMEPAGE_URL}")
|
||||
set(CPACK_PACKAGE_MAINTAINER "${CPACK_PACKAGE_VENDOR}")
|
||||
set(CPACK_DEBIAN_PACKAGE_DEPENDS "")
|
||||
set(CPACK_DEBIAN_PACKAGE_MAINTAINER "${CPACK_PACKAGE_MAINTAINER}")
|
||||
set(CPACK_PACKAGE_MAINTAINER "${CPACK_PACKAGE_VENDOR}")
|
||||
set(CPACK_DEB_COMPONENT_INSTALL ON)
|
||||
set(CPACK_RPM_COMPONENT_INSTALL ON)
|
||||
set(CPACK_NSIS_COMPONENT_INSTALL ON)
|
||||
set(CPACK_DEBIAN_COMPRESSION_TYPE "xz")
|
||||
|
||||
set(CPACK_RESOURCE_FILE_LICENSE "${CMAKE_CURRENT_SOURCE_DIR}/LICENSE")
|
||||
set(CPACK_RESOURCE_FILE_README "${CMAKE_CURRENT_SOURCE_DIR}/README.md")
|
||||
|
||||
set(CMAKE_CONFIG_FILE_BASENAME "${PROJECT_NAME}Config.cmake")
|
||||
set(CMAKE_EXPORT_FILE_BASENAME "${PROJECT_NAME}Export.cmake")
|
||||
set(CMAKE_CONFIG_VERSION_FILE_BASENAME "${PROJECT_NAME}ConfigVersion.cmake")
|
||||
set(CMAKE_CONFIG_VERSION_FILE_NAME "${CMAKE_CURRENT_BINARY_DIR}/${CMAKE_CONFIG_VERSION_FILE_BASENAME}")
|
||||
|
||||
export(TARGETS "${PROJECT_NAME}"
|
||||
NAMESPACE "${EXPORT_NAMESPACE}"
|
||||
FILE "${CMAKE_EXPORT_FILE_BASENAME}"
|
||||
EXPORT_LINK_INTERFACE_LIBRARIES
|
||||
)
|
||||
|
||||
install(EXPORT "${PROJECT_NAME}"
|
||||
FILE "${CMAKE_CONFIG_FILE_BASENAME}"
|
||||
NAMESPACE "${EXPORT_NAMESPACE}"
|
||||
DESTINATION "${CMAKE_INSTALL_LIBDIR_ARCHIND}/cmake/${PROJECT_NAME}"
|
||||
)
|
||||
|
||||
write_basic_package_version_file(
|
||||
"${CMAKE_CONFIG_VERSION_FILE_NAME}"
|
||||
#VERSION "100500.100500.100500" # any version of same bitness suits. CMake cannot compare to infinity, so use a large number we expect to be greater than any future version
|
||||
VERSION ${_VERSION}
|
||||
COMPATIBILITY AnyNewerVersion
|
||||
ARCH_INDEPENDENT
|
||||
)
|
||||
install(FILES "${CMAKE_CONFIG_VERSION_FILE_NAME}"
|
||||
DESTINATION "${CMAKE_INSTALL_LIBDIR_ARCHIND}/cmake/${PROJECT_NAME}"
|
||||
)
|
||||
|
||||
configure_pkg_config_file("${PROJECT_NAME}"
|
||||
NAME "${PROJECT_NAME}"
|
||||
VERSION "${PROJECT_VERSION}"
|
||||
DESCRIPTION "${CPACK_PACKAGE_DESCRIPTION}"
|
||||
URL "${CPACK_PACKAGE_HOMEPAGE_URL}"
|
||||
INSTALL_LIB_DIR "${CMAKE_INSTALL_LIBDIR_ARCHIND}"
|
||||
INSTALL_INCLUDE_DIR "${CMAKE_INSTALL_INCLUDEDIR}"
|
||||
)
|
||||
|
||||
include(CPack)
|
||||
endif()
|
||||
|
|
2
LICENSE
2
LICENSE
|
@ -1,6 +1,6 @@
|
|||
MIT License
|
||||
|
||||
Copyright (c) 2016 - 2022 Daniil Goncharov
|
||||
Copyright (c) 2016 - 2024 Daniil Goncharov
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
|
|
74
README.md
74
README.md
|
@ -1,37 +1,15 @@
|
|||
[](https://bit.ly/3OMysM8)
|
||||
|
||||
```text
|
||||
_ _ __ _____
|
||||
| \ | | / _| / ____|_ _
|
||||
| \| | __ _ _ __ ___ ___ ___ | |_ | | _| |_ _| |_
|
||||
| . ` |/ _` | '_ ` _ \ / _ \/ _ \| _| | | |_ _|_ _|
|
||||
| |\ | (_| | | | | | | __/ (_) | | | |____|_| |_|
|
||||
|_| \_|\__,_|_| |_| |_|\___|\___/|_| \_____|
|
||||
```
|
||||
|
||||
[](https://github.com/Neargye/nameof/releases)
|
||||
[](https://conan.io/center/nameof)
|
||||
[](https://conan.io/center/recipes/nameof)
|
||||
[](https://github.com/microsoft/vcpkg/tree/master/ports/nameof)
|
||||
[](LICENSE)
|
||||
[](https://wandbox.org/permlink/PBBzVKlbMIfC3WOk)
|
||||
[](https://godbolt.org/z/s_ecko)
|
||||
[](https://stand-with-ukraine.pp.ua)
|
||||
|
||||
# Nameof C++
|
||||
|
||||
Header-only C++17 library provides nameof macros and functions to simply obtain the name of a variable, type, function, macro, and enum.
|
||||
|
||||
## Features
|
||||
|
||||
* C++17
|
||||
* Header-only
|
||||
* Dependency-free
|
||||
* Compile-time
|
||||
* Name of variable, member variable
|
||||
* Name of type, variable type
|
||||
* Name of function, member function
|
||||
* Name of enum, enum variable
|
||||
* Name of macro
|
||||
* Enum to string
|
||||
If you like this project, please consider donating to one of the funds that help victims of the war in Ukraine: https://u24.gov.ua.
|
||||
|
||||
## Documentation
|
||||
|
||||
|
@ -39,7 +17,7 @@ Header-only C++17 library provides nameof macros and functions to simply obtain
|
|||
* [Limitations](doc/limitations.md)
|
||||
* [Integration](#Integration)
|
||||
|
||||
## [Examples](example/example.cpp)
|
||||
## [Features & Examples](example/example.cpp)
|
||||
|
||||
* Nameof
|
||||
|
||||
|
@ -60,6 +38,12 @@ Header-only C++17 library provides nameof macros and functions to simply obtain
|
|||
// Name of macro.
|
||||
NAMEOF(__LINE__) -> "__LINE__"
|
||||
NAMEOF(NAMEOF(structvar)) -> "NAMEOF"
|
||||
|
||||
// Obtains full name of variable, function, macro.
|
||||
NAMEOF_FULL(somevar.some_method<int>()) -> "some_method<int>"
|
||||
|
||||
// Obtains raw name of variable, function, macro.
|
||||
NAMEOF_RAW(somevar.some_method<int>()) -> "somevar.some_method<int>()"
|
||||
```
|
||||
|
||||
* Nameof enum
|
||||
|
@ -77,9 +61,13 @@ Header-only C++17 library provides nameof macros and functions to simply obtain
|
|||
NAMEOF_ENUM_CONST(Color::GREEN) -> "GREEN"
|
||||
nameof::nameof_enum<Color::GREEN>() -> "GREEN"
|
||||
|
||||
// Enum flag variable to string.
|
||||
// Enum flags variable to string.
|
||||
NAMEOF_ENUM_FLAG(Color::GREEN | Color::BLUE) -> "GREEN|BLUE"
|
||||
nameof::nameof_enum_flag<Color::GREEN>() -> "GREEN|BLUE"
|
||||
nameof::nameof_enum_flag<Color::GREEN | Color::BLUE>() -> "GREEN|BLUE"
|
||||
|
||||
// Obtains name of enum variable or default value if enum variable out of range.
|
||||
NAMEOF_ENUM_OR(Color::GREEN) -> "GREEN"
|
||||
NAMEOF_ENUM_OR((Color)0, "none") -> "none"
|
||||
```
|
||||
|
||||
* Nameof type
|
||||
|
@ -108,19 +96,22 @@ Header-only C++17 library provides nameof macros and functions to simply obtain
|
|||
NAMEOF_TYPE_RTTI(*ptr) -> "my::detail::Derived"
|
||||
NAMEOF_FULL_TYPE_RTTI(*ptr) -> "volatile const my::detail::Derived&"
|
||||
NAMEOF_SHORT_TYPE_RTTI(*ptr) -> "Derived"
|
||||
```
|
||||
|
||||
* Compile-time
|
||||
struct A {
|
||||
int this_is_the_name;
|
||||
};
|
||||
// Obtains name of member.
|
||||
NAMEOF_MEMBER(&A::this_is_the_name) -> "this_is_the_name"
|
||||
nameof::nameof_member(&A::this_is_the_name) -> "this_is_the_name"
|
||||
|
||||
```cpp
|
||||
constexpr auto somevar_name = NAMEOF(somevar);
|
||||
// somevar_name -> "somevar"
|
||||
constexpr auto color_name = NAMEOF_ENUM(Color::BLUE); // or nameof::nameof_enum(Color::BLUE)
|
||||
// color_name -> "BLUE"
|
||||
constexpr auto var_type_name = NAMEOF_TYPE_EXPR(var); // or nameof::nameof_type<decltype(var)>()
|
||||
// var_type_name -> "int"
|
||||
constexpr auto type_name = NAMEOF_TYPE(T); // or nameof::nameof_type<T>()
|
||||
// type_name -> "int"
|
||||
int someglobalvariable = 0;
|
||||
// Obtains name of a function, a global or class static variable.
|
||||
NAMEOF_POINTER(&someglobalconstvariable) == "someglobalconstvariable"
|
||||
nameof::nameof_pointer(&someglobalconstvariable) == "someglobalconstvariable"
|
||||
|
||||
constexpr auto global_ptr = &someglobalvariable;
|
||||
NAMEOF_POINTER(global_ptr) == "someglobalconstvariable"
|
||||
nameof::nameof_pointer(global_ptr) == "someglobalconstvariable"
|
||||
```
|
||||
|
||||
## Remarks
|
||||
|
@ -149,9 +140,6 @@ CPMAddPackage(
|
|||
|
||||
## Compiler compatibility
|
||||
|
||||
* Clang/LLVM >= 6
|
||||
* MSVC++ >= 14.11 / Visual Studio >= 2017
|
||||
* Xcode >= 10
|
||||
* GCC >= 7 (GCC >= 9 for NAMEOF_ENUM)
|
||||
Check in the [reference](doc/reference.md) for each features.
|
||||
|
||||
## Licensed under the [MIT License](LICENSE)
|
||||
|
|
239
cmake/GenPkgConfig/GenPkgConfig.cmake
Normal file
239
cmake/GenPkgConfig/GenPkgConfig.cmake
Normal file
|
@ -0,0 +1,239 @@
|
|||
#[=======================================================================[.rst:
|
||||
|
||||
GenPkgConfig
|
||||
------------
|
||||
|
||||
This is the library helping you to generate and install pkg-config files.
|
||||
|
||||
Unlicense
|
||||
^^^^^^^^^
|
||||
|
||||
This is free and unencumbered software released into the public domain.
|
||||
Anyone is free to copy, modify, publish, use, compile, sell, or distribute this software, either in source code form or as a compiled binary, for any purpose, commercial or non-commercial, and by any means.
|
||||
In jurisdictions that recognize copyright laws, the author or authors of this software dedicate any and all copyright interest in the software to the public domain. We make this dedication for the benefit of the public at large and to the detriment of our heirs and successors. We intend this dedication to be an overt act of relinquishment in perpetuity of all present and future rights to this software under copyright law.
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
For more information, please refer to <https://unlicense.org/>
|
||||
|
||||
Warning
|
||||
^^^^^^^
|
||||
|
||||
CMake is currently merging a built-in impl of pkg-config file generator! https://gitlab.kitware.com/cmake/cmake/-/merge_requests/6363
|
||||
|
||||
Functions
|
||||
^^^^^^^^^
|
||||
.. command:: configure_pkg_config_file
|
||||
|
||||
.. versionadded:: 3.22
|
||||
|
||||
Generates a pkg-config file for
|
||||
|
||||
::
|
||||
|
||||
configure_pkg_config_file(<targetName>
|
||||
NAME <name of the package>
|
||||
VERSION <version to be written into the package>
|
||||
DESCRIPTION <description to be written into the package>
|
||||
URL <homepage URL to be written into the package>
|
||||
COMPONENT <install as the component>
|
||||
INSTALL_LIB_DIR <path to something like CMAKE_INSTALL_LIBDIR>
|
||||
INSTALL_INCLUDE_DIR <path to something like CMAKE_INSTALL_INCLUDEDIR>
|
||||
REQUIRES ... <list of pkg-config packages this one depends on> ...
|
||||
REQUIRES ... <list of pkg-config packages this one conflicts with> ...
|
||||
)
|
||||
|
||||
The arguments are optional and usually are not needed to be set if global (not component-specific) CPACK vars have been set before.
|
||||
|
||||
Generation is done in build time using packaging expressions.
|
||||
|
||||
#]=======================================================================]
|
||||
|
||||
set("GEN_PKG_CONFIG_WORKAROUNDS_BUILD_TIME_SCRIPTS" "${CMAKE_CURRENT_LIST_DIR}/buildTimeScripts")
|
||||
|
||||
cmake_policy(SET CMP0070 NEW)
|
||||
|
||||
function(configure_pkg_config_file TARGET)
|
||||
cmake_parse_arguments(""
|
||||
"" # options
|
||||
"NAME;VERSION;DESCRIPTION;URL;COMPONENT;INSTALL_LIB_DIR;INSTALL_INCLUDE_DIR" # one_value_keywords
|
||||
"REQUIRES;CONFLICTS" # multi_value_keywords
|
||||
${ARGN}
|
||||
)
|
||||
|
||||
configure_pkg_config_file_vars("${TARGET}" "${_NAME}" "${_INSTALL_LIB_DIR}" "${_INSTALL_INCLUDE_DIR}" "${_COMPONENT}" "${_DESCRIPTION}" "${_URL}" "${_VERSION}" "${_REQUIRES}" "${_CONFLICTS}")
|
||||
endfunction()
|
||||
|
||||
function(ge_expr_basename inputExpr outVar)
|
||||
set("${outVar}" "$<TARGET_PROPERTY:${inputExpr}>" PARENT_SCOPE)
|
||||
endfunction()
|
||||
|
||||
function(configure_pkg_config_file_vars TARGET _NAME _INSTALL_LIB_DIR _INSTALL_INCLUDE_DIR _COMPONENT _DESCRIPTION _URL _VERSION _REQUIRES _CONFLICTS)
|
||||
#$<TARGET_PROPERTY:${TARGET},NAME>
|
||||
#INTERFACE_LINK_DIRECTORIES
|
||||
#INTERFACE_LINK_LIBRARIES
|
||||
#INTERFACE_LINK_OPTIONS
|
||||
|
||||
if(_NAME)
|
||||
else()
|
||||
set(_NAME "$<TARGET_PROPERTY:${TARGET},NAME>")
|
||||
endif()
|
||||
|
||||
if(_DESCRIPTION)
|
||||
else()
|
||||
set(_DESCRIPTION "${CPACK_PACKAGE_DESCRIPTION}")
|
||||
endif()
|
||||
|
||||
if(_VERSION)
|
||||
else()
|
||||
set(_VERSION "${CPACK_PACKAGE_VERSION}")
|
||||
endif()
|
||||
|
||||
if(_URL)
|
||||
else()
|
||||
set(_URL "${CPACK_PACKAGE_HOMEPAGE_URL}")
|
||||
endif()
|
||||
|
||||
if(INSTALL_INCLUDE_DIR)
|
||||
else()
|
||||
set(INSTALL_INCLUDE_DIR "${CMAKE_INSTALL_INCLUDEDIR}")
|
||||
endif()
|
||||
|
||||
if(INSTALL_LIB_DIR)
|
||||
else()
|
||||
set(INSTALL_LIB_DIR "${CMAKE_INSTALL_LIBDIR}")
|
||||
endif()
|
||||
|
||||
set(PKG_CONFIG_FILE_NAME "${CMAKE_CURRENT_BINARY_DIR}/${_NAME}.pc")
|
||||
|
||||
set(PUBLIC_INCLUDES "$<TARGET_PROPERTY:${TARGET},INTERFACE_INCLUDE_DIRECTORIES>")
|
||||
set(PUBLIC_LIBS "$<TARGET_PROPERTY:${TARGET},INTERFACE_LINK_LIBRARIES>")
|
||||
set(PUBLIC_COMPILE_FLAGS "$<TARGET_PROPERTY:${TARGET},INTERFACE_COMPILE_DEFINITIONS>")
|
||||
|
||||
set("IS_INTERFACE" "$<STREQUAL:$<TARGET_PROPERTY:${TARGET},TYPE>,INTERFACE_LIBRARY>")
|
||||
set("IS_OBJECT" "$<STREQUAL:$<TARGET_PROPERTY:${TARGET},TYPE>,OBJECT_LIBRARY>")
|
||||
get_target_property(CONFIGURE_TIME_TARGET_TYPE "${TARGET}" TYPE)
|
||||
if(CONFIGURE_TIME_TARGET_TYPE STREQUAL OBJECT_LIBRARY)
|
||||
set(CONFIGURE_TIME_IS_OBJECT ON) # Special measures have to be taken!!!
|
||||
endif()
|
||||
|
||||
set("NEEDS_LIBS" "$<AND:$<NOT:${IS_INTERFACE}>,$<NOT:${IS_OBJECT}>>")
|
||||
set("NEEDS_LIB_DIR" "$<NOT:${IS_INTERFACE}>")
|
||||
string(REPLACE "," "$<COMMA>" NEEDS_LIBS_ESCAPED "${NEEDS_LIBS}")
|
||||
string(REPLACE ">" "$<ANGLE-R>" NEEDS_LIBS_ESCAPED "${NEEDS_LIBS_ESCAPED}")
|
||||
|
||||
list(APPEND header "prefix=${CMAKE_INSTALL_PREFIX}")
|
||||
list(APPEND header "$<IF:$<OR:$<BOOL:${PUBLIC_LIBS}>,${NEEDS_LIB_DIR}>,libdir=\${prefix}/${INSTALL_LIB_DIR},>")
|
||||
list(APPEND header "$<IF:$<BOOL:${PUBLIC_INCLUDES}>,includedir=\${prefix}/${INSTALL_INCLUDE_DIR},>")
|
||||
|
||||
|
||||
list(APPEND libSpecific "Name: ${_NAME}")
|
||||
if(_DESCRIPTION)
|
||||
list(APPEND libSpecific "Description: ${_DESCRIPTION}")
|
||||
endif()
|
||||
if(_URL)
|
||||
list(APPEND libSpecific "URL: ${_URL}")
|
||||
endif()
|
||||
if(_VERSION)
|
||||
list(APPEND libSpecific "Version: ${_VERSION}")
|
||||
endif()
|
||||
if(_REQUIRES)
|
||||
list(APPEND libSpecific "Requires: ${_REQUIRES}")
|
||||
endif()
|
||||
if(_CONFLICTS)
|
||||
list(APPEND libSpecific "Conflicts: ${_CONFLICTS}")
|
||||
endif()
|
||||
|
||||
set(OTHER_INCLUDE_FLAGS "-I$<JOIN:$<REMOVE_DUPLICATES:${PUBLIC_INCLUDES}>, -I>") # Not needed, we can only get build interface flags here. Insert them after -I\${includedir} if you find a way to fix/workaround it
|
||||
|
||||
# Here is a workaround to inability to use TARGET_LINKER_FILE_NAME for targets not involving library generation.
|
||||
# Strangely $<IF evaluates both branches, not only the one taken, which causes an error
|
||||
# We workaround it by generating the subexpression source using $<IF and then evaluating it using $<GENEX_EVAL
|
||||
# Of course we could have used conditional expressions on CMake script part, but I have decided to implement it in generator expressions part, so hypthetically all the expressions can be merged into a single file and this function can be made simple
|
||||
|
||||
set(ESCAPED_GENEXPR_BEGINNING "$<1:$><") # A hack because there is no escape for `$` or `<` or `$<`. So we just disrupt $< into pieces
|
||||
set(CURRENT_LIB_ESCAPED_BINARY_NAME "${ESCAPED_GENEXPR_BEGINNING}TARGET_LINKER_FILE_NAME:${TARGET}$<ANGLE-R>")
|
||||
set(LINK_CURRENT_LIB_FLAG "$<GENEX_EVAL:$<IF:${NEEDS_LIBS},-l:${CURRENT_LIB_ESCAPED_BINARY_NAME},>>")
|
||||
|
||||
if(CONFIGURE_TIME_IS_OBJECT)
|
||||
set(IS_TARGET_OBJECTS_CONFIGURE_TIME_UNAVAILABLE ON)
|
||||
if(IS_TARGET_OBJECTS_CONFIGURE_TIME_UNAVAILABLE)
|
||||
message(WARNING "CMake is shit: There is (at least was at the time of writing of this code) no generator expression to get only basenames of object files. They are also unavailable at configure stage. And there were no CMake generator expressions for making replacements in strings. So we workaround this.")
|
||||
set(TARGET_OBJECTS_FILE "${TARGET}.obj_list")
|
||||
set(OBJECTS_FILE_RETRIEVAL_TARGET_NAME "${TARGET}_get_objects_list")
|
||||
|
||||
set(PKGCONFIG_DUMMY_UNFINISHED_GEN_TARGET_NAME "${TARGET}_pkgconfig_unfinished")
|
||||
set(PKGCONFIG_PATCH_TARGET_NAME "${TARGET}_patch_pkgconfig")
|
||||
|
||||
set(PKG_CONFIG_FILE_NAME_FINISHED "${PKG_CONFIG_FILE_NAME}")
|
||||
set(PKG_CONFIG_FILE_NAME_UNFINISHED "${PKG_CONFIG_FILE_NAME_FINISHED}.unfinished")
|
||||
|
||||
file(GENERATE OUTPUT "${TARGET_OBJECTS_FILE}" CONTENT "$<TARGET_OBJECTS:${TARGET}>")
|
||||
|
||||
add_custom_command(
|
||||
OUTPUT "${TARGET_OBJECTS_FILE}"
|
||||
COMMENT "A dummy command to workaround cmake limitations"
|
||||
)
|
||||
add_custom_target("${OBJECTS_FILE_RETRIEVAL_TARGET_NAME}"
|
||||
DEPENDS "${TARGET_OBJECTS_FILE}"
|
||||
)
|
||||
|
||||
add_custom_command(
|
||||
OUTPUT "${PKG_CONFIG_FILE_NAME_FINISHED}"
|
||||
PRE_BUILD COMMAND ${CMAKE_COMMAND} "-DobjectsFile=\"${TARGET_OBJECTS_FILE}\"" "-DpkgConfigFileUnlinished=\"${PKG_CONFIG_FILE_NAME_UNFINISHED}\"" "-DpkgConfigFileFinal=\"${PKG_CONFIG_FILE_NAME_FINISHED}\"" "-P" "${GEN_PKG_CONFIG_WORKAROUNDS_BUILD_TIME_SCRIPTS}/getObjectFilesBaseNames.cmake"
|
||||
MAIN_DEPENDENCY "${TARGET_OBJECTS_FILE}"
|
||||
DEPENDS "${PKG_CONFIG_FILE_NAME_UNFINISHED}"
|
||||
COMMENT "Working around CMake limitations about getting list of basenames of object files and about lack of generator expressions to modify strings: ${PKG_CONFIG_FILE_NAME_UNFINISHED} + ${TARGET_OBJECTS_FILE} -> ${PKG_CONFIG_FILE_NAME_FINISHED}"
|
||||
)
|
||||
add_custom_target("${PKGCONFIG_PATCH_TARGET_NAME}" ALL
|
||||
DEPENDS "${PKG_CONFIG_FILE_NAME_FINISHED}"
|
||||
)
|
||||
add_dependencies("${PKGCONFIG_PATCH_TARGET_NAME}" "${OBJECTS_FILE_RETRIEVAL_TARGET_NAME}" "${PKGCONFIG_DUMMY_UNFINISHED_GEN_TARGET_NAME}")
|
||||
|
||||
set(PROPERLY_JOINED_TARGET_OBJECTS "@PROPERLY_JOINED_TARGET_OBJECTS@")
|
||||
else()
|
||||
set("RAW_TARGET_OBJECTS" "$<TARGET_OBJECTS:${TARGET}>")
|
||||
message(FATAL_ERROR "This branch is unimplemented because CMake lacked the needed functionality at the time")
|
||||
set(PROPERLY_JOINED_TARGET_OBJECTS "${RAW_TARGET_OBJECTS}")
|
||||
endif()
|
||||
endif()
|
||||
|
||||
set(LINK_CURRENT_OBJECT_FLAG "$<IF:${IS_OBJECT},${PROPERLY_JOINED_TARGET_OBJECTS},>")
|
||||
|
||||
list(APPEND libSpecific "$<IF:$<OR:$<BOOL:${PUBLIC_LIBS}>,${NEEDS_LIBS},${IS_OBJECT}>,Libs: -L\${libdir} ${LINK_CURRENT_LIB_FLAG} ${LINK_CURRENT_OBJECT_FLAG} $<IF:$<BOOL:${PUBLIC_LIBS}>,-l$<JOIN:$<REMOVE_DUPLICATES:${PUBLIC_LIBS}>, -l>,>,>\n$<IF:$<OR:$<BOOL:${PUBLIC_INCLUDES}>,$<BOOL:${PUBLIC_COMPILE_FLAGS}>>,Cflags: -I\${includedir} $<JOIN:$<REMOVE_DUPLICATES:${PUBLIC_COMPILE_FLAGS}>,>,>")
|
||||
|
||||
|
||||
list(JOIN header "\n" header)
|
||||
list(JOIN libSpecific "\n" libSpecific)
|
||||
set(libSpecific "${header}\n\n${libSpecific}")
|
||||
|
||||
if(CONFIGURE_TIME_IS_OBJECT)
|
||||
file(GENERATE OUTPUT "${PKG_CONFIG_FILE_NAME_UNFINISHED}"
|
||||
CONTENT "${libSpecific}"
|
||||
)
|
||||
|
||||
# Dummy target for generated files
|
||||
add_custom_command(
|
||||
OUTPUT "${PKG_CONFIG_FILE_NAME_UNFINISHED}"
|
||||
COMMENT "A dummy command to workaround cmake limitations"
|
||||
)
|
||||
add_custom_target("${PKGCONFIG_DUMMY_UNFINISHED_GEN_TARGET_NAME}"
|
||||
DEPENDS "${PKG_CONFIG_FILE_NAME_UNFINISHED}"
|
||||
)
|
||||
|
||||
|
||||
install(FILES "${PKG_CONFIG_FILE_NAME_FINISHED}"
|
||||
DESTINATION "${_INSTALL_LIB_DIR}/pkgconfig"
|
||||
COMPONENT "${_COMPONENT}"
|
||||
)
|
||||
else()
|
||||
file(GENERATE OUTPUT "${PKG_CONFIG_FILE_NAME}"
|
||||
CONTENT "${libSpecific}"
|
||||
)
|
||||
|
||||
|
||||
install(FILES "${PKG_CONFIG_FILE_NAME}"
|
||||
DESTINATION "${_INSTALL_LIB_DIR}/pkgconfig"
|
||||
COMPONENT "${_COMPONENT}"
|
||||
)
|
||||
endif()
|
||||
endfunction()
|
||||
|
45
cmake/GenPkgConfig/ReadMe.md
Normal file
45
cmake/GenPkgConfig/ReadMe.md
Normal file
|
@ -0,0 +1,45 @@
|
|||
GenPkgConfig.cmake
|
||||
===================
|
||||
|
||||
A script generating pkg-config files.
|
||||
|
||||
WARNING: CMake [is currently merging own built-in pkgconfig generation implementation](https://gitlab.kitware.com/cmake/cmake/-/merge_requests/6363)!
|
||||
|
||||
If you require such a new version of CMake, you probably should use the built-in impl instead.
|
||||
|
||||
Syntax
|
||||
------
|
||||
|
||||
```cmake
|
||||
configure_pkg_config_file(<targetName>
|
||||
NAME <name of the pkgconfig package>
|
||||
VERSION <version to be written into the package>
|
||||
DESCRIPTION <description to be written into the package>
|
||||
URL <homepage URL to be written into the package>
|
||||
COMPONENT <install as the component>
|
||||
INSTALL_LIB_DIR <path to something like CMAKE_INSTALL_LIBDIR>
|
||||
INSTALL_INCLUDE_DIR <path to something like CMAKE_INSTALL_INCLUDEDIR>
|
||||
REQUIRES ... <list of pkg-config packages this one depends on> ...
|
||||
REQUIRES ... <list of pkg-config packages this one conflicts with> ...
|
||||
)
|
||||
```
|
||||
|
||||
Issuees
|
||||
-------
|
||||
|
||||
1. For `OBJECT` targets we have run into big issues. CMake
|
||||
1. Doesn't allow to get the list of object files at configure time
|
||||
2. Allows to get a list of object files as a generator exression ...
|
||||
3. BUT ... the path to them is full, but we need only file name!
|
||||
4. CMake doesn't allow to strip directory path via generator expression
|
||||
5. ... neither it allows string editing within generator expressions ...
|
||||
|
||||
so ... we have to create a custom target using a custom CMake script executed separately, but ...
|
||||
|
||||
6. `file(GENERATE` doesn't properly register dependencies
|
||||
... so we have to use `add_custom_command` to say CMake that these files are generated
|
||||
|
||||
7. And CMake `install(FILES` doesn't mean that the targets generating these files are automatically executed,
|
||||
|
||||
So we have to use `ALL` in `add_custom_target`.
|
||||
|
24
cmake/GenPkgConfig/UNLICENSE
Normal file
24
cmake/GenPkgConfig/UNLICENSE
Normal file
|
@ -0,0 +1,24 @@
|
|||
This is free and unencumbered software released into the public domain.
|
||||
|
||||
Anyone is free to copy, modify, publish, use, compile, sell, or
|
||||
distribute this software, either in source code form or as a compiled
|
||||
binary, for any purpose, commercial or non-commercial, and by any
|
||||
means.
|
||||
|
||||
In jurisdictions that recognize copyright laws, the author or authors
|
||||
of this software dedicate any and all copyright interest in the
|
||||
software to the public domain. We make this dedication for the benefit
|
||||
of the public at large and to the detriment of our heirs and
|
||||
successors. We intend this dedication to be an overt act of
|
||||
relinquishment in perpetuity of all present and future rights to this
|
||||
software under copyright law.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||
IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||||
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||
OTHER DEALINGS IN THE SOFTWARE.
|
||||
|
||||
For more information, please refer to <https://unlicense.org/>
|
|
@ -0,0 +1,18 @@
|
|||
|
||||
message(STATUS "objectsFile ${objectsFile}")
|
||||
message(STATUS "pkgConfigFileFinal ${pkgConfigFileFinal}")
|
||||
message(STATUS "pkgConfigFileUnlinished ${pkgConfigFileUnlinished}")
|
||||
|
||||
file(READ "${objectsFile}" TARGET_OBJECTS)
|
||||
|
||||
set(PROPERLY_JOINED_TARGET_OBJECTS "")
|
||||
|
||||
foreach(objFullPath ${TARGET_OBJECTS})
|
||||
get_filename_component(objFullPath "${objFullPath}" NAME)
|
||||
list(APPEND PROPERLY_JOINED_TARGET_OBJECTS "${objFullPath}")
|
||||
endforeach()
|
||||
list(JOIN PROPERLY_JOINED_TARGET_OBJECTS " " PROPERLY_JOINED_TARGET_OBJECTS)
|
||||
|
||||
message(STATUS "PROPERLY_JOINED_TARGET_OBJECTS AFTER ${PROPERLY_JOINED_TARGET_OBJECTS}")
|
||||
|
||||
configure_file("${pkgConfigFileUnlinished}" "${pkgConfigFileFinal}" @ONLY)
|
|
@ -2,19 +2,25 @@
|
|||
|
||||
## Nameof
|
||||
|
||||
* If argument does not have name, occurs the compilation error `"Expression does not have a name."`.
|
||||
* If the argument does not have a name, the compilation error `"Expression does not have a name."` occurs.
|
||||
|
||||
## Nameof Type
|
||||
|
||||
* This library uses a compiler-specific hack (based on `__PRETTY_FUNCTION__` / `__FUNCSIG__`), which works on Clang >= 5, MSVC >= 15.3 and GCC >= 7.
|
||||
* This library uses a compiler-specific hack (based on `__PRETTY_FUNCTION__` / `__FUNCSIG__`).
|
||||
|
||||
* nameof_type and nameof_type_rtti returns compiler-specific type name.
|
||||
* nameof_type and nameof_type_rtti return a compiler-specific type name.
|
||||
|
||||
* To check is nameof_type supported compiler use macro `NAMEOF_TYPE_SUPPORTED` or constexpr constant `nameof::is_nameof_type_supported`.</br>
|
||||
If nameof_type used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
* To check if nameof_type is supported by your compiler use the macro `NAMEOF_TYPE_SUPPORTED` or constexpr constant `nameof::is_nameof_type_supported`.</br>
|
||||
If nameof_type is used on an unsupported compiler, a compilation error occurs. To suppress the error define the macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_type_rtti supported compiler use macro `NAMEOF_TYPE_RTTI_SUPPORTED` or constexpr constant `nameof::is_nameof_type_rtti_supported`.</br>
|
||||
If nameof_type used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
* To check if nameof_type_rtti is supported by your compiler use macro `NAMEOF_TYPE_RTTI_SUPPORTED` or constexpr constant `nameof::is_nameof_type_rtti_supported`.</br>
|
||||
If nameof_type_rtti is used on an unsupported compiler, a compilation error occurs. To suppress the error define the macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_member supported compiler use macro `NAMEOF_MEMBER_SUPPORTED` or constexpr constant `nameof::is_nameof_member_supported`.</br>
|
||||
If nameof_member used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_pointer supported compiler use macro `NAMEOF_POINTER_SUPPORTED` or constexpr constant `nameof::is_nameof_pointer_supported`.</br>
|
||||
If nameof_pointer used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
## Nameof Enum
|
||||
|
||||
|
@ -22,18 +28,18 @@
|
|||
|
||||
* Do not use [nameof](https://github.com/Neargye/nameof) and [magic_enum](https://github.com/Neargye/magic_enum) in the same project to get enum name.
|
||||
|
||||
* To check is nameof_enum supported compiler use macro `NAMEOF_ENUM_SUPPORTED` or constexpr constant `nameof::is_nameof_enum_supported`.</br>
|
||||
If nameof_enum used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_ENUM_NO_CHECK_SUPPORT`.
|
||||
* To check if nameof_enum is supported by your compiler use the macro `NAMEOF_ENUM_SUPPORTED` or constexpr constant `nameof::is_nameof_enum_supported`.</br>
|
||||
If nameof_enum is used on an unsupported compiler, a compilation error occurs. To suppress the error define the macro `NAMEOF_ENUM_NO_CHECK_SUPPORT`.
|
||||
|
||||
* Enum value must be in range `[NAMEOF_ENUM_RANGE_MIN, NAMEOF_ENUM_RANGE_MAX]`.
|
||||
|
||||
* By default `NAMEOF_ENUM_RANGE_MIN = -128`, `NAMEOF_ENUM_RANGE_MAX = 128`.
|
||||
|
||||
* `NAMEOF_ENUM_RANGE_MIN` must be less or equals than `0` and must be greater than `INT16_MIN`.
|
||||
* `NAMEOF_ENUM_RANGE_MIN` must be less than or equal to `0` and must be greater than `INT16_MIN`.
|
||||
|
||||
* `NAMEOF_ENUM_RANGE_MAX` must be greater than `0` and must be less than `INT16_MAX`.
|
||||
|
||||
* If need another range for all enum types by default, redefine the macro `NAMEOF_ENUM_RANGE_MIN` and `NAMEOF_ENUM_RANGE_MAX`.
|
||||
* If another range is needed for all enum types by default, redefine the macro `NAMEOF_ENUM_RANGE_MIN` and `NAMEOF_ENUM_RANGE_MAX`.
|
||||
|
||||
```cpp
|
||||
#define NAMEOF_ENUM_RANGE_MIN 0
|
||||
|
@ -41,7 +47,7 @@
|
|||
#include <nameof.hpp>
|
||||
```
|
||||
|
||||
* If need another range for specific enum type, add specialization `enum_range` for necessary enum type. Specialization of `enum_range` must be injected in `namespace nameof::customize`.
|
||||
* If another range is needed for a specific enum type, add specialization `enum_range` for necessary enum type. Specialization of `enum_range` must be injected in `namespace nameof::customize`.
|
||||
|
||||
```cpp
|
||||
#include <nameof.hpp>
|
||||
|
@ -55,7 +61,7 @@
|
|||
};
|
||||
```
|
||||
|
||||
* Won't work if a value is aliased, work with enum-aliases is compiler-implementation-defined.
|
||||
* Won't work if a value is aliased, support for enum-aliases is compiler-implementation-defined.
|
||||
|
||||
* Won't work if the enum is a forward declaration.
|
||||
|
||||
|
|
140
doc/reference.md
140
doc/reference.md
|
@ -16,19 +16,27 @@
|
|||
* [`NAMEOF_TYPE_RTTI` obtains type name, using RTTI.](#nameof_type_rtti)
|
||||
* [`NAMEOF_FULL_TYPE_RTTI` obtains short type name, using RTTI.](#nameof_full_type_rtti)
|
||||
* [`NAMEOF_SHORT_TYPE_RTTI` obtains short type name, using RTTI.](#nameof_short_type_rtti)
|
||||
* [`NAMEOF_MEMBER` obtains name of member.](#nameof_member)
|
||||
* [`NAMEOF_POINTER` obtains name of a function, a global or class static variable.](#nameof_pointer)
|
||||
|
||||
## Synopsis
|
||||
|
||||
* Before use, read the [limitations](limitations.md) of functionality.
|
||||
|
||||
* To check is nameof_enum supported compiler use macro `NAMEOF_ENUM_SUPPORTED` or constexpr constant `nameof::is_nameof_enum_supported`.</br>
|
||||
If nameof_enum used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_ENUM_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_type supported compiler use macro `NAMEOF_TYPE_SUPPORTED` or constexpr constant `nameof::is_nameof_type_supported`.</br>
|
||||
If nameof_type used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_type_rtti supported compiler use macro `NAMEOF_TYPE_RTTI_SUPPORTED` or constexpr constant `nameof::is_nameof_type_rtti_supported`.</br>
|
||||
If nameof_type used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_enum supported compiler use macro `NAMEOF_ENUM_SUPPORTED` or constexpr constant `nameof::is_nameof_enum_supported`.</br>
|
||||
If nameof_enum used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_ENUM_NO_CHECK_SUPPORT`.
|
||||
* To check is nameof_member supported compiler use macro `NAMEOF_MEMBER_SUPPORTED` or constexpr constant `nameof::is_nameof_member_supported`.</br>
|
||||
If nameof_member used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To check is nameof_pointer supported compiler use macro `NAMEOF_POINTER_SUPPORTED` or constexpr constant `nameof::is_nameof_pointer_supported`.</br>
|
||||
If nameof_pointer used on unsupported compiler, occurs the compilation error. To suppress error define macro `NAMEOF_TYPE_NO_CHECK_SUPPORT`.
|
||||
|
||||
* To add custom enum or type names see the [example](../example/example_custom_name.cpp).
|
||||
|
||||
|
@ -71,6 +79,11 @@
|
|||
NAMEOF(NAMEOF(structvar)) -> "NAMEOF"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_FULL`
|
||||
|
||||
* Obtains full (with template suffix) name of variable, function, macro.
|
||||
|
@ -89,6 +102,11 @@
|
|||
NAMEOF_FULL(somevar.some_method<int>()) -> "some_method<int>"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_RAW`
|
||||
|
||||
* Obtains raw name of variable, function, macro.
|
||||
|
@ -104,6 +122,10 @@
|
|||
NAMEOF_RAW(&some_class::some_method<int>) -> "&some_class::some_method<int>"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_ENUM`
|
||||
|
||||
|
@ -121,6 +143,11 @@
|
|||
nameof::nameof_enum(color) -> "RED"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 9 and C++ >= 17</br>
|
||||
|
||||
# `NAMEOF_ENUM_OR`
|
||||
|
||||
* Obtains name of enum variable or default value if enum variable out of range.
|
||||
|
@ -129,6 +156,19 @@
|
|||
|
||||
* If argument does not have name or [out of range](limitations.md#nameof-enum), returns `default_value`.
|
||||
|
||||
```cpp
|
||||
auto color = Color::RED;
|
||||
NAMEOF_ENUM_OR(color, "none") -> "RED"
|
||||
NAMEOF_ENUM_OR((Color)-1, "none") -> "none"
|
||||
nameof::nameof_enum_or(color, "none") -> "RED"
|
||||
nameof::nameof_enum_or((Color)-1, "none") -> "none"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 9 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_ENUM_CONST`
|
||||
|
||||
* Obtains name of static storage enum variable.
|
||||
|
@ -146,6 +186,11 @@
|
|||
nameof::nameof_enum<Color::GREEN>() -> "GREEN"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 9 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_ENUM_FLAG`
|
||||
|
||||
* Obtains name of enum flag variable.
|
||||
|
@ -166,11 +211,17 @@
|
|||
flag = AnimalFlags::CanFly | AnimalFlags::Endangered;
|
||||
NAMEOF_ENUM_FLAG(flag) -> "CanFly|Endangered"
|
||||
nameof_enum_flag(flag) -> "CanFly|Endangered"
|
||||
nameof_enum_flag(flag, '$') -> "CanFly$Endangered"
|
||||
|
||||
NAMEOF_ENUM(HasClaws | CanFly) -> ""
|
||||
nameof_enum(HasClaws | CanFly) -> ""
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 9 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_TYPE`
|
||||
|
||||
* Obtains type name, reference and cv-qualifiers are ignored.
|
||||
|
@ -192,6 +243,11 @@
|
|||
nameof::nameof_type<T>() -> "int"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_FULL_TYPE`
|
||||
|
||||
* Obtains full type name, with reference and cv-qualifiers.
|
||||
|
@ -210,6 +266,11 @@
|
|||
nameof::nameof_full_type<T>() -> "const int&"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_SHORT_TYPE`
|
||||
|
||||
* Obtains short type name.
|
||||
|
@ -228,6 +289,11 @@
|
|||
nameof::nameof_short_type<T>() -> "SomeClass"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_TYPE_EXPR`
|
||||
|
||||
* Obtains string name type of expression, reference and cv-qualifiers are ignored.
|
||||
|
@ -249,6 +315,11 @@
|
|||
nameof::nameof_type<decltype(var)>() -> "int"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_FULL_TYPE_EXPR`
|
||||
|
||||
* Obtains full type name of expression, with reference and cv-qualifiers.
|
||||
|
@ -268,6 +339,11 @@
|
|||
nameof::nameof_full_type<decltype(var)>() -> "const int&"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_SHORT_TYPE_EXPR`
|
||||
|
||||
* Obtains short type name of expression.
|
||||
|
@ -286,6 +362,11 @@
|
|||
nameof::nameof_short_type<decltype(var)>() -> "SomeClass"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2017 and C++ >= 17</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_TYPE_RTTI`
|
||||
|
||||
* Obtains type name, using RTTI.
|
||||
|
@ -299,6 +380,11 @@
|
|||
NAMEOF_TYPE_RTTI(*ptr) -> "my::detail::Derived"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17 and RTTI enabled</br>
|
||||
Visual Studio >= 2017 and C++ >= 17 and RTTI enabled</br>
|
||||
GCC >= 7 and C++ >= 17 and RTTI enabled</br>
|
||||
|
||||
## `NAMEOF_FULL_TYPE_RTTI`
|
||||
|
||||
* Obtains full type name, using RTTI.
|
||||
|
@ -324,3 +410,53 @@
|
|||
volatile const my::detail::Base* ptr = new my::detail::Derived();
|
||||
NAMEOF_SHORT_TYPE_RTTI(*ptr) -> "Derived"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17 and RTTI enabled</br>
|
||||
Visual Studio >= 2017 and C++ >= 17 and RTTI enabled</br>
|
||||
GCC >= 7 and C++ >= 17 and RTTI enabled</br>
|
||||
|
||||
## `NAMEOF_MEMBER`
|
||||
|
||||
* Obtains name of member.
|
||||
|
||||
* Returns `string_view`.
|
||||
|
||||
* Examples
|
||||
|
||||
```cpp
|
||||
struct A {
|
||||
int this_is_the_name;
|
||||
};
|
||||
// ..
|
||||
NAMEOF_MEMBER(&A::this_is_the_name) -> "this_is_the_name"
|
||||
nameof::nameof_member(&A::this_is_the_name) -> "this_is_the_name"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2022 and C++ >= 20</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
||||
## `NAMEOF_POINTER`
|
||||
|
||||
* Obtains name of a function, a global or class static variable.
|
||||
|
||||
* Returns `string_view`.
|
||||
|
||||
* Examples
|
||||
```cpp
|
||||
int someglobalvariable = 0;
|
||||
// ..
|
||||
NAMEOF_POINTER(&someglobalconstvariable) == "someglobalconstvariable"
|
||||
nameof::nameof_pointer(&someglobalconstvariable) == "someglobalconstvariable"
|
||||
|
||||
constexpr auto global_ptr = &someglobalvariable;
|
||||
NAMEOF_POINTER(global_ptr) == "someglobalconstvariable"
|
||||
nameof::nameof_pointer(global_ptr) == "someglobalconstvariable"
|
||||
```
|
||||
|
||||
* Compiler compatibility
|
||||
Clang/LLVM >= 5 and C++ >= 17</br>
|
||||
Visual Studio >= 2022 and C++ >= 20</br>
|
||||
GCC >= 7 and C++ >= 17</br>
|
||||
|
|
|
@ -12,10 +12,12 @@ endif()
|
|||
function(make_example target)
|
||||
add_executable(${target} ${target}.cpp)
|
||||
set_target_properties(${target} PROPERTIES CXX_EXTENSIONS OFF)
|
||||
target_compile_features(${target} PRIVATE cxx_std_17)
|
||||
target_compile_options(${target} PRIVATE ${OPTIONS})
|
||||
target_link_libraries(${target} PRIVATE ${CMAKE_PROJECT_NAME})
|
||||
endfunction()
|
||||
|
||||
make_example(example)
|
||||
make_example(example_custom_name)
|
||||
if(NAMEOF_MODULE)
|
||||
make_example(example_module)
|
||||
endif()
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
// Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
||||
// SPDX-License-Identifier: MIT
|
||||
// Copyright (c) 2018 - 2022 Daniil Goncharov <neargye@gmail.com>.
|
||||
// Copyright (c) 2018 - 2024 Daniil Goncharov <neargye@gmail.com>.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
// Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
||||
// SPDX-License-Identifier: MIT
|
||||
// Copyright (c) 2020 - 2022 Daniil Goncharov <neargye@gmail.com>.
|
||||
// Copyright (c) 2020 - 2024 Daniil Goncharov <neargye@gmail.com>.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
|
@ -40,7 +40,7 @@ constexpr std::string_view nameof::customize::enum_name<Color>(Color value) noex
|
|||
case Color::GREEN:
|
||||
return {}; // Empty string for default value.
|
||||
}
|
||||
return {}; // Empty string for unknow value.
|
||||
return {}; // Empty string for unknown value.
|
||||
}
|
||||
|
||||
// Сustom definitions of names for enum.
|
||||
|
@ -51,7 +51,7 @@ constexpr std::string_view nameof::customize::enum_name<Numbers>(Numbers value)
|
|||
case Numbers::One:
|
||||
return "the one";
|
||||
default:
|
||||
return {}; // Empty string for default or unknow value.
|
||||
return {}; // Empty string for default or unknown value.
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
|
11
example/example_module.cpp
Normal file
11
example/example_module.cpp
Normal file
|
@ -0,0 +1,11 @@
|
|||
import nameof;
|
||||
#include <nameof_macro.hpp>
|
||||
|
||||
struct Test {};
|
||||
|
||||
auto main() -> int {
|
||||
auto _ = nameof::nameof_type<Test>();
|
||||
{
|
||||
auto _ = NAMEOF_TYPE(Test);
|
||||
}
|
||||
}
|
1073
include/nameof.hpp
1073
include/nameof.hpp
File diff suppressed because it is too large
Load diff
149
include/nameof_macro.hpp
Normal file
149
include/nameof_macro.hpp
Normal file
|
@ -0,0 +1,149 @@
|
|||
#ifndef NEARGYE_NAMEOF_MACRO_HPP
|
||||
#define NEARGYE_NAMEOF_MACRO_HPP
|
||||
|
||||
// Checks nameof_type compiler compatibility.
|
||||
#if defined(__clang__) && __clang_major__ >= 5 || defined(__GNUC__) && __GNUC__ >= 7 || defined(_MSC_VER) && _MSC_VER >= 1910
|
||||
#undef NAMEOF_TYPE_SUPPORTED
|
||||
#define NAMEOF_TYPE_SUPPORTED 1
|
||||
#endif
|
||||
|
||||
// Checks nameof_type_rtti compiler compatibility.
|
||||
#if defined(__clang__)
|
||||
#if __has_feature(cxx_rtti)
|
||||
#undef NAMEOF_TYPE_RTTI_SUPPORTED
|
||||
#define NAMEOF_TYPE_RTTI_SUPPORTED 1
|
||||
#endif
|
||||
#elif defined(__GNUC__)
|
||||
#if defined(__GXX_RTTI)
|
||||
#undef NAMEOF_TYPE_RTTI_SUPPORTED
|
||||
#define NAMEOF_TYPE_RTTI_SUPPORTED 1
|
||||
#endif
|
||||
#elif defined(_MSC_VER)
|
||||
#if defined(_CPPRTTI)
|
||||
#undef NAMEOF_TYPE_RTTI_SUPPORTED
|
||||
#define NAMEOF_TYPE_RTTI_SUPPORTED 1
|
||||
#endif
|
||||
#endif
|
||||
|
||||
// Checks nameof_member compiler compatibility.
|
||||
#if defined(__clang__) && __clang_major__ >= 5 || defined(__GNUC__) && __GNUC__ >= 7 || \
|
||||
defined(_MSC_VER) && defined(_MSVC_LANG) && _MSVC_LANG >= 202002L
|
||||
#undef NAMEOF_MEMBER_SUPPORTED
|
||||
#define NAMEOF_MEMBER_SUPPORTED 1
|
||||
#endif
|
||||
|
||||
// Checks nameof_pointer compiler compatibility.
|
||||
#if defined(__clang__) && __clang_major__ >= 5 || defined(__GNUC__) && __GNUC__ >= 7 || \
|
||||
defined(_MSC_VER) && defined(_MSVC_LANG) && _MSVC_LANG >= 202002L
|
||||
#undef NAMEOF_POINTER_SUPPORTED
|
||||
#define NAMEOF_POINTER_SUPPORTED 1
|
||||
#endif
|
||||
|
||||
// Checks nameof_enum compiler compatibility.
|
||||
#if defined(__clang__) && __clang_major__ >= 5 || defined(__GNUC__) && __GNUC__ >= 9 || defined(_MSC_VER) && _MSC_VER >= 1910
|
||||
#undef NAMEOF_ENUM_SUPPORTED
|
||||
#define NAMEOF_ENUM_SUPPORTED 1
|
||||
#endif
|
||||
|
||||
// Checks nameof_enum compiler aliases compatibility.
|
||||
#if defined(__clang__) && __clang_major__ >= 5 || defined(__GNUC__) && __GNUC__ >= 9 || defined(_MSC_VER) && _MSC_VER >= 1920
|
||||
#undef NAMEOF_ENUM_SUPPORTED_ALIASES
|
||||
#define NAMEOF_ENUM_SUPPORTED_ALIASES 1
|
||||
#endif
|
||||
|
||||
// Enum value must be greater or equals than NAMEOF_ENUM_RANGE_MIN. By default NAMEOF_ENUM_RANGE_MIN = -128.
|
||||
// If need another min range for all enum types by default, redefine the macro NAMEOF_ENUM_RANGE_MIN.
|
||||
#if !defined(NAMEOF_ENUM_RANGE_MIN)
|
||||
#define NAMEOF_ENUM_RANGE_MIN -128
|
||||
#endif
|
||||
|
||||
// Enum value must be less or equals than NAMEOF_ENUM_RANGE_MAX. By default NAMEOF_ENUM_RANGE_MAX = 128.
|
||||
// If need another max range for all enum types by default, redefine the macro NAMEOF_ENUM_RANGE_MAX.
|
||||
#if !defined(NAMEOF_ENUM_RANGE_MAX)
|
||||
#define NAMEOF_ENUM_RANGE_MAX 128
|
||||
#endif
|
||||
|
||||
#define NAMEOF_VERSION_MAJOR 0
|
||||
#define NAMEOF_VERSION_MINOR 10
|
||||
#define NAMEOF_VERSION_PATCH 4
|
||||
|
||||
// Obtains name of variable, function, macro.
|
||||
#define NAMEOF(...) \
|
||||
[]() constexpr noexcept { \
|
||||
::std::void_t<decltype(__VA_ARGS__)>(); \
|
||||
constexpr auto _name = ::nameof::detail::pretty_name(#__VA_ARGS__); \
|
||||
static_assert(!_name.empty(), "Expression does not have a name."); \
|
||||
constexpr auto _size = _name.size(); \
|
||||
constexpr auto _nameof = ::nameof::cstring<_size>{_name}; \
|
||||
return _nameof; \
|
||||
}()
|
||||
|
||||
// Obtains full name of variable, function, macro.
|
||||
#define NAMEOF_FULL(...) \
|
||||
[]() constexpr noexcept { \
|
||||
::std::void_t<decltype(__VA_ARGS__)>(); \
|
||||
constexpr auto _name = ::nameof::detail::pretty_name(#__VA_ARGS__, false); \
|
||||
static_assert(!_name.empty(), "Expression does not have a name."); \
|
||||
constexpr auto _size = _name.size(); \
|
||||
constexpr auto _nameof_full = ::nameof::cstring<_size>{_name}; \
|
||||
return _nameof_full; \
|
||||
}()
|
||||
|
||||
// Obtains raw name of variable, function, macro.
|
||||
#define NAMEOF_RAW(...) \
|
||||
[]() constexpr noexcept { \
|
||||
::std::void_t<decltype(__VA_ARGS__)>(); \
|
||||
constexpr auto _name = ::nameof::string_view{#__VA_ARGS__}; \
|
||||
static_assert(!_name.empty(), "Expression does not have a name."); \
|
||||
constexpr auto _size = _name.size(); \
|
||||
constexpr auto _nameof_raw = ::nameof::cstring<_size>{_name}; \
|
||||
return _nameof_raw; \
|
||||
}()
|
||||
|
||||
// Obtains name of enum variable.
|
||||
#define NAMEOF_ENUM(...) ::nameof::nameof_enum<::std::decay_t<decltype(__VA_ARGS__)>>(__VA_ARGS__)
|
||||
|
||||
// Obtains name of enum variable or default value if enum variable out of range.
|
||||
#define NAMEOF_ENUM_OR(...) ::nameof::nameof_enum_or(__VA_ARGS__)
|
||||
|
||||
// Obtains name of static storage enum variable.
|
||||
// This version is much lighter on the compile times and is not restricted to the enum_range limitation.
|
||||
#define NAMEOF_ENUM_CONST(...) ::nameof::nameof_enum<__VA_ARGS__>()
|
||||
|
||||
// Obtains name of enum-flags variable.
|
||||
#define NAMEOF_ENUM_FLAG(...) ::nameof::nameof_enum_flag<::std::decay_t<decltype(__VA_ARGS__)>>(__VA_ARGS__)
|
||||
|
||||
// Obtains type name, reference and cv-qualifiers are ignored.
|
||||
#define NAMEOF_TYPE(...) ::nameof::nameof_type<__VA_ARGS__>()
|
||||
|
||||
// Obtains full type name, with reference and cv-qualifiers.
|
||||
#define NAMEOF_FULL_TYPE(...) ::nameof::nameof_full_type<__VA_ARGS__>()
|
||||
|
||||
// Obtains short type name.
|
||||
#define NAMEOF_SHORT_TYPE(...) ::nameof::nameof_short_type<__VA_ARGS__>()
|
||||
|
||||
// Obtains type name of expression, reference and cv-qualifiers are ignored.
|
||||
#define NAMEOF_TYPE_EXPR(...) ::nameof::nameof_type<decltype(__VA_ARGS__)>()
|
||||
|
||||
// Obtains full type name of expression, with reference and cv-qualifiers.
|
||||
#define NAMEOF_FULL_TYPE_EXPR(...) ::nameof::nameof_full_type<decltype(__VA_ARGS__)>()
|
||||
|
||||
// Obtains short type name of expression.
|
||||
#define NAMEOF_SHORT_TYPE_EXPR(...) ::nameof::nameof_short_type<decltype(__VA_ARGS__)>()
|
||||
|
||||
// Obtains type name, with reference and cv-qualifiers, using RTTI.
|
||||
#define NAMEOF_TYPE_RTTI(...) ::nameof::detail::nameof_type_rtti<::std::void_t<decltype(__VA_ARGS__)>>(typeid(__VA_ARGS__).name())
|
||||
|
||||
// Obtains full type name, using RTTI.
|
||||
#define NAMEOF_FULL_TYPE_RTTI(...) ::nameof::detail::nameof_full_type_rtti<decltype(__VA_ARGS__)>(typeid(__VA_ARGS__).name())
|
||||
|
||||
// Obtains short type name, using RTTI.
|
||||
#define NAMEOF_SHORT_TYPE_RTTI(...) ::nameof::detail::nameof_short_type_rtti<decltype(__VA_ARGS__)>(typeid(__VA_ARGS__).name())
|
||||
|
||||
// Obtains name of member.
|
||||
#define NAMEOF_MEMBER(...) ::nameof::nameof_member<__VA_ARGS__>()
|
||||
|
||||
// Obtains name of a function, a global or class static variable.
|
||||
#define NAMEOF_POINTER(...) ::nameof::nameof_pointer<__VA_ARGS__>()
|
||||
|
||||
#endif
|
46
src/nameof.cpp
Normal file
46
src/nameof.cpp
Normal file
|
@ -0,0 +1,46 @@
|
|||
module;
|
||||
|
||||
#ifndef NAMEOF_IMPORT_STD
|
||||
|
||||
#include <array>
|
||||
#include <cassert>
|
||||
#include <cstddef>
|
||||
#include <cstdint>
|
||||
#include <iosfwd>
|
||||
#include <iterator>
|
||||
#include <limits>
|
||||
#include <type_traits>
|
||||
#include <utility>
|
||||
|
||||
#if !defined(NAMEOF_USING_ALIAS_STRING)
|
||||
#include <string>
|
||||
#endif
|
||||
#if !defined(NAMEOF_USING_ALIAS_STRING_VIEW)
|
||||
#include <string_view>
|
||||
#endif
|
||||
|
||||
#if __has_include(<cxxabi.h>)
|
||||
#include <cxxabi.h>
|
||||
|
||||
#include <cstdlib>
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
export module nameof;
|
||||
|
||||
#ifdef NAMEOF_IMPORT_STD
|
||||
|
||||
import std;
|
||||
|
||||
#endif
|
||||
|
||||
#ifdef NAMEOF_ATTACH_TO_GLOBAL_MODULE
|
||||
extern "C++" {
|
||||
#endif
|
||||
|
||||
#include <nameof.hpp>
|
||||
|
||||
#ifdef NAMEOF_ATTACH_TO_GLOBAL_MODULE
|
||||
} // extern "C++"
|
||||
#endif
|
|
@ -1,7 +1,5 @@
|
|||
include(CheckCXXCompilerFlag)
|
||||
|
||||
set(SOURCES test.cpp)
|
||||
|
||||
if(CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
|
||||
set(OPTIONS /W4 /WX)
|
||||
check_cxx_compiler_flag(/permissive HAS_PERMISSIVE_FLAG)
|
||||
|
@ -10,16 +8,18 @@ if(CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
|
|||
endif()
|
||||
|
||||
check_cxx_compiler_flag(/std:c++20 HAS_CPP20_FLAG)
|
||||
check_cxx_compiler_flag(/std:c++23 HAS_CPP23_FLAG)
|
||||
check_cxx_compiler_flag(/std:c++latest HAS_CPPLATEST_FLAG)
|
||||
elseif(CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
||||
set(CMAKE_VERBOSE_MAKEFILE ON)
|
||||
set(OPTIONS -Wall -Wextra -pedantic-errors -Werror)
|
||||
|
||||
check_cxx_compiler_flag(-std=c++20 HAS_CPP20_FLAG)
|
||||
check_cxx_compiler_flag(-std=c++23 HAS_CPP23_FLAG)
|
||||
endif()
|
||||
|
||||
function(make_test target std)
|
||||
add_executable(${target} ${SOURCES})
|
||||
function(make_test src target std)
|
||||
add_executable(${target} ${src})
|
||||
target_compile_options(${target} PRIVATE ${OPTIONS})
|
||||
target_include_directories(${target} PRIVATE 3rdparty/Catch2)
|
||||
target_link_libraries(${target} PRIVATE ${CMAKE_PROJECT_NAME})
|
||||
|
@ -34,12 +34,20 @@ function(make_test target std)
|
|||
add_test(NAME ${target} COMMAND ${target})
|
||||
endfunction()
|
||||
|
||||
make_test(${CMAKE_PROJECT_NAME}-cpp17.t c++17)
|
||||
make_test(test.cpp test-cpp17 c++17)
|
||||
make_test(test_aliases.cpp test_aliases-cpp17 c++17)
|
||||
|
||||
if(HAS_CPP20_FLAG)
|
||||
make_test(${CMAKE_PROJECT_NAME}-cpp20.t c++20)
|
||||
make_test(test.cpp test-cpp20 c++20)
|
||||
make_test(test_aliases.cpp test_aliases-cpp20 c++20)
|
||||
endif()
|
||||
|
||||
if(HAS_CPP23_FLAG)
|
||||
make_test(test.cpp test-cpp23 c++23)
|
||||
make_test(test_aliases.cpp test_aliases-cpp23 c++23)
|
||||
endif()
|
||||
|
||||
if(HAS_CPPLATEST_FLAG)
|
||||
make_test(${CMAKE_PROJECT_NAME}-cpplatest.t c++latest)
|
||||
make_test(test.cpp test-cpplatest c++latest)
|
||||
make_test(test_aliases.cpp test_aliases-cpplatest c++latest)
|
||||
endif()
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
// Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
||||
// SPDX-License-Identifier: MIT
|
||||
// Copyright (c) 2018 - 2022 Daniil Goncharov <neargye@gmail.com>.
|
||||
// Copyright (c) 2018 - 2024 Daniil Goncharov <neargye@gmail.com>.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
|
@ -46,8 +46,16 @@ struct SomeStruct {
|
|||
int SomeMethod2() const {
|
||||
throw std::runtime_error{"should not be called!"};
|
||||
}
|
||||
|
||||
static int somestaticfield;
|
||||
constexpr static int someotherstaticfield = 21;
|
||||
};
|
||||
|
||||
int SomeStruct::somestaticfield;
|
||||
|
||||
int someglobalvariable = 0;
|
||||
const int someglobalconstvariable = 42;
|
||||
|
||||
void SomeMethod3() {
|
||||
throw std::runtime_error{"should not be called!"};
|
||||
}
|
||||
|
@ -113,7 +121,7 @@ enum class OutOfRange {
|
|||
too_high = NAMEOF_ENUM_RANGE_MAX + 1
|
||||
};
|
||||
|
||||
struct TestRtti{
|
||||
struct TestRtti {
|
||||
struct Base { virtual ~Base() = default; };
|
||||
struct Derived : Base {};
|
||||
};
|
||||
|
@ -910,12 +918,23 @@ TEST_CASE("NAMEOF_SHORT_TYPE_RTTI") {
|
|||
|
||||
#if defined(NAMEOF_MEMBER_SUPPORTED) && NAMEOF_MEMBER_SUPPORTED
|
||||
|
||||
struct StructMemberInitializationUsingNameof {
|
||||
std::string teststringfield = std::string{nameof::nameof_member<&StructMemberInitializationUsingNameof::teststringfield>()};
|
||||
};
|
||||
|
||||
struct StructWithNonConstexprDestructor {
|
||||
~StructWithNonConstexprDestructor() {}
|
||||
int somefield;
|
||||
};
|
||||
|
||||
TEST_CASE("NAMEOF_MEMBER") {
|
||||
REQUIRE(NAMEOF_MEMBER(&SomeStruct::somefield) == "somefield");
|
||||
REQUIRE(NAMEOF_MEMBER(&SomeStruct::SomeMethod1) == "SomeMethod1");
|
||||
REQUIRE(NAMEOF_MEMBER(&Long::LL::field) == "field");
|
||||
constexpr auto member_ptr = &SomeStruct::somefield;
|
||||
REQUIRE(NAMEOF_MEMBER(member_ptr) == "somefield");
|
||||
REQUIRE(NAMEOF_MEMBER(&StructMemberInitializationUsingNameof::teststringfield) == "teststringfield");
|
||||
REQUIRE(NAMEOF_MEMBER(&StructWithNonConstexprDestructor::somefield) == "somefield");
|
||||
}
|
||||
|
||||
TEST_CASE("nameof_member") {
|
||||
|
@ -924,6 +943,36 @@ TEST_CASE("nameof_member") {
|
|||
REQUIRE(nameof::nameof_member<&Long::LL::field>() == "field");
|
||||
constexpr auto member_ptr = &SomeStruct::somefield;
|
||||
REQUIRE(nameof::nameof_member<member_ptr>() == "somefield");
|
||||
REQUIRE(nameof::nameof_member<&StructMemberInitializationUsingNameof::teststringfield>() == "teststringfield");
|
||||
REQUIRE(nameof::nameof_member<&StructWithNonConstexprDestructor::somefield>() == "somefield");
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
#if defined(NAMEOF_POINTER_SUPPORTED) && NAMEOF_POINTER_SUPPORTED
|
||||
|
||||
void somefunction() {}
|
||||
|
||||
TEST_CASE("NAMEOF_POINTER") {
|
||||
REQUIRE(NAMEOF_POINTER(&SomeStruct::somestaticfield) == "somestaticfield");
|
||||
REQUIRE(NAMEOF_POINTER(&SomeStruct::someotherstaticfield) == "someotherstaticfield");
|
||||
REQUIRE(NAMEOF_POINTER(static_cast<const char*>(nullptr)) == "nullptr");
|
||||
REQUIRE(NAMEOF_POINTER(static_cast<int***>(nullptr)) == "nullptr");
|
||||
constexpr auto global_ptr = &someglobalvariable;
|
||||
REQUIRE(NAMEOF_POINTER(global_ptr) == "someglobalvariable");
|
||||
REQUIRE(NAMEOF_POINTER(&someglobalconstvariable) == "someglobalconstvariable");
|
||||
REQUIRE(NAMEOF_POINTER(&somefunction) == "somefunction");
|
||||
}
|
||||
|
||||
TEST_CASE("nameof_pointer") {
|
||||
REQUIRE(nameof::nameof_pointer<&SomeStruct::somestaticfield>() == "somestaticfield");
|
||||
REQUIRE(nameof::nameof_pointer<&SomeStruct::someotherstaticfield>() == "someotherstaticfield");
|
||||
REQUIRE(nameof::nameof_pointer<static_cast<const char*>(nullptr)>() == "nullptr");
|
||||
REQUIRE(nameof::nameof_pointer<static_cast<int***>(nullptr)>() == "nullptr");
|
||||
constexpr auto global_ptr = &someglobalvariable;
|
||||
REQUIRE(nameof::nameof_pointer<global_ptr>() == "someglobalvariable");
|
||||
REQUIRE(nameof::nameof_pointer<&someglobalconstvariable>() == "someglobalconstvariable");
|
||||
REQUIRE(nameof::nameof_pointer<&somefunction>() == "somefunction");
|
||||
}
|
||||
|
||||
#endif
|
||||
|
|
97
test/test_aliases.cpp
Normal file
97
test/test_aliases.cpp
Normal file
|
@ -0,0 +1,97 @@
|
|||
// Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
||||
// SPDX-License-Identifier: MIT
|
||||
// Copyright (c) 2019 - 2024 Daniil Goncharov <neargye@gmail.com>.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
// in the Software without restriction, including without limitation the rights
|
||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
// copies of the Software, and to permit persons to whom the Software is
|
||||
// furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in all
|
||||
// copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
// SOFTWARE.
|
||||
|
||||
#include <string>
|
||||
#include <string_view>
|
||||
|
||||
#define CATCH_CONFIG_MAIN
|
||||
#include <catch.hpp>
|
||||
|
||||
struct MyString {
|
||||
MyString() : str{} {} // required
|
||||
MyString(const char* s) : str{s} {} // required
|
||||
MyString(const char* s, std::size_t l) : str{s, l} {} // required
|
||||
bool empty() const { return str.empty(); } // required
|
||||
MyString& append(std::size_t count, char c) { str.append(count, c); return *this; } // required
|
||||
MyString& append(const char* s, std::size_t count) { str.append(s, count); return *this; } // required
|
||||
MyString& append(const MyString& s) { str.append(s.str); return *this; } // required
|
||||
|
||||
std::size_t size() const { return str.size(); }
|
||||
int compare(const char* s) const { return str.compare(s); }
|
||||
|
||||
private:
|
||||
std::string str;
|
||||
};
|
||||
|
||||
struct MyStringView {
|
||||
using value_type = char; // required
|
||||
static constexpr auto npos = std::string_view::npos; // required
|
||||
|
||||
constexpr MyStringView() : str{} {} // required
|
||||
constexpr MyStringView(const char* s) : str{s} {} // required
|
||||
constexpr MyStringView(const char* s, std::size_t size) : str{s, size} {} // required
|
||||
constexpr bool empty() const { return str.empty(); } // required
|
||||
constexpr std::size_t size() const { return str.size(); } // required
|
||||
constexpr const char* data() const { return str.data(); } // required
|
||||
constexpr const char& operator[](std::size_t i) const { return str[i]; } // required
|
||||
constexpr void remove_prefix(std::size_t n) { str.remove_prefix(n); } // required
|
||||
constexpr void remove_suffix(std::size_t n) { str.remove_suffix(n); } // required
|
||||
constexpr int compare(MyStringView s) const { return str.compare(s.str); } // required
|
||||
|
||||
constexpr int compare(const char* s) const { return str.compare(s); }
|
||||
|
||||
private:
|
||||
std::string_view str;
|
||||
|
||||
constexpr MyStringView(std::string_view s) : str{s} {}
|
||||
};
|
||||
|
||||
#define NAMEOF_USING_ALIAS_STRING using string = MyString;
|
||||
#define NAMEOF_USING_ALIAS_STRING_VIEW using string_view = MyStringView;
|
||||
|
||||
#include <nameof.hpp>
|
||||
|
||||
enum class Color { RED = 1, GREEN = 2, BLUE = 4 };
|
||||
|
||||
TEST_CASE("string") {
|
||||
auto cr = nameof::nameof_enum_flag(Color::RED);
|
||||
REQUIRE_FALSE(cr.empty());
|
||||
REQUIRE(cr.compare("RED") == 0);
|
||||
|
||||
auto crg = nameof::nameof_enum_flag(static_cast<Color>(1 | 2));
|
||||
REQUIRE_FALSE(crg.empty());
|
||||
REQUIRE(crg.compare("RED|GREEN") == 0);
|
||||
|
||||
auto cn = nameof::nameof_enum_flag(Color{0});
|
||||
REQUIRE(cn.empty());
|
||||
REQUIRE(cn.size() == 0);
|
||||
}
|
||||
|
||||
TEST_CASE("string_view") {
|
||||
auto cr = nameof::nameof_enum(Color::RED);
|
||||
REQUIRE_FALSE(cr.empty());
|
||||
REQUIRE(cr.compare("RED") == 0);
|
||||
|
||||
auto cn = nameof::nameof_enum(Color{0});
|
||||
REQUIRE(cn.empty());
|
||||
REQUIRE(cn.size() == 0);
|
||||
}
|
Loading…
Reference in a new issue