diff options
author | Maciej W. Rozycki <macro@codesourcery.com> | 2013-09-09 22:36:57 +0100 |
---|---|---|
committer | Maciej W. Rozycki <macro@codesourcery.com> | 2013-09-09 22:36:57 +0100 |
commit | 95e7cf295e449c0aec3d7a76f01907bf300106bf (patch) | |
tree | aa626c4fbf2f591b73d27caf79c74c796d1dc672 /math | |
parent | a9f5ce099cb30fe6b8f2ba240cacffe7ecfbfef2 (diff) | |
download | glibc-95e7cf295e449c0aec3d7a76f01907bf300106bf.tar.gz glibc-95e7cf295e449c0aec3d7a76f01907bf300106bf.tar.xz glibc-95e7cf295e449c0aec3d7a76f01907bf300106bf.zip |
Fix static-binary lazy FPU context allocation
Long ago static startup did not parse the auxiliary vector and therefore could not get at any `AT_FPUCW' tag to check whether upon FPU context allocation the kernel would use a FPU control word setting different to that provided by the `__fpu_control' variable. Static startup therefore always initialized the FPU control word, forcing immediate FPU context allocation even for binaries that otherwise never used the FPU. As from GIT commit f8f900ecb9096ec47f5b7bb7626e29223c69061a static startup supports parsing the auxiliary vector, so now it can avoid explicit initialization of the FPU control word, just as can dynamic startup, in the usual case where the setting written to the FPU control word would be the same as the kernel uses. This defers FPU context allocation until the binary itself actually pokes at the FPU. Note that the `AT_FPUCW' tag is usually absent from the auxiliary vector in which case _FPU_DEFAULT is assumed to be the kernel default.
Diffstat (limited to 'math')
-rw-r--r-- | math/Makefile | 11 | ||||
-rw-r--r-- | math/test-fpucw-ieee-static.c | 1 | ||||
-rw-r--r-- | math/test-fpucw-ieee.c | 24 | ||||
-rw-r--r-- | math/test-fpucw-static.c | 1 | ||||
-rw-r--r-- | math/test-fpucw.c | 10 |
5 files changed, 40 insertions, 7 deletions
diff --git a/math/Makefile b/math/Makefile index 3ed78fc9c3..a9bd49baee 100644 --- a/math/Makefile +++ b/math/Makefile @@ -87,9 +87,11 @@ long-c-yes = $(calls:=l) # Rules for the test suite. tests = test-matherr test-fenv atest-exp atest-sincos atest-exp2 basic-test \ - test-misc test-fpucw tst-definitions test-tgmath test-tgmath-ret \ - bug-nextafter bug-nexttoward bug-tgmath1 test-tgmath-int \ - test-tgmath2 test-powl tst-CMPLX tst-CMPLX2 test-snan + test-misc test-fpucw test-fpucw-ieee tst-definitions test-tgmath \ + test-tgmath-ret bug-nextafter bug-nexttoward bug-tgmath1 \ + test-tgmath-int test-tgmath2 test-powl tst-CMPLX tst-CMPLX2 test-snan \ + $(tests-static) +tests-static = test-fpucw-static test-fpucw-ieee-static # We do the `long double' tests only if this data type is available and # distinct from `double'. test-longdouble-yes = test-ldouble test-ildoubl @@ -217,7 +219,8 @@ $(objpfx)libieee.a: $(objpfx)ieee-math.o $(LN_S) $(<F) $(@F) ifeq ($(build-shared),yes) -$(addprefix $(objpfx),$(tests)): $(objpfx)libm.so$(libm.so-version) +$(addprefix $(objpfx),$(filter-out $(tests-static),$(tests))): $(objpfx)libm.so$(libm.so-version) +$(addprefix $(objpfx),$(tests-static)): $(objpfx)libm.a else $(addprefix $(objpfx),$(tests)): $(objpfx)libm.a endif diff --git a/math/test-fpucw-ieee-static.c b/math/test-fpucw-ieee-static.c new file mode 100644 index 0000000000..3a35542060 --- /dev/null +++ b/math/test-fpucw-ieee-static.c @@ -0,0 +1 @@ +#include "test-fpucw-ieee.c" diff --git a/math/test-fpucw-ieee.c b/math/test-fpucw-ieee.c new file mode 100644 index 0000000000..ae5fc73f6e --- /dev/null +++ b/math/test-fpucw-ieee.c @@ -0,0 +1,24 @@ +/* FPU control word overridden initialization test. + Copyright (C) 2013 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 Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 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 + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with the GNU C Library; if not, see + <http://www.gnu.org/licenses/>. */ + +#define FPU_CONTROL _FPU_IEEE + +#include "test-fpucw.c" + +/* Preempt the library's definition of `__fpu_control'. */ +fpu_control_t __fpu_control = _FPU_IEEE; diff --git a/math/test-fpucw-static.c b/math/test-fpucw-static.c new file mode 100644 index 0000000000..bc1cfa8ce5 --- /dev/null +++ b/math/test-fpucw-static.c @@ -0,0 +1 @@ +#include "test-fpucw.c" diff --git a/math/test-fpucw.c b/math/test-fpucw.c index 1961a67481..ae6a2c9a71 100644 --- a/math/test-fpucw.c +++ b/math/test-fpucw.c @@ -19,6 +19,10 @@ #include <fpu_control.h> #include <stdio.h> +#ifndef FPU_CONTROL +# define FPU_CONTROL _FPU_DEFAULT +#endif + int main (void) { @@ -30,11 +34,11 @@ main (void) cw &= ~_FPU_RESERVED; - if (cw != (_FPU_DEFAULT & ~_FPU_RESERVED)) + if (cw != (FPU_CONTROL & ~_FPU_RESERVED)) printf ("control word is 0x%lx but should be 0x%lx.\n", - (long int) cw, (long int) (_FPU_DEFAULT & ~_FPU_RESERVED)); + (long int) cw, (long int) (FPU_CONTROL & ~_FPU_RESERVED)); - return cw != (_FPU_DEFAULT & ~_FPU_RESERVED); + return cw != (FPU_CONTROL & ~_FPU_RESERVED); #else return 0; |