📄 math.h
字号:
/*
* math.h: ANSI 'C' (X3J11 Oct 88) library header, section 4.5
* Copyright (C) Codemist Ltd., 1988
* Copyright 1991-1998,2004 ARM Limited. All rights reserved
*/
/*
* RCS $Revision: 1.59.2.1 $ Codemist 0.03
* Checkin $Date: 2004/10/07 15:06:07 $
* Revising $Author: sdouglas $
*/
/*
* Parts of this file are based upon fdlibm:
*
* ====================================================
* Copyright (C) 1993 by Sun Microsystems, Inc. All rights reserved.
*
* Developed at SunSoft, a Sun Microsystems, Inc. business.
* Permission to use, copy, modify, and distribute this
* software is freely granted, provided that this notice
* is preserved.
* ====================================================
*/
#ifndef __math_h
#define __math_h
#define _ARMABI __declspec(__nothrow)
#define _ARMABI_SOFTFP __declspec(__nothrow) __softfp
#define _ARMABI_PURE __declspec(__nothrow) __pure
/*
* Macros for our inline functions down below.
* unsigned& __FLT(float x) - returns the bit pattern of x
* unsigned& __HI(double x) - returns the bit pattern of the high part of x
* (high part has exponent & sign bit in it)
* unsigned& __LO(double x) - returns the bit pattern of the low part of x
*
* We can assign to __FLT, __HI, and __LO and the appropriate bits get set in
* the floating point variable used.
*
* __HI & __LO are affected by the endianness and the target FPU.
*/
#define __FLT(x) (*(unsigned *)&(x))
#ifdef __BIG_ENDIAN
# define __LO(x) (*(1 + (unsigned *)&(x)))
# define __HI(x) (*(unsigned *)&(x))
#else /* ndef __BIG_ENDIAN */
# define __HI(x) (*(1 + (unsigned *)&(x)))
# define __LO(x) (*(unsigned *)&(x))
#endif /* ndef __BIG_ENDIAN */
# ifndef __MATH_DECLS
# define __MATH_DECLS
/*
* A set of functions that we don't actually want to put in the standard
* namespace ever. These are all called by the C99 macros. As they're
* not specified by any standard they can't belong in ::std::. The
* macro #defines are below amongst the standard function declarations.
* We only include these if we actually need them later on
*/
#if !defined(__STRICT_ANSI__) || (defined(__STDC_VERSION__) && 199901L <= __STDC_VERSION__)
# ifdef __cplusplus
extern "C" {
# endif /* __cplusplus */
extern __softfp unsigned __ARM_dcmp4(double /*x*/, double /*y*/);
extern __softfp unsigned __ARM_fcmp4(float /*x*/, float /*y*/);
/*
* Compare x and y and return the CPSR in r0. These means we can test for
* result types with bit pattern matching.
*
* These are a copy of the declarations in rt_fp.h keep in sync.
*/
extern _ARMABI_SOFTFP int __ARM_fpclassifyf(float /*x*/);
extern _ARMABI_SOFTFP int __ARM_fpclassify(double /*x*/);
/* Classify x into NaN, infinite, normal, subnormal, zero */
/* Used by fpclassify macro */
extern __inline _ARMABI_SOFTFP int __ARM_isfinitef(float __x)
{
return ((__FLT(__x) >> 23) & 0xff) != 0xff;
}
extern __inline _ARMABI_SOFTFP int __ARM_isfinite(double __x)
{
return ((__HI(__x) >> 20) & 0x7ff) != 0x7ff;
}
/* Return 1 if __x is finite, 0 otherwise */
/* Used by isfinite macro */
extern __inline _ARMABI_SOFTFP int __ARM_isinff(float __x)
{
return (__FLT(__x) << 1) == 0xff000000;
}
extern __inline _ARMABI_SOFTFP int __ARM_isinf(double __x)
{
return ((__HI(__x) << 1) == 0xffe00000) && (__LO(__x) == 0);
}
/* Return 1 if __x is infinite, 0 otherwise */
/* Used by isinf macro */
extern __inline _ARMABI_SOFTFP int __ARM_islessgreaterf(float __x, float __y)
{
unsigned __f = __ARM_fcmp4(__x, __y) >> 28;
return (__f == 8) || (__f == 2); /* Just N set or Just Z set */
}
extern __inline _ARMABI_SOFTFP int __ARM_islessgreater(double __x, double __y)
{
unsigned __f = __ARM_dcmp4(__x, __y) >> 28;
return (__f == 8) || (__f == 2); /* Just N set or Just Z set */
}
/*
* Compare __x and __y and return 1 if __x < __y or __x > __y, 0 otherwise
* Used by islessgreater macro
*/
extern __inline _ARMABI_SOFTFP int __ARM_isnanf(float __x)
{
return (0x7f800000 - (__FLT(__x) & 0x7fffffff)) >> 31;
}
extern __inline _ARMABI_SOFTFP int __ARM_isnan(double __x)
{
unsigned __xf = __HI(__x) | ((__LO(__x) == 0) ? 0 : 1);
return (0x7ff00000 - (__xf & 0x7fffffff)) >> 31;
}
/* Return 1 if __x is a NaN, 0 otherwise */
/* Used by isnan macro */
extern __inline _ARMABI_SOFTFP int __ARM_isnormalf(float __x)
{
unsigned __xe = (__FLT(__x) >> 23) & 0xff;
return (__xe != 0xff) && (__xe != 0);
}
extern __inline _ARMABI_SOFTFP int __ARM_isnormal(double __x)
{
unsigned __xe = (__HI(__x) >> 20) & 0x7ff;
return (__xe != 0x7ff) && (__xe != 0);
}
/* Return 1 if __x is a normalised number, 0 otherwise */
/* used by isnormal macro */
extern __inline _ARMABI_SOFTFP int __ARM_signbitf(float __x)
{
return __FLT(__x) >> 31;
}
extern __inline _ARMABI_SOFTFP int __ARM_signbit(double __x)
{
return __HI(__x) >> 31;
}
/* Return signbit of __x */
/* Used by signbit macro */
# ifdef __cplusplus
} /* extern "C" */
# endif /* __cplusplus */
#endif /* Strict ANSI */
# undef __CLIBNS
# ifdef __cplusplus
namespace std {
# define __CLIBNS ::std::
extern "C" {
# else
# define __CLIBNS
# endif /* __cplusplus */
#if !defined(__STRICT_ANSI__) || (defined(__STDC_VERSION__) && 199901L <= __STDC_VERSION__)
/* C99 additions */
typedef float float_t;
typedef double double_t;
# ifdef __FP_FAST
# define FLT_EVAL_METHOD (-1)
# else
# define FLT_EVAL_METHOD 0
# endif
# define HUGE_VAL ((double)__INFINITY__)
# define HUGE_VALF ((float)__INFINITY__)
# define HUGE_VALL ((long double)__INFINITY__)
# define INFINITY ((float)__INFINITY__)
# define NAN (__ESCAPE__(0f_7FC00000))
#endif
extern _ARMABI double acos(double /*x*/);
/* computes the principal value of the arc cosine of x */
/* a domain error occurs for arguments not in the range -1 to 1 */
/* Returns: the arc cosine in the range 0 to Pi. */
extern _ARMABI double asin(double /*x*/);
/* computes the principal value of the arc sine of x */
/* a domain error occurs for arguments not in the range -1 to 1 */
/* and -HUGE_VAL is returned. */
/* Returns: the arc sine in the range -Pi/2 to Pi/2. */
extern _ARMABI_PURE double atan(double /*x*/);
/* computes the principal value of the arc tangent of x */
/* Returns: the arc tangent in the range -Pi/2 to Pi/2. */
extern _ARMABI double atan2(double /*y*/, double /*x*/);
/* computes the principal value of the arc tangent of y/x, using the */
/* signs of both arguments to determine the quadrant of the return value */
/* a domain error occurs if both args are zero, and -HUGE_VAL returned. */
/* Returns: the arc tangent of y/x, in the range -Pi to Pi. */
extern _ARMABI_PURE double cos(double /*x*/);
/* computes the cosine of x (measured in radians). A large magnitude */
/* argument may yield a result with little or no significance */
/* Returns: the cosine value. */
extern _ARMABI_PURE double sin(double /*x*/);
/* computes the sine of x (measured in radians). A large magnitude */
/* argument may yield a result with little or no significance */
/* Returns: the sine value. */
extern void __use_accurate_range_reduction(void);
/* reference this to select the larger, slower, but more accurate */
/* range reduction in sin, cos and tan */
extern _ARMABI double tan(double /*x*/);
/* computes the tangent of x (measured in radians). A large magnitude */
/* argument may yield a result with little or no significance */
/* Returns: the tangent value. */
/* if range error; returns HUGE_VAL. */
extern _ARMABI double cosh(double /*x*/);
/* computes the hyperbolic cosine of x. A range error occurs if the */
/* magnitude of x is too large. */
/* Returns: the hyperbolic cosine value. */
/* if range error; returns HUGE_VAL. */
extern _ARMABI double sinh(double /*x*/);
/* computes the hyperbolic sine of x. A range error occurs if the */
/* magnitude of x is too large. */
/* Returns: the hyperbolic sine value. */
/* if range error; returns -HUGE_VAL or HUGE_VAL depending */
/* on the sign of the argument */
extern _ARMABI_PURE double tanh(double /*x*/);
/* computes the hyperbolic tangent of x. */
/* Returns: the hyperbolic tangent value. */
extern _ARMABI double exp(double /*x*/);
/* computes the exponential function of x. A range error occurs if the */
/* magnitude of x is too large. */
/* Returns: the exponential value. */
/* if underflow range error; 0 is returned. */
/* if overflow range error; HUGE_VAL is returned. */
extern _ARMABI double frexp(double /*value*/, int * /*exp*/);
/* breaks a floating-point number into a normalised fraction and an */
/* integral power of 2. It stores the integer in the int object pointed */
/* to by exp. */
/* Returns: the value x, such that x is a double with magnitude in the */
/* interval 0.5 to 1.0 or zero, and value equals x times 2 raised to the */
/* power *exp. If value is zero, both parts of the result are zero. */
extern _ARMABI double ldexp(double /*x*/, int /*exp*/);
/* multiplies a floating-point number by an integral power of 2. */
/* A range error may occur. */
/* Returns: the value of x times 2 raised to the power of exp. */
/* if range error; HUGE_VAL is returned. */
extern _ARMABI double log(double /*x*/);
/* computes the natural logarithm of x. A domain error occurs if the */
/* argument is negative, and -HUGE_VAL is returned. A range error occurs */
/* if the argument is zero. */
/* Returns: the natural logarithm. */
/* if range error; -HUGE_VAL is returned. */
extern _ARMABI double log10(double /*x*/);
/* computes the base-ten logarithm of x. A domain error occurs if the */
/* argument is negative. A range error occurs if the argument is zero. */
/* Returns: the base-ten logarithm. */
extern _ARMABI double modf(double /*value*/, double * /*iptr*/);
/* breaks the argument value into integral and fraction parts, each of */
/* which has the same sign as the argument. It stores the integral part */
/* as a double in the object pointed to by iptr. */
/* Returns: the signed fractional part of value. */
extern _ARMABI double pow(double /*x*/, double /*y*/);
/* computes x raised to the power of y. A domain error occurs if x is */
/* zero and y is less than or equal to zero, or if x is negative and y */
/* is not an integer, and -HUGE_VAL returned. A range error may occur. */
/* Returns: the value of x raised to the power of y. */
/* if underflow range error; 0 is returned. */
/* if overflow range error; HUGE_VAL is returned. */
extern _ARMABI double sqrt(double /*x*/);
/* computes the non-negative square root of x. A domain error occurs */
/* if the argument is negative, and -HUGE_VAL returned. */
/* Returns: the value of the square root. */
#ifdef __TARGET_FPU_VFP
/* native VFP sqrt; not standard compliant: doesn't set errno */
extern _ARMABI_PURE double _sqrt(double);
extern _ARMABI_PURE float _sqrtf(float);
#else
__inline double _sqrt(double __x) { return sqrt(__x); }
__inline float _sqrtf(float __x) { return (float)sqrt(__x); }
#endif
extern _ARMABI_PURE double ceil(double /*x*/);
/* computes the smallest integer not less than x. */
/* Returns: the smallest integer not less than x, expressed as a double. */
extern _ARMABI_PURE double fabs(double /*x*/);
/* computes the absolute value of the floating-point number x. */
/* Returns: the absolute value of x. */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -