1 /* boost integer_traits.hpp header file
2  *
3  * Copyright Jens Maurer 2000
4  * Distributed under the Boost Software License, Version 1.0. (See
5  * accompanying file LICENSE_1_0.txt or copy at
6  * http://www.boost.org/LICENSE_1_0.txt)
7  *
8  * $Id$
9  *
10  * Idea by Beman Dawes, Ed Brey, Steve Cleary, and Nathan Myers
11  */
12 
13 //  See http://www.boost.org/libs/integer for documentation.
14 
15 
16 #ifndef BOOST_INTEGER_TRAITS_HPP
17 #define BOOST_INTEGER_TRAITS_HPP
18 
19 #include <boost/config.hpp>
20 #include <boost/limits.hpp>
21 
22 // These are an implementation detail and not part of the interface
23 #include <limits.h>
24 // we need wchar.h for WCHAR_MAX/MIN but not all platforms provide it,
25 // and some may have <wchar.h> but not <cwchar> ...
26 #if !defined(BOOST_NO_INTRINSIC_WCHAR_T) && (!defined(BOOST_NO_CWCHAR) || defined(sun) || defined(__sun) || defined(__QNX__))
27 #include <wchar.h>
28 #endif
29 
30 //
31 // We simply cannot include this header on gcc without getting copious warnings of the kind:
32 //
33 // ../../../boost/integer_traits.hpp:164:66: warning: use of C99 long long integer constant
34 //
35 // And yet there is no other reasonable implementation, so we declare this a system header
36 // to suppress these warnings.
37 //
38 #if defined(__GNUC__) && (__GNUC__ >= 4)
39 #pragma GCC system_header
40 #endif
41 
42 namespace boost {
43 template<class T>
44 class integer_traits : public std::numeric_limits<T>
45 {
46 public:
47   BOOST_STATIC_CONSTANT(bool, is_integral = false);
48 };
49 
50 namespace detail {
51 template<class T, T min_val, T max_val>
52 class integer_traits_base
53 {
54 public:
55   BOOST_STATIC_CONSTANT(bool, is_integral = true);
56   BOOST_STATIC_CONSTANT(T, const_min = min_val);
57   BOOST_STATIC_CONSTANT(T, const_max = max_val);
58 };
59 
60 #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION
61 //  A definition is required even for integral static constants
62 template<class T, T min_val, T max_val>
63 const bool integer_traits_base<T, min_val, max_val>::is_integral;
64 
65 template<class T, T min_val, T max_val>
66 const T integer_traits_base<T, min_val, max_val>::const_min;
67 
68 template<class T, T min_val, T max_val>
69 const T integer_traits_base<T, min_val, max_val>::const_max;
70 #endif
71 
72 } // namespace detail
73 
74 template<>
75 class integer_traits<bool>
76   : public std::numeric_limits<bool>,
77     public detail::integer_traits_base<bool, false, true>
78 { };
79 
80 template<>
81 class integer_traits<char>
82   : public std::numeric_limits<char>,
83     public detail::integer_traits_base<char, CHAR_MIN, CHAR_MAX>
84 { };
85 
86 template<>
87 class integer_traits<signed char>
88   : public std::numeric_limits<signed char>,
89     public detail::integer_traits_base<signed char, SCHAR_MIN, SCHAR_MAX>
90 { };
91 
92 template<>
93 class integer_traits<unsigned char>
94   : public std::numeric_limits<unsigned char>,
95     public detail::integer_traits_base<unsigned char, 0, UCHAR_MAX>
96 { };
97 
98 #ifndef BOOST_NO_INTRINSIC_WCHAR_T
99 template<>
100 class integer_traits<wchar_t>
101   : public std::numeric_limits<wchar_t>,
102     // Don't trust WCHAR_MIN and WCHAR_MAX with Mac OS X's native
103     // library: they are wrong!
104 #if defined(WCHAR_MIN) && defined(WCHAR_MAX) && !defined(__APPLE__)
105     public detail::integer_traits_base<wchar_t, WCHAR_MIN, WCHAR_MAX>
106 #elif defined(__BORLANDC__) || defined(__CYGWIN__) || defined(__MINGW32__) || (defined(__BEOS__) && defined(__GNUC__))
107     // No WCHAR_MIN and WCHAR_MAX, whar_t is short and unsigned:
108     public detail::integer_traits_base<wchar_t, 0, 0xffff>
109 #elif (defined(__sgi) && (!defined(__SGI_STL_PORT) || __SGI_STL_PORT < 0x400))\
110     || (defined __APPLE__)\
111     || (defined(__OpenBSD__) && defined(__GNUC__))\
112     || (defined(__NetBSD__) && defined(__GNUC__))\
113     || (defined(__FreeBSD__) && defined(__GNUC__))\
114     || (defined(__DragonFly__) && defined(__GNUC__))\
115     || (defined(__hpux) && defined(__GNUC__) && (__GNUC__ == 3) && !defined(__SGI_STL_PORT))
116     // No WCHAR_MIN and WCHAR_MAX, wchar_t has the same range as int.
117     //  - SGI MIPSpro with native library
118     //  - gcc 3.x on HP-UX
119     //  - Mac OS X with native library
120     //  - gcc on FreeBSD, OpenBSD and NetBSD
121     public detail::integer_traits_base<wchar_t, INT_MIN, INT_MAX>
122 #else
123 #error No WCHAR_MIN and WCHAR_MAX present, please adjust integer_traits<> for your compiler.
124 #endif
125 { };
126 #endif // BOOST_NO_INTRINSIC_WCHAR_T
127 
128 template<>
129 class integer_traits<short>
130   : public std::numeric_limits<short>,
131     public detail::integer_traits_base<short, SHRT_MIN, SHRT_MAX>
132 { };
133 
134 template<>
135 class integer_traits<unsigned short>
136   : public std::numeric_limits<unsigned short>,
137     public detail::integer_traits_base<unsigned short, 0, USHRT_MAX>
138 { };
139 
140 template<>
141 class integer_traits<int>
142   : public std::numeric_limits<int>,
143     public detail::integer_traits_base<int, INT_MIN, INT_MAX>
144 { };
145 
146 template<>
147 class integer_traits<unsigned int>
148   : public std::numeric_limits<unsigned int>,
149     public detail::integer_traits_base<unsigned int, 0, UINT_MAX>
150 { };
151 
152 template<>
153 class integer_traits<long>
154   : public std::numeric_limits<long>,
155     public detail::integer_traits_base<long, LONG_MIN, LONG_MAX>
156 { };
157 
158 template<>
159 class integer_traits<unsigned long>
160   : public std::numeric_limits<unsigned long>,
161     public detail::integer_traits_base<unsigned long, 0, ULONG_MAX>
162 { };
163 
164 #if !defined(BOOST_NO_INTEGRAL_INT64_T) && !defined(BOOST_NO_INT64_T)
165 #if defined(ULLONG_MAX) && defined(BOOST_HAS_LONG_LONG)
166 
167 template<>
168 class integer_traits< ::boost::long_long_type>
169   : public std::numeric_limits< ::boost::long_long_type>,
170     public detail::integer_traits_base< ::boost::long_long_type, LLONG_MIN, LLONG_MAX>
171 { };
172 
173 template<>
174 class integer_traits< ::boost::ulong_long_type>
175   : public std::numeric_limits< ::boost::ulong_long_type>,
176     public detail::integer_traits_base< ::boost::ulong_long_type, 0, ULLONG_MAX>
177 { };
178 
179 #elif defined(ULONG_LONG_MAX) && defined(BOOST_HAS_LONG_LONG)
180 
181 template<>
182 class integer_traits< ::boost::long_long_type>  : public std::numeric_limits< ::boost::long_long_type>,    public detail::integer_traits_base< ::boost::long_long_type, LONG_LONG_MIN, LONG_LONG_MAX>{ };
183 template<>
184 class integer_traits< ::boost::ulong_long_type>
185   : public std::numeric_limits< ::boost::ulong_long_type>,
186     public detail::integer_traits_base< ::boost::ulong_long_type, 0, ULONG_LONG_MAX>
187 { };
188 
189 #elif defined(ULONGLONG_MAX) && defined(BOOST_HAS_LONG_LONG)
190 
191 template<>
192 class integer_traits< ::boost::long_long_type>
193   : public std::numeric_limits< ::boost::long_long_type>,
194     public detail::integer_traits_base< ::boost::long_long_type, LONGLONG_MIN, LONGLONG_MAX>
195 { };
196 
197 template<>
198 class integer_traits< ::boost::ulong_long_type>
199   : public std::numeric_limits< ::boost::ulong_long_type>,
200     public detail::integer_traits_base< ::boost::ulong_long_type, 0, ULONGLONG_MAX>
201 { };
202 
203 #elif defined(_LLONG_MAX) && defined(_C2) && defined(BOOST_HAS_LONG_LONG)
204 
205 template<>
206 class integer_traits< ::boost::long_long_type>
207   : public std::numeric_limits< ::boost::long_long_type>,
208     public detail::integer_traits_base< ::boost::long_long_type, -_LLONG_MAX - _C2, _LLONG_MAX>
209 { };
210 
211 template<>
212 class integer_traits< ::boost::ulong_long_type>
213   : public std::numeric_limits< ::boost::ulong_long_type>,
214     public detail::integer_traits_base< ::boost::ulong_long_type, 0, _ULLONG_MAX>
215 { };
216 
217 #elif defined(BOOST_HAS_LONG_LONG)
218 //
219 // we have long long but no constants, this happens for example with gcc in -ansi mode,
220 // we'll just have to work out the values for ourselves (assumes 2's compliment representation):
221 //
222 template<>
223 class integer_traits< ::boost::long_long_type>
224   : public std::numeric_limits< ::boost::long_long_type>,
225     public detail::integer_traits_base< ::boost::long_long_type, (1LL << (sizeof(::boost::long_long_type) * CHAR_BIT - 1)), ~(1LL << (sizeof(::boost::long_long_type) * CHAR_BIT - 1))>
226 { };
227 
228 template<>
229 class integer_traits< ::boost::ulong_long_type>
230   : public std::numeric_limits< ::boost::ulong_long_type>,
231     public detail::integer_traits_base< ::boost::ulong_long_type, 0, ~0uLL>
232 { };
233 
234 #elif defined(BOOST_HAS_MS_INT64)
235 
236 template<>
237 class integer_traits< __int64>
238   : public std::numeric_limits< __int64>,
239     public detail::integer_traits_base< __int64, _I64_MIN, _I64_MAX>
240 { };
241 
242 template<>
243 class integer_traits< unsigned __int64>
244   : public std::numeric_limits< unsigned __int64>,
245     public detail::integer_traits_base< unsigned __int64, 0, _UI64_MAX>
246 { };
247 
248 #endif
249 #endif
250 
251 } // namespace boost
252 
253 #endif /* BOOST_INTEGER_TRAITS_HPP */
254 
255 
256 
257