/* * Copyright (c) 1997 * Silicon Graphics Computer Systems, Inc. * * Copyright (c) 1999 * Boris Fomitchev * * This material is provided "as is", with absolutely no warranty expressed * or implied. Any use is at your own risk. * * Permission to use or copy this software for any purpose is hereby granted * without fee, provided the above notices are retained on all copies. * Permission to modify the code and to distribute modified code is granted, * provided the above notices are retained, and a notice that the code was * modified is included with the above copyright notice. * */ /* NOTE: This may be not portable code. Parts of numeric_limits<> are * inherently machine-dependent. At present this file is suitable * for the MIPS, SPARC, Alpha and ia32 architectures. */ #ifndef _STLP_INTERNAL_LIMITS #define _STLP_INTERNAL_LIMITS #ifndef _STLP_CLIMITS # include #endif #ifndef _STLP_CFLOAT # include #endif #if defined (_STLP_HAS_WCHAR_T) && !defined (_STLP_INTERNAL_CWCHAR) # include #endif _STLP_BEGIN_NAMESPACE enum float_round_style { round_indeterminate = -1, round_toward_zero = 0, round_to_nearest = 1, round_toward_infinity = 2, round_toward_neg_infinity = 3 }; enum float_denorm_style { denorm_indeterminate = -1, denorm_absent = 0, denorm_present = 1 }; _STLP_MOVE_TO_PRIV_NAMESPACE // Base class for all specializations of numeric_limits. template class _Numeric_limits_base { public: static __number (_STLP_CALL min)() _STLP_NOTHROW { return __number(); } static __number (_STLP_CALL max)() _STLP_NOTHROW { return __number(); } _STLP_STATIC_CONSTANT(int, digits = 0); _STLP_STATIC_CONSTANT(int, digits10 = 0); _STLP_STATIC_CONSTANT(int, radix = 0); _STLP_STATIC_CONSTANT(int, min_exponent = 0); _STLP_STATIC_CONSTANT(int, min_exponent10 = 0); _STLP_STATIC_CONSTANT(int, max_exponent = 0); _STLP_STATIC_CONSTANT(int, max_exponent10 = 0); _STLP_STATIC_CONSTANT(float_denorm_style, has_denorm = denorm_absent); _STLP_STATIC_CONSTANT(float_round_style, round_style = round_toward_zero); _STLP_STATIC_CONSTANT(bool, is_specialized = false); _STLP_STATIC_CONSTANT(bool, is_signed = false); _STLP_STATIC_CONSTANT(bool, is_integer = false); _STLP_STATIC_CONSTANT(bool, is_exact = false); _STLP_STATIC_CONSTANT(bool, has_infinity = false); _STLP_STATIC_CONSTANT(bool, has_quiet_NaN = false); _STLP_STATIC_CONSTANT(bool, has_signaling_NaN = false); _STLP_STATIC_CONSTANT(bool, has_denorm_loss = false); _STLP_STATIC_CONSTANT(bool, is_iec559 = false); _STLP_STATIC_CONSTANT(bool, is_bounded = false); _STLP_STATIC_CONSTANT(bool, is_modulo = false); _STLP_STATIC_CONSTANT(bool, traps = false); _STLP_STATIC_CONSTANT(bool, tinyness_before = false); static __number _STLP_CALL epsilon() _STLP_NOTHROW { return __number(); } static __number _STLP_CALL round_error() _STLP_NOTHROW { return __number(); } static __number _STLP_CALL infinity() _STLP_NOTHROW { return __number(); } static __number _STLP_CALL quiet_NaN() _STLP_NOTHROW { return __number(); } static __number _STLP_CALL signaling_NaN() _STLP_NOTHROW { return __number(); } static __number _STLP_CALL denorm_min() _STLP_NOTHROW { return __number(); } }; // Base class for integers. #ifdef _STLP_LIMITED_DEFAULT_TEMPLATES # ifdef _STLP_LONG_LONG # define _STLP_LIMITS_MIN_TYPE _STLP_LONG_LONG # define _STLP_LIMITS_MAX_TYPE unsigned _STLP_LONG_LONG # else # define _STLP_LIMITS_MIN_TYPE long # define _STLP_LIMITS_MAX_TYPE unsigned long # endif #else # define _STLP_LIMITS_MIN_TYPE _Int # define _STLP_LIMITS_MAX_TYPE _Int #endif /* _STLP_LIMITED_DEFAULT_TEMPLATES */ template class _Integer_limits : public _Numeric_limits_base<_Int> { public: static _Int (_STLP_CALL min) () _STLP_NOTHROW { return (_Int)__imin; } static _Int (_STLP_CALL max) () _STLP_NOTHROW { return (_Int)__imax; } _STLP_STATIC_CONSTANT(int, digits = (__idigits < 0) ? ((int)((sizeof(_Int) * (CHAR_BIT))) - ((__imin == 0) ? 0 : 1)) : (__idigits)); _STLP_STATIC_CONSTANT(int, digits10 = (digits * 301UL) / 1000); _STLP_STATIC_CONSTANT(int, radix = 2); _STLP_STATIC_CONSTANT(bool, is_specialized = true); _STLP_STATIC_CONSTANT(bool, is_signed = (__imin != 0)); _STLP_STATIC_CONSTANT(bool, is_integer = true); _STLP_STATIC_CONSTANT(bool, is_exact = true); _STLP_STATIC_CONSTANT(bool, is_bounded = true); _STLP_STATIC_CONSTANT(bool, is_modulo = __ismod); }; // Base class for floating-point numbers. template class _Floating_limits : public _Numeric_limits_base<__number> { public: _STLP_STATIC_CONSTANT(int, digits = __Digits); _STLP_STATIC_CONSTANT(int, digits10 = __Digits10); _STLP_STATIC_CONSTANT(int, radix = FLT_RADIX); _STLP_STATIC_CONSTANT(int, min_exponent = __MinExp); _STLP_STATIC_CONSTANT(int, max_exponent = __MaxExp); _STLP_STATIC_CONSTANT(int, min_exponent10 = __MinExp10); _STLP_STATIC_CONSTANT(int, max_exponent10 = __MaxExp10); _STLP_STATIC_CONSTANT(float_denorm_style, has_denorm = __DenormStyle); _STLP_STATIC_CONSTANT(float_round_style, round_style = __RoundStyle); _STLP_STATIC_CONSTANT(bool, is_specialized = true); _STLP_STATIC_CONSTANT(bool, is_signed = true); _STLP_STATIC_CONSTANT(bool, has_infinity = true); #if (!defined (_STLP_MSVC) || (_STLP_MSVC > 1300)) && \ (!defined (__BORLANDC__) || (__BORLANDC__ >= 0x590)) && \ (!defined (_CRAY) || defined (_CRAYIEEE)) _STLP_STATIC_CONSTANT(bool, has_quiet_NaN = true); _STLP_STATIC_CONSTANT(bool, has_signaling_NaN = true); #else _STLP_STATIC_CONSTANT(bool, has_quiet_NaN = false); _STLP_STATIC_CONSTANT(bool, has_signaling_NaN = false); #endif _STLP_STATIC_CONSTANT(bool, is_iec559 = __IsIEC559 && has_infinity && has_quiet_NaN && has_signaling_NaN && (has_denorm == denorm_present)); _STLP_STATIC_CONSTANT(bool, has_denorm_loss = false); _STLP_STATIC_CONSTANT(bool, is_bounded = true); _STLP_STATIC_CONSTANT(bool, traps = true); _STLP_STATIC_CONSTANT(bool, tinyness_before = false); }; _STLP_MOVE_TO_STD_NAMESPACE // Class numeric_limits // The unspecialized class. template class numeric_limits : public _STLP_PRIV _Numeric_limits_base<_Tp> {}; // Specializations for all built-in integral types. #if !defined (_STLP_NO_BOOL) _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #endif /* _STLP_NO_BOOL */ _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #if !defined (_STLP_NO_SIGNED_BUILTINS) _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #endif _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #if !(defined (_STLP_NO_WCHAR_T) || (!defined (_STLP_SGX_CONFIG) && defined (_STLP_WCHAR_T_IS_USHORT))) _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #endif _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #if defined (__xlC__) && (__xlC__ == 0x500) # undef INT_MIN # define INT_MIN -2147483648 #endif _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; #if defined (_STLP_LONG_LONG) # if defined (_STLP_MSVC) || defined (__BORLANDC__) # define LONGLONG_MAX 0x7fffffffffffffffi64 # define LONGLONG_MIN (-LONGLONG_MAX-1i64) # define ULONGLONG_MAX 0xffffffffffffffffUi64 # else # ifndef LONGLONG_MAX # define LONGLONG_MAX 0x7fffffffffffffffLL # endif # ifndef LONGLONG_MIN # define LONGLONG_MIN (-LONGLONG_MAX-1LL) # endif # ifndef ULONGLONG_MAX # define ULONGLONG_MAX 0xffffffffffffffffULL # endif # endif # if !defined (__GNUC__) || (__GNUC__ == 2 && __GNUC_MINOR__ <= 96) || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1) || (__GNUC__ > 3) _STLP_TEMPLATE_NULL class numeric_limits<_STLP_LONG_LONG> : public _STLP_PRIV _Integer_limits<_STLP_LONG_LONG, LONGLONG_MIN, LONGLONG_MAX, -1, true> {}; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Integer_limits {}; # else /* gcc 2.97 (after 2000-11-01), 2.98, 3.0 */ /* newest gcc has new mangling scheme, that has problem with generating name [instantiated] of template specialization like _Integer_limits<_STLP_LONG_LONG, LONGLONG_MIN, LONGLONG_MAX, -1, true> ~~~~~~~~~~~~ ~~~~~~~~~~~~ Below is code that solve this problem. - ptr */ _STLP_TEMPLATE_NULL class numeric_limits<_STLP_LONG_LONG> : public _STLP_PRIV _Numeric_limits_base<_STLP_LONG_LONG> { public: static _STLP_LONG_LONG (_STLP_CALL min) () _STLP_NOTHROW { return LONGLONG_MIN; } static _STLP_LONG_LONG (_STLP_CALL max) () _STLP_NOTHROW { return LONGLONG_MAX; } _STLP_STATIC_CONSTANT(int, digits = ((int)((sizeof(_STLP_LONG_LONG) * (CHAR_BIT))) - 1)); _STLP_STATIC_CONSTANT(int, digits10 = (digits * 301UL) / 1000); _STLP_STATIC_CONSTANT(int, radix = 2); _STLP_STATIC_CONSTANT(bool, is_specialized = true); _STLP_STATIC_CONSTANT(bool, is_signed = true); _STLP_STATIC_CONSTANT(bool, is_integer = true); _STLP_STATIC_CONSTANT(bool, is_exact = true); _STLP_STATIC_CONSTANT(bool, is_bounded = true); _STLP_STATIC_CONSTANT(bool, is_modulo = true); }; _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Numeric_limits_base { public: static unsigned _STLP_LONG_LONG (_STLP_CALL min) () _STLP_NOTHROW { return 0ULL; } static unsigned _STLP_LONG_LONG (_STLP_CALL max) () _STLP_NOTHROW { return ULONGLONG_MAX; } _STLP_STATIC_CONSTANT(int, digits = ((int)((sizeof(unsigned _STLP_LONG_LONG) * (CHAR_BIT))))); _STLP_STATIC_CONSTANT(int, digits10 = (digits * 301UL) / 1000); _STLP_STATIC_CONSTANT(int, radix = 2); _STLP_STATIC_CONSTANT(bool, is_specialized = true); _STLP_STATIC_CONSTANT(bool, is_signed = false); _STLP_STATIC_CONSTANT(bool, is_integer = true); _STLP_STATIC_CONSTANT(bool, is_exact = true); _STLP_STATIC_CONSTANT(bool, is_bounded = true); _STLP_STATIC_CONSTANT(bool, is_modulo = true); }; # endif /* __GNUC__ > 2000-11-01 */ #endif /* _STLP_LONG_LONG */ _STLP_MOVE_TO_PRIV_NAMESPACE // Specializations for all built-in floating-point types. template class _LimG { public: static float _STLP_CALL get_F_inf(); static float _STLP_CALL get_F_qNaN(); static float _STLP_CALL get_F_sNaN(); static float _STLP_CALL get_F_denormMin(); static double _STLP_CALL get_D_inf(); static double _STLP_CALL get_D_qNaN(); static double _STLP_CALL get_D_sNaN(); static double _STLP_CALL get_D_denormMin(); #if !defined (_STLP_NO_LONG_DOUBLE) static long double _STLP_CALL get_LD_inf(); static long double _STLP_CALL get_LD_qNaN(); static long double _STLP_CALL get_LD_sNaN(); static long double _STLP_CALL get_LD_denormMin(); #endif }; #if defined (_STLP_USE_TEMPLATE_EXPORT) _STLP_EXPORT_TEMPLATE_CLASS _LimG; #endif #if defined (__GNUC__) # if defined (__FLT_DENORM_MIN__) # define _STLP_FLT_DENORM_MIN __FLT_DENORM_MIN__ # else # define _STLP_FLT_DENORM_STYLE denorm_absent # endif # if defined (__DBL_DENORM_MIN__) # define _STLP_DBL_DENORM_MIN __DBL_DENORM_MIN__ # else # define _STLP_DBL_DENORM_STYLE denorm_absent # endif # if defined (__LDBL_DENORM_MIN__) # define _STLP_LDBL_DENORM_MIN __LDBL_DENORM_MIN__ # else # define _STLP_LDBL_DENORM_STYLE denorm_absent # endif #endif /* If compiler do not expose thanks to some macro its status regarding * denormalized floating point numbers, we consider that denormalization * is present. Unit tests will tell us if compiler do not support them. */ #if !defined (_STLP_FLT_DENORM_STYLE) # define _STLP_FLT_DENORM_STYLE denorm_present #endif #if !defined (_STLP_DBL_DENORM_STYLE) # define _STLP_DBL_DENORM_STYLE denorm_present #endif #if !defined (_STLP_LDBL_DENORM_STYLE) # define _STLP_LDBL_DENORM_STYLE denorm_present #endif _STLP_MOVE_TO_STD_NAMESPACE _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Floating_limits { public: static float (_STLP_CALL min) () _STLP_NOTHROW { return FLT_MIN; } static float _STLP_CALL denorm_min() _STLP_NOTHROW #if defined (_STLP_FLT_DENORM_MIN) { return _STLP_FLT_DENORM_MIN; } #else { return _STLP_FLT_DENORM_STYLE ? _STLP_PRIV _LimG::get_F_denormMin() : FLT_MIN; } #endif static float (_STLP_CALL max) () _STLP_NOTHROW { return FLT_MAX; } static float _STLP_CALL epsilon() _STLP_NOTHROW { return FLT_EPSILON; } static float _STLP_CALL round_error() _STLP_NOTHROW { return 0.5f; } // Units: ulps. static float _STLP_CALL infinity() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_F_inf(); } static float _STLP_CALL quiet_NaN() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_F_qNaN(); } static float _STLP_CALL signaling_NaN() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_F_sNaN(); } }; #undef _STLP_FLT_DENORM_MIN #undef _STLP_FLT_DNORM_STYLE _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Floating_limits { public: static double (_STLP_CALL min)() _STLP_NOTHROW { return DBL_MIN; } static double _STLP_CALL denorm_min() _STLP_NOTHROW #if defined (_STLP_DBL_DENORM_MIN) { return _STLP_DBL_DENORM_MIN; } #else { return _STLP_DBL_DENORM_STYLE ? _STLP_PRIV _LimG::get_D_denormMin() : DBL_MIN; } #endif static double (_STLP_CALL max)() _STLP_NOTHROW { return DBL_MAX; } static double _STLP_CALL epsilon() _STLP_NOTHROW { return DBL_EPSILON; } static double _STLP_CALL round_error() _STLP_NOTHROW { return 0.5; } // Units: ulps. static double _STLP_CALL infinity() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_D_inf(); } static double _STLP_CALL quiet_NaN() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_D_qNaN(); } static double _STLP_CALL signaling_NaN() _STLP_NOTHROW { return _STLP_PRIV _LimG::get_D_sNaN(); } }; #if !defined (_STLP_NO_LONG_DOUBLE) _STLP_TEMPLATE_NULL class numeric_limits : public _STLP_PRIV _Floating_limits { public: static long double (_STLP_CALL min) () _STLP_NOTHROW { return LDBL_MIN; } static long double _STLP_CALL denorm_min() _STLP_NOTHROW #if defined (_STLP_LDBL_DENORM_MIN) { return _STLP_LDBL_DENORM_MIN; } #else { return _STLP_LDBL_DENORM_STYLE ? _STLP_PRIV _LimG::get_LD_denormMin() : LDBL_MIN; } #endif _STLP_STATIC_CONSTANT(bool, is_iec559 = false); static long double (_STLP_CALL max) () _STLP_NOTHROW { return LDBL_MAX; } static long double _STLP_CALL epsilon() _STLP_NOTHROW { return LDBL_EPSILON; } static long double _STLP_CALL round_error() _STLP_NOTHROW { return 0.5l; } static long double _STLP_CALL infinity() _STLP_NOTHROW //For MSVC, long double is nothing more than an alias for double. #if !defined (_STLP_MSVC) { return _STLP_PRIV _LimG::get_LD_inf(); } #else { return _STLP_PRIV _LimG::get_D_inf(); } #endif static long double _STLP_CALL quiet_NaN() _STLP_NOTHROW #if !defined (_STLP_MSVC) { return _STLP_PRIV _LimG::get_LD_qNaN(); } #else { return _STLP_PRIV _LimG::get_D_qNaN(); } #endif static long double _STLP_CALL signaling_NaN() _STLP_NOTHROW #if !defined (_STLP_MSVC) { return _STLP_PRIV _LimG::get_LD_sNaN(); } #else { return _STLP_PRIV _LimG::get_D_sNaN(); } #endif }; #endif // We write special values (Inf and NaN) as bit patterns and // cast the the appropriate floating-point types. _STLP_END_NAMESPACE #if !defined (_STLP_LINK_TIME_INSTANTIATION) # include #endif #endif // Local Variables: // mode:C++ // End: