2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
// FPEnvironment_C99.h
|
|
|
|
//
|
|
|
|
// Library: Foundation
|
|
|
|
// Package: Core
|
|
|
|
// Module: FPEnvironment
|
|
|
|
//
|
|
|
|
// Definitions of class FPEnvironmentImpl for C99.
|
|
|
|
//
|
|
|
|
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
2014-05-04 21:02:42 +02:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef Foundation_FPEnvironment_C99_INCLUDED
|
|
|
|
#define Foundation_FPEnvironment_C99_INCLUDED
|
|
|
|
|
|
|
|
|
|
|
|
#include "Poco/Foundation.h"
|
|
|
|
#include <fenv.h>
|
2013-06-16 18:12:32 +02:00
|
|
|
#include <cmath>
|
2012-04-29 20:52:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
namespace Poco {
|
|
|
|
|
|
|
|
|
|
|
|
class FPEnvironmentImpl
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
enum RoundingModeImpl
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_DOWNWARD)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_ROUND_DOWNWARD_IMPL = FE_DOWNWARD,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_ROUND_DOWNWARD_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_UPWARD)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_ROUND_UPWARD_IMPL = FE_UPWARD,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_ROUND_UPWARD_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_TONEAREST)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_ROUND_TONEAREST_IMPL = FE_TONEAREST,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_ROUND_TONEAREST_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_TOWARDZERO)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_ROUND_TOWARDZERO_IMPL = FE_TOWARDZERO
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_ROUND_TOWARDZERO_IMPL = 0
|
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
};
|
|
|
|
enum FlagImpl
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_DIVBYZERO)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_DIVIDE_BY_ZERO_IMPL = FE_DIVBYZERO,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_DIVIDE_BY_ZERO_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_INEXACT)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_INEXACT_IMPL = FE_INEXACT,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_INEXACT_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_OVERFLOW)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_OVERFLOW_IMPL = FE_OVERFLOW,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_OVERFLOW_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_UNDERFLOW)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_UNDERFLOW_IMPL = FE_UNDERFLOW,
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_UNDERFLOW_IMPL = 0,
|
|
|
|
#endif
|
2021-06-14 22:58:09 +02:00
|
|
|
#if defined(FE_INVALID)
|
2012-04-29 20:52:25 +02:00
|
|
|
FP_INVALID_IMPL = FE_INVALID
|
2021-06-14 18:50:02 +02:00
|
|
|
#else
|
|
|
|
FP_INVALID_IMPL = 0
|
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
};
|
|
|
|
FPEnvironmentImpl();
|
|
|
|
FPEnvironmentImpl(const FPEnvironmentImpl& env);
|
|
|
|
~FPEnvironmentImpl();
|
|
|
|
FPEnvironmentImpl& operator = (const FPEnvironmentImpl& env);
|
2021-06-14 18:50:02 +02:00
|
|
|
void keepCurrentImpl();
|
2012-04-29 20:52:25 +02:00
|
|
|
static void clearFlagsImpl();
|
2021-06-14 18:50:02 +02:00
|
|
|
static bool isFlagImpl(FlagImpl flag);
|
2012-04-29 20:52:25 +02:00
|
|
|
static void setRoundingModeImpl(RoundingModeImpl mode);
|
|
|
|
static RoundingModeImpl getRoundingModeImpl();
|
2021-06-14 18:50:02 +02:00
|
|
|
static bool isInfiniteImpl(float value);
|
2012-04-29 20:52:25 +02:00
|
|
|
static bool isInfiniteImpl(double value);
|
|
|
|
static bool isInfiniteImpl(long double value);
|
2021-06-14 18:50:02 +02:00
|
|
|
static bool isNaNImpl(float value);
|
2012-04-29 20:52:25 +02:00
|
|
|
static bool isNaNImpl(double value);
|
|
|
|
static bool isNaNImpl(long double value);
|
2021-06-14 18:50:02 +02:00
|
|
|
static float copySignImpl(float target, float source);
|
2012-04-29 20:52:25 +02:00
|
|
|
static double copySignImpl(double target, double source);
|
|
|
|
static long double copySignImpl(long double target, long double source);
|
|
|
|
|
|
|
|
private:
|
|
|
|
fenv_t _env;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// inlines
|
|
|
|
//
|
|
|
|
inline bool FPEnvironmentImpl::isInfiniteImpl(float value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isinf(value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isinf(value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool FPEnvironmentImpl::isInfiniteImpl(double value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isinf(value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isinf(value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool FPEnvironmentImpl::isInfiniteImpl(long double value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isinf((double) value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isinf((double) value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool FPEnvironmentImpl::isNaNImpl(float value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isnan(value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isnan(value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool FPEnvironmentImpl::isNaNImpl(double value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isnan(value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isnan(value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool FPEnvironmentImpl::isNaNImpl(long double value)
|
|
|
|
{
|
2021-06-14 22:58:09 +02:00
|
|
|
#if POCO_OS == POCO_OS_AIX
|
|
|
|
return ::isnan((double) value) != 0;
|
|
|
|
#else
|
2013-06-15 16:20:49 +02:00
|
|
|
return std::isnan((double) value) != 0;
|
2021-06-14 22:58:09 +02:00
|
|
|
#endif
|
2012-04-29 20:52:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline float FPEnvironmentImpl::copySignImpl(float target, float source)
|
|
|
|
{
|
|
|
|
return copysignf(target, source);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline double FPEnvironmentImpl::copySignImpl(double target, double source)
|
|
|
|
{
|
|
|
|
return copysign(target, source);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace Poco
|
|
|
|
|
|
|
|
|
|
|
|
#endif // Foundation_FPEnvironment_C99_INCLUDED
|