Blame sysdeps/aarch64/fpu/fraiseexcpt.c

Packit 6c4009
/* Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
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 License as
Packit 6c4009
   published by the Free Software Foundation; either version 2.1 of the
Packit 6c4009
   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
#include <fenv.h>
Packit 6c4009
#include <fpu_control.h>
Packit 6c4009
#include <float.h>
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
__feraiseexcept (int excepts)
Packit 6c4009
{
Packit 6c4009
  int fpsr;
Packit 6c4009
  const float fp_zero = 0.0;
Packit 6c4009
  const float fp_one = 1.0;
Packit 6c4009
  const float fp_max = FLT_MAX;
Packit 6c4009
  const float fp_min = FLT_MIN;
Packit 6c4009
  const float fp_1e32 = 1.0e32f;
Packit 6c4009
  const float fp_two = 2.0;
Packit 6c4009
  const float fp_three = 3.0;
Packit 6c4009
Packit 6c4009
  /* Raise exceptions represented by EXCEPTS.  But we must raise only
Packit 6c4009
     one signal at a time.  It is important that if the OVERFLOW or
Packit 6c4009
     UNDERFLOW exception and the inexact exception are given at the
Packit 6c4009
     same time, the OVERFLOW or UNDERFLOW exception precedes the
Packit 6c4009
     INEXACT exception.
Packit 6c4009
Packit 6c4009
     After each exception we read from the FPSR, to force the
Packit 6c4009
     exception to be raised immediately.  */
Packit 6c4009
Packit 6c4009
  if (FE_INVALID & excepts)
Packit 6c4009
    __asm__ __volatile__ (
Packit 6c4009
			  "ldr	s0, %1\n\t"
Packit 6c4009
			  "fdiv	s0, s0, s0\n\t"
Packit 6c4009
			  "mrs	%0, fpsr" : "=r" (fpsr)
Packit 6c4009
			  : "m" (fp_zero)
Packit 6c4009
			  : "d0");
Packit 6c4009
Packit 6c4009
  if (FE_DIVBYZERO & excepts)
Packit 6c4009
    __asm__ __volatile__ (
Packit 6c4009
			  "ldr	s0, %1\n\t"
Packit 6c4009
			  "ldr	s1, %2\n\t"
Packit 6c4009
			  "fdiv	s0, s0, s1\n\t"
Packit 6c4009
			  "mrs	%0, fpsr" : "=r" (fpsr)
Packit 6c4009
			  : "m" (fp_one), "m" (fp_zero)
Packit 6c4009
			  : "d0", "d1");
Packit 6c4009
Packit 6c4009
  if (FE_OVERFLOW & excepts)
Packit 6c4009
    /* There's no way to raise overflow without also raising inexact.  */
Packit 6c4009
    __asm__ __volatile__ (
Packit 6c4009
			  "ldr	s0, %1\n\t"
Packit 6c4009
			  "ldr	s1, %2\n\t"
Packit 6c4009
			  "fadd s0, s0, s1\n\t"
Packit 6c4009
			  "mrs	%0, fpsr" : "=r" (fpsr)
Packit 6c4009
			  : "m" (fp_max), "m" (fp_1e32)
Packit 6c4009
			  : "d0", "d1");
Packit 6c4009
Packit 6c4009
  if (FE_UNDERFLOW & excepts)
Packit 6c4009
    __asm__ __volatile__ (
Packit 6c4009
			  "ldr	s0, %1\n\t"
Packit 6c4009
			  "ldr	s1, %2\n\t"
Packit 6c4009
			  "fdiv s0, s0, s1\n\t"
Packit 6c4009
			  "mrs	%0, fpsr" : "=r" (fpsr)
Packit 6c4009
			  : "m" (fp_min), "m" (fp_three)
Packit 6c4009
			  : "d0", "d1");
Packit 6c4009
Packit 6c4009
  if (FE_INEXACT & excepts)
Packit 6c4009
    __asm__ __volatile__ (
Packit 6c4009
			  "ldr	s0, %1\n\t"
Packit 6c4009
			  "ldr	s1, %2\n\t"
Packit 6c4009
			  "fdiv s0, s0, s1\n\t"
Packit 6c4009
			  "mrs	%0, fpsr" : "=r" (fpsr)
Packit 6c4009
			  : "m" (fp_two), "m" (fp_three)
Packit 6c4009
			  : "d0", "d1");
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
libm_hidden_def (__feraiseexcept)
Packit 6c4009
weak_alias (__feraiseexcept, feraiseexcept)
Packit 6c4009
libm_hidden_weak (feraiseexcept)