backports.hpp
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2013-2021 Regents of the University of California.
4  *
5  * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
6  *
7  * ndn-cxx library is free software: you can redistribute it and/or modify it under the
8  * terms of the GNU Lesser General Public License as published by the Free Software
9  * Foundation, either version 3 of the License, or (at your option) any later version.
10  *
11  * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
12  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
13  * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
14  *
15  * You should have received copies of the GNU General Public License and GNU Lesser
16  * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
17  * <http://www.gnu.org/licenses/>.
18  *
19  * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
20  */
21 
22 #ifndef NDN_CXX_UTIL_BACKPORTS_HPP
23 #define NDN_CXX_UTIL_BACKPORTS_HPP
24 
26 
27 #include <boost/predef/compiler/clang.h>
28 #include <boost/predef/compiler/gcc.h>
29 #include <boost/predef/compiler/visualc.h>
30 
31 #ifdef __has_cpp_attribute
32 # define NDN_CXX_HAS_CPP_ATTRIBUTE(x) __has_cpp_attribute(x)
33 #else
34 # define NDN_CXX_HAS_CPP_ATTRIBUTE(x) 0
35 #endif
36 
37 #ifdef __has_include
38 # define NDN_CXX_HAS_INCLUDE(x) __has_include(x)
39 #else
40 # define NDN_CXX_HAS_INCLUDE(x) 0
41 #endif
42 
43 //
44 // http://wg21.link/P0188
45 // [[fallthrough]] attribute (C++17)
46 //
47 #if (__cplusplus > 201402L) && NDN_CXX_HAS_CPP_ATTRIBUTE(fallthrough)
48 # define NDN_CXX_FALLTHROUGH [[fallthrough]]
49 #elif NDN_CXX_HAS_CPP_ATTRIBUTE(clang::fallthrough)
50 # define NDN_CXX_FALLTHROUGH [[clang::fallthrough]]
51 #elif NDN_CXX_HAS_CPP_ATTRIBUTE(gnu::fallthrough)
52 # define NDN_CXX_FALLTHROUGH [[gnu::fallthrough]]
53 #elif BOOST_COMP_GNUC >= BOOST_VERSION_NUMBER(7,0,0)
54 # define NDN_CXX_FALLTHROUGH __attribute__((fallthrough))
55 #else
56 # define NDN_CXX_FALLTHROUGH ((void)0)
57 #endif
58 
59 //
60 // http://wg21.link/P0189
61 // [[nodiscard]] attribute (C++17)
62 //
63 #if (__cplusplus > 201402L) && NDN_CXX_HAS_CPP_ATTRIBUTE(nodiscard)
64 # define NDN_CXX_NODISCARD [[nodiscard]]
65 #elif NDN_CXX_HAS_CPP_ATTRIBUTE(gnu::warn_unused_result)
66 # define NDN_CXX_NODISCARD [[gnu::warn_unused_result]]
67 #else
68 # define NDN_CXX_NODISCARD
69 #endif
70 
71 #ifndef NDEBUG
72 # define NDN_CXX_UNREACHABLE BOOST_ASSERT(false)
73 #elif BOOST_COMP_GNUC || BOOST_COMP_CLANG
74 # define NDN_CXX_UNREACHABLE __builtin_unreachable()
75 #elif BOOST_COMP_MSVC
76 # define NDN_CXX_UNREACHABLE __assume(0)
77 #else
78 # include <cstdlib>
79 # define NDN_CXX_UNREACHABLE std::abort()
80 #endif
81 
82 #ifndef NDN_CXX_HAVE_STD_TO_STRING
83 #include <boost/lexical_cast.hpp>
84 #endif
85 
86 namespace ndn {
87 
88 //
89 // https://redmine.named-data.net/issues/2743
90 // std::to_string() (C++11)
91 //
92 #ifdef NDN_CXX_HAVE_STD_TO_STRING
93 using std::to_string;
94 #else
95 template<typename T>
96 inline std::string
97 to_string(const T& val)
98 {
99  return boost::lexical_cast<std::string>(val);
100 }
101 #endif // NDN_CXX_HAVE_STD_TO_STRING
102 
103 //
104 // https://wg21.link/P0025
105 // std::clamp() (C++17)
106 //
107 #if __cpp_lib_clamp >= 201603L
108 using std::clamp;
109 #else
110 template<typename T, typename Compare>
111 constexpr const T&
112 clamp(const T& v, const T& lo, const T& hi, Compare comp)
113 {
114  BOOST_ASSERT(!comp(hi, lo));
115  return comp(v, lo) ? lo : comp(hi, v) ? hi : v;
116 }
117 
118 template<typename T>
119 constexpr const T&
120 clamp(const T& v, const T& lo, const T& hi)
121 {
122  BOOST_ASSERT(!(hi < lo));
123  return (v < lo) ? lo : (hi < v) ? hi : v;
124 }
125 #endif // __cpp_lib_clamp
126 
127 //
128 // https://wg21.link/P1682
129 // std::to_underlying() (C++23)
130 //
131 #if __cpp_lib_to_underlying >= 202102L
132 using std::to_underlying;
133 #else
134 template<typename T>
135 NDN_CXX_NODISCARD constexpr std::underlying_type_t<T>
136 to_underlying(T val) noexcept
137 {
138  // instantiating underlying_type with a non-enum type is UB before C++20
139  static_assert(std::is_enum<T>::value, "");
140  return static_cast<std::underlying_type_t<T>>(val);
141 }
142 #endif // __cpp_lib_to_underlying
143 
144 } // namespace ndn
145 
146 #endif // NDN_CXX_UTIL_BACKPORTS_HPP
#define NDN_CXX_NODISCARD
Definition: backports.hpp:68
Common includes and macros used throughout the library.
std::string to_string(const errinfo_stacktrace &x)
Definition: exception.cpp:31
Definition: data.cpp:25
constexpr const T & clamp(const T &v, const T &lo, const T &hi)
Definition: backports.hpp:120
constexpr std::underlying_type_t< T > to_underlying(T val) noexcept
Definition: backports.hpp:136
constexpr const T & clamp(const T &v, const T &lo, const T &hi, Compare comp)
Definition: backports.hpp:112