mirror of
git://sourceware.org/git/glibc.git
synced 2024-12-21 04:31:04 +08:00
107e6a3c22
Support usable check for all CPU features with the following changes: 1. Change struct cpu_features to struct cpuid_features { struct cpuid_registers cpuid; struct cpuid_registers usable; }; struct cpu_features { struct cpu_features_basic basic; struct cpuid_features features[COMMON_CPUID_INDEX_MAX]; unsigned int preferred[PREFERRED_FEATURE_INDEX_MAX]; ... }; so that there is a usable bit for each cpuid bit. 2. After the cpuid bits have been initialized, copy the known bits to the usable bits. EAX/EBX from INDEX_1 and EAX from INDEX_7 aren't used for CPU feature detection. 3. Clear the usable bits which require OS support. 4. If the feature is supported by OS, copy its cpuid bit to its usable bit. 5. Replace HAS_CPU_FEATURE and CPU_FEATURES_CPU_P with CPU_FEATURE_USABLE and CPU_FEATURE_USABLE_P to check if a feature is usable. 6. Add DEPR_FPU_CS_DS for INDEX_7_EBX_13. 7. Unset MPX feature since it has been deprecated. The results are 1. If the feature is known and doesn't requre OS support, its usable bit is copied from the cpuid bit. 2. Otherwise, its usable bit is copied from the cpuid bit only if the feature is known to supported by OS. 3. CPU_FEATURE_USABLE/CPU_FEATURE_USABLE_P are used to check if the feature can be used. 4. HAS_CPU_FEATURE/CPU_FEATURE_CPU_P are used to check if CPU supports the feature.
70 lines
2.3 KiB
C
70 lines
2.3 KiB
C
/* Set floating-point environment exception handling.
|
|
Copyright (C) 1997-2020 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
|
|
|
|
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
|
|
<https://www.gnu.org/licenses/>. */
|
|
|
|
#include <fenv.h>
|
|
#include <math.h>
|
|
#include <unistd.h>
|
|
#include <ldsodefs.h>
|
|
#include <dl-procinfo.h>
|
|
|
|
int
|
|
__fesetexceptflag (const fexcept_t *flagp, int excepts)
|
|
{
|
|
fenv_t temp;
|
|
|
|
/* Get the current environment. We have to do this since we cannot
|
|
separately set the status word. */
|
|
__asm__ ("fnstenv %0" : "=m" (*&temp));
|
|
|
|
temp.__status_word &= ~(excepts & FE_ALL_EXCEPT);
|
|
temp.__status_word |= *flagp & excepts & FE_ALL_EXCEPT;
|
|
|
|
/* Store the new status word (along with the rest of the environment.
|
|
Possibly new exceptions are set but they won't get executed unless
|
|
the next floating-point instruction. */
|
|
__asm__ ("fldenv %0" : : "m" (*&temp));
|
|
|
|
/* If the CPU supports SSE, we set the MXCSR as well. */
|
|
if (CPU_FEATURE_USABLE (SSE))
|
|
{
|
|
unsigned int xnew_exc;
|
|
|
|
/* Get the current MXCSR. */
|
|
__asm__ ("stmxcsr %0" : "=m" (*&xnew_exc));
|
|
|
|
/* Set the relevant bits. */
|
|
xnew_exc &= ~(excepts & FE_ALL_EXCEPT);
|
|
xnew_exc |= *flagp & excepts & FE_ALL_EXCEPT;
|
|
|
|
/* Put the new data in effect. */
|
|
__asm__ ("ldmxcsr %0" : : "m" (*&xnew_exc));
|
|
}
|
|
|
|
/* Success. */
|
|
return 0;
|
|
}
|
|
|
|
#include <shlib-compat.h>
|
|
#if SHLIB_COMPAT (libm, GLIBC_2_1, GLIBC_2_2)
|
|
strong_alias (__fesetexceptflag, __old_fesetexceptflag)
|
|
compat_symbol (libm, __old_fesetexceptflag, fesetexceptflag, GLIBC_2_1);
|
|
#endif
|
|
|
|
versioned_symbol (libm, __fesetexceptflag, fesetexceptflag, GLIBC_2_2);
|