From 2ef3e843d89ad3f12de77e2b2fb9d5c4cbd5a99d Mon Sep 17 00:00:00 2001 From: fantasy-peak <1356346239@qq.com> Date: Fri, 1 Nov 2024 16:39:00 +0800 Subject: [PATCH] Add support C++20 modules --- .github/workflows/cpp20-modules.yml | 42 + CMakeLists.txt | 64 +- examples/cpp20_modules_example/CMakeLists.txt | 23 + .../cpp20_modules_example/module_client.cpp | 80 + .../cpp20_modules_example/module_server.cpp | 25 + examples/cpp20_modules_example/module_std.cpp | 30 + modules/drogon.cppm | 160 ++ modules/std.mock.cppm | 2267 +++++++++++++++++ 8 files changed, 2690 insertions(+), 1 deletion(-) create mode 100644 .github/workflows/cpp20-modules.yml create mode 100644 examples/cpp20_modules_example/CMakeLists.txt create mode 100644 examples/cpp20_modules_example/module_client.cpp create mode 100644 examples/cpp20_modules_example/module_server.cpp create mode 100644 examples/cpp20_modules_example/module_std.cpp create mode 100644 modules/drogon.cppm create mode 100644 modules/std.mock.cppm diff --git a/.github/workflows/cpp20-modules.yml b/.github/workflows/cpp20-modules.yml new file mode 100644 index 0000000000..9c5a7562c6 --- /dev/null +++ b/.github/workflows/cpp20-modules.yml @@ -0,0 +1,42 @@ +name: C++20 + +on: + push: + branches: [master] + pull_request: + +permissions: + contents: read + +jobs: + modules: + runs-on: ubuntu-24.04 + steps: + - uses: actions/checkout@v4 + + - name: Checkout Drogon source code + uses: actions/checkout@v4 + with: + submodules: true + fetch-depth: 0 + + - name: Install dependencies + run: | + sudo apt update + sudo apt-get install -y cmake ninja-build clang + sudo apt-get install -y libjsoncpp-dev uuid-dev libssl-dev zlib1g-dev libsqlite3-dev + sudo apt-get install -y ninja-build libbrotli-dev + cmake --version + ninja --version + clang++ --version + + - name: Clang Build Modules + run: | + mkdir build && cd build && pwd + CC=clang CXX=clang++ cmake .. \ + -DCMAKE_CXX_STANDARD=20 \ + -GNinja \ + -DDROGON_BUILD_MODULES=ON \ + -DDROGON_BUILD_MODULES_EXAMPLE=ON \ + -DBUILD_TESTING=OFF + ninja -j 9 diff --git a/CMakeLists.txt b/CMakeLists.txt index a46d3af1b6..15ce4196cd 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -54,6 +54,60 @@ if (CMAKE_CXX_COMPILER_ID MATCHES "MSVC") endif () add_library(${PROJECT_NAME}) + +option(DROGON_BUILD_MODULES "Build drogon library in C++20 Modules form" OFF) +option(DROGON_BUILD_MODULES_EXAMPLE "Build drogon modules example" OFF) +option(HAS_STD_MODULE "has c++ std module" OFF) + +if(${DROGON_BUILD_MODULES}) + message("-- use c++20 module") + + if(CMAKE_VERSION VERSION_GREATER_EQUAL 3.28) + cmake_policy(SET CMP0155 NEW) + elseif(CMAKE_VERSION VERSION_GREATER_EQUAL 3.27) + set(CMAKE_EXPERIMENTAL_CXX_MODULE_CMAKE_API "aa1f7df0-828a-4fcd-9afc-2dc80491aca7") + elseif(CMAKE_VERSION VERSION_GREATER_EQUAL 3.26) + set(CMAKE_EXPERIMENTAL_CXX_MODULE_CMAKE_API "2182bf5c-ef0d-489a-91da-49dbc3090d2a") + if (CMAKE_CXX_COMPILER_ID STREQUAL "Clang") + set(CMake_TEST_CXXModules_UUID "a246741c-d067-4019-a8fb-3d16b0c9d1d3") + set(CMAKE_EXPERIMENTAL_CXX_MODULE_DYNDEP 1) + string(CONCAT CMAKE_EXPERIMENTAL_CXX_SCANDEP_SOURCE + "${CMAKE_CXX_COMPILER_CLANG_SCAN_DEPS}" + " -format=p1689" + " --" + " " + " -x c++ -c -o " + " -MT " + " -MD -MF " + " > ") + set(CMAKE_EXPERIMENTAL_CXX_MODULE_MAP_FORMAT "clang") + set(CMAKE_EXPERIMENTAL_CXX_MODULE_MAP_FLAG "@") + + set(CMAKE_CXX_EXTENSIONS OFF) + endif() + if (CMAKE_CXX_COMPILER_ID MATCHES "MSVC") + set(CMAKE_EXPERIMENTAL_CXX_MODULE_DYNDEP 1) + endif() + endif() + + set(CMAKE_CXX_STANDARD_REQUIRED ON) + + set(MODULES_SRCS "") + list(APPEND MODULES_SRCS ${PROJECT_SOURCE_DIR}/modules/drogon.cppm) + if (NOT ${HAS_STD_MODULE}) + list(APPEND MODULES_SRCS ${PROJECT_SOURCE_DIR}/modules/std.mock.cppm) + endif() + + message("${MODULES_SRCS}") + + target_sources(${PROJECT_NAME} PUBLIC FILE_SET CXX_MODULES FILES ${MODULES_SRCS}) + target_compile_features(${PROJECT_NAME} PUBLIC cxx_std_20) + # build modules example + if(${DROGON_BUILD_MODULES_EXAMPLE}) + add_subdirectory(examples/cpp20_modules_example) + endif() +endif() + if (BUILD_SHARED_LIBS) find_package(Threads) # set(BUILD_EXAMPLES FALSE) @@ -678,12 +732,20 @@ if (BUILD_TESTING) endif (BUILD_TESTING) # Installation - +if(${DROGON_BUILD_MODULES}) +install(TARGETS ${PROJECT_NAME} + EXPORT DrogonTargets + RUNTIME DESTINATION "${INSTALL_BIN_DIR}" COMPONENT bin + ARCHIVE DESTINATION "${INSTALL_LIB_DIR}" COMPONENT lib + LIBRARY FILE_SET CXX_MODULES DESTINATION "${INSTALL_LIB_DIR}" COMPONENT lib) +else() install(TARGETS ${PROJECT_NAME} EXPORT DrogonTargets RUNTIME DESTINATION "${INSTALL_BIN_DIR}" COMPONENT bin ARCHIVE DESTINATION "${INSTALL_LIB_DIR}" COMPONENT lib LIBRARY DESTINATION "${INSTALL_LIB_DIR}" COMPONENT lib) +endif() + install(FILES ${DROGON_HEADERS} DESTINATION ${INSTALL_INCLUDE_DIR}/drogon) diff --git a/examples/cpp20_modules_example/CMakeLists.txt b/examples/cpp20_modules_example/CMakeLists.txt new file mode 100644 index 0000000000..99a902d838 --- /dev/null +++ b/examples/cpp20_modules_example/CMakeLists.txt @@ -0,0 +1,23 @@ +cmake_minimum_required(VERSION 3.28) + +project(modules_example) + +set(CMAKE_CXX_STANDARD 20) + +# Default to C++ extensions being off. Clang's modules support have trouble +# with extensions right now. +set(CMAKE_CXX_EXTENSIONS OFF) + +find_package(jsoncpp REQUIRED) +find_package(OpenSSL REQUIRED) +# set(CMAKE_PREFIX_PATH /root/github/drogon/build/nqf/lib/cmake/Drogon;/root/github/drogon/build/nqf/lib/cmake/Trantor) +# find_package(Drogon REQUIRED) + +add_executable(module_client module_client.cpp) +target_link_libraries(module_client drogon OpenSSL::SSL OpenSSL::Crypto uuid pthread dl) + +add_executable(module_std module_std.cpp) +target_link_libraries(module_std PUBLIC drogon OpenSSL::SSL OpenSSL::Crypto uuid pthread dl) + +add_executable(module_server module_server.cpp) +target_link_libraries(module_server PUBLIC drogon OpenSSL::SSL OpenSSL::Crypto uuid pthread dl) diff --git a/examples/cpp20_modules_example/module_client.cpp b/examples/cpp20_modules_example/module_client.cpp new file mode 100644 index 0000000000..07068cbdf5 --- /dev/null +++ b/examples/cpp20_modules_example/module_client.cpp @@ -0,0 +1,80 @@ +import drogon; +import std; + +#include + +#ifdef __linux__ +#include +#include +#endif + +using namespace drogon; + +int nth_resp = 0; + +int main() +{ + trantor::Logger::setLogLevel(trantor::Logger::kTrace); + { + auto client = HttpClient::newHttpClient("http://www.baidu.com"); + client->setSockOptCallback([](int fd) { + std::cout << "setSockOptCallback:" << fd << std::endl; +#ifdef __linux__ + int optval = 10; + ::setsockopt(fd, + SOL_TCP, + TCP_KEEPCNT, + &optval, + static_cast(sizeof optval)); + ::setsockopt(fd, + SOL_TCP, + TCP_KEEPIDLE, + &optval, + static_cast(sizeof optval)); + ::setsockopt(fd, + SOL_TCP, + TCP_KEEPINTVL, + &optval, + static_cast(sizeof optval)); +#endif + }); + + auto req = HttpRequest::newHttpRequest(); + req->setMethod(drogon::Get); + req->setPath("/s"); + req->setParameter("wd", "wx"); + req->setParameter("oq", "wx"); + + for (int i = 0; i < 10; ++i) + { + client->sendRequest( + req, [](ReqResult result, const HttpResponsePtr &response) { + if (result != ReqResult::Ok) + { + std::cout + << "error while sending request to server! result: " + << result << std::endl; + return; + } + + std::cout << "receive response!" << std::endl; + // auto headers=response. + ++nth_resp; + std::cout << response->getBody() << std::endl; + auto cookies = response->cookies(); + for (auto const &cookie : cookies) + { + std::cout << cookie.first << "=" + << cookie.second.value() + << ":domain=" << cookie.second.domain() + << std::endl; + } + std::cout << "count=" << nth_resp << std::endl; + }); + } + std::cout << "requestsBufferSize:" << client->requestsBufferSize() + << std::endl; + } + + app().run(); +} diff --git a/examples/cpp20_modules_example/module_server.cpp b/examples/cpp20_modules_example/module_server.cpp new file mode 100644 index 0000000000..e2d16e43e6 --- /dev/null +++ b/examples/cpp20_modules_example/module_server.cpp @@ -0,0 +1,25 @@ +import drogon; +import std; + +#include + +using namespace drogon; + +int main() +{ + trantor::Logger::setLogLevel(trantor::Logger::kTrace); + app().registerHandler( + "/", + [](const HttpRequestPtr &request, + std::function &&callback) { + LOG_INFO << "connected:" + << (request->connected() ? "true" : "false"); + auto resp = HttpResponse::newHttpResponse(); + resp->setBody("Hello, World!"); + callback(resp); + }, + {Get}); + + LOG_INFO << "Server running on 127.0.0.1:8848"; + app().addListener("127.0.0.1", 8848).run(); +} diff --git a/examples/cpp20_modules_example/module_std.cpp b/examples/cpp20_modules_example/module_std.cpp new file mode 100644 index 0000000000..bf193bee63 --- /dev/null +++ b/examples/cpp20_modules_example/module_std.cpp @@ -0,0 +1,30 @@ +import std; + +int main() +{ + std::cout << "hello world" << std::endl; + + std::vector vec; + vec.emplace_back("hello world"); + std::cout << vec.size() << std::endl; + + std::unordered_map mp; + mp.emplace(1, 2); + std::cout << mp.size() << std::endl; + + auto it = + std::ranges::find_if(vec, [](auto &&p) { return p == "hello world"; }); + if (it != vec.end()) + { + std::cout << "found" << std::endl; + } + + std::unordered_set s; + s.insert(100); + std::cout << s.size() << std::endl; + + std::shared_ptr ptr_1; + std::unique_ptr ptr_2; + + return 0; +} diff --git a/modules/drogon.cppm b/modules/drogon.cppm new file mode 100644 index 0000000000..14df6bde5b --- /dev/null +++ b/modules/drogon.cppm @@ -0,0 +1,160 @@ +module; + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +export module drogon; + +export namespace drogon +{ + +using drogon::HttpClient; +using drogon::HttpClientPtr; +using drogon::HttpRequest; +using drogon::HttpRequestPtr; +using drogon::HttpResponse; +using drogon::HttpResponsePtr; +using drogon::MultiPartParser; +using drogon::RequestStream; +using drogon::RequestStreamPtr; +using drogon::RequestStreamReader; +using drogon::ResponseStream; +using drogon::ResponseStreamPtr; + +using drogon::AdviceCallback; +using drogon::AdviceChainCallback; +using drogon::AdviceDestroySessionCallback; +using drogon::AdviceStartSessionCallback; +using drogon::FilterCallback; +using drogon::FilterChainCallback; +using drogon::HttpReqCallback; +using drogon::MiddlewareCallback; +using drogon::MiddlewareNextCallback; + +using drogon::ContentType; +using drogon::HttpMethod; +using drogon::HttpStatusCode; +using drogon::ReqResult; +using drogon::to_string; +using drogon::to_string_view; +using drogon::operator<<; +using drogon::FileType; +using drogon::Version; +using drogon::WebSocketMessageType; + +using drogon::HttpController; +using drogon::HttpSimpleController; +using drogon::HttpViewData; + +using drogon::PubSubService; +using drogon::SubscriberID; + +using drogon::WebSocketClient; +using drogon::WebSocketClientPtr; +using drogon::WebSocketConnectionPtr; +using drogon::WebSocketController; +using drogon::WebSocketRequestCallback; + +using drogon::app; + +using drogon::HttpCoroMiddleware; +using drogon::HttpFilter; +using drogon::HttpMiddleware; +using drogon::IOThreadStorage; +using drogon::Plugin; + +using drogon::async_run; +using drogon::AsyncTask; +using drogon::CallbackAwaiter; +using drogon::co_future; +using drogon::Mutex; +using drogon::queueInLoopCoro; +using drogon::sleepCoro; +using drogon::switchThreadCoro; +using drogon::Task; +using drogon::untilQuit; + +using drogon::statusCodeToString; + +} // namespace drogon + +export namespace drogon::utils +{ + +using drogon::utils::base64Decode; +using drogon::utils::base64DecodedLength; +using drogon::utils::base64DecodeToVector; +using drogon::utils::base64Encode; +using drogon::utils::base64EncodedLength; +using drogon::utils::base64EncodeUnpadded; +using drogon::utils::binaryStringToHex; +using drogon::utils::brotliCompress; +using drogon::utils::brotliDecompress; +using drogon::utils::createPath; +using drogon::utils::formattedString; +using drogon::utils::genRandomString; +using drogon::utils::getMd5; +using drogon::utils::getSha1; +using drogon::utils::getSha256; +using drogon::utils::getSha3; +using drogon::utils::getUuid; +using drogon::utils::gzipCompress; +using drogon::utils::gzipDecompress; +using drogon::utils::hexToBinaryString; +using drogon::utils::hexToBinaryVector; +using drogon::utils::isBase64; +using drogon::utils::isInteger; +using drogon::utils::needUrlDecoding; +using drogon::utils::replaceAll; +using drogon::utils::splitString; +using drogon::utils::splitStringToSet; +using drogon::utils::urlDecode; +using drogon::utils::urlEncode; +using drogon::utils::urlEncodeComponent; + +} // namespace drogon::utils + +export namespace drogon::nosql +{ + +using drogon::nosql::RedisClient; +using drogon::nosql::RedisClientPtr; +using drogon::nosql::RedisErrorCode; +using drogon::nosql::RedisException; +using drogon::nosql::RedisExceptionCallback; +using drogon::nosql::RedisMessageCallback; +using drogon::nosql::RedisResult; +using drogon::nosql::RedisResultCallback; +using drogon::nosql::RedisResultType; +using drogon::nosql::RedisSubscriber; +using drogon::nosql::RedisTransactionPtr; + +} // namespace drogon::nosql + +export namespace drogon::orm +{ + +using drogon::orm::ConstRowIterator; +using drogon::orm::DbClient; +using drogon::orm::DbListener; +using drogon::orm::DrogonDbException; +using drogon::orm::Field; +using drogon::orm::Mapper; +using drogon::orm::Result; + +} // namespace drogon::orm diff --git a/modules/std.mock.cppm b/modules/std.mock.cppm new file mode 100644 index 0000000000..f86b18114e --- /dev/null +++ b/modules/std.mock.cppm @@ -0,0 +1,2267 @@ +module; + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#if defined(__clang__) +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wunknown-warning-option" +#pragma clang diagnostic ignored "-Wreserved-module-identifier" +#endif + +export module std; + +#if defined(__clang__) +#pragma clang diagnostic pop +#endif + +export namespace std +{ +using std::unordered_map; +using std::unordered_multimap; +using std::operator==; +using std::erase_if; +using std::swap; +} // namespace std + +export namespace std +{ +using std::unordered_set; +using std::unordered_multiset; +using std::operator==; +using std::erase_if; +using std::swap; +} // namespace std + +export namespace std +{ +using std::vector; +using std::operator==; +using std::operator<=>; +using std::erase; +using std::erase_if; +using std::swap; + +namespace pmr +{ +using std::pmr::vector; +} + +using std::hash; +} // namespace std + +export namespace std +{ +using std::advance; +using std::bidirectional_iterator; +using std::bidirectional_iterator_tag; +using std::contiguous_iterator; +using std::contiguous_iterator_tag; +using std::disable_sized_sentinel_for; +using std::distance; +using std::forward_iterator; +using std::forward_iterator_tag; +using std::incrementable; +using std::incrementable_traits; +using std::indirect_binary_predicate; +using std::indirect_equivalence_relation; +using std::indirect_result_t; +using std::indirect_strict_weak_order; +using std::indirect_unary_predicate; +using std::indirectly_comparable; +using std::indirectly_copyable; +using std::indirectly_copyable_storable; +using std::indirectly_movable; +using std::indirectly_movable_storable; +using std::indirectly_readable; +using std::indirectly_readable_traits; +using std::indirectly_regular_unary_invocable; +using std::indirectly_swappable; +using std::indirectly_unary_invocable; +using std::indirectly_writable; +using std::input_iterator; +using std::input_iterator_tag; +using std::input_or_output_iterator; +using std::iter_common_reference_t; +using std::iter_difference_t; +using std::iter_reference_t; +using std::iter_rvalue_reference_t; +using std::iter_value_t; +using std::iterator_traits; +using std::mergeable; +using std::next; +using std::output_iterator; +using std::output_iterator_tag; +using std::permutable; +using std::prev; +using std::projected; +using std::random_access_iterator; +using std::random_access_iterator_tag; +using std::sentinel_for; +using std::sized_sentinel_for; +using std::sortable; +using std::weakly_incrementable; + +namespace ranges +{ +using std::ranges::advance; +using std::ranges::distance; +using std::ranges::next; +using std::ranges::prev; +} // namespace ranges + +using std::reverse_iterator; +using std::operator==; +using std::operator!=; +using std::operator<; +using std::operator>; +using std::operator<=; +using std::operator>=; +using std::operator<=>; +using std::operator-; +using std::operator+; +using std::back_insert_iterator; +using std::back_inserter; +using std::begin; +using std::cbegin; +using std::cend; +using std::common_iterator; +using std::counted_iterator; +using std::crbegin; +using std::crend; +using std::data; +using std::default_sentinel; +using std::default_sentinel_t; +using std::empty; +using std::end; +using std::front_insert_iterator; +using std::front_inserter; +using std::insert_iterator; +using std::inserter; +using std::istream_iterator; +using std::istreambuf_iterator; +using std::iterator; +using std::make_move_iterator; +using std::make_reverse_iterator; +using std::move_iterator; +using std::move_sentinel; +using std::ostream_iterator; +using std::ostreambuf_iterator; +using std::rbegin; +using std::rend; +using std::size; +using std::ssize; +using std::unreachable_sentinel; +using std::unreachable_sentinel_t; +} // namespace std + +export namespace std +{ +using std::memchr; +using std::memcmp; +using std::memcpy; +using std::memmove; +using std::memset; +using std::size_t; +using std::strcat; +using std::strchr; +using std::strcmp; +using std::strcoll; +using std::strcpy; +using std::strcspn; +using std::strerror; +using std::strlen; +using std::strncat; +using std::strncmp; +using std::strncpy; +using std::strpbrk; +using std::strrchr; +using std::strspn; +using std::strstr; +using std::strtok; +using std::strxfrm; +} // namespace std + +// Workaround to ease the use. +export +{ + using ::memchr; + using ::memcmp; + using ::memcpy; + using ::memmove; + using ::memset; + using ::size_t; + using ::strcat; + using ::strchr; + using ::strcmp; + using ::strcoll; + using ::strcpy; + using ::strcspn; + using ::strerror; + using ::strlen; + using ::strncat; + using ::strncmp; + using ::strncpy; + using ::strpbrk; + using ::strrchr; + using ::strspn; + using ::strstr; + using ::strtok; + using ::strxfrm; +} + +export namespace std +{ +using std::clearerr; +using std::fclose; +using std::feof; +using std::ferror; +using std::fflush; +using std::fgetc; +using std::fgetpos; +using std::fgets; +using std::FILE; +using std::fopen; +using std::fpos_t; +using std::fprintf; +using std::fputc; +using std::fputs; +using std::fread; +using std::freopen; +using std::fscanf; +using std::fseek; +using std::fsetpos; +using std::ftell; +using std::fwrite; +using std::getc; +using std::getchar; +using std::perror; +using std::printf; +using std::putc; +using std::putchar; +using std::puts; +using std::remove; +using std::rename; +using std::rewind; +using std::scanf; +using std::setbuf; +using std::setvbuf; +using std::size_t; +using std::snprintf; +using std::sprintf; +using std::sscanf; +using std::tmpfile; +using std::tmpnam; +using std::ungetc; +using std::vfprintf; +using std::vfscanf; +using std::vprintf; +using std::vscanf; +using std::vsnprintf; +using std::vsprintf; +using std::vsscanf; +} // namespace std + +export namespace std +{ +using std::get; +using std::get_if; +using std::holds_alternative; +using std::variant; +using std::variant_alternative; +using std::variant_alternative_t; +using std::variant_npos; +using std::variant_size; +using std::variant_size_v; +using std::operator==; +using std::operator!=; +using std::operator<; +using std::operator>; +using std::operator<=; +using std::operator>=; +using std::operator<=>; +using std::bad_variant_access; +using std::hash; +using std::monostate; +using std::swap; +using std::visit; +} // namespace std + +export namespace std +{ +using std::bind; +using std::bind_front; +using std::bit_and; +using std::bit_not; +using std::bit_or; +using std::bit_xor; +using std::compare_three_way; +using std::cref; +using std::divides; +using std::equal_to; +using std::greater; +using std::greater_equal; +using std::identity; +using std::invoke; +using std::is_bind_expression; +using std::is_bind_expression_v; +using std::is_placeholder; +using std::is_placeholder_v; +using std::less; +using std::less_equal; +using std::logical_and; +using std::logical_not; +using std::logical_or; +using std::minus; +using std::modulus; +using std::multiplies; +using std::negate; +using std::not_equal_to; +using std::not_fn; +using std::plus; +using std::ref; +using std::reference_wrapper; + +namespace placeholders +{ +using std::placeholders::_1; +using std::placeholders::_10; +using std::placeholders::_2; +using std::placeholders::_3; +using std::placeholders::_4; +using std::placeholders::_5; +using std::placeholders::_6; +using std::placeholders::_7; +using std::placeholders::_8; +using std::placeholders::_9; +} // namespace placeholders + +using std::bad_function_call; +using std::function; +using std::mem_fn; +using std::swap; +using std::operator==; +using std::boyer_moore_horspool_searcher; +using std::boyer_moore_searcher; +using std::default_searcher; +using std::hash; + +namespace ranges +{ +using std::ranges::equal_to; +using std::ranges::greater; +using std::ranges::greater_equal; +using std::ranges::less; +using std::ranges::less_equal; +using std::ranges::not_equal_to; +} // namespace ranges +} // namespace std + +export namespace std +{ +using std::align; +using std::allocator; +using std::allocator_arg; +using std::allocator_arg_t; +using std::allocator_traits; +using std::assume_aligned; +using std::make_obj_using_allocator; +using std::pointer_traits; +using std::to_address; +using std::uninitialized_construct_using_allocator; +using std::uses_allocator; +using std::uses_allocator_construction_args; +using std::uses_allocator_v; +using std::operator==; +using std::addressof; +using std::uninitialized_default_construct; +using std::uninitialized_default_construct_n; + +namespace ranges +{ +using std::ranges::uninitialized_default_construct; +using std::ranges::uninitialized_default_construct_n; +} // namespace ranges + +using std::uninitialized_value_construct; +using std::uninitialized_value_construct_n; + +namespace ranges +{ +using std::ranges::uninitialized_value_construct; +using std::ranges::uninitialized_value_construct_n; +} // namespace ranges + +using std::uninitialized_copy; +using std::uninitialized_copy_n; + +namespace ranges +{ +using std::ranges::uninitialized_copy; +using std::ranges::uninitialized_copy_n; +using std::ranges::uninitialized_copy_n_result; +using std::ranges::uninitialized_copy_result; +} // namespace ranges + +using std::uninitialized_move; +using std::uninitialized_move_n; + +namespace ranges +{ +using std::ranges::uninitialized_move; +using std::ranges::uninitialized_move_n; +using std::ranges::uninitialized_move_n_result; +using std::ranges::uninitialized_move_result; +} // namespace ranges + +using std::uninitialized_fill; +using std::uninitialized_fill_n; + +namespace ranges +{ +using std::ranges::uninitialized_fill; +using std::ranges::uninitialized_fill_n; +} // namespace ranges + +using std::construct_at; + +namespace ranges +{ +using std::ranges::construct_at; +} + +using std::destroy; +using std::destroy_at; +using std::destroy_n; + +namespace ranges +{ +using std::ranges::destroy; +using std::ranges::destroy_at; +using std::ranges::destroy_n; +} // namespace ranges + +using std::default_delete; +using std::make_unique; +using std::unique_ptr; +using std::operator<; +using std::operator>; +using std::operator<=; +using std::operator>=; +using std::operator<=>; +using std::operator<<; +using std::allocate_shared; +using std::atomic_compare_exchange_strong; +using std::atomic_compare_exchange_strong_explicit; +using std::atomic_compare_exchange_weak; +using std::atomic_compare_exchange_weak_explicit; +using std::atomic_exchange; +using std::atomic_exchange_explicit; +using std::atomic_is_lock_free; +using std::atomic_load; +using std::atomic_load_explicit; +using std::atomic_store; +using std::atomic_store_explicit; +using std::bad_weak_ptr; +using std::const_pointer_cast; +using std::dynamic_pointer_cast; +using std::enable_shared_from_this; +using std::get_deleter; +using std::hash; +using std::make_shared; +using std::owner_less; +using std::reinterpret_pointer_cast; +using std::shared_ptr; +using std::static_pointer_cast; +using std::swap; +using std::weak_ptr; +} // namespace std + +export namespace std +{ +using std::cerr; +using std::cin; +using std::clog; +using std::cout; +using std::endl; +using std::wcerr; +using std::wcin; +using std::wclog; +using std::wcout; +} // namespace std + +export namespace std +{ +using std::ifstream; +} + +export namespace std +{ +namespace ranges +{ +using std::ranges::in_found_result; +using std::ranges::in_fun_result; +using std::ranges::in_in_out_result; +using std::ranges::in_in_result; +using std::ranges::in_out_out_result; +using std::ranges::in_out_result; +using std::ranges::min_max_result; +} // namespace ranges + +using std::all_of; + +namespace ranges +{ +using std::ranges::all_of; +} + +using std::any_of; + +namespace ranges +{ +using std::ranges::any_of; +} + +using std::none_of; + +namespace ranges +{ +using std::ranges::none_of; +} + +using std::for_each; + +namespace ranges +{ +using std::ranges::for_each; +using std::ranges::for_each_result; +} // namespace ranges + +using std::for_each_n; + +namespace ranges +{ +using std::ranges::for_each_n; +using std::ranges::for_each_n_result; +} // namespace ranges + +using std::find; +using std::find_if; +using std::find_if_not; + +namespace ranges +{ +using std::ranges::find; +using std::ranges::find_if; +using std::ranges::find_if_not; +} // namespace ranges + +namespace ranges +{ +} + +using std::find_end; + +namespace ranges +{ +using std::ranges::find_end; +} + +using std::find_first_of; + +namespace ranges +{ +using std::ranges::find_first_of; +} + +using std::adjacent_find; + +namespace ranges +{ +using std::ranges::adjacent_find; +} + +using std::count; +using std::count_if; + +namespace ranges +{ +using std::ranges::count; +using std::ranges::count_if; +} // namespace ranges + +using std::mismatch; + +namespace ranges +{ +using std::ranges::mismatch; +using std::ranges::mismatch_result; +} // namespace ranges + +using std::equal; + +namespace ranges +{ +using std::ranges::equal; +} + +using std::is_permutation; + +namespace ranges +{ +using std::ranges::is_permutation; +} + +using std::search; + +namespace ranges +{ +using std::ranges::search; +} + +using std::search_n; + +namespace ranges +{ +using std::ranges::search_n; +} + +namespace ranges +{ +} + +using std::copy; + +namespace ranges +{ +using std::ranges::copy; +using std::ranges::copy_result; +} // namespace ranges + +using std::copy_n; + +namespace ranges +{ +using std::ranges::copy_n; +using std::ranges::copy_n_result; +} // namespace ranges + +using std::copy_if; + +namespace ranges +{ +using std::ranges::copy_if; +using std::ranges::copy_if_result; +} // namespace ranges + +using std::copy_backward; + +namespace ranges +{ +using std::ranges::copy_backward; +using std::ranges::copy_backward_result; +} // namespace ranges + +using std::move; + +namespace ranges +{ +using std::ranges::move; +using std::ranges::move_result; +} // namespace ranges + +using std::move_backward; + +namespace ranges +{ +using std::ranges::move_backward; +using std::ranges::move_backward_result; +} // namespace ranges + +using std::swap_ranges; + +namespace ranges +{ +using std::ranges::swap_ranges; +using std::ranges::swap_ranges_result; +} // namespace ranges + +using std::iter_swap; +using std::transform; + +namespace ranges +{ +using std::ranges::binary_transform_result; +using std::ranges::transform; +using std::ranges::unary_transform_result; +} // namespace ranges + +using std::replace; +using std::replace_if; + +namespace ranges +{ +using std::ranges::replace; +using std::ranges::replace_if; +} // namespace ranges + +using std::replace_copy; +using std::replace_copy_if; + +namespace ranges +{ +using std::ranges::replace_copy; +using std::ranges::replace_copy_if; +using std::ranges::replace_copy_if_result; +using std::ranges::replace_copy_result; +} // namespace ranges + +using std::fill; +using std::fill_n; + +namespace ranges +{ +using std::ranges::fill; +using std::ranges::fill_n; +} // namespace ranges + +using std::generate; +using std::generate_n; + +namespace ranges +{ +using std::ranges::generate; +using std::ranges::generate_n; +} // namespace ranges + +using std::remove; +using std::remove_if; + +namespace ranges +{ +using std::ranges::remove; +using std::ranges::remove_if; +} // namespace ranges + +using std::remove_copy; +using std::remove_copy_if; + +namespace ranges +{ +using std::ranges::remove_copy; +using std::ranges::remove_copy_if; +using std::ranges::remove_copy_if_result; +using std::ranges::remove_copy_result; +} // namespace ranges + +using std::unique; + +namespace ranges +{ +using std::ranges::unique; +} + +using std::unique_copy; + +namespace ranges +{ +using std::ranges::unique_copy; +using std::ranges::unique_copy_result; +} // namespace ranges + +using std::reverse; + +namespace ranges +{ +using std::ranges::reverse; +} + +using std::reverse_copy; + +namespace ranges +{ +using std::ranges::reverse_copy; +using std::ranges::reverse_copy_result; +} // namespace ranges + +using std::rotate; + +namespace ranges +{ +using std::ranges::rotate; +} + +using std::rotate_copy; + +namespace ranges +{ +using std::ranges::rotate_copy; +using std::ranges::rotate_copy_result; +} // namespace ranges + +using std::sample; + +namespace ranges +{ +using std::ranges::sample; +} + +using std::shuffle; + +namespace ranges +{ +using std::ranges::shuffle; +} + +using std::shift_left; + +namespace ranges +{ +} + +using std::shift_right; + +namespace ranges +{ +} + +using std::sort; + +namespace ranges +{ +using std::ranges::sort; +} + +using std::stable_sort; + +namespace ranges +{ +using std::ranges::stable_sort; +} + +using std::partial_sort; + +namespace ranges +{ +using std::ranges::partial_sort; +} + +using std::partial_sort_copy; + +namespace ranges +{ +using std::ranges::partial_sort_copy; +using std::ranges::partial_sort_copy_result; +} // namespace ranges + +using std::is_sorted; +using std::is_sorted_until; + +namespace ranges +{ +using std::ranges::is_sorted; +using std::ranges::is_sorted_until; +} // namespace ranges + +using std::nth_element; + +namespace ranges +{ +using std::ranges::nth_element; +} + +using std::lower_bound; + +namespace ranges +{ +using std::ranges::lower_bound; +} + +using std::upper_bound; + +namespace ranges +{ +using std::ranges::upper_bound; +} + +using std::equal_range; + +namespace ranges +{ +using std::ranges::equal_range; +} + +using std::binary_search; + +namespace ranges +{ +using std::ranges::binary_search; +} + +using std::is_partitioned; + +namespace ranges +{ +using std::ranges::is_partitioned; +} + +using std::partition; + +namespace ranges +{ +using std::ranges::partition; +} + +using std::stable_partition; + +namespace ranges +{ +using std::ranges::stable_partition; +} + +using std::partition_copy; + +namespace ranges +{ +using std::ranges::partition_copy; +using std::ranges::partition_copy_result; +} // namespace ranges + +using std::partition_point; + +namespace ranges +{ +using std::ranges::partition_point; +} + +using std::merge; + +namespace ranges +{ +using std::ranges::merge; +using std::ranges::merge_result; +} // namespace ranges + +using std::inplace_merge; + +namespace ranges +{ +using std::ranges::inplace_merge; +} + +using std::includes; + +namespace ranges +{ +using std::ranges::includes; +} + +using std::set_union; + +namespace ranges +{ +using std::ranges::set_union; +using std::ranges::set_union_result; +} // namespace ranges + +using std::set_intersection; + +namespace ranges +{ +using std::ranges::set_intersection; +using std::ranges::set_intersection_result; +} // namespace ranges + +using std::set_difference; + +namespace ranges +{ +using std::ranges::set_difference; +using std::ranges::set_difference_result; +} // namespace ranges + +using std::set_symmetric_difference; + +namespace ranges +{ +using std::ranges::set_symmetric_difference; +using std::ranges::set_symmetric_difference_result; +} // namespace ranges + +using std::push_heap; + +namespace ranges +{ +using std::ranges::push_heap; +} + +using std::pop_heap; + +namespace ranges +{ +using std::ranges::pop_heap; +} + +using std::make_heap; + +namespace ranges +{ +using std::ranges::make_heap; +} + +using std::sort_heap; + +namespace ranges +{ +using std::ranges::sort_heap; +} + +using std::is_heap; + +namespace ranges +{ +using std::ranges::is_heap; +} + +using std::is_heap_until; + +namespace ranges +{ +using std::ranges::is_heap_until; +} + +using std::min; + +namespace ranges +{ +using std::ranges::min; +} + +using std::max; + +namespace ranges +{ +using std::ranges::max; +} + +using std::minmax; + +namespace ranges +{ +using std::ranges::minmax; +using std::ranges::minmax_result; +} // namespace ranges + +using std::min_element; + +namespace ranges +{ +using std::ranges::min_element; +} + +using std::max_element; + +namespace ranges +{ +using std::ranges::max_element; +} + +using std::minmax_element; + +namespace ranges +{ +using std::ranges::minmax_element; +using std::ranges::minmax_element_result; +} // namespace ranges + +using std::clamp; + +namespace ranges +{ +using std::ranges::clamp; +} + +using std::lexicographical_compare; + +namespace ranges +{ +using std::ranges::lexicographical_compare; +} + +using std::lexicographical_compare_three_way; +using std::next_permutation; + +namespace ranges +{ +using std::ranges::next_permutation; +using std::ranges::next_permutation_result; +} // namespace ranges + +using std::prev_permutation; + +namespace ranges +{ +using std::ranges::prev_permutation; +using std::ranges::prev_permutation_result; +} // namespace ranges +} // namespace std + +export namespace std +{ +using std::adopt_lock; +using std::adopt_lock_t; +using std::call_once; +using std::defer_lock; +using std::defer_lock_t; +using std::lock; +using std::lock_guard; +using std::mutex; +using std::once_flag; +using std::recursive_mutex; +using std::recursive_timed_mutex; +using std::scoped_lock; +using std::swap; +using std::timed_mutex; +using std::try_lock; +using std::try_to_lock; +using std::try_to_lock_t; +using std::unique_lock; +} // namespace std + +export namespace std +{ +using std::condition_variable; +using std::condition_variable_any; +using std::cv_status; +using std::notify_all_at_thread_exit; +} // namespace std + +export namespace std +{ +using std::atomic; +using std::atomic_bool; +using std::atomic_char; +using std::atomic_char16_t; +using std::atomic_char32_t; +using std::atomic_char8_t; +using std::atomic_compare_exchange_strong; +using std::atomic_compare_exchange_strong_explicit; +using std::atomic_compare_exchange_weak; +using std::atomic_compare_exchange_weak_explicit; +using std::atomic_exchange; +using std::atomic_exchange_explicit; +using std::atomic_fetch_add; +using std::atomic_fetch_add_explicit; +using std::atomic_fetch_and; +using std::atomic_fetch_and_explicit; +using std::atomic_fetch_or; +using std::atomic_fetch_or_explicit; +using std::atomic_fetch_sub; +using std::atomic_fetch_sub_explicit; +using std::atomic_fetch_xor; +using std::atomic_fetch_xor_explicit; +using std::atomic_flag; +using std::atomic_flag_clear; +using std::atomic_flag_clear_explicit; +using std::atomic_flag_test_and_set; +using std::atomic_flag_test_and_set_explicit; +using std::atomic_init; +using std::atomic_int; +using std::atomic_int16_t; +using std::atomic_int32_t; +using std::atomic_int64_t; +using std::atomic_int8_t; +using std::atomic_int_fast16_t; +using std::atomic_int_fast32_t; +using std::atomic_int_fast64_t; +using std::atomic_int_fast8_t; +using std::atomic_int_least16_t; +using std::atomic_int_least32_t; +using std::atomic_int_least64_t; +using std::atomic_int_least8_t; +using std::atomic_intmax_t; +using std::atomic_intptr_t; +using std::atomic_is_lock_free; +using std::atomic_llong; +using std::atomic_load; +using std::atomic_load_explicit; +using std::atomic_long; +using std::atomic_notify_all; +using std::atomic_notify_one; +using std::atomic_ptrdiff_t; +using std::atomic_ref; +using std::atomic_schar; +using std::atomic_short; +using std::atomic_signal_fence; +using std::atomic_size_t; +using std::atomic_store; +using std::atomic_store_explicit; +using std::atomic_thread_fence; +using std::atomic_uchar; +using std::atomic_uint; +using std::atomic_uint16_t; +using std::atomic_uint32_t; +using std::atomic_uint64_t; +using std::atomic_uint8_t; +using std::atomic_uint_fast16_t; +using std::atomic_uint_fast32_t; +using std::atomic_uint_fast64_t; +using std::atomic_uint_fast8_t; +using std::atomic_uint_least16_t; +using std::atomic_uint_least32_t; +using std::atomic_uint_least64_t; +using std::atomic_uint_least8_t; +using std::atomic_uintmax_t; +using std::atomic_uintptr_t; +using std::atomic_ullong; +using std::atomic_ulong; +using std::atomic_ushort; +using std::atomic_wait; +using std::atomic_wait_explicit; +using std::atomic_wchar_t; +using std::kill_dependency; +using std::memory_order; +using std::memory_order_acq_rel; +using std::memory_order_acquire; +using std::memory_order_consume; +using std::memory_order_relaxed; +using std::memory_order_release; +using std::memory_order_seq_cst; +} // namespace std + +export namespace std +{ +using std::coroutine_handle; +using std::coroutine_traits; +using std::operator==; +using std::operator<=>; +using std::hash; +using std::noop_coroutine; +using std::noop_coroutine_handle; +using std::noop_coroutine_promise; +using std::suspend_always; +using std::suspend_never; +} // namespace std + +export namespace std +{ +using std::byte; +using std::max_align_t; +using std::nullptr_t; +using std::ptrdiff_t; +using std::size_t; +using std::operator<<=; +using std::operator<<; +using std::operator>>=; +using std::operator>>; +using std::operator|=; +using std::operator|; +using std::operator&=; +using std::operator&; +using std::operator^=; +using std::operator^; +using std::operator~; +using std::to_integer; +} // namespace std + +// Ease the use +export +{ + using ::max_align_t; + using ::ptrdiff_t; + using ::size_t; +} + +export namespace std +{ +using std::bad_exception; +using std::current_exception; +using std::exception; +using std::exception_ptr; +using std::get_terminate; +using std::make_exception_ptr; +using std::nested_exception; +using std::rethrow_exception; +using std::rethrow_if_nested; +using std::set_terminate; +using std::terminate; +using std::terminate_handler; +using std::throw_with_nested; +using std::uncaught_exception; +using std::uncaught_exceptions; +} // namespace std + +export namespace std +{ +using std::as_const; +using std::cmp_equal; +using std::cmp_greater; +using std::cmp_greater_equal; +using std::cmp_less; +using std::cmp_less_equal; +using std::cmp_not_equal; +using std::declval; +using std::exchange; +using std::forward; +using std::in_range; +using std::index_sequence; +using std::index_sequence_for; +using std::integer_sequence; +using std::make_index_sequence; +using std::make_integer_sequence; +using std::move; +using std::move_if_noexcept; +using std::pair; +using std::swap; +using std::operator==; +using std::operator<=>; +using std::get; +using std::in_place; +using std::in_place_index; +using std::in_place_index_t; +using std::in_place_t; +using std::in_place_type; +using std::in_place_type_t; +using std::make_pair; +using std::piecewise_construct; +using std::piecewise_construct_t; +using std::tuple_element; +using std::tuple_size; + +namespace rel_ops +{ +using rel_ops::operator!=; +using rel_ops::operator>; +using rel_ops::operator<=; +using rel_ops::operator>=; +} // namespace rel_ops +} // namespace std + +export namespace std +{ +namespace ranges +{ +using std::ranges::begin; +using std::ranges::bidirectional_range; +using std::ranges::borrowed_range; +using std::ranges::cbegin; +using std::ranges::cdata; +using std::ranges::cend; +using std::ranges::common_range; +using std::ranges::contiguous_range; +using std::ranges::crbegin; +using std::ranges::crend; +using std::ranges::data; +using std::ranges::disable_sized_range; +using std::ranges::empty; +using std::ranges::enable_borrowed_range; +using std::ranges::enable_view; +using std::ranges::end; +using std::ranges::forward_range; +using std::ranges::get; +using std::ranges::input_range; +using std::ranges::iterator_t; +using std::ranges::output_range; +using std::ranges::random_access_range; +using std::ranges::range; +using std::ranges::range_difference_t; +using std::ranges::range_reference_t; +using std::ranges::range_rvalue_reference_t; +using std::ranges::range_size_t; +using std::ranges::range_value_t; +using std::ranges::rbegin; +using std::ranges::rend; +using std::ranges::sentinel_t; +using std::ranges::size; +using std::ranges::sized_range; +using std::ranges::ssize; +using std::ranges::subrange; +using std::ranges::subrange_kind; +using std::ranges::view; +using std::ranges::view_base; +using std::ranges::view_interface; +using std::ranges::viewable_range; +} // namespace ranges + +using std::ranges::get; + +namespace ranges +{ +using std::ranges::borrowed_iterator_t; +using std::ranges::borrowed_subrange_t; +using std::ranges::dangling; +using std::ranges::empty_view; + +namespace views +{ +using std::ranges::views::empty; +} + +using std::ranges::single_view; + +namespace views +{ +using std::ranges::views::single; +} + +using std::ranges::iota_view; + +namespace views +{ +using std::ranges::views::iota; +} + +using std::ranges::basic_istream_view; +using std::ranges::istream_view; + +namespace views +{ +using std::ranges::views::all; +using std::ranges::views::all_t; +} // namespace views + +using std::ranges::filter_view; +using std::ranges::ref_view; + +namespace views +{ +using std::ranges::views::filter; +} + +using std::ranges::transform_view; + +namespace views +{ +using std::ranges::views::transform; +} + +using std::ranges::take_view; + +namespace views +{ +using std::ranges::views::take; +} + +using std::ranges::take_while_view; + +namespace views +{ +using std::ranges::views::take_while; +} + +using std::ranges::drop_view; + +namespace views +{ +using std::ranges::views::drop; +} + +using std::ranges::drop_while_view; + +namespace views +{ +using std::ranges::views::drop_while; +} + +using std::ranges::join_view; + +namespace views +{ +using std::ranges::views::join; +} + +using std::ranges::lazy_split_view; +using std::ranges::split_view; + +namespace views +{ +using std::ranges::views::lazy_split; +using std::ranges::views::split; +} // namespace views + +namespace views +{ +using std::ranges::views::counted; +} + +using std::ranges::common_view; + +namespace views +{ +using std::ranges::views::common; +} + +using std::ranges::reverse_view; + +namespace views +{ +using std::ranges::views::reverse; +} + +using std::ranges::elements_view; +using std::ranges::keys_view; +using std::ranges::values_view; + +namespace views +{ +using std::ranges::views::elements; +using std::ranges::views::keys; +using std::ranges::views::values; +} // namespace views +} // namespace ranges +namespace views = ranges::views; +using std::tuple_element; +using std::tuple_size; +} // namespace std + +export namespace std +{ +using std::atto; +using std::centi; +using std::deca; +using std::deci; +using std::exa; +using std::femto; +using std::giga; +using std::hecto; +using std::kilo; +using std::mega; +using std::micro; +using std::milli; +using std::nano; +using std::peta; +using std::pico; +using std::ratio; +using std::ratio_add; +using std::ratio_divide; +using std::ratio_equal; +using std::ratio_equal_v; +using std::ratio_greater; +using std::ratio_greater_equal; +using std::ratio_greater_equal_v; +using std::ratio_greater_v; +using std::ratio_less; +using std::ratio_less_equal; +using std::ratio_less_equal_v; +using std::ratio_less_v; +using std::ratio_multiply; +using std::ratio_not_equal; +using std::ratio_not_equal_v; +using std::ratio_subtract; +using std::tera; +} // namespace std + +export namespace std +{ +using std::float_denorm_style; +using std::float_round_style; +using std::numeric_limits; +} // namespace std + +export namespace std +{ +using std::binary_semaphore; +using std::counting_semaphore; +} // namespace std + +export namespace std +{ +using std::domain_error; +using std::invalid_argument; +using std::length_error; +using std::logic_error; +using std::out_of_range; +using std::overflow_error; +using std::range_error; +using std::runtime_error; +using std::underflow_error; +} // namespace std + +export namespace std +{ +using std::_Exit; +using std::abort; +using std::abs; +using std::aligned_alloc; +using std::at_quick_exit; +using std::atexit; +using std::atof; +using std::atoi; +using std::atol; +using std::atoll; +using std::bsearch; +using std::calloc; +using std::div; +using std::div_t; +using std::exit; +using std::free; +using std::getenv; +using std::labs; +using std::ldiv; +using std::ldiv_t; +using std::llabs; +using std::lldiv; +using std::lldiv_t; +using std::malloc; +using std::mblen; +using std::mbstowcs; +using std::mbtowc; +using std::qsort; +using std::quick_exit; +using std::rand; +using std::realloc; +using std::size_t; +using std::srand; +using std::strtod; +using std::strtof; +using std::strtol; +using std::strtold; +using std::strtoll; +using std::strtoul; +using std::strtoull; +using std::system; +using std::wcstombs; +using std::wctomb; +} // namespace std + +export namespace std +{ +namespace chrono +{ +using std::chrono::duration; +using std::chrono::time_point; +} // namespace chrono + +using std::common_type; + +namespace chrono +{ +using std::chrono::duration_values; +using std::chrono::treat_as_floating_point; +using std::chrono::treat_as_floating_point_v; +using std::chrono::operator+; +using std::chrono::operator-; +using std::chrono::operator*; +using std::chrono::operator/; +using std::chrono::operator%; +using std::chrono::operator==; +using std::chrono::operator<; +using std::chrono::operator>; +using std::chrono::operator<=; +using std::chrono::operator>=; +using std::chrono::operator<=>; +using std::chrono::abs; +using std::chrono::ceil; +using std::chrono::day; +using std::chrono::days; +using std::chrono::duration_cast; +using std::chrono::file_clock; +using std::chrono::file_time; +using std::chrono::floor; +using std::chrono::hh_mm_ss; +using std::chrono::high_resolution_clock; +using std::chrono::hours; +using std::chrono::is_am; +using std::chrono::is_pm; +using std::chrono::last_spec; +using std::chrono::local_days; +using std::chrono::local_seconds; +using std::chrono::local_t; +using std::chrono::local_time; +using std::chrono::make12; +using std::chrono::make24; +using std::chrono::microseconds; +using std::chrono::milliseconds; +using std::chrono::minutes; +using std::chrono::month; +using std::chrono::month_day; +using std::chrono::month_day_last; +using std::chrono::month_weekday; +using std::chrono::month_weekday_last; +using std::chrono::months; +using std::chrono::nanoseconds; +using std::chrono::round; +using std::chrono::seconds; +using std::chrono::steady_clock; +using std::chrono::sys_days; +using std::chrono::sys_seconds; +using std::chrono::sys_time; +using std::chrono::system_clock; +using std::chrono::time_point_cast; +using std::chrono::weekday; +using std::chrono::weekday_indexed; +using std::chrono::weekday_last; +using std::chrono::weeks; +using std::chrono::year; +using std::chrono::year_month; +using std::chrono::year_month_day; +using std::chrono::year_month_day_last; +using std::chrono::year_month_weekday; +using std::chrono::year_month_weekday_last; +using std::chrono::years; +} // namespace chrono + +namespace chrono +{ +using std::chrono::April; +using std::chrono::August; +using std::chrono::December; +using std::chrono::February; +using std::chrono::Friday; +using std::chrono::January; +using std::chrono::July; +using std::chrono::June; +using std::chrono::last; +using std::chrono::March; +using std::chrono::May; +using std::chrono::Monday; +using std::chrono::November; +using std::chrono::October; +using std::chrono::Saturday; +using std::chrono::September; +using std::chrono::Sunday; +using std::chrono::Thursday; +using std::chrono::Tuesday; +using std::chrono::Wednesday; +} // namespace chrono +} // namespace std + +export namespace std::inline literals::inline chrono_literals +{ +using std::literals::chrono_literals::operator""h; +using std::literals::chrono_literals::operator""min; +using std::literals::chrono_literals::operator""s; +using std::literals::chrono_literals::operator""ms; +using std::literals::chrono_literals::operator""us; +using std::literals::chrono_literals::operator""ns; +using std::literals::chrono_literals::operator""d; +using std::literals::chrono_literals::operator""y; +} // namespace std::inline literals::inline chrono_literals + +export namespace std +{ +using std::basic_string; +using std::char_traits; +using std::operator+; +using std::operator==; +using std::operator<=>; +using std::swap; +using std::operator>>; +using std::operator<<; +using std::erase; +using std::erase_if; +using std::getline; +using std::stod; +using std::stof; +using std::stoi; +using std::stol; +using std::stold; +using std::stoll; +using std::stoul; +using std::stoull; +using std::string; +using std::to_string; +using std::to_wstring; +using std::u16string; +using std::u32string; +using std::u8string; +using std::wstring; + +namespace pmr +{ +using std::pmr::basic_string; +using std::pmr::string; +using std::pmr::u16string; +using std::pmr::u32string; +using std::pmr::u8string; +using std::pmr::wstring; +} // namespace pmr + +using std::hash; + +inline namespace literals +{ +inline namespace string_literals +{ +using std::literals::string_literals::operator""s; +} +} // namespace literals +} // namespace std + +export namespace std +{ +using std::array; +using std::operator==; +using std::operator<=>; +using std::get; +using std::swap; +using std::to_array; +using std::tuple_element; +using std::tuple_size; +} // namespace std + +export namespace std +{ +using std::bad_optional_access; +using std::nullopt; +using std::nullopt_t; +using std::optional; +using std::operator==; +using std::operator!=; +using std::operator<; +using std::operator>; +using std::operator<=; +using std::operator>=; +using std::operator<=>; +using std::hash; +using std::make_optional; +using std::swap; +} // namespace std + +export namespace std +{ +using std::assignable_from; +using std::common_reference_with; +using std::common_with; +using std::constructible_from; +using std::convertible_to; +using std::copy_constructible; +using std::copyable; +using std::default_initializable; +using std::derived_from; +using std::destructible; +using std::equality_comparable; +using std::equality_comparable_with; +using std::equivalence_relation; +using std::floating_point; +using std::integral; +using std::invocable; +using std::movable; +using std::move_constructible; +using std::predicate; +using std::regular; +using std::regular_invocable; +using std::relation; +using std::same_as; +using std::semiregular; +using std::signed_integral; +using std::strict_weak_order; +using std::swappable; +using std::swappable_with; +using std::totally_ordered; +using std::totally_ordered_with; +using std::unsigned_integral; +} // namespace std + +export namespace std +{ +using std::int16_t; +using std::int32_t; +using std::int64_t; +using std::int8_t; +using std::int_fast16_t; +using std::int_fast32_t; +using std::int_fast64_t; +using std::int_fast8_t; +using std::int_least16_t; +using std::int_least32_t; +using std::int_least64_t; +using std::int_least8_t; +using std::intmax_t; +using std::intptr_t; +using std::uint16_t; +using std::uint32_t; +using std::uint64_t; +using std::uint8_t; +using std::uint_fast16_t; +using std::uint_fast32_t; +using std::uint_fast64_t; +using std::uint_fast8_t; +using std::uint_least16_t; +using std::uint_least32_t; +using std::uint_least64_t; +using std::uint_least8_t; +using std::uintmax_t; +using std::uintptr_t; +} // namespace std + +// FIXME: Ease the use. +export +{ + using ::int16_t; + using ::int32_t; + using ::int64_t; + using ::int8_t; + using ::int_fast16_t; + using ::int_fast32_t; + using ::int_fast64_t; + using ::int_fast8_t; + using ::int_least16_t; + using ::int_least32_t; + using ::int_least64_t; + using ::int_least8_t; + using ::intmax_t; + using ::intptr_t; + using ::uint16_t; + using ::uint32_t; + using ::uint64_t; + using ::uint8_t; + using ::uint_fast16_t; + using ::uint_fast32_t; + using ::uint_fast64_t; + using ::uint_fast8_t; + using ::uint_least16_t; + using ::uint_least32_t; + using ::uint_least64_t; + using ::uint_least8_t; + using ::uintmax_t; + using ::uintptr_t; +} + +export namespace std +{ +using std::queue; +using std::operator==; +using std::operator!=; +using std::operator<; +using std::operator>; +using std::operator<=; +using std::operator>=; +using std::operator<=>; +using std::priority_queue; +using std::swap; +using std::uses_allocator; +} // namespace std + +export namespace std +{ +using std::align_val_t; +using std::bad_alloc; +using std::bad_array_new_length; +using std::destroying_delete; +using std::destroying_delete_t; +using std::get_new_handler; +using std::launder; +using std::new_handler; +using std::nothrow; +using std::nothrow_t; +using std::set_new_handler; +} // namespace std + +export +{ + using ::operator new; + using ::operator delete; + using ::operator new[]; + using ::operator delete[]; +} + +export namespace std +{ +using std::apply; +using std::forward_as_tuple; +using std::get; +using std::ignore; +using std::make_from_tuple; +using std::make_tuple; +using std::tie; +using std::tuple; +using std::tuple_cat; +using std::tuple_element; +using std::tuple_element_t; +using std::tuple_size; +using std::operator==; +using std::operator<=>; +using std::swap; +using std::tuple_size_v; +using std::uses_allocator; +} // namespace std + +export namespace std +{ +using std::add_const; +using std::add_const_t; +using std::add_cv; +using std::add_cv_t; +using std::add_lvalue_reference; +using std::add_lvalue_reference_t; +using std::add_pointer; +using std::add_pointer_t; +using std::add_rvalue_reference; +using std::add_rvalue_reference_t; +using std::add_volatile; +using std::add_volatile_t; +using std::aligned_storage; +using std::aligned_storage_t; +using std::aligned_union; +using std::aligned_union_t; +using std::alignment_of; +using std::alignment_of_v; +using std::basic_common_reference; +using std::bool_constant; +using std::common_reference; +using std::common_reference_t; +using std::common_type; +using std::common_type_t; +using std::conditional; +using std::conditional_t; +using std::conjunction; +using std::conjunction_v; +using std::decay; +using std::decay_t; +using std::disjunction; +using std::disjunction_v; +using std::enable_if; +using std::enable_if_t; +using std::extent; +using std::extent_v; +using std::false_type; +using std::has_unique_object_representations; +using std::has_unique_object_representations_v; +using std::has_virtual_destructor; +using std::has_virtual_destructor_v; +using std::integral_constant; +using std::invoke_result; +using std::invoke_result_t; +using std::is_abstract; +using std::is_abstract_v; +using std::is_aggregate; +using std::is_aggregate_v; +using std::is_arithmetic; +using std::is_arithmetic_v; +using std::is_array; +using std::is_array_v; +using std::is_assignable; +using std::is_assignable_v; +using std::is_base_of; +using std::is_base_of_v; +using std::is_bounded_array; +using std::is_bounded_array_v; +using std::is_class; +using std::is_class_v; +using std::is_compound; +using std::is_compound_v; +using std::is_const; +using std::is_const_v; +using std::is_constant_evaluated; +using std::is_constructible; +using std::is_constructible_v; +using std::is_convertible; +using std::is_convertible_v; +using std::is_copy_assignable; +using std::is_copy_assignable_v; +using std::is_copy_constructible; +using std::is_copy_constructible_v; +using std::is_default_constructible; +using std::is_default_constructible_v; +using std::is_destructible; +using std::is_destructible_v; +using std::is_empty; +using std::is_empty_v; +using std::is_enum; +using std::is_enum_v; +using std::is_final; +using std::is_final_v; +using std::is_floating_point; +using std::is_floating_point_v; +using std::is_function; +using std::is_function_v; +using std::is_fundamental; +using std::is_fundamental_v; +using std::is_integral; +using std::is_integral_v; +using std::is_invocable; +using std::is_invocable_r; +using std::is_invocable_r_v; +using std::is_invocable_v; +using std::is_lvalue_reference; +using std::is_lvalue_reference_v; +using std::is_member_function_pointer; +using std::is_member_function_pointer_v; +using std::is_member_object_pointer; +using std::is_member_object_pointer_v; +using std::is_member_pointer; +using std::is_member_pointer_v; +using std::is_move_assignable; +using std::is_move_assignable_v; +using std::is_move_constructible; +using std::is_move_constructible_v; +using std::is_nothrow_assignable; +using std::is_nothrow_assignable_v; +using std::is_nothrow_constructible; +using std::is_nothrow_constructible_v; +using std::is_nothrow_convertible; +using std::is_nothrow_convertible_v; +using std::is_nothrow_copy_assignable; +using std::is_nothrow_copy_assignable_v; +using std::is_nothrow_copy_constructible; +using std::is_nothrow_copy_constructible_v; +using std::is_nothrow_default_constructible; +using std::is_nothrow_default_constructible_v; +using std::is_nothrow_destructible; +using std::is_nothrow_destructible_v; +using std::is_nothrow_invocable; +using std::is_nothrow_invocable_r; +using std::is_nothrow_invocable_r_v; +using std::is_nothrow_invocable_v; +using std::is_nothrow_move_assignable; +using std::is_nothrow_move_assignable_v; +using std::is_nothrow_move_constructible; +using std::is_nothrow_move_constructible_v; +using std::is_nothrow_swappable; +using std::is_nothrow_swappable_v; +using std::is_nothrow_swappable_with; +using std::is_nothrow_swappable_with_v; +using std::is_null_pointer; +using std::is_null_pointer_v; +using std::is_object; +using std::is_object_v; +using std::is_pod; +using std::is_pod_v; +using std::is_pointer; +using std::is_pointer_v; +using std::is_polymorphic; +using std::is_polymorphic_v; +using std::is_reference; +using std::is_reference_v; +using std::is_rvalue_reference; +using std::is_rvalue_reference_v; +using std::is_same; +using std::is_same_v; +using std::is_scalar; +using std::is_scalar_v; +using std::is_signed; +using std::is_signed_v; +using std::is_standard_layout; +using std::is_standard_layout_v; +using std::is_swappable; +using std::is_swappable_v; +using std::is_swappable_with; +using std::is_swappable_with_v; +using std::is_trivial; +using std::is_trivial_v; +using std::is_trivially_assignable; +using std::is_trivially_assignable_v; +using std::is_trivially_constructible; +using std::is_trivially_constructible_v; +using std::is_trivially_copy_assignable; +using std::is_trivially_copy_assignable_v; +using std::is_trivially_copy_constructible; +using std::is_trivially_copy_constructible_v; +using std::is_trivially_copyable; +using std::is_trivially_copyable_v; +using std::is_trivially_default_constructible; +using std::is_trivially_default_constructible_v; +using std::is_trivially_destructible; +using std::is_trivially_destructible_v; +using std::is_trivially_move_assignable; +using std::is_trivially_move_assignable_v; +using std::is_trivially_move_constructible; +using std::is_trivially_move_constructible_v; +using std::is_unbounded_array; +using std::is_unbounded_array_v; +using std::is_union; +using std::is_union_v; +using std::is_unsigned; +using std::is_unsigned_v; +using std::is_void; +using std::is_void_v; +using std::is_volatile; +using std::is_volatile_v; +using std::make_signed; +using std::make_signed_t; +using std::make_unsigned; +using std::make_unsigned_t; +using std::negation; +using std::negation_v; +using std::rank; +using std::rank_v; +using std::remove_all_extents; +using std::remove_all_extents_t; +using std::remove_const; +using std::remove_const_t; +using std::remove_cv; +using std::remove_cv_t; +using std::remove_cvref; +using std::remove_cvref_t; +using std::remove_extent; +using std::remove_extent_t; +using std::remove_pointer; +using std::remove_pointer_t; +using std::remove_reference; +using std::remove_reference_t; +using std::remove_volatile; +using std::remove_volatile_t; +using std::true_type; +using std::type_identity; +using std::type_identity_t; +using std::underlying_type; +using std::underlying_type_t; +using std::unwrap_ref_decay; +using std::unwrap_ref_decay_t; +using std::unwrap_reference; +using std::unwrap_reference_t; +using std::void_t; +} // namespace std + +export namespace std +{ +using std::jthread; +using std::swap; +using std::thread; + +namespace this_thread +{ +using std::this_thread::get_id; +using std::this_thread::sleep_for; +using std::this_thread::sleep_until; +using std::this_thread::yield; +} // namespace this_thread + +using std::operator==; +using std::operator<=>; +using std::operator<<; +using std::hash; +} // namespace std + +#if defined(__GLIBCXX__) || defined(__GLIBCPP__) +export namespace __gnu_cxx +{ +using __gnu_cxx::operator==; +using __gnu_cxx::operator-; +} // namespace __gnu_cxx +#endif