// -*- C++ -*-
// Copyright (C) 2005-2024 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
// .
/** @file ext/type_traits.h
* This file is a GNU extension to the Standard C++ Library.
*/
#ifndef _EXT_TYPE_TRAITS
#define _EXT_TYPE_TRAITS 1
#pragma GCC system_header
#include
#include
extern "C++" {
namespace __gnu_cxx _GLIBCXX_VISIBILITY(default)
{
_GLIBCXX_BEGIN_NAMESPACE_VERSION
// Define a nested type if some predicate holds.
template
struct __enable_if
{ };
template
struct __enable_if
{ typedef _Tp __type; };
// Conditional expression for types. If true, first, if false, second.
template
struct __conditional_type
{ typedef _Iftrue __type; };
template
struct __conditional_type
{ typedef _Iffalse __type; };
// Given an integral builtin type, return the corresponding unsigned type.
template
struct __add_unsigned
{
private:
typedef __enable_if::__value, _Tp> __if_type;
public:
typedef typename __if_type::__type __type;
};
template<>
struct __add_unsigned
{ typedef unsigned char __type; };
template<>
struct __add_unsigned
{ typedef unsigned char __type; };
template<>
struct __add_unsigned
{ typedef unsigned short __type; };
template<>
struct __add_unsigned
{ typedef unsigned int __type; };
template<>
struct __add_unsigned
{ typedef unsigned long __type; };
template<>
struct __add_unsigned
{ typedef unsigned long long __type; };
// Declare but don't define.
template<>
struct __add_unsigned;
template<>
struct __add_unsigned;
// Given an integral builtin type, return the corresponding signed type.
template
struct __remove_unsigned
{
private:
typedef __enable_if::__value, _Tp> __if_type;
public:
typedef typename __if_type::__type __type;
};
template<>
struct __remove_unsigned
{ typedef signed char __type; };
template<>
struct __remove_unsigned
{ typedef signed char __type; };
template<>
struct __remove_unsigned
{ typedef short __type; };
template<>
struct __remove_unsigned
{ typedef int __type; };
template<>
struct __remove_unsigned
{ typedef long __type; };
template<>
struct __remove_unsigned
{ typedef long long __type; };
// Declare but don't define.
template<>
struct __remove_unsigned;
template<>
struct __remove_unsigned;
// For use in string and vstring.
template
_GLIBCXX_CONSTEXPR
inline bool
__is_null_pointer(_Type* __ptr)
{ return __ptr == 0; }
template
_GLIBCXX_CONSTEXPR
inline bool
__is_null_pointer(_Type)
{ return false; }
#if __cplusplus >= 201103L
constexpr bool
__is_null_pointer(std::nullptr_t)
{ return true; }
#endif
// For arithmetic promotions in and
template::__value>
struct __promote
{ typedef double __type; };
// No nested __type member for non-integer non-floating point types,
// allows this type to be used for SFINAE to constrain overloads in
// and to only the intended types.
template
struct __promote<_Tp, false>
{ };
template<>
struct __promote
{ typedef long double __type; };
template<>
struct __promote
{ typedef double __type; };
template<>
struct __promote
{ typedef float __type; };
#ifdef __STDCPP_FLOAT16_T__
template<>
struct __promote<_Float16>
{ typedef _Float16 __type; };
#endif
#ifdef __STDCPP_FLOAT32_T__
template<>
struct __promote<_Float32>
{ typedef _Float32 __type; };
#endif
#ifdef __STDCPP_FLOAT64_T__
template<>
struct __promote<_Float64>
{ typedef _Float64 __type; };
#endif
#ifdef __STDCPP_FLOAT128_T__
template<>
struct __promote<_Float128>
{ typedef _Float128 __type; };
#endif
#ifdef __STDCPP_BFLOAT16_T__
template<>
struct __promote<__gnu_cxx::__bfloat16_t>
{ typedef __gnu_cxx::__bfloat16_t __type; };
#endif
#if __cpp_fold_expressions
template
using __promoted_t = decltype((typename __promote<_Tp>::__type(0) + ...));
// Deducing the promoted type is done by __promoted_t<_Tp...>,
// then __promote is used to provide the nested __type member.
template
using __promote_2 = __promote<__promoted_t<_Tp, _Up>>;
template
using __promote_3 = __promote<__promoted_t<_Tp, _Up, _Vp>>;
template
using __promote_4 = __promote<__promoted_t<_Tp, _Up, _Vp, _Wp>>;
#else
template::__type,
typename _Up2 = typename __promote<_Up>::__type>
struct __promote_2
{
typedef __typeof__(_Tp2() + _Up2()) __type;
};
template::__type,
typename _Up2 = typename __promote<_Up>::__type,
typename _Vp2 = typename __promote<_Vp>::__type>
struct __promote_3
{
typedef __typeof__(_Tp2() + _Up2() + _Vp2()) __type;
};
template::__type,
typename _Up2 = typename __promote<_Up>::__type,
typename _Vp2 = typename __promote<_Vp>::__type,
typename _Wp2 = typename __promote<_Wp>::__type>
struct __promote_4
{
typedef __typeof__(_Tp2() + _Up2() + _Vp2() + _Wp2()) __type;
};
#endif
_GLIBCXX_END_NAMESPACE_VERSION
} // namespace
} // extern "C++"
#endif