/* * Copyright 2016 Nikolay Sivov for CodeWeavers * * This 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. * * This 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 this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "wine/test.h" #define DEFINE_EXPECT(func) \ static BOOL expect_ ## func = FALSE, called_ ## func = FALSE #define SET_EXPECT(func) \ expect_ ## func = TRUE #define CHECK_EXPECT2(func) \ do { \ ok(expect_ ##func, "unexpected call " #func "\n"); \ called_ ## func = TRUE; \ }while(0) #define CHECK_EXPECT(func) \ do { \ CHECK_EXPECT2(func); \ expect_ ## func = FALSE; \ }while(0) #define CHECK_CALLED(func) \ do { \ ok(called_ ## func, "expected " #func "\n"); \ expect_ ## func = called_ ## func = FALSE; \ }while(0) static inline double __port_min_pos_double(void) { static const UINT64 __min_pos_double = 0x10000000000000; return *(const double *)&__min_pos_double; } static inline double __port_max_double(void) { static const UINT64 __max_double = 0x7FEFFFFFFFFFFFFF; return *(const double *)&__max_double; } DEFINE_EXPECT(global_invalid_parameter_handler); DEFINE_EXPECT(thread_invalid_parameter_handler); typedef struct { const char *short_wday[7]; const char *wday[7]; const char *short_mon[12]; const char *mon[12]; const char *am; const char *pm; const char *short_date; const char *date; const char *time; int unk; int refcount; const wchar_t *short_wdayW[7]; const wchar_t *wdayW[7]; const wchar_t *short_monW[12]; const wchar_t *monW[12]; const wchar_t *amW; const wchar_t *pmW; const wchar_t *short_dateW; const wchar_t *dateW; const wchar_t *timeW; const wchar_t *locnameW; } __lc_time_data; typedef void (__cdecl *_se_translator_function)(unsigned int code, struct _EXCEPTION_POINTERS *info); static LONGLONG crt_init_end; _ACRTIMP int __cdecl _o__initialize_onexit_table(_onexit_table_t *table); _ACRTIMP int __cdecl _o__register_onexit_function(_onexit_table_t *table, _onexit_t func); _ACRTIMP int __cdecl _o__execute_onexit_table(_onexit_table_t *table); _ACRTIMP void *__cdecl _o_malloc(size_t); _se_translator_function __cdecl _set_se_translator(_se_translator_function func); void** __cdecl __current_exception(void); int* __cdecl __processing_throw(void); static void test__initialize_onexit_table(void) { _onexit_table_t table, table2; int ret; ret = _initialize_onexit_table(NULL); ok(ret == -1, "got %d\n", ret); memset(&table, 0, sizeof(table)); ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(table._first == table._last && table._first == table._end, "got first %p, last %p, end %p\n", table._first, table._last, table._end); memset(&table2, 0, sizeof(table2)); ret = _initialize_onexit_table(&table2); ok(ret == 0, "got %d\n", ret); ok(table2._first == table._first, "got %p, %p\n", table2._first, table._first); ok(table2._last == table._last, "got %p, %p\n", table2._last, table._last); ok(table2._end == table._end, "got %p, %p\n", table2._end, table._end); memset(&table2, 0, sizeof(table2)); ret = _o__initialize_onexit_table(&table2); ok(ret == 0, "got %d\n", ret); ok(table2._first == table._first, "got %p, %p\n", table2._first, table._first); ok(table2._last == table._last, "got %p, %p\n", table2._last, table._last); ok(table2._end == table._end, "got %p, %p\n", table2._end, table._end); /* uninitialized table */ table._first = table._last = table._end = (void*)0x123; ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(table._first == table._last && table._first == table._end, "got first %p, last %p, end %p\n", table._first, table._last, table._end); ok(table._first != (void*)0x123, "got %p\n", table._first); table._first = (void*)0x123; table._last = (void*)0x456; table._end = (void*)0x123; ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(table._first == table._last && table._first == table._end, "got first %p, last %p, end %p\n", table._first, table._last, table._end); ok(table._first != (void*)0x123, "got %p\n", table._first); table._first = (void*)0x123; table._last = (void*)0x456; table._end = (void*)0x789; ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(table._first == (void*)0x123, "got %p\n", table._first); ok(table._last == (void*)0x456, "got %p\n", table._last); ok(table._end == (void*)0x789, "got %p\n", table._end); table._first = NULL; table._last = (void*)0x456; table._end = NULL; ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(table._first == table._last && table._first == table._end, "got first %p, last %p, end %p\n", table._first, table._last, table._end); } static int g_onexit_called; static int CDECL onexit_func(void) { g_onexit_called++; return 0; } static int CDECL onexit_func2(void) { ok(g_onexit_called == 0, "got %d\n", g_onexit_called); g_onexit_called++; return 0; } static void test__register_onexit_function(void) { _onexit_table_t table; _PVFV *f; int ret; memset(&table, 0, sizeof(table)); ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(NULL, NULL); ok(ret == -1, "got %d\n", ret); ret = _register_onexit_function(NULL, onexit_func); ok(ret == -1, "got %d\n", ret); f = table._last; ret = _register_onexit_function(&table, NULL); ok(ret == 0, "got %d\n", ret); ok(f != table._last, "got %p, initial %p\n", table._last, f); ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); f = table._last; ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ok(f != table._last, "got %p, initial %p\n", table._last, f); f = table._last; ret = _o__register_onexit_function(&table, NULL); ok(ret == 0, "got %d\n", ret); ok(f != table._last, "got %p, initial %p\n", table._last, f); f = table._last; ret = _o__register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ok(f != table._last, "got %p, initial %p\n", table._last, f); ret = _execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); } static void test__execute_onexit_table(void) { _onexit_table_t table; int ret; ret = _execute_onexit_table(NULL); ok(ret == -1, "got %d\n", ret); memset(&table, 0, sizeof(table)); ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); /* execute empty table */ ret = _execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); /* same function registered multiple times */ ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, NULL); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ret = _o__register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ok(table._first != table._end, "got %p, %p\n", table._first, table._end); g_onexit_called = 0; ret = _execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(g_onexit_called == 3, "got %d\n", g_onexit_called); ok(table._first == table._end, "got %p, %p\n", table._first, table._end); ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, NULL); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ret = _o__register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ok(table._first != table._end, "got %p, %p\n", table._first, table._end); g_onexit_called = 0; ret = _o__execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(g_onexit_called == 3, "got %d\n", g_onexit_called); ok(table._first == table._end, "got %p, %p\n", table._first, table._end); /* execute again, table is already empty */ g_onexit_called = 0; ret = _execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(g_onexit_called == 0, "got %d\n", g_onexit_called); /* check call order */ memset(&table, 0, sizeof(table)); ret = _initialize_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, onexit_func); ok(ret == 0, "got %d\n", ret); ret = _register_onexit_function(&table, onexit_func2); ok(ret == 0, "got %d\n", ret); g_onexit_called = 0; ret = _execute_onexit_table(&table); ok(ret == 0, "got %d\n", ret); ok(g_onexit_called == 2, "got %d\n", g_onexit_called); } static void test___fpe_flt_rounds(void) { unsigned int cfp = _controlfp(0, 0); int ret; if(!cfp) { skip("_controlfp not supported\n"); return; } ok((_controlfp(_RC_NEAR, _RC_CHOP) & _RC_CHOP) == _RC_NEAR, "_controlfp(_RC_NEAR, _RC_CHOP) failed\n"); ret = __fpe_flt_rounds(); ok(ret == 1, "__fpe_flt_rounds returned %d\n", ret); ok((_controlfp(_RC_UP, _RC_CHOP) & _RC_CHOP) == _RC_UP, "_controlfp(_RC_UP, _RC_CHOP) failed\n"); ret = __fpe_flt_rounds(); ok(ret == 2 || broken(ret == 3) /* w1064v1507 */, "__fpe_flt_rounds returned %d\n", ret); ok((_controlfp(_RC_DOWN, _RC_CHOP) & _RC_CHOP) == _RC_DOWN, "_controlfp(_RC_DOWN, _RC_CHOP) failed\n"); ret = __fpe_flt_rounds(); ok(ret == 3 || broken(ret == 2) /* w1064v1507 */, "__fpe_flt_rounds returned %d\n", ret); ok((_controlfp(_RC_CHOP, _RC_CHOP) & _RC_CHOP) == _RC_CHOP, "_controlfp(_RC_CHOP, _RC_CHOP) failed\n"); ret = __fpe_flt_rounds(); ok(ret == 0, "__fpe_flt_rounds returned %d\n", ret); _controlfp(cfp, _MCW_EM | _MCW_RC | _MCW_PC); } static void test__control87_2(void) { #ifdef __i386__ unsigned int x86_cw_init, sse2_cw_init, x86_cw, sse2_cw, r; r = __control87_2(0, 0, &x86_cw_init, &sse2_cw_init); ok(r == 1, "__control87_2 returned %d\n", r); r = __control87_2(0, _EM_INVALID, &x86_cw, NULL); ok(r == 1, "__control87_2 returned %d\n", r); ok(x86_cw == (x86_cw_init & ~_EM_INVALID), "x86_cw = %x, x86_cw_init = %x\n", x86_cw, x86_cw_init); r = __control87_2(0, 0, &x86_cw, &sse2_cw); ok(r == 1, "__control87_2 returned %d\n", r); ok(x86_cw == (x86_cw_init & ~_EM_INVALID), "x86_cw = %x, x86_cw_init = %x\n", x86_cw, x86_cw_init); ok(sse2_cw == sse2_cw_init, "sse2_cw = %x, sse2_cw_init = %x\n", sse2_cw, sse2_cw_init); r = _control87(0, 0); ok(r == (x86_cw | sse2_cw | _EM_AMBIGUOUS), "r = %x, expected %x\n", r, x86_cw | sse2_cw | _EM_AMBIGUOUS); _control87(x86_cw_init, ~0); #endif } static void __cdecl global_invalid_parameter_handler( const wchar_t *expression, const wchar_t *function, const wchar_t *file, unsigned line, uintptr_t arg) { CHECK_EXPECT2(global_invalid_parameter_handler); } static void __cdecl thread_invalid_parameter_handler( const wchar_t *expression, const wchar_t *function, const wchar_t *file, unsigned line, uintptr_t arg) { CHECK_EXPECT(thread_invalid_parameter_handler); } static void test_invalid_parameter_handler(void) { _invalid_parameter_handler ret; ret = _get_invalid_parameter_handler(); ok(!ret, "ret != NULL\n"); ret = _get_thread_local_invalid_parameter_handler(); ok(!ret, "ret != NULL\n"); ret = _set_thread_local_invalid_parameter_handler(thread_invalid_parameter_handler); ok(!ret, "ret != NULL\n"); ret = _get_thread_local_invalid_parameter_handler(); ok(ret == thread_invalid_parameter_handler, "ret = %p\n", ret); ret = _get_invalid_parameter_handler(); ok(!ret, "ret != NULL\n"); ret = _set_invalid_parameter_handler(global_invalid_parameter_handler); ok(!ret, "ret != NULL\n"); ret = _get_invalid_parameter_handler(); ok(ret == global_invalid_parameter_handler, "ret = %p\n", ret); ret = _get_thread_local_invalid_parameter_handler(); ok(ret == thread_invalid_parameter_handler, "ret = %p\n", ret); SET_EXPECT(thread_invalid_parameter_handler); _ltoa_s(0, NULL, 0, 0); CHECK_CALLED(thread_invalid_parameter_handler); ret = _set_thread_local_invalid_parameter_handler(NULL); ok(ret == thread_invalid_parameter_handler, "ret = %p\n", ret); SET_EXPECT(global_invalid_parameter_handler); _ltoa_s(0, NULL, 0, 0); CHECK_CALLED(global_invalid_parameter_handler); ret = _set_invalid_parameter_handler(NULL); ok(ret == global_invalid_parameter_handler, "ret = %p\n", ret); ret = _set_invalid_parameter_handler(global_invalid_parameter_handler); ok(!ret, "ret != NULL\n"); } static void test__get_narrow_winmain_command_line(char *path) { PROCESS_INFORMATION proc; STARTUPINFOA startup; char cmd[MAX_PATH+32]; char *ret, *cmdline, *name; int len; ret = _get_narrow_winmain_command_line(); cmdline = GetCommandLineA(); len = strlen(cmdline); ok(ret>cmdline && ret0; len--) if(path[len-1]=='\\' || path[len-1]=='/') break; if(len) name = path+len; else name = path; sprintf(cmd, "\"\"%c\"\"\"%s\" \t \"misc\" cmd", name[0], name+1); memset(&startup, 0, sizeof(startup)); startup.cb = sizeof(startup); CreateProcessA(path, cmd, NULL, NULL, TRUE, CREATE_DEFAULT_ERROR_MODE|NORMAL_PRIORITY_CLASS, NULL, NULL, &startup, &proc); wait_child_process(proc.hProcess); CloseHandle(proc.hProcess); CloseHandle(proc.hThread); } static void test__sopen_dispatch(void) { int ret, fd; char *tempf; tempf = _tempnam(".", "wne"); fd = 0; ret = _sopen_dispatch(tempf, _O_CREAT, _SH_DENYWR, 0xff, &fd, 0); ok(!ret, "got %d\n", ret); ok(fd > 0, "got fd %d\n", fd); _close(fd); unlink(tempf); SET_EXPECT(global_invalid_parameter_handler); fd = 0; ret = _sopen_dispatch(tempf, _O_CREAT, _SH_DENYWR, 0xff, &fd, 1); ok(ret == EINVAL, "got %d\n", ret); ok(fd == -1, "got fd %d\n", fd); CHECK_CALLED(global_invalid_parameter_handler); if (fd > 0) { _close(fd); unlink(tempf); } free(tempf); } static void test__sopen_s(void) { int ret, fd; char *tempf; tempf = _tempnam(".", "wne"); fd = 0; ret = _sopen_s(&fd, tempf, _O_CREAT, _SH_DENYWR, 0); ok(!ret, "got %d\n", ret); ok(fd > 0, "got fd %d\n", fd); _close(fd); unlink(tempf); /* _open() does not validate pmode */ fd = _open(tempf, _O_CREAT, 0xff); ok(fd > 0, "got fd %d\n", fd); _close(fd); unlink(tempf); /* _sopen_s() invokes invalid parameter handler on invalid pmode */ SET_EXPECT(global_invalid_parameter_handler); fd = 0; ret = _sopen_s(&fd, tempf, _O_CREAT, _SH_DENYWR, 0xff); ok(ret == EINVAL, "got %d\n", ret); ok(fd == -1, "got fd %d\n", fd); CHECK_CALLED(global_invalid_parameter_handler); free(tempf); } static void test_lldiv(void) { lldiv_t r; r = lldiv(((LONGLONG)0x111 << 32) + 0x222, (LONGLONG)1 << 32); ok(r.quot == 0x111, "quot = %s\n", wine_dbgstr_longlong(r.quot)); ok(r.rem == 0x222, "rem = %s\n", wine_dbgstr_longlong(r.rem)); r = lldiv(((LONGLONG)0x69CF0012 << 32) + 0x0033E78A, 0x30); ok(r.quot == ((LONGLONG)0x02345000 << 32) + 0x600114D2, "quot = %s\n", wine_dbgstr_longlong(r.quot)); ok(r.rem == 0x2A, "rem = %s\n", wine_dbgstr_longlong(r.rem)); r = lldiv(((LONGLONG)0x243A5678 << 32) + 0x9ABCDEF0, (LONGLONG)0x12 << 48); ok(r.quot == 0x0203, "quot = %s\n", wine_dbgstr_longlong(r.quot)); ok(r.rem == ((LONGLONG)0x00045678 << 32) + 0x9ABCDEF0, "rem = %s\n", wine_dbgstr_longlong(r.rem)); } static void test_isblank(void) { int c; for(c = 0; c <= 0xff; c++) { if(c == '\t' || c == ' ') { if(c == '\t') ok(!_isctype(c, _BLANK), "tab shouldn't be blank\n"); else ok(_isctype(c, _BLANK), "space should be blank\n"); ok(isblank(c), "%d should be blank\n", c); ok(_isblank_l(c, NULL), "%d should be blank\n", c); } else { ok(!_isctype(c, _BLANK), "%d shouldn't be blank\n", c); ok(!isblank(c), "%d shouldn't be blank\n", c); ok(!_isblank_l(c, NULL), "%d shouldn't be blank\n", c); } } for(c = 0; c <= 0xffff; c++) { if(c == '\t' || c == ' ' || c == 0x3000 || c == 0xfeff) { if(c == '\t') ok(!_iswctype_l(c, _BLANK, NULL), "tab shouldn't be blank\n"); else ok(_iswctype_l(c, _BLANK, NULL), "%d should be blank\n", c); ok(iswblank(c), "%d should be blank\n", c); ok(_iswblank_l(c, NULL), "%d should be blank\n", c); } else { ok(!_iswctype_l(c, _BLANK, NULL), "%d shouldn't be blank\n", c); ok(!iswblank(c), "%d shouldn't be blank\n", c); ok(!_iswblank_l(c, NULL), "%d shouldn't be blank\n", c); } } } static struct _exception exception; static int CDECL matherr_callback(struct _exception *e) { exception = *e; if (!strcmp(e->name, "acos") && e->arg1 == 2) e->retval = -1; return 0; } static void test_math_errors(void) { const struct { char func[16]; double x; int error; int exception; } testsd[] = { {"_logb", -INFINITY, -1, -1}, {"_logb", -1, -1, -1}, {"_logb", 0, ERANGE, _SING}, {"_logb", INFINITY, -1, -1}, {"acos", -INFINITY, EDOM, _DOMAIN}, {"acos", -2, EDOM, _DOMAIN}, {"acos", -1, -1, -1}, {"acos", 1, -1, -1}, {"acos", 2, EDOM, _DOMAIN}, {"acos", INFINITY, EDOM, _DOMAIN}, {"acosh", -INFINITY, EDOM, -1}, {"acosh", 0, EDOM, -1}, {"acosh", 1, -1, -1}, {"acosh", INFINITY, -1, -1}, {"asin", -INFINITY, EDOM, _DOMAIN}, {"asin", -2, EDOM, _DOMAIN}, {"asin", -1, -1, -1}, {"asin", 1, -1, -1}, {"asin", 2, EDOM, _DOMAIN}, {"asin", INFINITY, EDOM, _DOMAIN}, {"asinh", -INFINITY, -1, -1}, {"asinh", INFINITY, -1, -1}, {"atan", -INFINITY, -1, -1}, {"atan", 0, -1, -1}, {"atan", INFINITY, -1, -1}, {"atanh", -INFINITY, EDOM, -1}, {"atanh", -2, EDOM, -1}, {"atanh", -1, ERANGE, -1}, {"atanh", 1, ERANGE, -1}, {"atanh", 2, EDOM, -1}, {"atanh", INFINITY, EDOM, -1}, {"cos", -INFINITY, EDOM, _DOMAIN}, {"cos", INFINITY, EDOM, _DOMAIN}, {"cosh", -INFINITY, -1, -1}, {"cosh", 0, -1, -1}, {"cosh", INFINITY, -1, -1}, {"exp", -INFINITY, -1, -1}, {"exp", -1e100, -1, _UNDERFLOW}, {"exp", 1e100, ERANGE, _OVERFLOW}, {"exp", INFINITY, -1, -1}, {"exp2", -INFINITY, -1, -1}, {"exp2", -1e100, -1, -1}, {"exp2", 1e100, ERANGE, -1}, {"exp2", INFINITY, -1, -1}, {"expm1", -INFINITY, -1, -1}, {"expm1", INFINITY, -1, -1}, {"log", -INFINITY, EDOM, _DOMAIN}, {"log", -1, EDOM, _DOMAIN}, {"log", 0, ERANGE, _SING}, {"log", INFINITY, -1, -1}, {"log10", -INFINITY, EDOM, _DOMAIN}, {"log10", -1, EDOM, _DOMAIN}, {"log10", 0, ERANGE, _SING}, {"log10", INFINITY, -1, -1}, {"log1p", -INFINITY, EDOM, -1}, {"log1p", -2, EDOM, -1}, {"log1p", -1, ERANGE, -1}, {"log1p", INFINITY, -1, -1}, {"log2", INFINITY, -1, -1}, {"sin", -INFINITY, EDOM, _DOMAIN}, {"sin", INFINITY, EDOM, _DOMAIN}, {"sinh", -INFINITY, -1, -1}, {"sinh", 0, -1, -1}, {"sinh", INFINITY, -1, -1}, {"sqrt", -INFINITY, EDOM, _DOMAIN}, {"sqrt", -1, EDOM, _DOMAIN}, {"sqrt", 0, -1, -1}, {"sqrt", INFINITY, -1, -1}, {"tan", -INFINITY, EDOM, _DOMAIN}, {"tan", -M_PI_2, -1, -1}, {"tan", M_PI_2, -1, -1}, {"tan", INFINITY, EDOM, _DOMAIN}, {"tanh", -INFINITY, -1, -1}, {"tanh", 0, -1, -1}, {"tanh", INFINITY, -1, -1}, }; const struct { char func[16]; double a; double b; int error; int exception; } tests2d[] = { {"atan2", -INFINITY, 0, -1, -1}, {"atan2", 0, 0, -1, -1}, {"atan2", INFINITY, 0, -1, -1}, {"atan2", 0, -INFINITY, -1, -1}, {"atan2", 0, INFINITY, -1, -1}, {"pow", -INFINITY, -2, -1, -1}, {"pow", -INFINITY, -1, -1, -1}, {"pow", -INFINITY, 0, -1, -1}, {"pow", -INFINITY, 1, -1, -1}, {"pow", -INFINITY, 2, -1, -1}, {"pow", -1e100, -10, -1, _UNDERFLOW}, {"pow", -1e100, 10, ERANGE, _OVERFLOW}, {"pow", -1, 1.5, EDOM, _DOMAIN}, {"pow", 0, -2, ERANGE, _SING}, {"pow", 0, -1, ERANGE, _SING}, {"pow", 0.5, -INFINITY, -1, -1}, {"pow", 0.5, INFINITY, -1, -1}, {"pow", 2, -INFINITY, -1, -1}, {"pow", 2, -1e100, -1, _UNDERFLOW}, {"pow", 2, 1e100, ERANGE, _OVERFLOW}, {"pow", 2, INFINITY, -1, -1}, {"pow", 1e100, -10, -1, _UNDERFLOW}, {"pow", 1e100, 10, ERANGE, _OVERFLOW}, {"pow", INFINITY, -2, -1, -1}, {"pow", INFINITY, -1, -1, -1}, {"pow", INFINITY, 0, -1, -1}, {"pow", INFINITY, 1, -1, -1}, {"pow", INFINITY, 2, -1, -1}, }; const struct { char func[16]; double a; double b; double c; int error; int exception; } tests3d[] = { /* 0 * inf --> EDOM */ {"fma", INFINITY, 0, 0, EDOM, -1}, {"fma", 0, INFINITY, 0, EDOM, -1}, /* inf - inf -> EDOM */ {"fma", INFINITY, 1, -INFINITY, EDOM, -1}, {"fma", -INFINITY, 1, INFINITY, EDOM, -1}, {"fma", 1, INFINITY, -INFINITY, EDOM, -1}, {"fma", 1, -INFINITY, INFINITY, EDOM, -1}, /* NaN */ {"fma", NAN, 0, 0, -1, -1}, {"fma", 0, NAN, 0, -1, -1}, {"fma", 0, 0, NAN, -1, -1}, /* over/underflow */ {"fma", __port_max_double(), __port_max_double(), __port_max_double(), -1, -1}, {"fma", __port_min_pos_double(), __port_min_pos_double(), 1, -1, -1}, }; const struct { char func[16]; double a; long b; int error; int exception; } testsdl[] = { {"_scalb", -INFINITY, 1, -1, -1}, {"_scalb", -1e100, 1, -1, -1}, {"_scalb", 0, 1, -1, -1}, {"_scalb", 1e100, 1, -1, -1}, {"_scalb", INFINITY, 1, -1, -1}, {"_scalb", 1, 1e9, ERANGE, _OVERFLOW}, {"ldexp", -INFINITY, 1, -1, -1}, {"ldexp", -1e100, 1, -1, -1}, {"ldexp", 0, 1, -1, -1}, {"ldexp", 1e100, 1, -1, -1}, {"ldexp", INFINITY, 1, -1, -1}, {"ldexp", 1, -1e9, -1, _UNDERFLOW}, {"ldexp", 1, 1e9, ERANGE, _OVERFLOW}, }; double (CDECL *p_funcd)(double); double (CDECL *p_func2d)(double, double); double (CDECL *p_func3d)(double, double, double); double (CDECL *p_funcdl)(double, long); HMODULE module; double d; int i; __setusermatherr(matherr_callback); module = GetModuleHandleW(L"ucrtbase.dll"); /* necessary so that exp(1e100)==INFINITY on glibc, we can remove this if we change our implementation */ fesetround(FE_TONEAREST); for(i = 0; i < ARRAY_SIZE(testsd); i++) { p_funcd = (void*)GetProcAddress(module, testsd[i].func); errno = -1; exception.type = -1; p_funcd(testsd[i].x); ok(errno == testsd[i].error, "%s(%f) got errno %d\n", testsd[i].func, testsd[i].x, errno); ok(exception.type == testsd[i].exception, "%s(%f) got exception type %d\n", testsd[i].func, testsd[i].x, exception.type); if(exception.type == -1) continue; ok(exception.arg1 == testsd[i].x, "%s(%f) got exception arg1 %f\n", testsd[i].func, testsd[i].x, exception.arg1); } for(i = 0; i < ARRAY_SIZE(tests2d); i++) { p_func2d = (void*)GetProcAddress(module, tests2d[i].func); errno = -1; exception.type = -1; p_func2d(tests2d[i].a, tests2d[i].b); ok(errno == tests2d[i].error, "%s(%f, %f) got errno %d\n", tests2d[i].func, tests2d[i].a, tests2d[i].b, errno); ok(exception.type == tests2d[i].exception, "%s(%f, %f) got exception type %d\n", tests2d[i].func, tests2d[i].a, tests2d[i].b, exception.type); if(exception.type == -1) continue; ok(exception.arg1 == tests2d[i].a, "%s(%f, %f) got exception arg1 %f\n", tests2d[i].func, tests2d[i].a, tests2d[i].b, exception.arg1); ok(exception.arg2 == tests2d[i].b, "%s(%f, %f) got exception arg2 %f\n", tests2d[i].func, tests2d[i].a, tests2d[i].b, exception.arg2); } for(i = 0; i < ARRAY_SIZE(tests3d); i++) { p_func3d = (void*)GetProcAddress(module, tests3d[i].func); errno = -1; exception.type = -1; p_func3d(tests3d[i].a, tests3d[i].b, tests3d[i].c); ok(errno == tests3d[i].error, "%s(%f, %f, %f) got errno %d\n", tests3d[i].func, tests3d[i].a, tests3d[i].b, tests3d[i].c, errno); ok(exception.type == tests3d[i].exception, "%s(%f, %f, %f) got exception type %d\n", tests3d[i].func, tests3d[i].a, tests3d[i].b, tests3d[i].c, exception.type); if(exception.type == -1) continue; ok(exception.arg1 == tests3d[i].a, "%s(%f, %f, %f) got exception arg1 %f\n", tests3d[i].func, tests3d[i].a, tests3d[i].b, tests3d[i].c, exception.arg1); ok(exception.arg2 == tests3d[i].b, "%s(%f, %f, %f) got exception arg2 %f\n", tests3d[i].func, tests3d[i].a, tests3d[i].b, tests3d[i].c, exception.arg2); } for(i = 0; i < ARRAY_SIZE(testsdl); i++) { p_funcdl = (void*)GetProcAddress(module, testsdl[i].func); errno = -1; exception.type = -1; p_funcdl(testsdl[i].a, testsdl[i].b); ok(errno == testsdl[i].error, "%s(%f, %ld) got errno %d\n", testsdl[i].func, testsdl[i].a, testsdl[i].b, errno); ok(exception.type == testsdl[i].exception, "%s(%f, %ld) got exception type %d\n", testsdl[i].func, testsdl[i].a, testsdl[i].b, exception.type); if(exception.type == -1) continue; ok(exception.arg1 == testsdl[i].a, "%s(%f, %ld) got exception arg1 %f\n", testsdl[i].func, testsdl[i].a, testsdl[i].b, exception.arg1); ok(exception.arg2 == testsdl[i].b, "%s(%f, %ld) got exception arg2 %f\n", testsdl[i].func, testsdl[i].a, testsdl[i].b, exception.arg2); } d = acos(2.0); ok(d == -1.0, "failed to change log10 return value: %e\n", d); } static void test_asctime(void) { const struct tm epoch = { 0, 0, 0, 1, 0, 70, 4, 0, 0 }; char *ret; ret = asctime(&epoch); ok(!strcmp(ret, "Thu Jan 1 00:00:00 1970\n"), "asctime returned %s\n", ret); } static void test_strftime(void) { const struct { const char *format; const char *ret; struct tm tm; BOOL todo_value; BOOL todo_handler; } tests[] = { {"%C", "", { 0, 0, 0, 1, 0, -2000, 4, 0, 0 }}, {"%C", "", { 0, 0, 0, 1, 0, -1901, 4, 0, 0 }}, {"%C", "00", { 0, 0, 0, 1, 0, -1900, 4, 0, 0 }}, {"%C", "18", { 0, 0, 0, 1, 0, -1, 4, 0, 0 }}, {"%C", "19", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%C", "99", { 0, 0, 0, 1, 0, 8099, 4, 0, 0 }}, {"%C", "", { 0, 0, 0, 1, 0, 8100, 4, 0, 0 }}, {"%d", "", { 0, 0, 0, 0, 0, 70, 4, 0, 0 }}, {"%d", "01", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%d", "31", { 0, 0, 0, 31, 0, 70, 4, 0, 0 }}, {"%d", "", { 0, 0, 0, 32, 0, 70, 4, 0, 0 }}, {"%D", "", { 0, 0, 0, 1, 0, -1901, 4, 0, 0 }}, {"%D", "01/01/00", { 0, 0, 0, 1, 0, -1900, 4, 0, 0 }}, {"%D", "01/01/99", { 0, 0, 0, 1, 0, -1, 4, 0, 0 }}, {"%D", "01/01/70", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%D", "01/01/99", { 0, 0, 0, 1, 0, 8099, 4, 0, 0 }}, {"%D", "", { 0, 0, 0, 1, 0, 8100, 4, 0, 0 }}, {"%#D", "1/1/70", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%e", "", { 0, 0, 0, 0, 0, 70, 4, 0, 0 }}, {"%e", " 1", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%e", "31", { 0, 0, 0, 31, 0, 70, 4, 0, 0 }}, {"%e", "", { 0, 0, 0, 32, 0, 70, 4, 0, 0 }}, {"%#e", "1", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%F", "1970-01-01", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#F", "1970-1-1", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%R", "00:00", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#R", "0:0", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%T", "00:00:00", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#T", "0:0:0", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%u", "", { 0, 0, 0, 1, 0, 117, -1, 0, 0 }}, {"%u", "7", { 0, 0, 0, 1, 0, 117, 0, 0, 0 }}, {"%u", "1", { 0, 0, 0, 1, 0, 117, 1, 0, 0 }}, {"%u", "6", { 0, 0, 0, 1, 0, 117, 6, 0, 0 }}, {"%u", "", { 0, 0, 0, 1, 0, 117, 7, 0, 0 }}, {"%h", "Jan", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%I", "", { 0, 0, -1, 1, 0, 70, 4, 0, 0 }}, {"%I", "12", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%I", "01", { 0, 0, 1, 1, 0, 70, 4, 0, 0 }}, {"%I", "11", { 0, 0, 11, 1, 0, 70, 4, 0, 0 }}, {"%I", "12", { 0, 0, 12, 1, 0, 70, 4, 0, 0 }}, {"%I", "01", { 0, 0, 13, 1, 0, 70, 4, 0, 0 }}, {"%I", "11", { 0, 0, 23, 1, 0, 70, 4, 0, 0 }}, {"%I", "", { 0, 0, 24, 1, 0, 70, 4, 0, 0 }}, {"%n", "\n", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%r", "12:00:00 AM", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%r", "02:00:00 PM", { 0, 0, 14, 1, 0, 121, 6, 0, 0 }}, {"%#r", "12:0:0 AM", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#r", "2:0:0 PM", { 0, 0, 14, 1, 0, 121, 6, 0, 0 }}, {"%t", "\t", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%g", "", { 0, 0, 0, 1, 0, -1901, 4, 0, 0 }}, {"%g", "", { 0, 0, 0, 1, 0, -1901, 3, 364, 0 }}, {"%g", "00", { 0, 0, 0, 1, 0, -1900, 4, 0, 0 }}, {"%g", "70", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%g", "71", { 0, 0, 0, 2, 0, 72, 0, 1, 0 }}, {"%g", "72", { 0, 0, 0, 3, 0, 72, 1, 2, 0 }}, {"%g", "16", { 0, 0, 0, 1, 0, 117, 0, 0, 0 }}, {"%g", "99", { 0, 0, 0, 1, 0, 8099, 4, 0, 0 }}, {"%g", "00", { 0, 0, 0, 1, 0, 8099, 3, 364, 0 }}, {"%g", "", { 0, 0, 0, 1, 0, 8100, 0, 0, 0 }}, {"%g", "", { 0, 0, 0, 1, 0, 8100, 4, 0, 0 }}, {"%G", "1970", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%G", "1971", { 0, 0, 0, 2, 0, 72, 0, 1, 0 }}, {"%G", "1972", { 0, 0, 0, 3, 0, 72, 1, 2, 0 }}, {"%G", "2016", { 0, 0, 0, 1, 0, 117, 0, 0, 0 }}, {"%V", "01", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%V", "52", { 0, 0, 0, 1, 0, 117, 0, 0, 0 }}, {"%V", "53", { 0, 0, 14, 1, 0, 121, 6, 0, 0 }}, {"%y", "", { 0, 0, 0, 0, 0, -1901, 0, 0, 0 }}, {"%y", "00", { 0, 0, 0, 0, 0, -1900, 0, 0, 0 }}, {"%y", "99", { 0, 0, 0, 0, 0, 8099, 0, 0, 0 }}, {"%y", "", { 0, 0, 0, 0, 0, 8100, 0, 0, 0 }}, {"%c", "Thu Jan 1 00:00:00 1970", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%c", "Thu Feb 30 00:00:00 1970", { 0, 0, 0, 30, 1, 70, 4, 0, 0 }}, {"%#c", "Thursday, January 01, 1970 00:00:00", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#c", "Thursday, February 30, 1970 00:00:00", { 0, 0, 0, 30, 1, 70, 4, 0, 0 }}, {"%x", "01/01/70", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%x", "02/30/70", { 0, 0, 0, 30, 1, 70, 4, 0, 0 }}, {"%#x", "Thursday, January 01, 1970", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%#x", "Thursday, February 30, 1970", { 0, 0, 0, 30, 1, 70, 4, 0, 0 }}, {"%#x", "", { 0, 0, 0, 30, 1, 70, 7, 0, 0 }}, {"%#x", "", { 0, 0, 0, 30, 12, 70, 4, 0, 0 }}, {"%X", "00:00:00", { 0, 0, 0, 1, 0, 70, 4, 0, 0 }}, {"%X", "14:00:00", { 0, 0, 14, 1, 0, 70, 4, 0, 0 }}, {"%X", "23:59:60", { 60, 59, 23, 1, 0, 70, 4, 0, 0 }}, }; const struct { const char *format; const char *ret; const wchar_t *short_date; const wchar_t *date; const wchar_t *time; struct tm tm; BOOL todo; } tests_td[] = { { "%c", "x z", L"x", L"y", L"z", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#c", "y z", L"x", L"y", L"z", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "M1", 0, 0, L"MMM", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "1", 0, 0, L"h", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "01", 0, 0, L"hh", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "h01", 0, 0, L"hhh", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "hh01", 0, 0, L"hhhh", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "1", 0, 0, L"H", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "01", 0, 0, L"HH", { 0, 0, 1, 1, 0, 70, 0, 0, 0 }}, { "%X", "H13", 0, 0, L"HHH", { 0, 0, 13, 1, 0, 70, 0, 0, 0 }}, { "%X", "0", 0, 0, L"m", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "00", 0, 0, L"mm", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "m00", 0, 0, L"mmm", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "0", 0, 0, L"s", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "00", 0, 0, L"ss", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "s00", 0, 0, L"sss", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "T", 0, 0, L"t", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"tt", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"ttttttttt", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"a", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"aaaaa", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"A", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%X", "TAM", 0, 0, L"AAAAA", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "1", L"d", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "01", L"dd", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "D1", L"ddd", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "Day1", L"dddd", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "dDay1", L"ddddd", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "1", L"M", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "01", L"MM", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "M1", L"MMM", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "Mon1", L"MMMM", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "MMon1", L"MMMMM", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "y", L"y", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "70", L"yy", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "y70", L"yyy", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "1970", L"yyyy", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "y1970", L"yyyyy", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%x", "ggggggggggg", L"ggggggggggg", 0, 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "1", 0, L"d", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "01", 0, L"dd", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "D1", 0, L"ddd", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "Day1", 0, L"dddd", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "dDay1", 0, L"ddddd", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "1", 0, L"M", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "01", 0, L"MM", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "M1", 0, L"MMM", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "Mon1", 0, L"MMMM", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "MMon1", 0, L"MMMMM", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "y", 0, L"y", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "70", 0, L"yy", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "y70", 0, L"yyy", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "1970", 0, L"yyyy", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%#x", "y1970", 0, L"yyyyy", 0, { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, { "%r", "z", L"x", L"y", L"z", { 0, 0, 0, 1, 0, 70, 0, 0, 0 }}, }; const struct { int year; int yday; const char *ret[7]; } tests_yweek[] = { { 100, 0, { "99 52", "00 01", "00 01", "00 01", "00 01", "99 53", "99 52" }}, { 100, 1, { "99 52", "00 01", "00 01", "00 01", "00 01", "00 01", "99 53" }}, { 100, 2, { "99 53", "00 01", "00 01", "00 01", "00 01", "00 01", "00 01" }}, { 100, 3, { "00 01", "00 01", "00 01", "00 01", "00 01", "00 01", "00 01" }}, { 100, 4, { "00 01", "00 02", "00 01", "00 01", "00 01", "00 01", "00 01" }}, { 100, 5, { "00 01", "00 02", "00 02", "00 01", "00 01", "00 01", "00 01" }}, { 100, 6, { "00 01", "00 02", "00 02", "00 02", "00 01", "00 01", "00 01" }}, { 100, 358, { "00 51", "00 52", "00 52", "00 52", "00 52", "00 52", "00 51" }}, { 100, 359, { "00 51", "00 52", "00 52", "00 52", "00 52", "00 52", "00 52" }}, { 100, 360, { "00 52", "00 52", "00 52", "00 52", "00 52", "00 52", "00 52" }}, { 100, 361, { "00 52", "00 53", "00 52", "00 52", "00 52", "00 52", "00 52" }}, { 100, 362, { "00 52", "00 53", "00 53", "00 52", "00 52", "00 52", "00 52" }}, { 100, 363, { "00 52", "01 01", "00 53", "00 53", "00 52", "00 52", "00 52" }}, { 100, 364, { "00 52", "01 01", "01 01", "00 53", "00 53", "00 52", "00 52" }}, { 100, 365, { "00 52", "01 01", "01 01", "01 01", "00 53", "00 53", "00 52" }}, { 101, 0, { "00 52", "01 01", "01 01", "01 01", "01 01", "00 53", "00 53" }}, { 101, 1, { "00 53", "01 01", "01 01", "01 01", "01 01", "01 01", "00 53" }}, { 101, 2, { "00 53", "01 01", "01 01", "01 01", "01 01", "01 01", "01 01" }}, { 101, 3, { "01 01", "01 01", "01 01", "01 01", "01 01", "01 01", "01 01" }}, { 101, 4, { "01 01", "01 02", "01 01", "01 01", "01 01", "01 01", "01 01" }}, { 101, 5, { "01 01", "01 02", "01 02", "01 01", "01 01", "01 01", "01 01" }}, { 101, 6, { "01 01", "01 02", "01 02", "01 02", "01 01", "01 01", "01 01" }}, { 101, 358, { "01 51", "01 52", "01 52", "01 52", "01 52", "01 52", "01 51" }}, { 101, 359, { "01 51", "01 52", "01 52", "01 52", "01 52", "01 52", "01 52" }}, { 101, 360, { "01 52", "01 52", "01 52", "01 52", "01 52", "01 52", "01 52" }}, { 101, 361, { "01 52", "01 53", "01 52", "01 52", "01 52", "01 52", "01 52" }}, { 101, 362, { "01 52", "02 01", "01 53", "01 52", "01 52", "01 52", "01 52" }}, { 101, 363, { "01 52", "02 01", "02 01", "01 53", "01 52", "01 52", "01 52" }}, { 101, 364, { "01 52", "02 01", "02 01", "02 01", "01 53", "01 52", "01 52" }}, }; __lc_time_data time_data = { { "d1", "d2", "d3", "d4", "d5", "d6", "d7" }, { "day1", "day2", "day3", "day4", "day5", "day6", "day7" }, { "m1", "m2", "m3", "m4", "m5", "m6", "m7", "m8", "m9", "m10", "m11", "m12" }, { "mon1", "mon2", "mon3", "mon4", "mon5", "mon6", "mon7", "mon8", "mon9", "mon10", "mon11", "mon12" }, "tam", "tpm", 0, 0, 0, 1, 0, { L"D1", L"D2", L"D3", L"D4", L"D5", L"D6", L"D7" }, { L"Day1", L"Day2", L"Day3", L"Day4", L"Day5", L"Day6", L"Day7" }, { L"M1", L"M2", L"M3", L"M4", L"M5", L"M6", L"M7", L"M8", L"M9", L"M10", L"M11", L"M12" }, { L"Mon1", L"Mon2", L"Mon3", L"Mon4", L"Mon5", L"Mon6", L"Mon7", L"Mon8", L"Mon9", L"Mon10", L"Mon11", L"Mon12" }, L"TAM", L"TPM" }; const struct tm epoch = { 0, 0, 0, 1, 0, 70, 4, 0, 0 }; struct tm tm_yweek = { 0, 0, 0, 1, 0, 70, 0, 0, 0 }; char buf[256]; int i, ret=0; for (i=0; i= 0) { ret = _stat32(path, &buf); ok(!ret, "_stat32('%s') returned %d\n", path, ret); strcat(path, "\\"); ret = _stat32(path, &buf); todo_wine ok(ret, "_stat32('%s') returned %d\n", path, ret); close(fd); remove(path); } memcpy(path+len, test_dir, sizeof(test_dir)); if(!mkdir(path)) { ret = _stat32(path, &buf); ok(!ret, "_stat32('%s') returned %d\n", path, ret); strcat(path, "\\"); ret = _stat32(path, &buf); ok(!ret, "_stat32('%s') returned %d\n", path, ret); rmdir(path); } } static void test__o_malloc(void) { void *m; size_t s; m = _o_malloc(1); ok(m != NULL, "p__o_malloc(1) returned NULL\n"); s = _msize(m); ok(s == 1, "_msize returned %d\n", (int)s); free(m); } static void test_clock(void) { static const int thresh = 100, max_load_delay = 1000; int c, expect_min; FILETIME cur; GetSystemTimeAsFileTime(&cur); c = clock(); expect_min = (((LONGLONG)cur.dwHighDateTime << 32) + cur.dwLowDateTime - crt_init_end) / 10000; ok(c >= expect_min - thresh && c < expect_min + max_load_delay, "clock() = %d, expected range [%d, %d]\n", c, expect_min - thresh, expect_min + max_load_delay); } static void __cdecl se_translator(unsigned int u, EXCEPTION_POINTERS *ep) { } static void test_thread_storage(void) { void **current_exception; void *processing_throw; _set_se_translator(se_translator); current_exception = __current_exception(); processing_throw = __processing_throw(); ok(current_exception+2 == processing_throw, "current_exception = %p, processing_throw = %p\n", current_exception, processing_throw); ok(current_exception[-2] == se_translator, "can't find se_translator in thread storage\n"); } static unsigned long fenv_encode(unsigned int e) { ok(!(e & ~FE_ALL_EXCEPT), "incorrect argument: %x\n", e); #if defined(__i386__) return e<<24 | e<<16 | e; #else return e<<24 | e; #endif } static void test_fenv(void) { #if defined(__i386__) || defined(__x86_64__) static const int tests[] = { 0, FE_INEXACT, FE_UNDERFLOW, FE_OVERFLOW, FE_DIVBYZERO, FE_INVALID, FE_ALL_EXCEPT, }; static const struct { fexcept_t except; unsigned int flag; unsigned int get; fexcept_t expect; } tests2[] = { /* except flag get expect */ { 0, 0, 0, 0 }, { FE_ALL_EXCEPT, FE_INEXACT, 0, 0 }, { FE_ALL_EXCEPT, FE_INEXACT, FE_ALL_EXCEPT, FE_INEXACT }, { FE_ALL_EXCEPT, FE_INEXACT, FE_INEXACT, FE_INEXACT }, { FE_ALL_EXCEPT, FE_INEXACT, FE_OVERFLOW, 0 }, { FE_ALL_EXCEPT, FE_ALL_EXCEPT, FE_ALL_EXCEPT, FE_ALL_EXCEPT }, { FE_ALL_EXCEPT, FE_ALL_EXCEPT, FE_INEXACT, FE_INEXACT }, { FE_ALL_EXCEPT, FE_ALL_EXCEPT, 0, 0 }, { FE_ALL_EXCEPT, FE_ALL_EXCEPT, ~0, FE_ALL_EXCEPT }, { FE_ALL_EXCEPT, FE_ALL_EXCEPT, ~FE_ALL_EXCEPT, 0 }, { FE_INEXACT, FE_ALL_EXCEPT, FE_ALL_EXCEPT, FE_INEXACT }, { FE_INEXACT, FE_UNDERFLOW, FE_ALL_EXCEPT, 0 }, { FE_UNDERFLOW, FE_INEXACT, FE_ALL_EXCEPT, 0 }, { FE_INEXACT|FE_UNDERFLOW, FE_UNDERFLOW, FE_ALL_EXCEPT, FE_UNDERFLOW }, { FE_UNDERFLOW, FE_INEXACT|FE_UNDERFLOW, FE_ALL_EXCEPT, FE_UNDERFLOW }, }; fenv_t env, env2; fexcept_t except; int i, ret, flags; _clearfp(); ret = fegetenv(&env); ok(!ret, "fegetenv returned %x\n", ret); if (env._Fe_ctl >> 24 != (env._Fe_ctl & 0xff)) { win_skip("fenv_t format not supported (too old ucrtbase)\n"); return; } fesetround(FE_UPWARD); ok(!env._Fe_stat, "env._Fe_stat = %lx\n", env._Fe_stat); ret = fegetenv(&env2); ok(!ret, "fegetenv returned %x\n", ret); ok(env._Fe_ctl != env2._Fe_ctl, "fesetround didn't change _Fe_ctl (%lx).\n", env._Fe_ctl); ret = fesetenv(&env); ok(!ret, "fesetenv returned %x\n", ret); ret = fegetround(); ok(ret == FE_TONEAREST, "Got unexpected round mode %#x.\n", ret); except = fenv_encode(FE_ALL_EXCEPT); ret = fesetexceptflag(&except, FE_INEXACT|FE_UNDERFLOW); ok(!ret, "fesetexceptflag returned %x\n", ret); except = fetestexcept(FE_ALL_EXCEPT); ok(except == (FE_INEXACT|FE_UNDERFLOW), "expected %x, got %lx\n", FE_INEXACT|FE_UNDERFLOW, except); ret = feclearexcept(~FE_ALL_EXCEPT); ok(!ret, "feclearexceptflag returned %x\n", ret); except = fetestexcept(FE_ALL_EXCEPT); ok(except == (FE_INEXACT|FE_UNDERFLOW), "expected %x, got %lx\n", FE_INEXACT|FE_UNDERFLOW, except); /* no crash, but no-op */ ret = fesetexceptflag(NULL, 0); ok(!ret, "fesetexceptflag returned %x\n", ret); except = fetestexcept(FE_ALL_EXCEPT); ok(except == (FE_INEXACT|FE_UNDERFLOW), "expected %x, got %lx\n", FE_INEXACT|FE_UNDERFLOW, except); /* zero clears all */ except = 0; ret = fesetexceptflag(&except, FE_ALL_EXCEPT); ok(!ret, "fesetexceptflag returned %x\n", ret); except = fetestexcept(FE_ALL_EXCEPT); ok(!except, "expected 0, got %lx\n", except); ret = fetestexcept(FE_ALL_EXCEPT); ok(!ret, "fetestexcept returned %x\n", ret); flags = 0; /* adding bits with flags */ for(i=0; i