[newlib-cygwin/fp-zeuch] Cygwin: initialize FP environment for each applicaiton thread
Corinna Vinschen
corinna@sourceware.org
Thu Feb 28 14:14:00 GMT 2019
https://sourceware.org/git/gitweb.cgi?p=newlib-cygwin.git;h=76a9cb772b6d8e98dc2fc8ef1ea197e0851e1037
commit 76a9cb772b6d8e98dc2fc8ef1ea197e0851e1037
Author: Corinna Vinschen <corinna@vinschen.de>
Date: Thu Feb 28 15:13:19 2019 +0100
Cygwin: initialize FP environment for each applicaiton thread
Also, initialize FP environment in timerfd thread before calling
RegisterClassW to avoid a spurious STATUS_FLOAT_INEXACT_RESULT
from msvcrt.dll.
Signed-off-by: Corinna Vinschen <corinna@vinschen.de>
Diff:
---
winsup/cygwin/dcrt0.cc | 2 +-
winsup/cygwin/fenv.cc | 22 +++++++++++++---------
winsup/cygwin/include/fenv.h | 2 +-
winsup/cygwin/miscfuncs.cc | 3 +++
winsup/cygwin/timerfd.cc | 13 ++++---------
5 files changed, 22 insertions(+), 20 deletions(-)
diff --git a/winsup/cygwin/dcrt0.cc b/winsup/cygwin/dcrt0.cc
index 11edcdf..bc7eeb4 100644
--- a/winsup/cygwin/dcrt0.cc
+++ b/winsup/cygwin/dcrt0.cc
@@ -1079,7 +1079,7 @@ _dll_crt0 ()
fork_info->alloc_stack ();
#endif
- _feinitialise ();
+ _feinitialise (true);
_main_tls = &_my_tls;
_main_tls->call ((DWORD (*) (void *, void *)) dll_crt0_1, NULL);
}
diff --git a/winsup/cygwin/fenv.cc b/winsup/cygwin/fenv.cc
index 396f98a..a2c8359 100644
--- a/winsup/cygwin/fenv.cc
+++ b/winsup/cygwin/fenv.cc
@@ -420,7 +420,7 @@ fesetprec (int prec)
/* Set up the FPU and SSE environment at the start of execution. */
void
-_feinitialise (void)
+_feinitialise (bool pre_main)
{
unsigned int edx, eax;
@@ -442,14 +442,18 @@ _feinitialise (void)
if (use_sse)
__asm__ volatile ("ldmxcsr %0" :: "m" (mxcsr));
- /* Setup unmasked environment, but leave __FE_DENORM masked. */
- feenableexcept (FE_ALL_EXCEPT);
- fegetenv (&fe_nomask_env);
+ if (pre_main)
+ {
+ /* Setup unmasked environment, but leave __FE_DENORM masked. */
+ feenableexcept (FE_ALL_EXCEPT);
+ fegetenv (&fe_nomask_env);
- /* Restore default exception masking (all masked). */
- fedisableexcept (FE_ALL_EXCEPT);
+ /* Restore default exception masking (all masked). */
+ fedisableexcept (FE_ALL_EXCEPT);
- /* Finally cache state as default environment. */
- fegetenv (&fe_dfl_env);
+ /* Finally cache state as default environment. */
+ fegetenv (&fe_dfl_env);
+ }
+ else
+ fedisableexcept (FE_ALL_EXCEPT);
}
-
diff --git a/winsup/cygwin/include/fenv.h b/winsup/cygwin/include/fenv.h
index a355c47..3c9b538 100644
--- a/winsup/cygwin/include/fenv.h
+++ b/winsup/cygwin/include/fenv.h
@@ -168,7 +168,7 @@ extern int fesetprec (int __prec);
#ifdef __INSIDE_CYGWIN__
/* This is Cygwin-custom, not from the standard, for use in the Cygwin CRT. */
-extern void _feinitialise ();
+extern void _feinitialise (bool);
#endif
#ifdef __cplusplus
diff --git a/winsup/cygwin/miscfuncs.cc b/winsup/cygwin/miscfuncs.cc
index b5dfffc..ed32a50 100644
--- a/winsup/cygwin/miscfuncs.cc
+++ b/winsup/cygwin/miscfuncs.cc
@@ -16,6 +16,7 @@ details. */
#include "fhandler.h"
#include "exception.h"
#include "tls_pbuf.h"
+#include "fenv.h"
int __reg2
check_invalid_virtual_addr (const void *s, unsigned sz)
@@ -397,6 +398,8 @@ pthread_wrapper (PVOID arg)
wrapper_arg.guardsize -= wincap.page_size ();
SetThreadStackGuarantee (&wrapper_arg.guardsize);
}
+ /* FP initialization. */
+ _feinitialise (false);
/* Initialize new _cygtls. */
_my_tls.init_thread (wrapper_arg.stackbase - CYGTLS_PADSIZE,
(DWORD (*)(void*, void*)) wrapper_arg.func);
diff --git a/winsup/cygwin/timerfd.cc b/winsup/cygwin/timerfd.cc
index 8e4c94e..c4faf30 100644
--- a/winsup/cygwin/timerfd.cc
+++ b/winsup/cygwin/timerfd.cc
@@ -15,6 +15,7 @@ details. */
#include "cygerrno.h"
#include <sys/timerfd.h>
#include "timerfd.h"
+#include "fenv.h"
#define TFD_CANCEL_FLAGS (TFD_TIMER_ABSTIME | TFD_TIMER_CANCEL_ON_SET)
@@ -32,15 +33,9 @@ timerfd_tracker::create_timechange_window ()
wclass.lpfnWndProc = DefWindowProcW;
wclass.hInstance = user_data->hmodule;
wclass.lpszClassName = cname;
- /* This sleep is required on Windows 10 64 bit only, and only when running
- under strace. One of the child processes inheriting the timerfd
- descriptor will get a STATUS_FLOAT_INEXACT_RESULT exception inside of
- msvcrt.dll. While this is completely crazy in itself, it's apparently
- some timing problem. It occurs in 4 out of 5 runs under strace only.
- The sleep is required before calling RegisterClassW. Moving it before
- CreateWindowExW does not work. What the heck? */
- if (being_debugged ())
- Sleep (1L);
+ /* Avoid a potential STATUS_FLOAT_INEXACT_RESULT in msvcrt.dll.
+ Only observed on x86_64 W10 under strace. */
+ _feinitialise (false);
atom = RegisterClassW (&wclass);
if (!atom)
debug_printf ("RegisterClass %E");
More information about the Cygwin-cvs
mailing list