mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-25 07:36:06 +00:00

fixes https://github.com/llvm/llvm-project/issues/105190 --------- Co-authored-by: Hui Xie <huixie@Mac.broadband> Co-authored-by: Hui Xie <huixie@Huis-MacBook-Pro.local>
2289 lines
75 KiB
Python
Executable File
2289 lines
75 KiB
Python
Executable File
#!/usr/bin/env python
|
||
|
||
import os
|
||
from builtins import range
|
||
from functools import reduce
|
||
from typing import Any, Dict, List # Needed for python 3.8 compatibility.
|
||
import functools
|
||
import json
|
||
|
||
|
||
def get_libcxx_paths():
|
||
utils_path = os.path.dirname(os.path.abspath(__file__))
|
||
script_name = os.path.basename(__file__)
|
||
assert os.path.exists(utils_path)
|
||
src_root = os.path.dirname(utils_path)
|
||
include_path = os.path.join(src_root, "include")
|
||
assert os.path.exists(include_path)
|
||
docs_path = os.path.join(src_root, "docs")
|
||
assert os.path.exists(docs_path)
|
||
macro_test_path = os.path.join(
|
||
src_root,
|
||
"test",
|
||
"std",
|
||
"language.support",
|
||
"support.limits",
|
||
"support.limits.general",
|
||
)
|
||
assert os.path.exists(macro_test_path)
|
||
assert os.path.exists(
|
||
os.path.join(macro_test_path, "version.version.compile.pass.cpp")
|
||
)
|
||
return script_name, src_root, include_path, docs_path, macro_test_path
|
||
|
||
|
||
script_name, source_root, include_path, docs_path, macro_test_path = get_libcxx_paths()
|
||
|
||
|
||
def has_header(h):
|
||
h_path = os.path.join(include_path, h)
|
||
return os.path.exists(h_path)
|
||
|
||
|
||
def add_version_header(tc):
|
||
tc["headers"].append("version")
|
||
return tc
|
||
|
||
|
||
# ================ ============================================================
|
||
# Field Description
|
||
# ================ ============================================================
|
||
# name The name of the feature-test macro.
|
||
# values A dict whose keys are C++ versions and whose values are the
|
||
# value of the feature-test macro for that C++ version.
|
||
# (TODO: This isn't a very clean model for feature-test
|
||
# macros affected by multiple papers.)
|
||
# headers An array with the headers that should provide the
|
||
# feature-test macro.
|
||
# test_suite_guard An optional string field. When this field is provided,
|
||
# `libcxx_guard` must also be provided. This field is used
|
||
# only to generate the unit tests for the feature-test macros.
|
||
# It can't depend on macros defined in <__config> because the
|
||
# `test/std/` parts of the test suite are intended to be
|
||
# portable to any C++ standard library implementation, not
|
||
# just libc++. It may depend on
|
||
# * macros defined by the compiler itself, or
|
||
# * macros generated by CMake.
|
||
# In some cases we add also depend on macros defined in
|
||
# <__configuration/availability.h>.
|
||
# libcxx_guard An optional string field. When this field is provided,
|
||
# `test_suite_guard` must also be provided. This field is used
|
||
# only to guard the feature-test macro in <version>. It may
|
||
# be the same as `test_suite_guard`, or it may depend on
|
||
# macros defined in <__config>.
|
||
# unimplemented An optional Boolean field with the value `True`. This field
|
||
# is only used when a feature isn't fully implemented. Once
|
||
# you've fully implemented the feature, you should remove
|
||
# this field.
|
||
# ================ ============================================================
|
||
feature_test_macros = [
|
||
add_version_header(x)
|
||
for x in [
|
||
{
|
||
"name": "__cpp_lib_adaptor_iterator_pair_constructor",
|
||
"values": {"c++23": 202106},
|
||
"headers": ["queue", "stack"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_addressof_constexpr",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_allocate_at_least",
|
||
"values": {
|
||
# Note LWG3887 Version macro for allocate_at_least
|
||
"c++23": 202302, # P2652R2 Disallow User Specialization of allocator_traits
|
||
},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_allocator_traits_is_always_equal",
|
||
"values": {"c++17": 201411},
|
||
"headers": [
|
||
"deque",
|
||
"forward_list",
|
||
"list",
|
||
"map",
|
||
"memory",
|
||
"scoped_allocator",
|
||
"set",
|
||
"string",
|
||
"unordered_map",
|
||
"unordered_set",
|
||
"vector",
|
||
],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_any",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["any"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_apply",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["tuple"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_array_constexpr",
|
||
"values": {"c++17": 201603, "c++20": 201811},
|
||
"headers": ["array", "iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_as_const",
|
||
"values": {"c++17": 201510},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_associative_heterogeneous_erasure",
|
||
"values": {"c++23": 202110},
|
||
"headers": ["map", "set", "unordered_map", "unordered_set"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_associative_heterogeneous_insertion",
|
||
"values": {
|
||
"c++26": 202306 # P2363R5 Extending associative containers with the remaining heterogeneous overloads
|
||
},
|
||
"headers": ["map", "set", "unordered_map", "unordered_set"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_assume_aligned",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_flag_test",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["atomic"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_float",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["atomic"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_is_always_lock_free",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["atomic"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_lock_free_type_aliases",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["atomic"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_min_max",
|
||
"values": {"c++26": 202403}, # P0493R5: Atomic minimum/maximum
|
||
"headers": ["atomic"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_ref",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["atomic"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_shared_ptr",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["atomic"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_value_initialization",
|
||
"values": {"c++20": 201911},
|
||
"headers": ["atomic", "memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_atomic_wait",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["atomic"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
"libcxx_guard": "_LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_barrier",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["barrier"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC)",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bind_back",
|
||
"values": {
|
||
"c++23": 202202,
|
||
# "c++26": 202306, # P2714R1 Bind front and back to NTTP callables
|
||
},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bind_front",
|
||
"values": {
|
||
"c++20": 201907,
|
||
"c++26": 202306, # P2714R1 Bind front and back to NTTP callables
|
||
},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bit_cast",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["bit"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bitops",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["bit"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bitset",
|
||
"values": {"c++26": 202306}, # P2697R1 Interfacing bitset with string_view
|
||
"headers": ["bitset"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bool_constant",
|
||
"values": {"c++17": 201505},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_bounded_array_traits",
|
||
"values": {"c++20": 201902},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_boyer_moore_searcher",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_byte",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["cstddef"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_byteswap",
|
||
"values": {"c++23": 202110},
|
||
"headers": ["bit"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_char8_t",
|
||
"values": {"c++20": 201907},
|
||
"headers": [
|
||
"atomic",
|
||
"filesystem",
|
||
"istream",
|
||
"limits",
|
||
"locale",
|
||
"ostream",
|
||
"string",
|
||
"string_view",
|
||
],
|
||
"test_suite_guard": "defined(__cpp_char8_t)",
|
||
"libcxx_guard": "_LIBCPP_HAS_CHAR8_T",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_chrono",
|
||
"values": {
|
||
"c++17": 201611,
|
||
# "c++26": 202306, # P2592R3 Hashing support for std::chrono value classes
|
||
},
|
||
"headers": ["chrono"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_chrono_udls",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["chrono"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_clamp",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_complex_udls",
|
||
"values": {"c++14": 201309},
|
||
"headers": ["complex"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_concepts",
|
||
"values": {"c++20": 202002},
|
||
"headers": ["concepts"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_algorithms",
|
||
"values": {
|
||
"c++20": 201806,
|
||
# "c++26": 202306, # P2562R1 constexpr Stable Sorting
|
||
},
|
||
"headers": ["algorithm", "utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_bitset",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["bitset"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_charconv",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["charconv"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_cmath",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["cmath", "cstdlib"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_complex",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["complex"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_dynamic_alloc",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_functional",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_iterator",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_memory",
|
||
"values": {"c++20": 201811, "c++23": 202202},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_new",
|
||
"values": {"c++26": 202406}, # P2747R2 constexpr placement new
|
||
"headers": ["new"],
|
||
"test_suite_guard": "!defined(_LIBCPP_ABI_VCRUNTIME)",
|
||
"libcxx_guard": "!defined(_LIBCPP_ABI_VCRUNTIME)",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_numeric",
|
||
"values": {"c++20": 201911},
|
||
"headers": ["numeric"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_string",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["string"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_string_view",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["string_view"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_tuple",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["tuple"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_typeinfo",
|
||
"values": {"c++23": 202106},
|
||
"headers": ["typeinfo"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_utility",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constexpr_vector",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["vector"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_constrained_equality",
|
||
"values": {"c++26": 202403}, # P2944R3: Comparisons for reference_wrapper
|
||
"headers": ["optional", "tuple", "utility", "variant"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_containers_ranges",
|
||
"values": {"c++23": 202202},
|
||
"headers": [
|
||
"deque",
|
||
"forward_list",
|
||
"list",
|
||
"map",
|
||
"queue",
|
||
"set",
|
||
"stack",
|
||
"string",
|
||
"unordered_map",
|
||
"unordered_set",
|
||
"vector",
|
||
],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_copyable_function",
|
||
"values": {"c++26": 202306}, # P2548R6 copyable_function
|
||
"headers": ["functional"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_coroutine",
|
||
"values": {"c++20": 201902},
|
||
"headers": ["coroutine"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_debugging",
|
||
"values": {
|
||
"c++26": 202311, # P2546R5 Debugging Support
|
||
# "c++26": 202403, # P2810R4: is_debugger_present is_replaceable
|
||
},
|
||
"headers": ["debugging"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_default_template_type_for_algorithm_values",
|
||
"values": {"c++26": 202403}, # P2248R8: Enabling list-initialization for algorithms
|
||
"headers": ["algorithm", "deque", "forward_list", "list", "ranges", "string", "vector"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_destroying_delete",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["new"],
|
||
"test_suite_guard": "TEST_STD_VER > 17 && defined(__cpp_impl_destroying_delete) && __cpp_impl_destroying_delete >= 201806L",
|
||
"libcxx_guard": "_LIBCPP_STD_VER >= 20 && defined(__cpp_impl_destroying_delete) && __cpp_impl_destroying_delete >= 201806L",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_enable_shared_from_this",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_endian",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["bit"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_erase_if",
|
||
"values": {"c++20": 202002},
|
||
"headers": [
|
||
"deque",
|
||
"forward_list",
|
||
"list",
|
||
"map",
|
||
"set",
|
||
"string",
|
||
"unordered_map",
|
||
"unordered_set",
|
||
"vector",
|
||
],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_exchange_function",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_execution",
|
||
"values": {"c++17": 201603, "c++20": 201902},
|
||
"headers": ["execution"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_expected",
|
||
"values": {"c++23": 202211},
|
||
"headers": ["expected"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_filesystem",
|
||
"values": {"c++17": 201703},
|
||
"headers": ["filesystem"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_FILESYSTEM && _LIBCPP_AVAILABILITY_HAS_FILESYSTEM_LIBRARY)",
|
||
"libcxx_guard": "_LIBCPP_HAS_FILESYSTEM && _LIBCPP_AVAILABILITY_HAS_FILESYSTEM_LIBRARY",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_flat_map",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["flat_map"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_flat_set",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["flat_set"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_format",
|
||
"values": {
|
||
"c++20": 202110,
|
||
# "c++23": 202207, Not implemented P2419R2 Clarify handling of encodings in localized formatting of chrono types
|
||
# "c++26": 202306, P2637R3 Member Visit (implemented)
|
||
# "c++26": 202311, P2918R2 Runtime format strings II (implemented)
|
||
},
|
||
# Note these three papers are adopted at the June 2023 meeting and have sequential numbering
|
||
# 202304 P2510R3 Formatting pointers (Implemented)
|
||
# 202305 P2757R3 Type-checking format args
|
||
# 202306 P2637R3 Member Visit
|
||
"headers": ["format"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_format_path",
|
||
"values": {"c++26": 202403}, # P2845R8: Formatting of std::filesystem::path
|
||
"headers": ["filesystem"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_format_ranges",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["format"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_format_uchar",
|
||
"values": {
|
||
"c++20": 202311 # DR P2909R4 Fix formatting of code units as integers
|
||
},
|
||
"headers": [
|
||
"format" # TODO verify this entry since the paper was underspecified.
|
||
],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_formatters",
|
||
"values": {"c++23": 202302},
|
||
"headers": ["stacktrace", "thread"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_forward_like",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_algorithm",
|
||
"values": {
|
||
"c++26": 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["algorithm"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_array",
|
||
"values": {
|
||
"c++26": 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["array"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_cstring",
|
||
"values": {
|
||
"c++26": 202306 # P2338R4 Freestanding Library: Character primitives and the C library
|
||
# 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["cstring"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_expected",
|
||
"values": {
|
||
"c++26": 202311 # P2833R2 Freestanding Library: inout expected span
|
||
},
|
||
"headers": ["expected"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_mdspan",
|
||
"values": {
|
||
"c++26": 202311 # P2833R2 Freestanding Library: inout expected span
|
||
},
|
||
"headers": ["mdspan"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_optional",
|
||
"values": {
|
||
"c++26": 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["optional"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_string_view",
|
||
"values": {
|
||
"c++26": 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["string_view"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_freestanding_variant",
|
||
"values": {
|
||
"c++26": 202311 # P2407R5 Freestanding Library: Partial Classes
|
||
},
|
||
"headers": ["variant"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_fstream_native_handle",
|
||
"values": {"c++26": 202306}, # P1759R6 Native handles and file streams
|
||
"headers": ["fstream"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_FILESYSTEM && _LIBCPP_HAS_LOCALIZATION)",
|
||
"libcxx_guard": "_LIBCPP_HAS_FILESYSTEM && _LIBCPP_HAS_LOCALIZATION",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_function_ref",
|
||
"values": {
|
||
"c++26": 202306 # P0792R14 function_ref: a type-erased callable reference
|
||
},
|
||
"headers": ["functional"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_gcd_lcm",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["numeric"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_generate_random",
|
||
"values": {"c++26": 202403}, # P1068R11: Vector API for random number generation
|
||
"headers": ["random"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_generic_associative_lookup",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["map", "set"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_generic_unordered_lookup",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["unordered_map", "unordered_set"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_hardware_interference_size",
|
||
"values": {"c++17": 201703},
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (defined(__GCC_DESTRUCTIVE_SIZE) && defined(__GCC_CONSTRUCTIVE_SIZE))",
|
||
"libcxx_guard": "defined(__GCC_DESTRUCTIVE_SIZE) && defined(__GCC_CONSTRUCTIVE_SIZE)",
|
||
"headers": ["new"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_has_unique_object_representations",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_hazard_pointer",
|
||
"values": {"c++26": 202306}, # P2530R3 Hazard Pointers for C++26
|
||
"headers": [
|
||
"hazard_pointer" # TODO verify this entry since the paper was underspecified.
|
||
],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_hypot",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["cmath"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_incomplete_container_elements",
|
||
"values": {"c++17": 201505},
|
||
"headers": ["forward_list", "list", "vector"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_inplace_vector",
|
||
"values": {"c++26": 202406}, # P0843R14 inplace_vector
|
||
"headers": ["inplace_vector"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_int_pow2",
|
||
"values": {"c++20": 202002},
|
||
"headers": ["bit"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_integer_comparison_functions",
|
||
"values": {"c++20": 202002},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_integer_sequence",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_integral_constant_callable",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_interpolate",
|
||
"values": {"c++20": 201902},
|
||
"headers": ["cmath", "numeric"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_invoke",
|
||
"values": {"c++17": 201411},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_invoke_r",
|
||
"values": {"c++23": 202106},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ios_noreplace",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["ios"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_aggregate",
|
||
"values": {"c++17": 201703},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_constant_evaluated",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_final",
|
||
"values": {"c++14": 201402},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_implicit_lifetime",
|
||
"values": {"c++23": 202302},
|
||
"headers": ["type_traits"],
|
||
"test_suite_guard": "__has_builtin(__builtin_is_implicit_lifetime)",
|
||
"libcxx_guard": "__has_builtin(__builtin_is_implicit_lifetime)",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_invocable",
|
||
"values": {"c++17": 201703},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_layout_compatible",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["type_traits"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_nothrow_convertible",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_null_pointer",
|
||
"values": {"c++14": 201309},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_pointer_interconvertible",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["type_traits"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_scoped_enum",
|
||
"values": {"c++23": 202011},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_swappable",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_virtual_base_of",
|
||
"values": {
|
||
"c++26": 202406 # P2985R0 A type trait for detecting virtual base classes
|
||
},
|
||
"headers": ["type_traits"],
|
||
"test_suite_guard": "__has_builtin(__builtin_is_virtual_base_of)",
|
||
"libcxx_guard": "__has_builtin(__builtin_is_virtual_base_of)",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_is_within_lifetime",
|
||
# Note this name was changed from "__cpp_lib_within_lifetime" when the paper was adopted
|
||
# https://github.com/cplusplus/draft/commit/0facada4cadd97e1ba15bfaea76a804f1dc5c309
|
||
"values": {
|
||
"c++26": 202306 # P2641R4 Checking if a union alternative is active
|
||
},
|
||
"headers": ["type_traits"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_jthread",
|
||
"values": {"c++20": 201911},
|
||
"headers": ["stop_token", "thread"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC)",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_latch",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["latch"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC)",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_launder",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["new"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_linalg",
|
||
"values": {
|
||
"c++26": 202311 # P1673 A free function linear algebra interface based on the BLAS
|
||
},
|
||
"headers": ["linalg"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_list_remove_return_type",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["forward_list", "list"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_logical_traits",
|
||
"values": {"c++17": 201510},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_make_from_tuple",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["tuple"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_make_reverse_iterator",
|
||
"values": {"c++14": 201402},
|
||
"headers": ["iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_make_unique",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_map_try_emplace",
|
||
"values": {"c++17": 201411},
|
||
"headers": ["map"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_math_constants",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["numbers"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_math_special_functions",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["cmath"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_mdspan",
|
||
"values": {
|
||
"c++23": 202207,
|
||
"c++26": 202406, # P2389R2 dextents Index Type Parameter
|
||
},
|
||
"headers": ["mdspan"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_memory_resource",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["memory_resource"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_AVAILABILITY_HAS_PMR",
|
||
"libcxx_guard": "_LIBCPP_AVAILABILITY_HAS_PMR",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_modules",
|
||
"values": {"c++23": 202207},
|
||
"headers": [],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_move_iterator_concept",
|
||
"values": {"c++20": 202207},
|
||
"headers": ["iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_move_only_function",
|
||
"values": {"c++23": 202110},
|
||
"headers": ["functional"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_node_extract",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["map", "set", "unordered_map", "unordered_set"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_nonmember_container_access",
|
||
"values": {"c++17": 201411},
|
||
"headers": [
|
||
"array",
|
||
"deque",
|
||
"forward_list",
|
||
"iterator",
|
||
"list",
|
||
"map",
|
||
"regex",
|
||
"set",
|
||
"string",
|
||
"unordered_map",
|
||
"unordered_set",
|
||
"vector",
|
||
],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_not_fn",
|
||
"values": {
|
||
"c++17": 201603,
|
||
"c++26": 202306, # P2714R1 Bind front and back to NTTP callables
|
||
},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_null_iterators",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_optional",
|
||
"values": {
|
||
"c++17": 201606,
|
||
"c++20": 202106, # P2231R1 Missing constexpr in std::optional and std::variant
|
||
"c++23": 202110, # P0798R8 Monadic operations for std::optional + LWG3621 Remove feature-test macro __cpp_lib_monadic_optional
|
||
},
|
||
"headers": ["optional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_optional_range_support",
|
||
"values": {"c++26": 202406}, # P3168R2 Give std::optional Range Support
|
||
"headers": ["optional"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_out_ptr",
|
||
"values": {
|
||
"c++23": 202106,
|
||
"c++26": 202311, # P2833R2 Freestanding Library: inout expected span
|
||
},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_parallel_algorithm",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["algorithm", "numeric"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_philox_engine",
|
||
"values": {
|
||
"c++26": 202406
|
||
}, # P2075R6 Philox as an extension of the C++ RNG engines
|
||
# Note the paper mentions 202310L as value, which differs from the typical procedure.
|
||
"headers": ["random"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_polymorphic_allocator",
|
||
"values": {"c++20": 201902},
|
||
"headers": ["memory_resource"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_AVAILABILITY_HAS_PMR",
|
||
"libcxx_guard": "_LIBCPP_AVAILABILITY_HAS_PMR",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_print",
|
||
"values": {
|
||
"c++23": 202207,
|
||
# "c++26": 202403, # P3107R5: Permit an efficient implementation of std::print
|
||
# "c++26": 202406, # P3235R3 std::print more types faster with less memory
|
||
},
|
||
"headers": ["ostream", "print"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_quoted_string_io",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["iomanip"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_HAS_LOCALIZATION",
|
||
"libcxx_guard": "_LIBCPP_HAS_LOCALIZATION",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges",
|
||
"values": {
|
||
"c++20": 202110, # P2415R2 What is a view?
|
||
"c++23": 202406, # P2997R1 Removing the common reference requirement from the indirectly invocable concepts (implemented as a DR against C++20)
|
||
},
|
||
"headers": ["algorithm", "functional", "iterator", "memory", "ranges"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_as_const",
|
||
"values": {
|
||
"c++23": 202207 # P2278R4 cbegin should always return a constant iterator
|
||
# 202311 # DR P2836R1 std::basic_const_iterator should follow its underlying type’s convertibility
|
||
},
|
||
"headers": ["ranges"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_as_rvalue",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["ranges"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_chunk",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["ranges"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_chunk_by",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["ranges"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_concat",
|
||
"values": {"c++26": 202403}, # P2542R8: views::concat
|
||
"headers": ["ranges"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_contains",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_find_last",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_iota",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["numeric"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_join_with",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["ranges"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_repeat",
|
||
"values": {"c++23": 202207},
|
||
"headers": ["ranges"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_slide",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["ranges"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_starts_ends_with",
|
||
"values": {"c++23": 202106},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_to_container",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["ranges"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ranges_zip",
|
||
"values": {"c++23": 202110},
|
||
"headers": ["ranges", "tuple", "utility"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ratio",
|
||
"values": {"c++26": 202306}, # P2734R0 Adding the new SI prefixes
|
||
"headers": ["ratio"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_raw_memory_algorithms",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_rcu",
|
||
"values": {"c++26": 202306}, # P2545R4 Read-Copy Update (RCU)
|
||
"headers": [
|
||
"rcu" # TODO verify this entry since the paper was underspecified.
|
||
],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_reference_from_temporary",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["type_traits"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_reference_wrapper",
|
||
"values": {"c++26": 202403}, # P2944R3: Comparisons for reference_wrapper
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_remove_cvref",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_result_of_sfinae",
|
||
"values": {"c++14": 201210},
|
||
"headers": ["functional", "type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_robust_nonmodifying_seq_ops",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_sample",
|
||
"values": {"c++17": 201603},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_saturation_arithmetic",
|
||
"values": {"c++26": 202311}, # P0543R3 Saturation arithmetic
|
||
"headers": ["numeric"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_scoped_lock",
|
||
"values": {"c++17": 201703},
|
||
"headers": ["mutex"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_HAS_THREADS",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_semaphore",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["semaphore"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || (_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC)",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS && _LIBCPP_AVAILABILITY_HAS_SYNC",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_senders",
|
||
"values": {"c++26": 202406}, # P2300R10 std::execution
|
||
"headers": ["execution"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_shared_mutex",
|
||
"values": {"c++17": 201505},
|
||
"headers": ["shared_mutex"],
|
||
"test_suite_guard": "_LIBCPP_HAS_THREADS",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_shared_ptr_arrays",
|
||
"values": {"c++17": 201611, "c++20": 201707},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_shared_ptr_weak_type",
|
||
"values": {"c++17": 201606},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_shared_timed_mutex",
|
||
"values": {"c++14": 201402},
|
||
"headers": ["shared_mutex"],
|
||
"test_suite_guard": "_LIBCPP_HAS_THREADS",
|
||
"libcxx_guard": "_LIBCPP_HAS_THREADS",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_shift",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["algorithm"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_smart_ptr_for_overwrite",
|
||
"values": {"c++20": 202002},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_smart_ptr_owner_equality",
|
||
"values": {
|
||
"c++26": 202306 # P1901R2 Enabling the Use of weak_ptr as Keys in Unordered Associative Containers
|
||
},
|
||
"headers": ["memory"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_source_location",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["source_location"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_span",
|
||
"values": {
|
||
"c++20": 202002,
|
||
# "c++26": 202311, # P2821R5 span.at()
|
||
# 202311 # P2833R2 Freestanding Library: inout expected span
|
||
},
|
||
"headers": ["span"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_span_at",
|
||
"values": {"c++26": 202311}, # P2821R3 span.at()
|
||
"headers": ["span"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_span_initializer_list",
|
||
"values": {"c++26": 202311}, # P2447R6 std::span over an initializer list
|
||
"headers": ["span"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_spanstream",
|
||
"values": {"c++23": 202106},
|
||
"headers": ["spanstream"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_ssize",
|
||
"values": {"c++20": 201902},
|
||
"headers": ["iterator"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_sstream_from_string_view",
|
||
"values": {
|
||
"c++26": 202306 # P2495R3 Interfacing stringstreams with string_view
|
||
},
|
||
"headers": ["sstream"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_stacktrace",
|
||
"values": {"c++23": 202011},
|
||
"headers": ["stacktrace"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_starts_ends_with",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["string", "string_view"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_stdatomic_h",
|
||
"values": {"c++23": 202011},
|
||
"headers": ["stdatomic.h"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_string_contains",
|
||
"values": {"c++23": 202011},
|
||
"headers": ["string", "string_view"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_string_resize_and_overwrite",
|
||
"values": {"c++23": 202110},
|
||
"headers": ["string"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_string_udls",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["string"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_string_view",
|
||
"values": {
|
||
"c++17": 201606,
|
||
"c++20": 201803,
|
||
"c++26": 202403, # P2591R5: Concatenation of strings and string views
|
||
},
|
||
"headers": ["string", "string_view"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_submdspan",
|
||
"values": {
|
||
"c++26": 202306, # P2630R4: submdspan
|
||
# "c++26": 202403, # P2642R6: Padded mdspan layouts
|
||
},
|
||
"headers": ["mdspan"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_syncbuf",
|
||
"values": {"c++20": 201803},
|
||
"headers": ["syncstream"],
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_HAS_EXPERIMENTAL_SYNCSTREAM",
|
||
"libcxx_guard": "_LIBCPP_HAS_EXPERIMENTAL_SYNCSTREAM",
|
||
},
|
||
{
|
||
"name": "__cpp_lib_text_encoding",
|
||
"values": {
|
||
"c++26": 202306 # P1885R12 Naming Text Encodings to Demystify Them
|
||
},
|
||
"headers": ["text_encoding"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_three_way_comparison",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["compare"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_to_address",
|
||
"values": {"c++20": 201711},
|
||
"headers": ["memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_to_array",
|
||
"values": {"c++20": 201907},
|
||
"headers": ["array"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_to_chars",
|
||
"values": {
|
||
"c++17": 201611,
|
||
"c++26": 202306, # P2497R0 Testing for success or failure of <charconv> functions
|
||
},
|
||
"headers": ["charconv"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_to_string",
|
||
"values": {"c++26": 202306}, # P2587R3 to_string or not to_string
|
||
"headers": ["string"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_to_underlying",
|
||
"values": {"c++23": 202102},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_transformation_trait_aliases",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_transparent_operators",
|
||
"values": {"c++14": 201210, "c++17": 201510},
|
||
"headers": ["functional", "memory"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_tuple_element_t",
|
||
"values": {"c++14": 201402},
|
||
"headers": ["tuple"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_tuple_like",
|
||
"values": {
|
||
"c++23": 202207, # P2165R4 Compatibility between tuple, pair and tuple-like objects
|
||
"c++26": 202311, # P2819R2 Add tuple protocol to complex (implemented)
|
||
},
|
||
"headers": ["map", "tuple", "unordered_map", "utility"],
|
||
"unimplemented": True,
|
||
},
|
||
{
|
||
"name": "__cpp_lib_tuples_by_type",
|
||
"values": {"c++14": 201304},
|
||
"headers": ["tuple", "utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_type_identity",
|
||
"values": {"c++20": 201806},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_type_trait_variable_templates",
|
||
"values": {"c++17": 201510},
|
||
"headers": ["type_traits"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_uncaught_exceptions",
|
||
"values": {"c++17": 201411},
|
||
"headers": ["exception"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_unordered_map_try_emplace",
|
||
"values": {"c++17": 201411},
|
||
"headers": ["unordered_map"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_unreachable",
|
||
"values": {"c++23": 202202},
|
||
"headers": ["utility"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_unwrap_ref",
|
||
"values": {"c++20": 201811},
|
||
"headers": ["functional"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_variant",
|
||
"values": {
|
||
"c++17": 202102, # std::visit for classes derived from std::variant
|
||
"c++20": 202106, # P2231R1 Missing constexpr in std::optional and std::variant
|
||
"c++26": 202306, # P2637R3 Member visit
|
||
},
|
||
"headers": ["variant"],
|
||
},
|
||
{
|
||
"name": "__cpp_lib_void_t",
|
||
"values": {"c++17": 201411},
|
||
"headers": ["type_traits"],
|
||
},
|
||
]
|
||
]
|
||
|
||
assert feature_test_macros == sorted(feature_test_macros, key=lambda tc: tc["name"])
|
||
for tc in feature_test_macros:
|
||
assert tc["headers"] == sorted(tc["headers"]), tc
|
||
assert ("libcxx_guard" in tc) == ("test_suite_guard" in tc), tc
|
||
valid_keys = ["name", "values", "headers", "libcxx_guard", "test_suite_guard", "unimplemented"]
|
||
assert all(key in valid_keys for key in tc.keys()), tc
|
||
|
||
# Map from each header to the Lit annotations that should be used for
|
||
# tests that include that header.
|
||
#
|
||
# For example, when threads are not supported, any test that includes
|
||
# <thread> should be marked as UNSUPPORTED, because including <thread>
|
||
# is a hard error in that case.
|
||
lit_markup = {
|
||
"barrier": ["UNSUPPORTED: no-threads"],
|
||
"filesystem": ["UNSUPPORTED: no-filesystem"],
|
||
"fstream": ["UNSUPPORTED: no-localization"],
|
||
"iomanip": ["UNSUPPORTED: no-localization"],
|
||
"ios": ["UNSUPPORTED: no-localization"],
|
||
"iostream": ["UNSUPPORTED: no-localization"],
|
||
"istream": ["UNSUPPORTED: no-localization"],
|
||
"latch": ["UNSUPPORTED: no-threads"],
|
||
"locale": ["UNSUPPORTED: no-localization"],
|
||
"mutex": ["UNSUPPORTED: no-threads"],
|
||
"ostream": ["UNSUPPORTED: no-localization"],
|
||
"print": ["UNSUPPORTED: no-filesystem"],
|
||
"regex": ["UNSUPPORTED: no-localization"],
|
||
"semaphore": ["UNSUPPORTED: no-threads"],
|
||
"shared_mutex": ["UNSUPPORTED: no-threads"],
|
||
"sstream": ["UNSUPPORTED: no-localization"],
|
||
"syncstream": ["UNSUPPORTED: no-localization"],
|
||
"stdatomic.h": ["UNSUPPORTED: no-threads"],
|
||
"stop_token": ["UNSUPPORTED: no-threads"],
|
||
"thread": ["UNSUPPORTED: no-threads"],
|
||
}
|
||
|
||
|
||
def get_std_dialects():
|
||
std_dialects = ["c++14", "c++17", "c++20", "c++23", "c++26"]
|
||
return list(std_dialects)
|
||
|
||
|
||
def get_first_std(d):
|
||
for s in get_std_dialects():
|
||
if s in d.keys():
|
||
return s
|
||
return None
|
||
|
||
|
||
def get_last_std(d):
|
||
rev_dialects = get_std_dialects()
|
||
rev_dialects.reverse()
|
||
for s in rev_dialects:
|
||
if s in d.keys():
|
||
return s
|
||
return None
|
||
|
||
|
||
def get_std_before(d, std):
|
||
std_dialects = get_std_dialects()
|
||
candidates = std_dialects[0 : std_dialects.index(std)]
|
||
candidates.reverse()
|
||
for cand in candidates:
|
||
if cand in d.keys():
|
||
return cand
|
||
return None
|
||
|
||
|
||
def get_value_before(d, std):
|
||
new_std = get_std_before(d, std)
|
||
if new_std is None:
|
||
return None
|
||
return d[new_std]
|
||
|
||
|
||
def get_for_std(d, std):
|
||
# This catches the C++11 case for which there should be no defined feature
|
||
# test macros.
|
||
std_dialects = get_std_dialects()
|
||
if std not in std_dialects:
|
||
return None
|
||
# Find the value for the newest C++ dialect between C++14 and std
|
||
std_list = list(std_dialects[0 : std_dialects.index(std) + 1])
|
||
std_list.reverse()
|
||
for s in std_list:
|
||
if s in d.keys():
|
||
return d[s]
|
||
return None
|
||
|
||
|
||
def get_std_number(std):
|
||
return std.replace("c++", "")
|
||
|
||
|
||
"""
|
||
Functions to produce the <version> header
|
||
"""
|
||
|
||
|
||
def produce_macros_definition_for_std(std):
|
||
result = ""
|
||
indent = 55
|
||
for tc in feature_test_macros:
|
||
if std not in tc["values"]:
|
||
continue
|
||
inner_indent = 1
|
||
if "test_suite_guard" in tc.keys():
|
||
result += "# if %s\n" % tc["libcxx_guard"]
|
||
inner_indent += 2
|
||
if get_value_before(tc["values"], std) is not None:
|
||
assert "test_suite_guard" not in tc.keys()
|
||
result += "# undef %s\n" % tc["name"]
|
||
line = "#%sdefine %s" % ((" " * inner_indent), tc["name"])
|
||
line += " " * (indent - len(line))
|
||
line += " %sL" % tc["values"][std]
|
||
if "unimplemented" in tc.keys():
|
||
line = "// " + line
|
||
result += line
|
||
result += "\n"
|
||
if "test_suite_guard" in tc.keys():
|
||
result += "# endif\n"
|
||
return result.strip()
|
||
|
||
|
||
def produce_macros_definitions():
|
||
macro_definition_template = """#if _LIBCPP_STD_VER >= {std_number}
|
||
{macro_definition}
|
||
#endif"""
|
||
|
||
macros_definitions = []
|
||
for std in get_std_dialects():
|
||
macros_definitions.append(
|
||
macro_definition_template.format(
|
||
std_number=get_std_number(std),
|
||
macro_definition=produce_macros_definition_for_std(std),
|
||
)
|
||
)
|
||
|
||
return "\n\n".join(macros_definitions)
|
||
|
||
|
||
def chunks(l, n):
|
||
"""Yield successive n-sized chunks from l."""
|
||
for i in range(0, len(l), n):
|
||
yield l[i : i + n]
|
||
|
||
|
||
def produce_version_synopsis():
|
||
indent = 56
|
||
header_indent = 56 + len("20XXYYL ")
|
||
result = ""
|
||
|
||
def indent_to(s, val):
|
||
if len(s) >= val:
|
||
return s
|
||
s += " " * (val - len(s))
|
||
return s
|
||
|
||
line = indent_to("Macro name", indent) + "Value"
|
||
line = indent_to(line, header_indent) + "Headers"
|
||
result += line + "\n"
|
||
for tc in feature_test_macros:
|
||
prev_defined_std = get_last_std(tc["values"])
|
||
line = "{name: <{indent}}{value}L ".format(
|
||
name=tc["name"], indent=indent, value=tc["values"][prev_defined_std]
|
||
)
|
||
headers = list(tc["headers"])
|
||
headers.remove("version")
|
||
for chunk in chunks(headers, 3):
|
||
line = indent_to(line, header_indent)
|
||
chunk = ["<%s>" % header for header in chunk]
|
||
line += " ".join(chunk)
|
||
result += line
|
||
result += "\n"
|
||
line = ""
|
||
while True:
|
||
prev_defined_std = get_std_before(tc["values"], prev_defined_std)
|
||
if prev_defined_std is None:
|
||
break
|
||
result += "%s%sL // %s\n" % (
|
||
indent_to("", indent),
|
||
tc["values"][prev_defined_std],
|
||
prev_defined_std.replace("c++", "C++"),
|
||
)
|
||
return result
|
||
|
||
|
||
def produce_version_header():
|
||
template = """// -*- C++ -*-
|
||
//===----------------------------------------------------------------------===//
|
||
//
|
||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
// See https://llvm.org/LICENSE.txt for license information.
|
||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
|
||
#ifndef _LIBCPP_VERSIONH
|
||
#define _LIBCPP_VERSIONH
|
||
|
||
/*
|
||
version synopsis
|
||
|
||
{synopsis}
|
||
|
||
*/
|
||
|
||
#if __cplusplus < 201103L && defined(_LIBCPP_USE_FROZEN_CXX03_HEADERS)
|
||
# include <__cxx03/version>
|
||
#else
|
||
# include <__config>
|
||
|
||
# if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
||
# pragma GCC system_header
|
||
# endif
|
||
|
||
// clang-format off
|
||
|
||
{cxx_macros}
|
||
|
||
#endif // __cplusplus < 201103L && defined(_LIBCPP_USE_FROZEN_CXX03_HEADERS)
|
||
|
||
// clang-format on
|
||
|
||
#endif // _LIBCPP_VERSIONH
|
||
"""
|
||
|
||
version_str = template.format(
|
||
synopsis=produce_version_synopsis().strip(),
|
||
cxx_macros=produce_macros_definitions(),
|
||
)
|
||
version_header_path = os.path.join(include_path, "version")
|
||
with open(version_header_path, "w", newline="\n") as f:
|
||
f.write(version_str)
|
||
|
||
|
||
"""
|
||
Functions to produce test files
|
||
"""
|
||
|
||
test_types = {
|
||
"undefined": """
|
||
# ifdef {name}
|
||
# error "{name} should not be defined before {std_first}"
|
||
# endif
|
||
""",
|
||
"test_suite_guard": """
|
||
# if {test_suite_guard}
|
||
# ifndef {name}
|
||
# error "{name} should be defined in {std}"
|
||
# endif
|
||
# if {name} != {value}
|
||
# error "{name} should have the value {value} in {std}"
|
||
# endif
|
||
# else
|
||
# ifdef {name}
|
||
# error "{name} should not be defined when the requirement '{test_suite_guard}' is not met!"
|
||
# endif
|
||
# endif
|
||
""",
|
||
"unimplemented": """
|
||
# if !defined(_LIBCPP_VERSION)
|
||
# ifndef {name}
|
||
# error "{name} should be defined in {std}"
|
||
# endif
|
||
# if {name} != {value}
|
||
# error "{name} should have the value {value} in {std}"
|
||
# endif
|
||
# else // _LIBCPP_VERSION
|
||
# ifdef {name}
|
||
# error "{name} should not be defined because it is unimplemented in libc++!"
|
||
# endif
|
||
# endif
|
||
""",
|
||
"defined": """
|
||
# ifndef {name}
|
||
# error "{name} should be defined in {std}"
|
||
# endif
|
||
# if {name} != {value}
|
||
# error "{name} should have the value {value} in {std}"
|
||
# endif
|
||
""",
|
||
}
|
||
|
||
|
||
def generate_std_test(test_list, std):
|
||
result = ""
|
||
for tc in test_list:
|
||
val = get_for_std(tc["values"], std)
|
||
if val is not None:
|
||
val = "%sL" % val
|
||
if val is None:
|
||
result += test_types["undefined"].format(
|
||
name=tc["name"], std_first=get_first_std(tc["values"])
|
||
)
|
||
elif "unimplemented" in tc.keys():
|
||
result += test_types["unimplemented"].format(
|
||
name=tc["name"], value=val, std=std
|
||
)
|
||
elif "test_suite_guard" in tc.keys():
|
||
result += test_types["test_suite_guard"].format(
|
||
name=tc["name"],
|
||
value=val,
|
||
std=std,
|
||
test_suite_guard=tc["test_suite_guard"],
|
||
)
|
||
else:
|
||
result += test_types["defined"].format(name=tc["name"], value=val, std=std)
|
||
return result.strip()
|
||
|
||
|
||
def generate_std_tests(test_list):
|
||
std_tests_template = """#if TEST_STD_VER < {first_std_number}
|
||
|
||
{pre_std_test}
|
||
|
||
{other_std_tests}
|
||
|
||
#elif TEST_STD_VER > {penultimate_std_number}
|
||
|
||
{last_std_test}
|
||
|
||
#endif // TEST_STD_VER > {penultimate_std_number}"""
|
||
|
||
std_dialects = get_std_dialects()
|
||
|
||
other_std_tests = []
|
||
for std in std_dialects[:-1]:
|
||
other_std_tests.append("#elif TEST_STD_VER == " + get_std_number(std))
|
||
other_std_tests.append(generate_std_test(test_list, std))
|
||
|
||
std_tests = std_tests_template.format(
|
||
first_std_number=get_std_number(std_dialects[0]),
|
||
pre_std_test=generate_std_test(test_list, "c++11"),
|
||
other_std_tests="\n\n".join(other_std_tests),
|
||
penultimate_std_number=get_std_number(std_dialects[-2]),
|
||
last_std_test=generate_std_test(test_list, std_dialects[-1]),
|
||
)
|
||
|
||
return std_tests
|
||
|
||
|
||
def generate_synopsis(test_list):
|
||
max_name_len = max([len(tc["name"]) for tc in test_list])
|
||
indent = max_name_len + 8
|
||
|
||
def mk_line(prefix, suffix):
|
||
return "{prefix: <{max_len}}{suffix}\n".format(
|
||
prefix=prefix, suffix=suffix, max_len=indent
|
||
)
|
||
|
||
result = ""
|
||
result += mk_line("/* Constant", "Value")
|
||
for tc in test_list:
|
||
prefix = " %s" % tc["name"]
|
||
for std in [s for s in get_std_dialects() if s in tc["values"].keys()]:
|
||
result += mk_line(
|
||
prefix, "%sL [%s]" % (tc["values"][std], std.replace("c++", "C++"))
|
||
)
|
||
prefix = ""
|
||
result += "*/"
|
||
return result
|
||
|
||
|
||
def produce_tests():
|
||
headers = set([h for tc in feature_test_macros for h in tc["headers"]])
|
||
for h in headers:
|
||
test_list = [tc for tc in feature_test_macros if h in tc["headers"]]
|
||
if not has_header(h):
|
||
for tc in test_list:
|
||
assert "unimplemented" in tc.keys()
|
||
continue
|
||
markup = "\n".join("// " + tag for tag in lit_markup.get(h, []))
|
||
test_body = """//===----------------------------------------------------------------------===//
|
||
//
|
||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
// See https://llvm.org/LICENSE.txt for license information.
|
||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
//
|
||
// WARNING: This test was generated by {script_name}
|
||
// and should not be edited manually.
|
||
//
|
||
// clang-format off
|
||
{markup}
|
||
// <{header}>
|
||
|
||
// Test the feature test macros defined by <{header}>
|
||
|
||
{synopsis}
|
||
|
||
#include <{header}>
|
||
#include "test_macros.h"
|
||
|
||
{cxx_tests}
|
||
|
||
""".format(
|
||
script_name=script_name,
|
||
header=h,
|
||
markup=("\n{}\n".format(markup) if markup else ""),
|
||
synopsis=generate_synopsis(test_list),
|
||
cxx_tests=generate_std_tests(test_list),
|
||
)
|
||
test_name = "{header}.version.compile.pass.cpp".format(header=h)
|
||
out_path = os.path.join(macro_test_path, test_name)
|
||
with open(out_path, "w", newline="\n") as f:
|
||
f.write(test_body)
|
||
|
||
|
||
"""
|
||
Produce documentation for the feature test macros
|
||
"""
|
||
|
||
|
||
def make_widths(grid):
|
||
widths = []
|
||
for i in range(0, len(grid[0])):
|
||
cell_width = 2 + max(
|
||
reduce(lambda x, y: x + y, [[len(row[i])] for row in grid], [])
|
||
)
|
||
widths += [cell_width]
|
||
return widths
|
||
|
||
|
||
def create_table(grid, indent):
|
||
indent_str = " " * indent
|
||
col_widths = make_widths(grid)
|
||
result = [indent_str + add_divider(col_widths, 2)]
|
||
header_flag = 2
|
||
for row_i in range(0, len(grid)):
|
||
row = grid[row_i]
|
||
line = indent_str + " ".join(
|
||
[pad_cell(row[i], col_widths[i]) for i in range(0, len(row))]
|
||
)
|
||
result.append(line.rstrip())
|
||
if row_i == len(grid) - 1:
|
||
header_flag = 2
|
||
if row[0].startswith("**"):
|
||
header_flag += 1
|
||
separator = indent_str + add_divider(col_widths, header_flag)
|
||
result.append(separator.rstrip())
|
||
header_flag = 0
|
||
return "\n".join(result)
|
||
|
||
|
||
def add_divider(widths, header_flag):
|
||
if header_flag == 3:
|
||
return "=".join(["=" * w for w in widths])
|
||
if header_flag == 2:
|
||
return " ".join(["=" * w for w in widths])
|
||
if header_flag == 1:
|
||
return "-".join(["-" * w for w in widths])
|
||
else:
|
||
return " ".join(["-" * w for w in widths])
|
||
|
||
|
||
def pad_cell(s, length, left_align=True):
|
||
padding = (length - len(s)) * " "
|
||
return s + padding
|
||
|
||
|
||
def get_status_table():
|
||
table = [["Macro Name", "Value"]]
|
||
for std in get_std_dialects():
|
||
table += [["**" + std.replace("c++", "C++") + "**", ""]]
|
||
for tc in feature_test_macros:
|
||
if std not in tc["values"].keys():
|
||
continue
|
||
value = "``%sL``" % tc["values"][std]
|
||
if "unimplemented" in tc.keys():
|
||
value = "*unimplemented*"
|
||
table += [["``%s``" % tc["name"], value]]
|
||
return table
|
||
|
||
|
||
def produce_docs():
|
||
doc_str = """.. _FeatureTestMacroTable:
|
||
|
||
==========================
|
||
Feature Test Macro Support
|
||
==========================
|
||
|
||
.. contents::
|
||
:local:
|
||
|
||
Overview
|
||
========
|
||
|
||
This file documents the feature test macros currently supported by libc++.
|
||
|
||
.. _feature-status:
|
||
|
||
Status
|
||
======
|
||
|
||
.. table:: Current Status
|
||
:name: feature-status-table
|
||
:widths: auto
|
||
|
||
{status_tables}
|
||
|
||
""".format(
|
||
status_tables=create_table(get_status_table(), 4)
|
||
)
|
||
|
||
table_doc_path = os.path.join(docs_path, "FeatureTestMacroTable.rst")
|
||
with open(table_doc_path, "w", newline="\n") as f:
|
||
f.write(doc_str)
|
||
|
||
|
||
def get_ftms(
|
||
data, std_dialects: List[str], use_implemented_status: bool
|
||
) -> Dict[str, Dict[str, Any]]:
|
||
"""Impementation for FeatureTestMacros.(standard|implemented)_ftms()."""
|
||
result = dict()
|
||
for feature in data:
|
||
last = None
|
||
entry = dict()
|
||
implemented = True
|
||
for std in std_dialects:
|
||
if std not in feature["values"].keys():
|
||
if last == None:
|
||
continue
|
||
else:
|
||
entry[std] = last
|
||
else:
|
||
if implemented:
|
||
values = feature["values"][std]
|
||
assert len(values) > 0, f"{feature['name']}[{std}] has no entries"
|
||
for value in values:
|
||
papers = list(values[value])
|
||
assert (
|
||
len(papers) > 0
|
||
), f"{feature['name']}[{std}][{value}] has no entries"
|
||
for paper in papers:
|
||
if use_implemented_status and not paper["implemented"]:
|
||
implemented = False
|
||
break
|
||
if implemented:
|
||
last = f"{value}L"
|
||
else:
|
||
break
|
||
|
||
entry[std] = last
|
||
result[feature["name"]] = entry
|
||
|
||
return result
|
||
|
||
|
||
def generate_version_header_dialect_block(data: Dict[str, Any]) -> str:
|
||
"""Generates the contents of the version header for a dialect.
|
||
|
||
This generates the contents of a
|
||
#if _LIBCPP_STD_VER >= XY
|
||
#endif // _LIBCPP_STD_VER >= XY
|
||
block.
|
||
"""
|
||
result = ""
|
||
for element in data:
|
||
for ftm, entry in element.items():
|
||
if not entry["implemented"]:
|
||
# When a FTM is not implemented don't add the guards
|
||
# or undefine the (possibly) defined macro.
|
||
result += f'// define {ftm} {entry["value"]}\n'
|
||
else:
|
||
need_undef = entry["need_undef"]
|
||
if entry["condition"]:
|
||
result += f'# if {entry["condition"]}\n'
|
||
if entry["need_undef"]:
|
||
result += f"# undef {ftm}\n"
|
||
result += f'# define {ftm} {entry["value"]}\n'
|
||
result += f"# endif\n"
|
||
else:
|
||
if entry["need_undef"]:
|
||
result += f"# undef {ftm}\n"
|
||
result += f'# define {ftm} {entry["value"]}\n'
|
||
|
||
return result
|
||
|
||
|
||
def generate_version_header_implementation(data: Dict[str, Dict[str, Any]]) -> str:
|
||
"""Generates the body of the version header."""
|
||
|
||
template = """#if _LIBCPP_STD_VER >= {dialect}
|
||
{feature_test_macros}#endif // _LIBCPP_STD_VER >= {dialect}"""
|
||
|
||
result = []
|
||
for std, ftms in data.items():
|
||
result.append(
|
||
template.format(
|
||
dialect=std,
|
||
feature_test_macros=generate_version_header_dialect_block(ftms),
|
||
)
|
||
)
|
||
|
||
return "\n\n".join(result)
|
||
|
||
|
||
class FeatureTestMacros:
|
||
"""Provides all feature-test macro (FTM) output components.
|
||
|
||
The class has several generators to use the feature-test macros in libc++:
|
||
- FTM status page
|
||
- The version header and its tests
|
||
|
||
This class is not intended to duplicate
|
||
https://isocpp.org/std/standing-documents/sd-6-sg10-feature-test-recommendations#library-feature-test-macros
|
||
SD-FeatureTest: Feature-Test Macros and Policies
|
||
|
||
Historically libc++ did not list all papers affecting a FTM, the new data
|
||
structure is able to do that. However there is no intention to add the
|
||
historical data. After papers have been implemented this information can be
|
||
removed. For example, __cpp_lib_format's value 201907 requires 3 papers,
|
||
once implemented it can be reduced to 1 paper and remove the paper number
|
||
and title. This would reduce the size of the data.
|
||
|
||
The input data is stored in the following JSON format:
|
||
[ # A list with multiple feature-test macro entries.
|
||
{
|
||
# required
|
||
# The name of the feature test macro. These names should be unique and
|
||
# sorted in the list.
|
||
"name": "__cpp_lib_any",
|
||
|
||
# required
|
||
# A map with the value of the FTM based on the language standard. Only
|
||
# the versions in which the value of the FTM changes are listed. For
|
||
# example, this macro's value does not change in C++20 so it does not
|
||
# list C++20. If it changes in C++26, it will have entries for C++17 and
|
||
# C++26.
|
||
"values": {
|
||
|
||
# required
|
||
# The language standard, also named dialect in this class.
|
||
"c++17": {
|
||
|
||
# required
|
||
# The value of the feature test macro. This contains an array with
|
||
# one or more papers that need to be implemented before this value
|
||
# is considered implemented.
|
||
"201606": [
|
||
{
|
||
# optional
|
||
# Contains the paper number that is part of the FTM version.
|
||
"number": "P0220R1",
|
||
|
||
# optional
|
||
# Contains the title of the paper that is part of the FTM
|
||
# version.
|
||
"title": "Adopt Library Fundamentals V1 TS Components for C++17"
|
||
|
||
# required
|
||
# The implementation status of the paper.
|
||
"implemented": true
|
||
}
|
||
]
|
||
}
|
||
},
|
||
|
||
# required
|
||
# A sorted list of headers that should provide the FTM. The header
|
||
# <version> is automatically added to this list. This list could be
|
||
# empty. For example, __cpp_lib_modules is only present in version.
|
||
# Requiring the field makes it easier to detect accidental omission.
|
||
"headers": [
|
||
"any"
|
||
],
|
||
|
||
# optional, required when libcxx_guard is present
|
||
# This field is used only to generate the unit tests for the
|
||
# feature-test macros. It can't depend on macros defined in <__config>
|
||
# because the `test/std/` parts of the test suite are intended to be
|
||
# portable to any C++ standard library implementation, not just libc++.
|
||
# It may depend on
|
||
# * macros defined by the compiler itself, or
|
||
# * macros generated by CMake.
|
||
# In some cases we add also depend on macros defined in
|
||
# <__availability>.
|
||
"test_suite_guard": "!defined(_LIBCPP_VERSION) || _LIBCPP_AVAILABILITY_HAS_PMR"
|
||
|
||
# optional, required when test_suite_guard is present
|
||
# This field is used only to guard the feature-test macro in
|
||
# <version>. It may be the same as `test_suite_guard`, or it may
|
||
# depend on macros defined in <__config>.
|
||
"libcxx_guard": "_LIBCPP_AVAILABILITY_HAS_PMR"
|
||
},
|
||
]
|
||
"""
|
||
|
||
# The JSON data structure.
|
||
__data = None
|
||
|
||
def __init__(self, filename: str):
|
||
"""Initializes the class with the JSON data in the file 'filename'."""
|
||
with open(filename) as f:
|
||
self.__data = json.load(f)
|
||
|
||
@functools.cached_property
|
||
def std_dialects(self) -> List[str]:
|
||
"""Returns the C++ dialects avaiable.
|
||
|
||
The available dialects are based on the 'c++xy' keys found the 'values'
|
||
entries in '__data'. So when WG21 starts to feature-test macros for a
|
||
future C++ Standard this dialect will automatically be available.
|
||
|
||
The return value is a sorted list with the C++ dialects used. Since FTM
|
||
were added in C++14 the list will not contain C++03 or C++11.
|
||
"""
|
||
dialects = set()
|
||
for feature in self.__data:
|
||
keys = feature["values"].keys()
|
||
assert len(keys) > 0, "'values' is empty"
|
||
dialects |= keys
|
||
|
||
return sorted(list(dialects))
|
||
|
||
@functools.cached_property
|
||
def standard_ftms(self) -> Dict[str, Dict[str, Any]]:
|
||
"""Returns the FTM versions per dialect in the Standard.
|
||
|
||
This function does not use the 'implemented' flag. The output contains
|
||
the versions used in the Standard. When a FTM in libc++ is not
|
||
implemented according to the Standard to output may opt to show the
|
||
expected value.
|
||
|
||
The result is a dict with the following content
|
||
- key: Name of the feature test macro.
|
||
- value: A dict with the following content:
|
||
* key: The version of the C++ dialect.
|
||
* value: The value of the feature-test macro.
|
||
"""
|
||
return get_ftms(self.__data, self.std_dialects, False)
|
||
|
||
@functools.cached_property
|
||
def implemented_ftms(self) -> Dict[str, Dict[str, Any]]:
|
||
"""Returns the FTM versions per dialect implemented in libc++.
|
||
|
||
Unlike `get_std_dialect_versions` this function uses the 'implemented'
|
||
flag. This returns the actual implementation status in libc++.
|
||
|
||
The result is a dict with the following content
|
||
- key: Name of the feature test macro.
|
||
- value: A dict with the following content:
|
||
* key: The version of the C++ dialect.
|
||
* value: The value of the feature-test macro. When a feature-test
|
||
macro is not implemented its value is None.
|
||
"""
|
||
|
||
return get_ftms(self.__data, self.std_dialects, True)
|
||
|
||
@functools.cached_property
|
||
def ftm_metadata(self) -> Dict[str, Dict[str, Any]]:
|
||
"""Returns the metadata of the FTMs defined in the Standard.
|
||
|
||
The metadata does not depend on the C++ dialect used.
|
||
The result is a dict with the following contents:
|
||
- key: Name of the feature test macro.
|
||
- value: A dict with the following content:
|
||
* headers: The list of headers that should provide the FTM
|
||
* test_suite_guard: The condition for testing the FTM in the test suite.
|
||
* test_suite_guard: The condition for testing the FTM in the version header.
|
||
"""
|
||
result = dict()
|
||
for feature in self.__data:
|
||
entry = dict()
|
||
entry["headers"] = feature["headers"]
|
||
entry["test_suite_guard"] = feature.get("test_suite_guard", None)
|
||
entry["libcxx_guard"] = feature.get("libcxx_guard", None)
|
||
result[feature["name"]] = entry
|
||
|
||
return result
|
||
|
||
@property
|
||
def version_header_implementation(self) -> Dict[str, List[Dict[str, Any]]]:
|
||
"""Generates the body of the version header."""
|
||
result = dict()
|
||
for std in self.std_dialects:
|
||
result[get_std_number(std)] = list()
|
||
|
||
for ftm, values in self.standard_ftms.items():
|
||
last_value = None
|
||
last_entry = None
|
||
for std, value in values.items():
|
||
# When a newer Standard does not change the value of the macro
|
||
# there is no need to redefine it with the same value.
|
||
if last_value and value == last_value:
|
||
continue
|
||
last_value = value
|
||
|
||
entry = dict()
|
||
entry["value"] = value
|
||
entry["implemented"] = self.implemented_ftms[ftm][std] == self.standard_ftms[ftm][std]
|
||
entry["need_undef"] = last_entry is not None and last_entry["implemented"] and entry["implemented"]
|
||
entry["condition"] = self.ftm_metadata[ftm]["libcxx_guard"]
|
||
|
||
last_entry = entry
|
||
result[get_std_number(std)].append(dict({ftm: entry}))
|
||
|
||
return result
|
||
|
||
@property
|
||
def version_header(self) -> str:
|
||
"""Generates the version header."""
|
||
template = """// -*- C++ -*-
|
||
//===----------------------------------------------------------------------===//
|
||
//
|
||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||
// See https://llvm.org/LICENSE.txt for license information.
|
||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||
//
|
||
//===----------------------------------------------------------------------===//
|
||
|
||
#ifndef _LIBCPP_VERSIONH
|
||
#define _LIBCPP_VERSIONH
|
||
|
||
#include <__config>
|
||
|
||
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
||
# pragma GCC system_header
|
||
#endif
|
||
|
||
{feature_test_macros}
|
||
|
||
#endif // _LIBCPP_VERSIONH
|
||
"""
|
||
return template.format(
|
||
feature_test_macros=generate_version_header_implementation(
|
||
self.version_header_implementation
|
||
)
|
||
)
|
||
|
||
|
||
def main():
|
||
produce_version_header()
|
||
produce_tests()
|
||
produce_docs()
|
||
|
||
# Example how to use the new version header generation function to generate
|
||
# the file.
|
||
if False:
|
||
ftm = FeatureTestMacros(
|
||
os.path.join(
|
||
source_root, "test", "libcxx", "feature_test_macro", "test_data.json"
|
||
)
|
||
)
|
||
version_header_path = os.path.join(include_path, "version")
|
||
with open(version_header_path, "w", newline="\n") as f:
|
||
f.write(ftm.version_header)
|
||
|
||
|
||
if __name__ == "__main__":
|
||
main()
|