mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-26 23:10:06 +00:00
6cac323c8d
This patches consolidates all the powerpc ceil{f} implementations on the generic sysdeps/powerpc/fpu/s_ceil{f}. The generic implementation uses either the compiler builts for ISA 2.03+ (which generates the frip instruction) or a generic implementation which uses FP only operations. It adds a generic implementation (round_to_integer.h) which is shared with other rounding to integer routines. The resulting code should be similar in term os performance to previous assembly one. The IFUNC organization for powerpc64 is also change to be enabled only for powerpc64 and not for powerpc64le (since minium ISA of 2.08 does not require the fallback generic implementation). Checked on powerpc-linux-gnu (built without --with-cpu, with --with-cpu=power4 and with --with-cpu=power5+ and --disable-multi-arch), powerpc64-linux-gnu (built without --with-cp and with --with-cpu=power5+ and --disable-multi-arch). * sysdeps/powerpc/fpu/fenv_libc.h (__fesetround_inline_nocheck): New function. * sysdeps/powerpc/fpu/round_to_integer.h: New file. * sysdeps/powerpc/fpu/s_ceil.c: Likewise. * sysdeps/powerpc/fpu/s_ceilf.c: Likewise. * sysdeps/powerpc/powerpc32/fpu/s_ceil.S: Remove file. * sysdeps/powerpc/powerpc32/fpu/s_ceilf.S: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/Makefile (CFLAGS-s_ceil-power5+.c, CFLAGS-s_ceilf-power5+.c): New rule. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceil-power5+.S: Remove file. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceil-ppc32.S: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceilf-power5+.S: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceilf-ppc32.S: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceil-power5+.c: New file. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceil-ppc32.c: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceilf-power5+.c: Likewise. * sysdeps/powerpc/powerpc32/power4/fpu/multiarch/s_ceilf-ppc32.c: Likewise. * sysdeps/powerpc/powerpc32/power5+/fpu/s_ceil.S: Remove file. * sysdeps/powerpc/powerpc32/power5+/fpu/s_ceilf.S: Likewise. * sysdeps/powerpc/powerpc64/be/fpu/multiarch/Makefile: New file. * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceil-power5+.c: Likewise. * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceil-ppc64.c: Likewise. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceil.c: Move to ... * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceil.c: ... here. * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceilf-power5+.c: New file. * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceilf-ppc64.c: Likewise. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceilf.c: Move to ... * sysdeps/powerpc/powerpc64/be/fpu/multiarch/s_ceilf.c: ... * here. * sysdeps/powerpc/powerpc64/fpu/multiarch/Makefile (libm-sysdep_routines): Remove s_ceil-power5+, s_ceil-ppc64, s_ceilf-power5+, and s_ceilf-ppc64. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceil-power5+.S: Remove file. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceil-ppc64.S: Likewise. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceilf-power5+.S: Likewise. * sysdeps/powerpc/powerpc64/fpu/multiarch/s_ceilf-ppc64.S: Likewise. * sysdeps/powerpc/powerpc64/fpu/s_ceil.S: Likewise. * sysdeps/powerpc/powerpc64/fpu/s_ceilf.S: Likewise. * sysdeps/powerpc/powerpc64/power5+/fpu/s_ceil.S: Likewise. * sysdeps/powerpc/powerpc64/power5+/fpu/s_ceilf.S: Likewise. Reviewed-by: Gabriel F. T. Gomes <gabriel@inconstante.eti.br>
106 lines
2.3 KiB
C
106 lines
2.3 KiB
C
/* Round to integer generic implementation.
|
|
Copyright (C) 2019 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public License as
|
|
published by the Free Software Foundation; either version 2 of the
|
|
License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with the GNU C Library; see the file COPYING.LIB. If
|
|
not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#ifndef _ROUND_TO_INTEGER_H
|
|
#define _ROUND_TO_INTEGER_H
|
|
|
|
#include <fenv_private.h>
|
|
|
|
enum round_mode
|
|
{
|
|
CEIL
|
|
};
|
|
|
|
static inline void
|
|
set_fenv_mode (enum round_mode mode)
|
|
{
|
|
int rmode;
|
|
switch (mode)
|
|
{
|
|
case CEIL: rmode = FE_UPWARD; break;
|
|
default: rmode = FE_TONEAREST; break;
|
|
}
|
|
__fesetround_inline_nocheck (rmode);
|
|
}
|
|
|
|
static inline float
|
|
round_to_integer_float (enum round_mode mode, float x)
|
|
{
|
|
/* Ensure sNaN input is converted to qNaN. */
|
|
if (__glibc_unlikely (isnan (x)))
|
|
return x + x;
|
|
|
|
if (fabs (x) > 0x1p+23)
|
|
return x;
|
|
|
|
float r = x;
|
|
|
|
/* Save current FPU rounding mode and inexact state. */
|
|
fenv_t fe = fegetenv_register ();
|
|
set_fenv_mode (mode);
|
|
if (x > 0.0)
|
|
{
|
|
r += 0x1p+23;
|
|
r -= 0x1p+23;
|
|
r = fabs (r);
|
|
}
|
|
else if (x < 0.0)
|
|
{
|
|
r -= 0x1p+23;
|
|
r += 0x1p+23;
|
|
r = -fabs (r);
|
|
}
|
|
__builtin_mtfsf (0xff, fe);
|
|
|
|
return r;
|
|
}
|
|
|
|
static inline double
|
|
round_to_integer_double (enum round_mode mode, double x)
|
|
{
|
|
/* Ensure sNaN input is converted to qNaN. */
|
|
if (__glibc_unlikely (isnan (x)))
|
|
return x + x;
|
|
|
|
if (fabs (x) > 0x1p+52)
|
|
return x;
|
|
|
|
double r = x;
|
|
|
|
/* Save current FPU rounding mode and inexact state. */
|
|
fenv_t fe = fegetenv_register ();
|
|
set_fenv_mode (mode);
|
|
if (x > 0.0)
|
|
{
|
|
r += 0x1p+52;
|
|
r -= 0x1p+52;
|
|
r = fabs (r);
|
|
}
|
|
else if (x < 0.0)
|
|
{
|
|
r -= 0x1p+52;
|
|
r += 0x1p+52;
|
|
r = -fabs (r);
|
|
}
|
|
__builtin_mtfsf (0xff, fe);
|
|
|
|
return r;
|
|
}
|
|
|
|
#endif
|