libstdc++: Implement more maintainable <version> header

This commit replaces the ad-hoc logic in <version> with an AutoGen
database that (mostly) declaratively generates a version.h bit which
combines all of the FTM logic across all headers together.

This generated header defines macros of the form __glibcxx_foo,
equivalent to their __cpp_lib_foo variants, according to rules specified
in version.def and, optionally, if __glibcxx_want_foo or
__glibcxx_want_all are defined, also defines __cpp_lib_foo forms with
the same definition.

libstdc++-v3/ChangeLog:

	* include/Makefile.am (bits_freestanding): Add version.h.
	(allcreated): Add version.h.
	(${bits_srcdir}/version.h): New rule.  Regenerates
	version.h out of version.{def,tpl}.
	* include/Makefile.in: Regenerate.
	* include/bits/version.def: New file.  Declares a list of
	all feature test macros, their values and their preconditions.
	* include/bits/version.tpl: New file.  Turns version.def
	into a sequence of #if blocks.
	* include/bits/version.h: New file.  Generated from
	version.def.
	* include/std/version: Replace with a __glibcxx_want_all define
	and bits/version.h include.
This commit is contained in:
Arsen Arsenović 2023-03-27 15:29:08 +02:00 committed by Arsen Arsenović
parent ff572e8c9d
commit b7b801b2f5
No known key found for this signature in database
GPG key ID: 52C294301EA2C493
6 changed files with 3770 additions and 349 deletions

View file

@ -154,6 +154,7 @@ bits_freestanding = \
${bits_srcdir}/stl_raw_storage_iter.h \
${bits_srcdir}/stl_relops.h \
${bits_srcdir}/stl_uninitialized.h \
${bits_srcdir}/version.h \
${bits_srcdir}/string_view.tcc \
${bits_srcdir}/uniform_int_dist.h \
${bits_srcdir}/unique_ptr.h \
@ -1113,7 +1114,8 @@ allcreated = \
${host_builddir}/c++config.h \
${host_builddir}/largefile-config.h \
${thread_host_headers} \
${pch_build}
${pch_build} \
${bits_srcdir}/version.h
# Here are the rules for building the headers
all-local: ${allstamped} ${allcreated}
@ -1463,6 +1465,12 @@ ${pch3_output}: ${pch3_source} ${pch2_output}
-mkdir -p ${pch3_output_builddir}
$(CXX) $(PCHFLAGS) $(AM_CPPFLAGS) -O2 -g ${pch3_source} -o $@
# AutoGen <bits/version.h>.
.PHONY: update-version
update-version:
cd ${bits_srcdir} && \
autogen version.def
# The real deal.
install-data-local: install-headers
install-headers:

View file

@ -509,6 +509,7 @@ bits_freestanding = \
${bits_srcdir}/stl_raw_storage_iter.h \
${bits_srcdir}/stl_relops.h \
${bits_srcdir}/stl_uninitialized.h \
${bits_srcdir}/version.h \
${bits_srcdir}/string_view.tcc \
${bits_srcdir}/uniform_int_dist.h \
${bits_srcdir}/unique_ptr.h \
@ -1441,7 +1442,8 @@ allcreated = \
${host_builddir}/c++config.h \
${host_builddir}/largefile-config.h \
${thread_host_headers} \
${pch_build}
${pch_build} \
${bits_srcdir}/version.h
# Host includes for threads
@ -1937,6 +1939,12 @@ ${pch3_output}: ${pch3_source} ${pch2_output}
-mkdir -p ${pch3_output_builddir}
$(CXX) $(PCHFLAGS) $(AM_CPPFLAGS) -O2 -g ${pch3_source} -o $@
# AutoGen <bits/version.h>.
${bits_srcdir}/version.h: ${bits_srcdir}/version.def \
${bits_srcdir}/version.tpl
cd $(@D) && \
autogen version.def
# The real deal.
install-data-local: install-headers
install-headers:

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,210 @@
/*{ AutoGen5 template -*- C++ -*-
h
(use-modules (srfi srfi-1))
}*/
// Copyright (C) 2023 Free Software Foundation, Inc.
// This file is part of the GNU ISO C++ Library. This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// Under Section 7 of GPL version 3, you are granted additional
// permissions described in the GCC Runtime Library Exception, version
// 3.1, as published by the Free Software Foundation.
// You should have received a copy of the GNU General Public License and
// a copy of the GCC Runtime Library Exception along with this program;
// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
// <http://www.gnu.org/licenses/>.
/*{ (dne "// ") }*/
/** @file bits/version.h
* This is an internal header file, included by other library headers.
* Do not attempt to use it directly. @headername{version}
*/
// Usage guide:
//
// In your usual header, do something like:
//
// #define __glibcxx_want_ranges
// #define __glibcxx_want_concepts
// #include <bits/version.h>
//
// This will generate the FTMs you named, and let you use them in your code as
// if it was user code. All macros are also exposed under __glibcxx_NAME even
// if unwanted, to permit bits and other FTMs to depend on them for condtional
// computation without exposing extra FTMs to user code.
#pragma GCC system_header
#include <bits/c++config.h>
/*{
;; Helper for dealing with autogens redefined (error)
(define (ferror msg . args)
(error (apply format (cons* #f msg args))))
;; Helper function that, in the context of a single FTM value, generates the
;; condition expression that fulfills the specification of this value. See the
;; comment block in version.def for an explanation of the format this
;; function parses.
(define (generate-cond)
(define c++min #f)
(define gnu++min #f)
(define gthread #f)
(define hosted #f)
(define c++11-abi #f)
(define extra-cond (get "extra_cond"))
(define conds '())
(define (prepend x)
(if x (set! conds (cons x conds))))
(if (string-null? extra-cond)
(set! extra-cond #f))
(let ((cxxmin (get "cxxmin"))
(gnuxxmin (get "gnuxxmin"))
(getstdcond
(lambda (var)
(let ((ver (get var)))
(if (not (string-null? ver))
(let ((std-cond (get (format #f "stds[~a]" ver))))
(if (string-null? std-cond)
(ferror "Standard ~s invalid." ver))
std-cond)
#f)))))
(set! c++min (getstdcond "cxxmin"))
(set! gnu++min (getstdcond "gnuxxmin")))
(let ((process-tristate
(lambda (name)
(let ((val (get name)))
(cond
((equal? val "") #f) ; Don't test
((equal? val "yes") "") ; Test directly
((equal? val "no") "!") ; Invert
(else (ferror "Bad ~a value ~a." name val)))))))
(set! gthread (process-tristate "gthread"))
(set! c++11-abi (process-tristate "cxx11abi"))
(set! hosted (process-tristate "hosted")))
(prepend (if extra-cond (format #f "(~a)" extra-cond) #f))
(prepend (if hosted (format #f "~a~a" hosted "_GLIBCXX_HOSTED") #f))
(prepend (if gthread (format #f "~a~a" gthread "defined(_GLIBCXX_HAS_GTHREADS)") #f))
(prepend (if c++11-abi (format #f "~a~a" c++11-abi "_GLIBCXX_USE_CXX11_ABI") #f))
(prepend
(let ((strict "defined(__STRICT_ANSI__)")
(c++ "__cplusplus"))
(cond
((or (and (equal? c++min gnu++min) c++min)
(and (not gnu++min) c++min))
;; If we have both values, and they are equal, or we only have gnu++min,
;; we want to output a simple check.
(format #f "(~a ~a)" c++ c++min))
((and gnu++min c++min)
;; We have differing cases for strict and non-strict modes.
(format #f "((~a && ~a ~a) || (!~a && ~a ~a))"
strict c++ c++min
strict c++ gnu++min))
((and gnu++min (not c++min))
(format #f "(!~a && (~a ~a))" strict c++ gnu++min))
(else #f))))
(string-join conds " && " 'strict-infix))
}*/
/*{ FOR ftms
}*/// /*{ (def-file-line "name") }*/
#if !defined(__cpp_lib_/*{name}*/)
/*{ FOR values }*//*{ #
This macro block defines two versions of each FTM:
1. __glibcxx_NAME, which is defined unconditionally, and
2. __cpp_lib_NAME, which is defined only if marked as wanted.
This allows FTMs to depend on eachother in their definitions without messing
with the exported values.
This can also be used by bits that do not want to expose FTMs that they can't
implement.
}*/# /*{(unless (first-for?) "el")}*/if /*{(generate-cond)}*/
# define __glibcxx_/*{name}*/ /*{v}*/L
# if defined(__glibcxx_want_all) || defined(__glibcxx_want_/*{name}*/)
# define /*{
;; Compute the name for this FTM based on stdname/name.
(if (exist? "stdname")
(get "stdname")
(format #f "__cpp_lib_~a" (get "name")))
}*/ /*{v}*/L
# endif
/*{ ENDFOR values
}*/# endif
#endif /* !defined(__cpp_lib_/*{name}*/) && defined(__glibcxx_want_/*{name}*/) */
#undef __glibcxx_want_/*{name
}*//*{ (unless (last-for?) "\n\n" "\n")}*/
/*{ ENDFOR ftms }*//*{
;; Helper that generates [LO, HI].
(define (closed-int-set lo hi)
(iota (+ (- hi lo) 1) lo))
;; Sanity checking for duplicates and for value order.
(let ((ht (make-hash-table (count "ftms"))))
(for-each
(lambda (idx)
(let ((name (get (format #f "ftms[~a].name" idx))))
(if (string-null? name) (ferror "No name for FTM ~a" idx))
(let ((cur (cdr (or (hash-get-handle ht name) '(1 . 0)))))
(hash-set! ht name (+ cur 1)))))
(closed-int-set (low-lim "ftms") (high-lim "ftms")))
(if (hash-fold
(lambda (name count prior)
(if (= 1 count)
prior
(begin
(printf "FTM %s appears %d times.\n" name count)
#t)))
#f ht)
(error "Duplicates found.")))
(define (check-value-order ftm key order)
(let ((valmin 999999)) ; TODO(arsen): bump before year 10000
(for-each
(lambda (vidx)
(let* ((sval (get (format #f "values[~a].~a" vidx key)))
(val (string->number sval)))
(unless (string-null? sval)
(unless val (ferror "Bad value in FTM ~a" ftm))
(if (order val valmin)
(ferror "Found inverted ~s value in FTM ~a: ~a" key ftm sval))
(set! valmin val))))
(closed-int-set (low-lim "values") (high-lim "values")))))
}*//*{ FOR ftms }*//*{#
Check for values that are in ascending order. Because it is generally the
case that FTMs increase as the values of tests they probe for do, we check
them to prevent simple, silly errors.
We're iterating in a separate FOR block rather than in pure Guile since, for
some reason, high-lim and low-lim do not work with complex names that include
periods and indices (whereas exist? and others do).
}*//*{
(let ((ftm (get "name")))
(check-value-order (get "name") "v" >)
(check-value-order (get "name") "cxxmin" >)
(check-value-order (get "name") "gnuxxmin" >))
}*//*{ ENDFOR ftms }*/
#undef __glibcxx_want_all

View file

@ -30,353 +30,9 @@
#ifndef _GLIBCXX_VERSION_INCLUDED
#define _GLIBCXX_VERSION_INCLUDED
// To facilitate testsuite/17_intro/versionconflict.cc
#ifndef _GLIBCXX_TESTING_SYSHDR
# pragma GCC system_header
#endif
#pragma GCC system_header
#include <bits/c++config.h>
// c++03
#if _GLIBCXX_HOSTED
# define __cpp_lib_incomplete_container_elements 201505L
#endif
#if !defined(__STRICT_ANSI__)
// gnu++03
# define __cpp_lib_uncaught_exceptions 201411L
#endif
#if __cplusplus >= 201103L
// c++11
#define __cpp_lib_allocator_traits_is_always_equal 201411L
#define __cpp_lib_is_null_pointer 201309L
#define __cpp_lib_result_of_sfinae 201210L
#if _GLIBCXX_HOSTED
#if __cplusplus <= 201703L // N.B. updated value in C++20
# define __cpp_lib_shared_ptr_arrays 201611L
#endif
#endif
#if !defined(__STRICT_ANSI__)
// gnu++11
# define __cpp_lib_is_swappable 201603L
# define __cpp_lib_void_t 201411L
# if _GLIBCXX_HOSTED
# define __cpp_lib_enable_shared_from_this 201603L
# endif
#endif
// For C++11 and later we support ISO/IEC 29124 Mathematical Special Functions
#define __STDCPP_MATH_SPEC_FUNCS__ 201003L
#if __cplusplus >= 201402L
// c++14
#if __cpp_impl_coroutine
# define __cpp_lib_coroutine 201902L
#endif
#define __cpp_lib_exchange_function 201304L
#define __cpp_lib_integer_sequence 201304L
#define __cpp_lib_integral_constant_callable 201304L
#define __cpp_lib_is_final 201402L
#define __cpp_lib_make_reverse_iterator 201402L
#ifndef _GLIBCXX_DEBUG // PR libstdc++/70303
# define __cpp_lib_null_iterators 201304L
#endif
#define __cpp_lib_robust_nonmodifying_seq_ops 201304L
#define __cpp_lib_transformation_trait_aliases 201304L
#define __cpp_lib_transparent_operators 201510L
#define __cpp_lib_tuple_element_t 201402L
#define __cpp_lib_tuples_by_type 201304L
#if _GLIBCXX_FLOAT_IS_IEEE_BINARY32 && _GLIBCXX_DOUBLE_IS_IEEE_BINARY64 \
&& __SIZE_WIDTH__ >= 32
# define __cpp_lib_to_chars 201611L
#endif
#if _GLIBCXX_HOSTED
# define __cpp_lib_chrono_udls 201304L
# define __cpp_lib_complex_udls 201309L
# define __cpp_lib_generic_associative_lookup 201304L
# define __cpp_lib_make_unique 201304L
# define __cpp_lib_quoted_string_io 201304L
# ifdef _GLIBCXX_HAS_GTHREADS
# define __cpp_lib_shared_timed_mutex 201402L
# endif
# define __cpp_lib_string_udls 201304L
#endif
#if __cplusplus >= 201703L
// c++17
#define __cpp_lib_addressof_constexpr 201603L
#define __cpp_lib_any 201606L
#define __cpp_lib_apply 201603L
#define __cpp_lib_as_const 201510L
#define __cpp_lib_atomic_is_always_lock_free 201603L
#define __cpp_lib_bool_constant 201505L
#define __cpp_lib_byte 201603L
#define __cpp_lib_clamp 201603L
#define __cpp_lib_gcd 201606L
#define __cpp_lib_gcd_lcm 201606L
#ifdef _GLIBCXX_HAVE_BUILTIN_HAS_UNIQ_OBJ_REP
# define __cpp_lib_has_unique_object_representations 201606L
#endif
#ifdef __GCC_DESTRUCTIVE_SIZE
# define __cpp_lib_hardware_interference_size 201703L
#endif
#define __cpp_lib_invoke 201411L
#ifdef _GLIBCXX_HAVE_BUILTIN_IS_AGGREGATE
# define __cpp_lib_is_aggregate 201703L
#endif
#define __cpp_lib_is_invocable 201703L
#define __cpp_lib_is_swappable 201603L
#ifdef _GLIBCXX_HAVE_BUILTIN_LAUNDER
# define __cpp_lib_launder 201606L
#endif
#define __cpp_lib_lcm 201606L
#define __cpp_lib_logical_traits 201510L
#define __cpp_lib_make_from_tuple 201606L
#define __cpp_lib_nonmember_container_access 201411L
#define __cpp_lib_not_fn 201603L
#if __cplusplus == 201703L // N.B. updated value in C++20
# define __cpp_lib_optional 201606L
#endif
#define __cpp_lib_raw_memory_algorithms 201606L
#define __cpp_lib_sample 201603L
#define __cpp_lib_type_trait_variable_templates 201510L
#define __cpp_lib_uncaught_exceptions 201411L
#if !(__cpp_concepts >= 202002L && __cpp_constexpr >= 201811L)
// N.B. updated value in C++20
# define __cpp_lib_variant 202102L
#endif
#define __cpp_lib_void_t 201411L
#if __cplusplus == 201703L // N.B. updated value in C++20
# define __cpp_lib_array_constexpr 201803L
#endif
#if __cplusplus == 201703L // N.B. updated value in C++20
# if _GLIBCXX_HAVE_IS_CONSTANT_EVALUATED
# define __cpp_lib_constexpr_char_traits 201611L
# define __cpp_lib_constexpr_string 201611L
# endif
#endif
#if _GLIBCXX_HOSTED
#define __cpp_lib_boyer_moore_searcher 201603L
#define __cpp_lib_chrono 201611L
#define __cpp_lib_enable_shared_from_this 201603L
#define __cpp_lib_execution 201902L // FIXME: should be 201603L
#define __cpp_lib_filesystem 201703L
#define __cpp_lib_hypot 201603L
#define __cpp_lib_map_try_emplace 201411L
#define __cpp_lib_math_special_functions 201603L
#ifdef _GLIBCXX_HAS_GTHREADS
# define __cpp_lib_memory_resource 201603L
#else
# define __cpp_lib_memory_resource 1
#endif
#define __cpp_lib_node_extract 201606L
#define __cpp_lib_parallel_algorithm 201603L
#ifdef _GLIBCXX_HAS_GTHREADS
# define __cpp_lib_scoped_lock 201703L
# define __cpp_lib_shared_mutex 201505L
#endif
#define __cpp_lib_shared_ptr_weak_type 201606L
#define __cpp_lib_string_view 201803L
#define __cpp_lib_unordered_map_try_emplace 201411L
#endif
#if __cplusplus >= 202002L
// c++20
#define __cpp_lib_assume_aligned 201811L
#define __cpp_lib_atomic_flag_test 201907L
#define __cpp_lib_atomic_float 201711L
#define __cpp_lib_atomic_lock_free_type_aliases 201907L
#define __cpp_lib_atomic_ref 201806L
#define __cpp_lib_atomic_value_initialization 201911L
#define __cpp_lib_bind_front 201907L
#define __cpp_lib_starts_ends_with 201711L
#if __has_builtin(__builtin_bit_cast)
# define __cpp_lib_bit_cast 201806L
#endif
#define __cpp_lib_bitops 201907L
#define __cpp_lib_bounded_array_traits 201902L
// __cpp_lib_char8_t is defined in <bits/c++config.h>
#if __cpp_concepts >= 201907L
# define __cpp_lib_concepts 202002L
#endif
#if __cpp_impl_destroying_delete
# define __cpp_lib_destroying_delete 201806L
#endif
#define __cpp_lib_constexpr_string_view 201811L
#define __cpp_lib_endian 201907L
#define __cpp_lib_int_pow2 202002L
#define __cpp_lib_integer_comparison_functions 202002L
#ifdef _GLIBCXX_HAVE_IS_CONSTANT_EVALUATED
# define __cpp_lib_is_constant_evaluated 201811L
#endif
#if __has_builtin(__is_layout_compatible) \
&& __has_builtin(__builtin_is_corresponding_member)
# define __cpp_lib_is_layout_compatible 201907L
#endif
#define __cpp_lib_is_nothrow_convertible 201806L
#if __has_builtin(__is_pointer_interconvertible_base_of) \
&& __has_builtin(__builtin_is_pointer_interconvertible_with_class)
# define __cpp_lib_is_pointer_interconvertible 201907L
#endif
#define __cpp_lib_math_constants 201907L
#if __cpp_lib_concepts
# define __cpp_lib_make_obj_using_allocator 201811L
#endif
#if __cplusplus <= 202002L // N.B. updated value in C++23
# define __cpp_lib_optional 202106L
#endif
#define __cpp_lib_remove_cvref 201711L
#if __has_builtin(__builtin_source_location)
# define __cpp_lib_source_location 201907L
#endif
#if __cpp_lib_concepts
# define __cpp_lib_span 202002L
#endif
#define __cpp_lib_ssize 201902L
#if __cpp_impl_three_way_comparison >= 201907L && __cpp_lib_concepts
# define __cpp_lib_three_way_comparison 201907L
#endif
#define __cpp_lib_to_address 201711L
#define __cpp_lib_to_array 201907L
#define __cpp_lib_type_identity 201806L
#define __cpp_lib_unwrap_ref 201811L
#if __cpp_concepts >= 202002L && __cpp_constexpr >= 201811L
# define __cpp_lib_variant 202106L
#endif
#define __cpp_lib_constexpr_algorithms 201806L
#define __cpp_lib_constexpr_functional 201907L
#if __cplusplus <= 202002L // N.B. updated value in C++23
# define __cpp_lib_constexpr_memory 201811L
#endif // C++20
#define __cpp_lib_constexpr_numeric 201911L
#define __cpp_lib_constexpr_tuple 201811L
#define __cpp_lib_array_constexpr 201811L
#ifdef __cpp_lib_is_constant_evaluated
# define __cpp_lib_constexpr_char_traits 201811L
#endif
#define __cpp_lib_constexpr_iterator 201811L
#define __cpp_lib_constexpr_utility 201811L
#define __cpp_lib_interpolate 201902L
#if __cpp_lib_concepts
# define __cpp_lib_move_iterator_concept 202207L
#if __cplusplus <= 202002L // N.B. updated value in C++23
# define __cpp_lib_ranges 202110L
#endif
#endif
#define __cpp_lib_shift 201806L
#if _GLIBCXX_HOSTED
#define __cpp_lib_atomic_shared_ptr 201711L
#if defined _GLIBCXX_HAS_GTHREADS || defined _GLIBCXX_HAVE_LINUX_FUTEX
# define __cpp_lib_atomic_wait 201907L
# if __cpp_aligned_new
# define __cpp_lib_barrier 201907L
# endif
#endif
#undef __cpp_lib_chrono
#define __cpp_lib_chrono 201907L
// FIXME: #define __cpp_lib_execution 201902L
#define __cpp_lib_constexpr_complex 201711L
#define __cpp_lib_constexpr_dynamic_alloc 201907L
#define __cpp_lib_constexpr_tuple 201811L
#ifdef __cpp_lib_is_constant_evaluated
# if _GLIBCXX_USE_CXX11_ABI
# define __cpp_lib_constexpr_string 201907L
# else
# define __cpp_lib_constexpr_string 201811L
# endif
#endif
#define __cpp_lib_constexpr_vector 201907L
#define __cpp_lib_erase_if 202002L
#define __cpp_lib_format 202106L
#define __cpp_lib_generic_unordered_lookup 201811L
#ifdef _GLIBCXX_HAS_GTHREADS
# define __cpp_lib_jthread 201911L
#endif
#if __cpp_lib_atomic_wait
# define __cpp_lib_latch 201907L
#endif
#define __cpp_lib_list_remove_return_type 201806L
#define __cpp_lib_polymorphic_allocator 201902L
#if __cpp_lib_atomic_wait || _GLIBCXX_HAVE_POSIX_SEMAPHORE
# define __cpp_lib_semaphore 201907L
#endif
#define __cpp_lib_shared_ptr_arrays 201707L
#define __cpp_lib_smart_ptr_for_overwrite 202002L
# if _GLIBCXX_USE_CXX11_ABI
// Only supported with cxx11-abi
# define __cpp_lib_syncbuf 201803L
# endif
#endif
#if __cplusplus > 202002L
// c++23
#define __cpp_lib_byteswap 202110L
#define __cpp_lib_constexpr_charconv 202207L
#define __cpp_lib_constexpr_typeinfo 202106L
#if __cpp_concepts >= 202002L
# define __cpp_lib_expected 202211L
#endif
#define __cpp_lib_invoke_r 202106L
#define __cpp_lib_is_scoped_enum 202011L
#if __cpp_lib_concepts
# undef __cpp_lib_optional
# define __cpp_lib_optional 202110L
#endif
#define __cpp_lib_reference_from_temporary 202202L
#define __cpp_lib_to_underlying 202102L
#define __cpp_lib_unreachable 202202L
#define __cpp_lib_ranges 202202L
#define __cpp_lib_ranges_zip 202110L
#define __cpp_lib_ranges_chunk 202202L
#define __cpp_lib_ranges_slide 202202L
#define __cpp_lib_ranges_chunk_by 202202L
#define __cpp_lib_ranges_join_with 202202L
#define __cpp_lib_ranges_repeat 202207L
#define __cpp_lib_ranges_stride 202207L
#define __cpp_lib_ranges_cartesian_product 202207L
#define __cpp_lib_ranges_as_rvalue 202207L
#define __cpp_lib_ranges_as_const 202207L
#define __cpp_lib_ranges_enumerate 202302L
#define __cpp_lib_ranges_contains 202207L
#define __cpp_lib_ranges_iota 202202L
#define __cpp_lib_ranges_find_last 202207L
#define __cpp_lib_ranges_fold 202207L
#if __cpp_constexpr_dynamic_alloc
# if _GLIBCXX_HOSTED
# define __cpp_lib_constexpr_bitset 202202L
# endif
# undef __cpp_lib_constexpr_memory
# define __cpp_lib_constexpr_memory 202202L
#endif
#define __cpp_lib_stdatomic_h 202011L
#if _GLIBCXX_HOSTED
#define __cpp_lib_adaptor_iterator_pair_constructor 202106L
#define __cpp_lib_ios_noreplace 202207L
#define __cpp_lib_move_only_function 202110L
#if __cpp_lib_span
# define __cpp_lib_spanstream 202106L
#endif
#if _GLIBCXX_HAVE_STACKTRACE
# define __cpp_lib_stacktrace 202011L
#endif
#define __cpp_lib_stdatomic_h 202011L
#define __cpp_lib_string_contains 202011L
#if _GLIBCXX_USE_CXX11_ABI // Only supported with cxx11-abi
# define __cpp_lib_string_resize_and_overwrite 202110L
#endif
#endif
#endif // C++23
#endif // C++20
#endif // C++17
#endif // C++14
#endif // C++11
#define __glibcxx_want_all
#include <bits/version.h>
#endif // _GLIBCXX_VERSION_INCLUDED