Module: wine Branch: master Commit: 117faee955919598c85c575139fd0fc903364db4 URL: http://source.winehq.org/git/wine.git/?a=commit;h=117faee955919598c85c575139...
Author: Piotr Caban piotr@codeweavers.com Date: Sun May 9 15:07:33 2010 +0200
msvcrt: Fix strtod and wcstod implementation.
---
dlls/msvcrt/msvcrt.h | 2 ++ dlls/msvcrt/string.c | 7 +++++++ dlls/msvcrt/wcs.c | 7 +++++++ 3 files changed, 16 insertions(+), 0 deletions(-)
diff --git a/dlls/msvcrt/msvcrt.h b/dlls/msvcrt/msvcrt.h index 8d66eef..ed11d35 100644 --- a/dlls/msvcrt/msvcrt.h +++ b/dlls/msvcrt/msvcrt.h @@ -170,6 +170,8 @@ extern void msvcrt_free_signals(void);
extern unsigned msvcrt_create_io_inherit_block(WORD*, BYTE**);
+extern unsigned int __cdecl _control87(unsigned int, unsigned int); + /* run-time error codes */ #define _RT_STACK 0 #define _RT_NULLPTR 1 diff --git a/dlls/msvcrt/string.c b/dlls/msvcrt/string.c index e1d9557..21dc29f 100644 --- a/dlls/msvcrt/string.c +++ b/dlls/msvcrt/string.c @@ -164,6 +164,7 @@ void CDECL MSVCRT__swab(char* src, char* dst, int len) double CDECL MSVCRT_strtod_l( const char *str, char **end, MSVCRT__locale_t locale) { unsigned __int64 d=0, hlp; + unsigned fpcontrol; int exp=0, sign=1; const char *p; double ret; @@ -249,11 +250,17 @@ double CDECL MSVCRT_strtod_l( const char *str, char **end, MSVCRT__locale_t loca } }
+ fpcontrol = _control87(0, 0); + _control87(MSVCRT__EM_DENORMAL|MSVCRT__EM_INVALID|MSVCRT__EM_ZERODIVIDE + |MSVCRT__EM_OVERFLOW|MSVCRT__EM_UNDERFLOW|MSVCRT__EM_INEXACT, 0xffffffff); + if(exp>0) ret = (double)sign*d*pow(10, exp); else ret = (double)sign*d/pow(10, -exp);
+ _control87(fpcontrol, 0xffffffff); + if((d && ret==0.0) || isinf(ret)) *MSVCRT__errno() = MSVCRT_ERANGE;
diff --git a/dlls/msvcrt/wcs.c b/dlls/msvcrt/wcs.c index 96a25e5..79be91a 100644 --- a/dlls/msvcrt/wcs.c +++ b/dlls/msvcrt/wcs.c @@ -129,6 +129,7 @@ double CDECL MSVCRT__wcstod_l(const MSVCRT_wchar_t* str, MSVCRT_wchar_t** end, MSVCRT__locale_t locale) { unsigned __int64 d=0, hlp; + unsigned fpcontrol; int exp=0, sign=1; const MSVCRT_wchar_t *p; double ret; @@ -212,11 +213,17 @@ double CDECL MSVCRT__wcstod_l(const MSVCRT_wchar_t* str, MSVCRT_wchar_t** end, } }
+ fpcontrol = _control87(0, 0); + _control87(MSVCRT__EM_DENORMAL|MSVCRT__EM_INVALID|MSVCRT__EM_ZERODIVIDE + |MSVCRT__EM_OVERFLOW|MSVCRT__EM_UNDERFLOW|MSVCRT__EM_INEXACT, 0xffffffff); + if(exp>0) ret = (double)sign*d*pow(10, exp); else ret = (double)sign*d/pow(10, -exp);
+ _control87(fpcontrol, 0xffffffff); + if((d && ret==0.0) || isinf(ret)) *MSVCRT__errno() = MSVCRT_ERANGE;