Blame sysdeps/generic/get-rounding-mode.h

Packit 6c4009
/* Determine floating-point rounding mode within libc.  Generic version.
Packit 6c4009
   Copyright (C) 2012-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#ifndef _GET_ROUNDING_MODE_H
Packit 6c4009
#define _GET_ROUNDING_MODE_H	1
Packit 6c4009
Packit 6c4009
#include <fpu_control.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
Packit 6c4009
/* Define values for FE_* modes not defined for this architecture.  */
Packit 6c4009
#ifdef FE_DOWNWARD
Packit 6c4009
# define ORIG_FE_DOWNWARD FE_DOWNWARD
Packit 6c4009
#else
Packit 6c4009
# define ORIG_FE_DOWNWARD 0
Packit 6c4009
#endif
Packit 6c4009
#ifdef FE_TONEAREST
Packit 6c4009
# define ORIG_FE_TONEAREST FE_TONEAREST
Packit 6c4009
#else
Packit 6c4009
# define ORIG_FE_TONEAREST 0
Packit 6c4009
#endif
Packit 6c4009
#ifdef FE_TOWARDZERO
Packit 6c4009
# define ORIG_FE_TOWARDZERO FE_TOWARDZERO
Packit 6c4009
#else
Packit 6c4009
# define ORIG_FE_TOWARDZERO 0
Packit 6c4009
#endif
Packit 6c4009
#ifdef FE_UPWARD
Packit 6c4009
# define ORIG_FE_UPWARD FE_UPWARD
Packit 6c4009
#else
Packit 6c4009
# define ORIG_FE_UPWARD 0
Packit 6c4009
#endif
Packit 6c4009
#define FE_CONSTRUCT_DISTINCT_VALUE(X, Y, Z) \
Packit 6c4009
  ((((X) & 1) | ((Y) & 2) | ((Z) & 4)) ^ 7)
Packit 6c4009
#ifndef FE_DOWNWARD
Packit 6c4009
# define FE_DOWNWARD FE_CONSTRUCT_DISTINCT_VALUE (ORIG_FE_TONEAREST,	\
Packit 6c4009
						  ORIG_FE_TOWARDZERO,	\
Packit 6c4009
						  ORIG_FE_UPWARD)
Packit 6c4009
#endif
Packit 6c4009
#ifndef FE_TONEAREST
Packit 6c4009
# define FE_TONEAREST FE_CONSTRUCT_DISTINCT_VALUE (FE_DOWNWARD,		\
Packit 6c4009
						   ORIG_FE_TOWARDZERO,	\
Packit 6c4009
						   ORIG_FE_UPWARD)
Packit 6c4009
#endif
Packit 6c4009
#ifndef FE_TOWARDZERO
Packit 6c4009
# define FE_TOWARDZERO FE_CONSTRUCT_DISTINCT_VALUE (FE_DOWNWARD,	\
Packit 6c4009
						    FE_TONEAREST,	\
Packit 6c4009
						    ORIG_FE_UPWARD)
Packit 6c4009
#endif
Packit 6c4009
#ifndef FE_UPWARD
Packit 6c4009
# define FE_UPWARD FE_CONSTRUCT_DISTINCT_VALUE (FE_DOWNWARD,	\
Packit 6c4009
						FE_TONEAREST,	\
Packit 6c4009
						FE_TOWARDZERO)
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
/* Return the floating-point rounding mode.  */
Packit 6c4009
Packit 6c4009
static inline int
Packit 6c4009
get_rounding_mode (void)
Packit 6c4009
{
Packit 6c4009
#if (defined _FPU_RC_DOWN			\
Packit 6c4009
     || defined _FPU_RC_NEAREST			\
Packit 6c4009
     || defined _FPU_RC_ZERO			\
Packit 6c4009
     || defined _FPU_RC_UP)
Packit 6c4009
  fpu_control_t fc;
Packit 6c4009
  const fpu_control_t mask = (0
Packit 6c4009
# ifdef _FPU_RC_DOWN
Packit 6c4009
			      | _FPU_RC_DOWN
Packit 6c4009
# endif
Packit 6c4009
# ifdef _FPU_RC_NEAREST
Packit 6c4009
			      | _FPU_RC_NEAREST
Packit 6c4009
# endif
Packit 6c4009
# ifdef _FPU_RC_ZERO
Packit 6c4009
			      | _FPU_RC_ZERO
Packit 6c4009
# endif
Packit 6c4009
# ifdef _FPU_RC_UP
Packit 6c4009
			      | _FPU_RC_UP
Packit 6c4009
# endif
Packit 6c4009
			      );
Packit 6c4009
Packit 6c4009
  _FPU_GETCW (fc);
Packit 6c4009
  switch (fc & mask)
Packit 6c4009
    {
Packit 6c4009
# ifdef _FPU_RC_DOWN
Packit 6c4009
    case _FPU_RC_DOWN:
Packit 6c4009
      return FE_DOWNWARD;
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
# ifdef _FPU_RC_NEAREST
Packit 6c4009
    case _FPU_RC_NEAREST:
Packit 6c4009
      return FE_TONEAREST;
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
# ifdef _FPU_RC_ZERO
Packit 6c4009
    case _FPU_RC_ZERO:
Packit 6c4009
      return FE_TOWARDZERO;
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
# ifdef _FPU_RC_UP
Packit 6c4009
    case _FPU_RC_UP:
Packit 6c4009
      return FE_UPWARD;
Packit 6c4009
# endif
Packit 6c4009
Packit 6c4009
    default:
Packit 6c4009
      abort ();
Packit 6c4009
    }
Packit 6c4009
#else
Packit 6c4009
  return FE_TONEAREST;
Packit 6c4009
#endif
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
#endif /* get-rounding-mode.h */