2015-09-19 02:43:31 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2015 YongHao Hu
|
|
|
|
*
|
|
|
|
* 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 <locale.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "wine/test.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
|
|
|
|
typedef unsigned char MSVCP_bool;
|
|
|
|
|
|
|
|
#define SECSPERDAY 86400
|
|
|
|
/* 1601 to 1970 is 369 years plus 89 leap days */
|
|
|
|
#define SECS_1601_TO_1970 ((369 * 365 + 89) * (ULONGLONG)SECSPERDAY)
|
|
|
|
#define TICKSPERSEC 10000000
|
|
|
|
#define TICKS_1601_TO_1970 (SECS_1601_TO_1970 * TICKSPERSEC)
|
|
|
|
|
|
|
|
static int (__cdecl *p_tr2_sys__Make_dir)(char const*);
|
|
|
|
static MSVCP_bool (__cdecl *p_tr2_sys__Remove_dir)(char const*);
|
|
|
|
static __int64 (__cdecl *p_tr2_sys__Last_write_time)(char const*);
|
2018-08-15 12:43:41 +02:00
|
|
|
static __int64 (__cdecl *p_tr2_sys__Last_write_time_wchar)(WCHAR const*);
|
2015-09-19 02:43:31 +02:00
|
|
|
static void (__cdecl *p_tr2_sys__Last_write_time_set)(char const*, __int64);
|
2018-08-15 12:43:41 +02:00
|
|
|
static void (__cdecl *p_tr2_sys__Last_write_time_set_wchar)(WCHAR const*, __int64);
|
2015-09-19 02:43:31 +02:00
|
|
|
|
2022-01-19 17:50:33 +01:00
|
|
|
static const BYTE *p_byte_reverse_table;
|
|
|
|
|
2015-09-19 02:43:31 +02:00
|
|
|
static HMODULE msvcp;
|
|
|
|
#define SETNOFAIL(x,y) x = (void*)GetProcAddress(msvcp,y)
|
|
|
|
#define SET(x,y) do { SETNOFAIL(x,y); ok(x != NULL, "Export '%s' not found\n", y); } while(0)
|
|
|
|
static BOOL init(void)
|
|
|
|
{
|
|
|
|
msvcp = LoadLibraryA("msvcp110.dll");
|
|
|
|
if(!msvcp)
|
|
|
|
{
|
|
|
|
win_skip("msvcp110.dll not installed\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(sizeof(void*) == 8) { /* 64-bit initialization */
|
|
|
|
SET(p_tr2_sys__Make_dir,
|
|
|
|
"?_Make_dir@sys@tr2@std@@YAHPEBD@Z");
|
|
|
|
SET(p_tr2_sys__Remove_dir,
|
|
|
|
"?_Remove_dir@sys@tr2@std@@YA_NPEBD@Z");
|
|
|
|
SET(p_tr2_sys__Last_write_time,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YA_JPEBD@Z");
|
2018-08-15 12:43:41 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_wchar,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YA_JPEB_W@Z");
|
2015-09-19 02:43:31 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_set,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YAXPEBD_J@Z");
|
2018-08-15 12:43:41 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_set_wchar,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YAXPEB_W_J@Z");
|
2015-09-19 02:43:31 +02:00
|
|
|
} else {
|
|
|
|
SET(p_tr2_sys__Make_dir,
|
|
|
|
"?_Make_dir@sys@tr2@std@@YAHPBD@Z");
|
|
|
|
SET(p_tr2_sys__Remove_dir,
|
|
|
|
"?_Remove_dir@sys@tr2@std@@YA_NPBD@Z");
|
|
|
|
SET(p_tr2_sys__Last_write_time,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YA_JPBD@Z");
|
2018-08-15 12:43:41 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_wchar,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YA_JPB_W@Z");
|
2015-09-19 02:43:31 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_set,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YAXPBD_J@Z");
|
2018-08-15 12:43:41 +02:00
|
|
|
SET(p_tr2_sys__Last_write_time_set_wchar,
|
|
|
|
"?_Last_write_time@sys@tr2@std@@YAXPB_W_J@Z");
|
2015-09-19 02:43:31 +02:00
|
|
|
}
|
2022-01-19 17:50:33 +01:00
|
|
|
SET(p_byte_reverse_table, "?_Byte_reverse_table@details@Concurrency@@3QBEB");
|
2015-09-19 02:43:31 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_tr2_sys__Last_write_time(void)
|
|
|
|
{
|
|
|
|
HANDLE file;
|
|
|
|
int ret;
|
|
|
|
FILETIME lwt;
|
|
|
|
__int64 last_write_time, newtime, margin_of_error = 10 * TICKSPERSEC;
|
|
|
|
ret = p_tr2_sys__Make_dir("tr2_test_dir");
|
|
|
|
ok(ret == 1, "test_tr2_sys__Make_dir(): expect 1 got %d\n", ret);
|
|
|
|
file = CreateFileA("tr2_test_dir/f1", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, NULL);
|
|
|
|
ok(file != INVALID_HANDLE_VALUE, "CreateFileA failed: INVALID_HANDLE_VALUE\n");
|
|
|
|
CloseHandle(file);
|
|
|
|
|
|
|
|
last_write_time = p_tr2_sys__Last_write_time("tr2_test_dir/f1");
|
|
|
|
newtime = last_write_time + 222222;
|
|
|
|
p_tr2_sys__Last_write_time_set("tr2_test_dir/f1", newtime);
|
2018-08-15 12:43:41 +02:00
|
|
|
newtime = p_tr2_sys__Last_write_time("tr2_test_dir/f1");
|
|
|
|
ok(last_write_time != newtime, "last_write_time should have changed: %s\n",
|
2017-07-04 00:29:20 +02:00
|
|
|
wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
|
2020-10-07 23:34:29 +02:00
|
|
|
last_write_time = p_tr2_sys__Last_write_time_wchar(L"tr2_test_dir/f1");
|
2018-08-15 12:43:41 +02:00
|
|
|
ok(last_write_time == newtime,
|
|
|
|
"last_write_time and last_write_time_wchar returned different times (%s != %s)\n",
|
|
|
|
wine_dbgstr_longlong(last_write_time), wine_dbgstr_longlong(newtime));
|
|
|
|
|
2015-09-19 02:43:31 +02:00
|
|
|
/* test the formula */
|
|
|
|
file = CreateFileA("tr2_test_dir/f1", 0, FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, 0);
|
|
|
|
ok(file != INVALID_HANDLE_VALUE, "create file failed: INVALID_HANDLE_VALUE\n");
|
|
|
|
ok(GetFileTime(file, 0, 0, &lwt), "GetFileTime failed\n");
|
|
|
|
CloseHandle(file);
|
|
|
|
last_write_time = (((__int64)lwt.dwHighDateTime)<< 32) + lwt.dwLowDateTime;
|
|
|
|
last_write_time -= TICKS_1601_TO_1970;
|
|
|
|
last_write_time /= TICKSPERSEC;
|
|
|
|
ok(newtime-margin_of_error<=last_write_time && last_write_time<=newtime+margin_of_error,
|
2017-07-04 00:29:20 +02:00
|
|
|
"don't fit the formula, last_write_time is %s\n", wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
|
|
|
|
newtime = 0;
|
|
|
|
p_tr2_sys__Last_write_time_set("tr2_test_dir/f1", newtime);
|
|
|
|
newtime = p_tr2_sys__Last_write_time("tr2_test_dir/f1");
|
|
|
|
file = CreateFileA("tr2_test_dir/f1", 0, FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, 0);
|
|
|
|
ok(file != INVALID_HANDLE_VALUE, "create file failed: INVALID_HANDLE_VALUE\n");
|
|
|
|
ok(GetFileTime(file, 0, 0, &lwt), "GetFileTime failed\n");
|
|
|
|
CloseHandle(file);
|
|
|
|
last_write_time = (((__int64)lwt.dwHighDateTime)<< 32) + lwt.dwLowDateTime;
|
|
|
|
last_write_time -= TICKS_1601_TO_1970;
|
|
|
|
last_write_time /= TICKSPERSEC;
|
|
|
|
ok(newtime-margin_of_error<=last_write_time && last_write_time<=newtime+margin_of_error,
|
2017-07-04 00:29:20 +02:00
|
|
|
"don't fit the formula, last_write_time is %s\n", wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
|
|
|
|
newtime = 123456789;
|
2020-10-07 23:34:29 +02:00
|
|
|
p_tr2_sys__Last_write_time_set_wchar(L"tr2_test_dir/f1", newtime);
|
2015-09-19 02:43:31 +02:00
|
|
|
newtime = p_tr2_sys__Last_write_time("tr2_test_dir/f1");
|
|
|
|
file = CreateFileA("tr2_test_dir/f1", 0, FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
|
|
NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, 0);
|
|
|
|
ok(file != INVALID_HANDLE_VALUE, "create file failed: INVALID_HANDLE_VALUE\n");
|
|
|
|
ok(GetFileTime(file, 0, 0, &lwt), "GetFileTime failed\n");
|
|
|
|
CloseHandle(file);
|
|
|
|
last_write_time = (((__int64)lwt.dwHighDateTime)<< 32) + lwt.dwLowDateTime;
|
|
|
|
last_write_time -= TICKS_1601_TO_1970;
|
|
|
|
last_write_time /= TICKSPERSEC;
|
|
|
|
ok(newtime-margin_of_error<=last_write_time && last_write_time<=newtime+margin_of_error,
|
2017-07-04 00:29:20 +02:00
|
|
|
"don't fit the formula, last_write_time is %s\n", wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
|
|
|
|
errno = 0xdeadbeef;
|
|
|
|
last_write_time = p_tr2_sys__Last_write_time("not_exist");
|
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time(): errno expect 0xdeadbeef, got %d\n", errno);
|
2017-07-04 00:29:20 +02:00
|
|
|
ok(last_write_time == 0, "expect 0 got %s\n", wine_dbgstr_longlong(last_write_time));
|
2020-10-07 23:34:29 +02:00
|
|
|
last_write_time = p_tr2_sys__Last_write_time_wchar(L"not_exist");
|
2018-08-15 12:43:41 +02:00
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time_wchar(): errno expect 0xdeadbeef, got %d\n", errno);
|
|
|
|
ok(last_write_time == 0, "expect 0 got %s\n", wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
last_write_time = p_tr2_sys__Last_write_time(NULL);
|
2017-07-04 00:29:20 +02:00
|
|
|
ok(last_write_time == 0, "expect 0 got %s\n", wine_dbgstr_longlong(last_write_time));
|
2018-08-15 12:43:41 +02:00
|
|
|
last_write_time = p_tr2_sys__Last_write_time_wchar(NULL);
|
|
|
|
ok(last_write_time == 0, "expect 0 got %s\n", wine_dbgstr_longlong(last_write_time));
|
2015-09-19 02:43:31 +02:00
|
|
|
|
|
|
|
errno = 0xdeadbeef;
|
2018-08-15 12:43:41 +02:00
|
|
|
p_tr2_sys__Last_write_time_set("not_exist", newtime);
|
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time(): errno expect 0xdeadbeef, got %d\n", errno);
|
2020-10-07 23:34:29 +02:00
|
|
|
p_tr2_sys__Last_write_time_set_wchar(L"not_exist", newtime);
|
2018-08-15 12:43:41 +02:00
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time(): errno expect 0xdeadbeef, got %d\n", errno);
|
2015-09-19 02:43:31 +02:00
|
|
|
p_tr2_sys__Last_write_time_set(NULL, newtime);
|
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time(): errno expect 0xdeadbeef, got %d\n", errno);
|
2018-08-15 12:43:41 +02:00
|
|
|
p_tr2_sys__Last_write_time_set_wchar(NULL, newtime);
|
|
|
|
ok(errno == 0xdeadbeef, "tr2_sys__Last_write_time(): errno expect 0xdeadbeef, got %d\n", errno);
|
2015-09-19 02:43:31 +02:00
|
|
|
|
|
|
|
ok(DeleteFileA("tr2_test_dir/f1"), "expect tr2_test_dir/f1 to exist\n");
|
|
|
|
ret = p_tr2_sys__Remove_dir("tr2_test_dir");
|
|
|
|
ok(ret == 1, "test_tr2_sys__Remove_dir(): expect 1 got %d\n", ret);
|
|
|
|
}
|
|
|
|
|
2016-01-13 22:54:11 +01:00
|
|
|
static struct {
|
|
|
|
int value[2];
|
|
|
|
const char* export_name;
|
|
|
|
} vbtable_size_exports_list[] = {
|
|
|
|
{{0x20, 0x20}, "??_8?$basic_iostream@DU?$char_traits@D@std@@@std@@7B?$basic_istream@DU?$char_traits@D@std@@@1@@"},
|
|
|
|
{{0x10, 0x10}, "??_8?$basic_iostream@DU?$char_traits@D@std@@@std@@7B?$basic_ostream@DU?$char_traits@D@std@@@1@@"},
|
|
|
|
{{0x20, 0x20}, "??_8?$basic_iostream@GU?$char_traits@G@std@@@std@@7B?$basic_istream@GU?$char_traits@G@std@@@1@@"},
|
|
|
|
{{0x10, 0x10}, "??_8?$basic_iostream@GU?$char_traits@G@std@@@std@@7B?$basic_ostream@GU?$char_traits@G@std@@@1@@"},
|
|
|
|
{{0x20, 0x20}, "??_8?$basic_iostream@_WU?$char_traits@_W@std@@@std@@7B?$basic_istream@_WU?$char_traits@_W@std@@@1@@"},
|
|
|
|
{{0x10, 0x10}, "??_8?$basic_iostream@_WU?$char_traits@_W@std@@@std@@7B?$basic_ostream@_WU?$char_traits@_W@std@@@1@@"},
|
|
|
|
{{0x18, 0x18}, "??_8?$basic_istream@DU?$char_traits@D@std@@@std@@7B@"},
|
|
|
|
{{0x18, 0x18}, "??_8?$basic_istream@GU?$char_traits@G@std@@@std@@7B@"},
|
|
|
|
{{0x18, 0x18}, "??_8?$basic_istream@_WU?$char_traits@_W@std@@@std@@7B@"},
|
|
|
|
{{ 0x8, 0x10}, "??_8?$basic_ostream@DU?$char_traits@D@std@@@std@@7B@"},
|
|
|
|
{{ 0x8, 0x10}, "??_8?$basic_ostream@GU?$char_traits@G@std@@@std@@7B@"},
|
|
|
|
{{ 0x8, 0x10}, "??_8?$basic_ostream@_WU?$char_traits@_W@std@@@std@@7B@"},
|
|
|
|
{{ 0x0, 0x0}, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
static void test_vbtable_size_exports(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
const int *p_vbtable;
|
|
|
|
int arch_idx = (sizeof(void*) == 8);
|
|
|
|
|
|
|
|
for (i = 0; vbtable_size_exports_list[i].export_name; i++)
|
|
|
|
{
|
|
|
|
SET(p_vbtable, vbtable_size_exports_list[i].export_name);
|
|
|
|
|
|
|
|
ok(p_vbtable[0] == 0, "vbtable[0] wrong, got 0x%x\n", p_vbtable[0]);
|
|
|
|
ok(p_vbtable[1] == vbtable_size_exports_list[i].value[arch_idx],
|
|
|
|
"%d: %s[1] wrong, got 0x%x\n", i, vbtable_size_exports_list[i].export_name, p_vbtable[1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-19 17:50:33 +01:00
|
|
|
static BYTE byte_reverse(BYTE b)
|
|
|
|
{
|
|
|
|
b = ((b & 0xf0) >> 4) | ((b & 0x0f) << 4);
|
|
|
|
b = ((b & 0xcc) >> 2) | ((b & 0x33) << 2);
|
|
|
|
b = ((b & 0xaa) >> 1) | ((b & 0x55) << 1);
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_data_exports(void)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
ok(IsBadWritePtr((BYTE *)p_byte_reverse_table, 256), "byte_reverse_table is writeable.\n");
|
|
|
|
for (i = 0; i < 256; ++i)
|
|
|
|
{
|
|
|
|
ok(p_byte_reverse_table[i] == byte_reverse(i), "Got unexpected byte %#x, expected %#x.\n",
|
|
|
|
p_byte_reverse_table[i], byte_reverse(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-19 02:43:31 +02:00
|
|
|
START_TEST(msvcp110)
|
|
|
|
{
|
|
|
|
if(!init()) return;
|
|
|
|
test_tr2_sys__Last_write_time();
|
2016-01-13 22:54:11 +01:00
|
|
|
test_vbtable_size_exports();
|
2022-01-19 17:50:33 +01:00
|
|
|
test_data_exports();
|
2015-09-19 02:43:31 +02:00
|
|
|
FreeLibrary(msvcp);
|
|
|
|
}
|