/* * Wine Conformance Test EXE * * Copyright 2003, 2004 Jakob Eriksson (for Solid Form Sweden AB) * Copyright 2003 Dimitrie O. Paun * Copyright 2003 Ferenc Wagner * * 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 * * This program is dedicated to Anna Lindh, * Swedish Minister of Foreign Affairs. * Anna was murdered September 11, 2003. * */ #include "config.h" #include "wine/port.h" #define COBJMACROS #include #include #include #include #include "winetest.h" #include "resource.h" struct wine_test { char *name; int resource; int subtest_count; char **subtests; char *exename; char *maindllpath; }; char *tag = NULL; static struct wine_test *wine_tests; static int nr_of_files, nr_of_tests; static const char whitespace[] = " \t\r\n"; static const char testexe[] = "_test.exe"; static char build_id[64]; /* filters for running only specific tests */ static char *filters[64]; static unsigned int nb_filters = 0; /* Needed to check for .NET dlls */ static HMODULE hmscoree; static HRESULT (WINAPI *pLoadLibraryShim)(LPCWSTR, LPCWSTR, LPVOID, HMODULE *); /* To store the current PATH setting (related to .NET only provided dlls) */ static char *curpath; /* check if test is being filtered out */ static BOOL test_filtered_out( LPCSTR module, LPCSTR testname ) { char *p, dllname[MAX_PATH]; unsigned int i, len; strcpy( dllname, module ); CharLowerA( dllname ); p = strstr( dllname, testexe ); if (p) *p = 0; len = strlen(dllname); if (!nb_filters) return FALSE; for (i = 0; i < nb_filters; i++) { if (!strncmp( dllname, filters[i], len )) { if (!filters[i][len]) return FALSE; if (filters[i][len] != ':') continue; if (!testname || !strcmp( testname, &filters[i][len+1] )) return FALSE; } } return TRUE; } static char * get_file_version(char * file_name) { static char version[32]; DWORD size; DWORD handle; size = GetFileVersionInfoSizeA(file_name, &handle); if (size) { char * data = heap_alloc(size); if (data) { if (GetFileVersionInfoA(file_name, handle, size, data)) { static char backslash[] = "\\"; VS_FIXEDFILEINFO *pFixedVersionInfo; UINT len; if (VerQueryValueA(data, backslash, (LPVOID *)&pFixedVersionInfo, &len)) { sprintf(version, "%d.%d.%d.%d", pFixedVersionInfo->dwFileVersionMS >> 16, pFixedVersionInfo->dwFileVersionMS & 0xffff, pFixedVersionInfo->dwFileVersionLS >> 16, pFixedVersionInfo->dwFileVersionLS & 0xffff); } else sprintf(version, "version not available"); } else sprintf(version, "unknown"); heap_free(data); } else sprintf(version, "failed"); } else sprintf(version, "version not available"); return version; } static int running_under_wine (void) { HMODULE module = GetModuleHandleA("ntdll.dll"); if (!module) return 0; return (GetProcAddress(module, "wine_server_call") != NULL); } static int running_on_visible_desktop (void) { HWND desktop; HMODULE huser32 = GetModuleHandle("user32.dll"); HWINSTA (WINAPI *pGetProcessWindowStation)(void); BOOL (WINAPI *pGetUserObjectInformationA)(HANDLE,INT,LPVOID,DWORD,LPDWORD); pGetProcessWindowStation = (void *)GetProcAddress(huser32, "GetProcessWindowStation"); pGetUserObjectInformationA = (void *)GetProcAddress(huser32, "GetUserObjectInformationA"); desktop = GetDesktopWindow(); if (!GetWindowLongPtrW(desktop, GWLP_WNDPROC)) /* Win9x */ return IsWindowVisible(desktop); if (pGetProcessWindowStation && pGetUserObjectInformationA) { DWORD len; HWINSTA wstation; USEROBJECTFLAGS uoflags; wstation = (HWINSTA)pGetProcessWindowStation(); assert(pGetUserObjectInformationA(wstation, UOI_FLAGS, &uoflags, sizeof(uoflags), &len)); return (uoflags.dwFlags & WSF_VISIBLE) != 0; } return IsWindowVisible(desktop); } /* check for wine fake dll module */ static BOOL is_fake_dll( HMODULE module ) { static const char fakedll_signature[] = "Wine placeholder DLL"; const IMAGE_DOS_HEADER *dos; if (!((ULONG_PTR)module & 1)) return FALSE; /* not loaded as datafile */ dos = (const IMAGE_DOS_HEADER *)((const char *)module - 1); if (dos->e_magic != IMAGE_DOS_SIGNATURE) return FALSE; if (dos->e_lfanew >= sizeof(*dos) + sizeof(fakedll_signature) && !memcmp( dos + 1, fakedll_signature, sizeof(fakedll_signature) )) return TRUE; return FALSE; } /* check if Gecko is present, trying to trigger the install if not */ static BOOL gecko_check(void) { IHTMLDocument2 *doc; IHTMLElement *body; BOOL ret = FALSE; CoInitialize( NULL ); if (FAILED( CoCreateInstance( &CLSID_HTMLDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IHTMLDocument2, (void **)&doc ))) return FALSE; if ((ret = SUCCEEDED( IHTMLDocument2_get_body( doc, &body )))) IHTMLElement_Release( body ); IHTMLDocument_Release( doc ); return ret; } static void print_version (void) { #ifdef __i386__ static const char platform[] = "i386"; #elif defined(__x86_64__) static const char platform[] = "x86_64"; #elif defined(__sparc__) static const char platform[] = "sparc"; #elif defined(__ALPHA__) static const char platform[] = "alpha"; #elif defined(__powerpc__) static const char platform[] = "powerpc"; #endif OSVERSIONINFOEX ver; BOOL ext, wow64; int is_win2k3_r2; const char *(CDECL *wine_get_build_id)(void); void (CDECL *wine_get_host_version)( const char **sysname, const char **release ); BOOL (WINAPI *pIsWow64Process)(HANDLE hProcess, PBOOL Wow64Process); ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX); if (!(ext = GetVersionEx ((OSVERSIONINFO *) &ver))) { ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); if (!GetVersionEx ((OSVERSIONINFO *) &ver)) report (R_FATAL, "Can't get OS version."); } pIsWow64Process = (void *)GetProcAddress(GetModuleHandleA("kernel32.dll"),"IsWow64Process"); if (!pIsWow64Process || !pIsWow64Process( GetCurrentProcess(), &wow64 )) wow64 = FALSE; xprintf (" Platform=%s%s\n", platform, wow64 ? " (WOW64)" : ""); xprintf (" bRunningUnderWine=%d\n", running_under_wine ()); xprintf (" bRunningOnVisibleDesktop=%d\n", running_on_visible_desktop ()); xprintf (" dwMajorVersion=%u\n dwMinorVersion=%u\n" " dwBuildNumber=%u\n PlatformId=%u\n szCSDVersion=%s\n", ver.dwMajorVersion, ver.dwMinorVersion, ver.dwBuildNumber, ver.dwPlatformId, ver.szCSDVersion); wine_get_build_id = (void *)GetProcAddress(GetModuleHandleA("ntdll.dll"), "wine_get_build_id"); wine_get_host_version = (void *)GetProcAddress(GetModuleHandleA("ntdll.dll"), "wine_get_host_version"); if (wine_get_build_id) xprintf( " WineBuild=%s\n", wine_get_build_id() ); if (wine_get_host_version) { const char *sysname, *release; wine_get_host_version( &sysname, &release ); xprintf( " Host system=%s\n Host version=%s\n", sysname, release ); } is_win2k3_r2 = GetSystemMetrics(SM_SERVERR2); if(is_win2k3_r2) xprintf(" R2 build number=%d\n", is_win2k3_r2); if (!ext) return; xprintf (" wServicePackMajor=%d\n wServicePackMinor=%d\n" " wSuiteMask=%d\n wProductType=%d\n wReserved=%d\n", ver.wServicePackMajor, ver.wServicePackMinor, ver.wSuiteMask, ver.wProductType, ver.wReserved); } static inline int is_dot_dir(const char* x) { return ((x[0] == '.') && ((x[1] == 0) || ((x[1] == '.') && (x[2] == 0)))); } static void remove_dir (const char *dir) { HANDLE hFind; WIN32_FIND_DATA wfd; char path[MAX_PATH]; size_t dirlen = strlen (dir); /* Make sure the directory exists before going further */ memcpy (path, dir, dirlen); strcpy (path + dirlen++, "\\*"); hFind = FindFirstFile (path, &wfd); if (hFind == INVALID_HANDLE_VALUE) return; do { char *lp = wfd.cFileName; if (!lp[0]) lp = wfd.cAlternateFileName; /* ? FIXME not (!lp) ? */ if (is_dot_dir (lp)) continue; strcpy (path + dirlen, lp); if (FILE_ATTRIBUTE_DIRECTORY & wfd.dwFileAttributes) remove_dir(path); else if (!DeleteFile (path)) report (R_WARNING, "Can't delete file %s: error %d", path, GetLastError ()); } while (FindNextFile (hFind, &wfd)); FindClose (hFind); if (!RemoveDirectory (dir)) report (R_WARNING, "Can't remove directory %s: error %d", dir, GetLastError ()); } static const char* get_test_source_file(const char* test, const char* subtest) { static const char* special_dirs[][2] = { { 0, 0 } }; static char buffer[MAX_PATH]; int i; for (i = 0; special_dirs[i][0]; i++) { if (strcmp(test, special_dirs[i][0]) == 0) { test = special_dirs[i][1]; break; } } snprintf(buffer, sizeof(buffer), "dlls/%s/tests/%s.c", test, subtest); return buffer; } static void* extract_rcdata (LPCTSTR name, LPCTSTR type, DWORD* size) { HRSRC rsrc; HGLOBAL hdl; LPVOID addr; if (!(rsrc = FindResource (NULL, name, type)) || !(*size = SizeofResource (0, rsrc)) || !(hdl = LoadResource (0, rsrc)) || !(addr = LockResource (hdl))) return NULL; return addr; } /* Fills in the name and exename fields */ static void extract_test (struct wine_test *test, const char *dir, LPTSTR res_name) { BYTE* code; DWORD size; char *exepos; HANDLE hfile; DWORD written; code = extract_rcdata (res_name, "TESTRES", &size); if (!code) report (R_FATAL, "Can't find test resource %s: %d", res_name, GetLastError ()); test->name = heap_strdup( res_name ); test->exename = strmake (NULL, "%s\\%s", dir, test->name); exepos = strstr (test->name, testexe); if (!exepos) report (R_FATAL, "Not an .exe file: %s", test->name); *exepos = 0; test->name = heap_realloc (test->name, exepos - test->name + 1); report (R_STEP, "Extracting: %s", test->name); hfile = CreateFileA(test->exename, GENERIC_READ | GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); if (hfile == INVALID_HANDLE_VALUE) report (R_FATAL, "Failed to open file %s.", test->exename); if (!WriteFile(hfile, code, size, &written, NULL)) report (R_FATAL, "Failed to write file %s.", test->exename); CloseHandle(hfile); } static DWORD wait_process( HANDLE process, DWORD timeout ) { DWORD wait, diff = 0, start = GetTickCount(); MSG msg; while (diff < timeout) { wait = MsgWaitForMultipleObjects( 1, &process, FALSE, timeout - diff, QS_ALLINPUT ); if (wait != WAIT_OBJECT_0 + 1) return wait; while (PeekMessageA( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg ); diff = GetTickCount() - start; } return WAIT_TIMEOUT; } static void append_path( const char *path) { char *newpath; newpath = heap_alloc(strlen(curpath) + 1 + strlen(path) + 1); strcpy(newpath, curpath); strcat(newpath, ";"); strcat(newpath, path); SetEnvironmentVariableA("PATH", newpath); heap_free(newpath); } /* Run a command for MS milliseconds. If OUT != NULL, also redirect stdout to there. Return the exit status, -2 if can't create process or the return value of WaitForSingleObject. */ static int run_ex (char *cmd, HANDLE out_file, const char *tempdir, DWORD ms) { STARTUPINFO si; PROCESS_INFORMATION pi; DWORD wait, status; GetStartupInfo (&si); si.dwFlags = STARTF_USESTDHANDLES; si.hStdInput = GetStdHandle( STD_INPUT_HANDLE ); si.hStdOutput = out_file ? out_file : GetStdHandle( STD_OUTPUT_HANDLE ); si.hStdError = out_file ? out_file : GetStdHandle( STD_ERROR_HANDLE ); if (!CreateProcessA (NULL, cmd, NULL, NULL, TRUE, CREATE_DEFAULT_ERROR_MODE, NULL, tempdir, &si, &pi)) return -2; CloseHandle (pi.hThread); status = wait_process( pi.hProcess, ms ); switch (status) { case WAIT_OBJECT_0: GetExitCodeProcess (pi.hProcess, &status); CloseHandle (pi.hProcess); return status; case WAIT_FAILED: report (R_ERROR, "Wait for '%s' failed: %d", cmd, GetLastError ()); break; case WAIT_TIMEOUT: break; default: report (R_ERROR, "Wait returned %d", status); break; } if (!TerminateProcess (pi.hProcess, 257)) report (R_ERROR, "TerminateProcess failed: %d", GetLastError ()); wait = wait_process( pi.hProcess, 5000 ); switch (wait) { case WAIT_OBJECT_0: break; case WAIT_FAILED: report (R_ERROR, "Wait for termination of '%s' failed: %d", cmd, GetLastError ()); break; case WAIT_TIMEOUT: report (R_ERROR, "Can't kill process '%s'", cmd); break; default: report (R_ERROR, "Waiting for termination: %d", wait); break; } CloseHandle (pi.hProcess); return status; } static DWORD get_subtests (const char *tempdir, struct wine_test *test, LPTSTR res_name) { char *cmd; HANDLE subfile; DWORD err, total; char buffer[8192], *index; static const char header[] = "Valid test names:"; int status, allocated; char tmpdir[MAX_PATH], subname[MAX_PATH]; SECURITY_ATTRIBUTES sa; test->subtest_count = 0; if (!GetTempPathA( MAX_PATH, tmpdir ) || !GetTempFileNameA( tmpdir, "sub", 0, subname )) report (R_FATAL, "Can't name subtests file."); /* make handle inheritable */ sa.nLength = sizeof(sa); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; subfile = CreateFileA( subname, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, &sa, CREATE_ALWAYS, 0, NULL ); if ((subfile == INVALID_HANDLE_VALUE) && (GetLastError() == ERROR_INVALID_PARAMETER)) { /* FILE_SHARE_DELETE not supported on win9x */ subfile = CreateFileA( subname, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, &sa, CREATE_ALWAYS, 0, NULL ); } if (subfile == INVALID_HANDLE_VALUE) { err = GetLastError(); report (R_ERROR, "Can't open subtests output of %s: %u", test->name, GetLastError()); goto quit; } extract_test (test, tempdir, res_name); cmd = strmake (NULL, "%s --list", test->exename); if (test->maindllpath) { /* We need to add the path (to the main dll) to PATH */ append_path(test->maindllpath); } status = run_ex (cmd, subfile, tempdir, 5000); err = GetLastError(); if (test->maindllpath) { /* Restore PATH again */ SetEnvironmentVariableA("PATH", curpath); } heap_free (cmd); if (status == -2) { report (R_ERROR, "Cannot run %s error %u", test->exename, err); goto quit; } SetFilePointer( subfile, 0, NULL, FILE_BEGIN ); ReadFile( subfile, buffer, sizeof(buffer), &total, NULL ); CloseHandle( subfile ); if (sizeof buffer == total) { report (R_ERROR, "Subtest list of %s too big.", test->name, sizeof buffer); err = ERROR_OUTOFMEMORY; goto quit; } buffer[total] = 0; index = strstr (buffer, header); if (!index) { report (R_ERROR, "Can't parse subtests output of %s", test->name); err = ERROR_INTERNAL_ERROR; goto quit; } index += sizeof header; allocated = 10; test->subtests = heap_alloc (allocated * sizeof(char*)); index = strtok (index, whitespace); while (index) { if (test->subtest_count == allocated) { allocated *= 2; test->subtests = heap_realloc (test->subtests, allocated * sizeof(char*)); } if (!test_filtered_out( test->name, index )) test->subtests[test->subtest_count++] = heap_strdup(index); index = strtok (NULL, whitespace); } test->subtests = heap_realloc (test->subtests, test->subtest_count * sizeof(char*)); err = 0; quit: if (!DeleteFileA (subname)) report (R_WARNING, "Can't delete file '%s': %u", subname, GetLastError()); return err; } static void run_test (struct wine_test* test, const char* subtest, HANDLE out_file, const char *tempdir) { int status; const char* file = get_test_source_file(test->name, subtest); char *cmd = strmake (NULL, "%s %s", test->exename, subtest); xprintf ("%s:%s start %s -\n", test->name, subtest, file); status = run_ex (cmd, out_file, tempdir, 120000); heap_free (cmd); xprintf ("%s:%s done (%d)\n", test->name, subtest, status); } static BOOL CALLBACK EnumTestFileProc (HMODULE hModule, LPCTSTR lpszType, LPTSTR lpszName, LONG_PTR lParam) { if (!test_filtered_out( lpszName, NULL )) (*(int*)lParam)++; return TRUE; } static const struct clsid_mapping { const char *name; CLSID clsid; } clsid_list[] = { {"oledb32", {0xc8b522d1, 0x5cf3, 0x11ce, {0xad, 0xe5, 0x00, 0xaa, 0x00, 0x44, 0x77, 0x3d}}}, {NULL, {0, 0, 0, {0,0,0,0,0,0,0,0}}} }; static BOOL get_main_clsid(const char *name, CLSID *clsid) { const struct clsid_mapping *mapping; for(mapping = clsid_list; mapping->name; mapping++) { if(!strcasecmp(name, mapping->name)) { *clsid = mapping->clsid; return TRUE; } } return FALSE; } static HMODULE load_com_dll(const char *name, char **path, char *filename) { HMODULE dll = NULL; HKEY hkey; char keyname[100]; char dllname[MAX_PATH]; char *p; CLSID clsid; if(!get_main_clsid(name, &clsid)) return NULL; sprintf(keyname, "CLSID\\{%08x-%04x-%04x-%02x%2x-%02x%2x%02x%2x%02x%2x}\\InprocServer32", clsid.Data1, clsid.Data2, clsid.Data3, clsid.Data4[0], clsid.Data4[1], clsid.Data4[2], clsid.Data4[3], clsid.Data4[4], clsid.Data4[5], clsid.Data4[6], clsid.Data4[7]); if(RegOpenKeyA(HKEY_CLASSES_ROOT, keyname, &hkey) == ERROR_SUCCESS) { LONG size = sizeof(dllname); if(RegQueryValueA(hkey, NULL, dllname, &size) == ERROR_SUCCESS) { if ((dll = LoadLibraryExA(dllname, NULL, LOAD_LIBRARY_AS_DATAFILE))) { strcpy( filename, dllname ); p = strrchr(dllname, '\\'); if (p) *p = 0; *path = heap_strdup( dllname ); } } RegCloseKey(hkey); } return dll; } static void get_dll_path(HMODULE dll, char **path, char *filename) { char dllpath[MAX_PATH]; GetModuleFileNameA(dll, dllpath, MAX_PATH); strcpy(filename, dllpath); *strrchr(dllpath, '\\') = '\0'; *path = heap_strdup( dllpath ); } static BOOL CALLBACK extract_test_proc (HMODULE hModule, LPCTSTR lpszType, LPTSTR lpszName, LONG_PTR lParam) { const char *tempdir = (const char *)lParam; char dllname[MAX_PATH]; char filename[MAX_PATH]; WCHAR dllnameW[MAX_PATH]; HMODULE dll; DWORD err; if (test_filtered_out( lpszName, NULL )) return TRUE; /* Check if the main dll is present on this system */ CharLowerA(lpszName); strcpy(dllname, lpszName); *strstr(dllname, testexe) = 0; wine_tests[nr_of_files].maindllpath = NULL; strcpy(filename, dllname); dll = LoadLibraryExA(dllname, NULL, LOAD_LIBRARY_AS_DATAFILE); if (!dll) dll = load_com_dll(dllname, &wine_tests[nr_of_files].maindllpath, filename); if (dll && running_under_wine() && ((ULONG_PTR)dll & 1)) { /* builtin dlls can't be loaded as datafile, so we must have native or fake dll */ if (!is_fake_dll(dll)) { FreeLibrary(dll); xprintf (" %s=load error Configured as native\n", dllname); return TRUE; } } if (!dll && pLoadLibraryShim) { MultiByteToWideChar(CP_ACP, 0, dllname, -1, dllnameW, MAX_PATH); if (FAILED( pLoadLibraryShim(dllnameW, NULL, NULL, &dll) )) dll = 0; else get_dll_path(dll, &wine_tests[nr_of_files].maindllpath, filename); } if (!dll) { xprintf (" %s=dll is missing\n", dllname); return TRUE; } if (!strcmp( dllname, "mshtml" ) && running_under_wine() && !gecko_check()) { FreeLibrary(dll); xprintf (" %s=load error Gecko is not installed\n", dllname); return TRUE; } FreeLibrary(dll); if (!(err = get_subtests( tempdir, &wine_tests[nr_of_files], lpszName ))) { xprintf (" %s=%s\n", dllname, get_file_version(filename)); nr_of_tests += wine_tests[nr_of_files].subtest_count; nr_of_files++; } else { xprintf (" %s=load error %u\n", dllname, err); } return TRUE; } static char * run_tests (char *logname, char *outdir) { int i; char *strres, *eol, *nextline; DWORD strsize; SECURITY_ATTRIBUTES sa; char tmppath[MAX_PATH], tempdir[MAX_PATH+4]; DWORD needed; /* Get the current PATH only once */ needed = GetEnvironmentVariableA("PATH", NULL, 0); curpath = heap_alloc(needed); GetEnvironmentVariableA("PATH", curpath, needed); SetErrorMode (SEM_FAILCRITICALERRORS | SEM_NOGPFAULTERRORBOX); if (!GetTempPathA( MAX_PATH, tmppath )) report (R_FATAL, "Can't name temporary dir (check %%TEMP%%)."); if (!logname) { static char tmpname[MAX_PATH]; if (!GetTempFileNameA( tmppath, "res", 0, tmpname )) report (R_FATAL, "Can't name logfile."); logname = tmpname; } report (R_OUT, logname); /* make handle inheritable */ sa.nLength = sizeof(sa); sa.lpSecurityDescriptor = NULL; sa.bInheritHandle = TRUE; logfile = CreateFileA( logname, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, &sa, CREATE_ALWAYS, 0, NULL ); if ((logfile == INVALID_HANDLE_VALUE) && (GetLastError() == ERROR_INVALID_PARAMETER)) { /* FILE_SHARE_DELETE not supported on win9x */ logfile = CreateFileA( logname, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, &sa, CREATE_ALWAYS, 0, NULL ); } if (logfile == INVALID_HANDLE_VALUE) report (R_FATAL, "Could not open logfile: %u", GetLastError()); /* try stable path for ZoneAlarm */ if (!outdir) { strcpy( tempdir, tmppath ); strcat( tempdir, "wct" ); if (!CreateDirectoryA( tempdir, NULL )) { if (!GetTempFileNameA( tmppath, "wct", 0, tempdir )) report (R_FATAL, "Can't name temporary dir (check %%TEMP%%)."); DeleteFileA( tempdir ); if (!CreateDirectoryA( tempdir, NULL )) report (R_FATAL, "Could not create directory: %s", tempdir); } } else strcpy( tempdir, outdir); report (R_DIR, tempdir); xprintf ("Version 4\n"); xprintf ("Tests from build %s\n", build_id[0] ? build_id : "-" ); xprintf ("Archive: -\n"); /* no longer used */ xprintf ("Tag: %s\n", tag); xprintf ("Build info:\n"); strres = extract_rcdata ("BUILD_INFO", "STRINGRES", &strsize); while (strres) { eol = memchr (strres, '\n', strsize); if (!eol) { nextline = NULL; eol = strres + strsize; } else { strsize -= eol - strres + 1; nextline = strsize?eol+1:NULL; if (eol > strres && *(eol-1) == '\r') eol--; } xprintf (" %.*s\n", eol-strres, strres); strres = nextline; } xprintf ("Operating system version:\n"); print_version (); xprintf ("Dll info:\n" ); report (R_STATUS, "Counting tests"); if (!EnumResourceNames (NULL, "TESTRES", EnumTestFileProc, (LPARAM)&nr_of_files)) report (R_FATAL, "Can't enumerate test files: %d", GetLastError ()); wine_tests = heap_alloc (nr_of_files * sizeof wine_tests[0]); /* Do this only once during extraction (and version checking) */ hmscoree = LoadLibraryA("mscoree.dll"); pLoadLibraryShim = NULL; if (hmscoree) pLoadLibraryShim = (void *)GetProcAddress(hmscoree, "LoadLibraryShim"); report (R_STATUS, "Extracting tests"); report (R_PROGRESS, 0, nr_of_files); nr_of_files = 0; nr_of_tests = 0; if (!EnumResourceNames (NULL, "TESTRES", extract_test_proc, (LPARAM)tempdir)) report (R_FATAL, "Can't enumerate test files: %d", GetLastError ()); FreeLibrary(hmscoree); xprintf ("Test output:\n" ); report (R_DELTA, 0, "Extracting: Done"); report (R_STATUS, "Running tests"); report (R_PROGRESS, 1, nr_of_tests); for (i = 0; i < nr_of_files; i++) { struct wine_test *test = wine_tests + i; int j; if (test->maindllpath) { /* We need to add the path (to the main dll) to PATH */ append_path(test->maindllpath); } for (j = 0; j < test->subtest_count; j++) { report (R_STEP, "Running: %s:%s", test->name, test->subtests[j]); run_test (test, test->subtests[j], logfile, tempdir); } if (test->maindllpath) { /* Restore PATH again */ SetEnvironmentVariableA("PATH", curpath); } } report (R_DELTA, 0, "Running: Done"); report (R_STATUS, "Cleaning up"); CloseHandle( logfile ); logfile = 0; if (!outdir) remove_dir (tempdir); heap_free(wine_tests); heap_free(curpath); return logname; } static BOOL WINAPI ctrl_handler(DWORD ctrl_type) { if (ctrl_type == CTRL_C_EVENT) { printf("Ignoring Ctrl-C, use Ctrl-Break if you really want to terminate\n"); return TRUE; } return FALSE; } static BOOL CALLBACK extract_only_proc (HMODULE hModule, LPCTSTR lpszType, LPTSTR lpszName, LONG_PTR lParam) { const char *target_dir = (const char *)lParam; char filename[MAX_PATH]; if (test_filtered_out( lpszName, NULL )) return TRUE; strcpy(filename, lpszName); CharLowerA(filename); extract_test( &wine_tests[nr_of_files], target_dir, filename ); nr_of_files++; return TRUE; } static void extract_only (const char *target_dir) { BOOL res; report (R_DIR, target_dir); res = CreateDirectoryA( target_dir, NULL ); if (!res && GetLastError() != ERROR_ALREADY_EXISTS) report (R_FATAL, "Could not create directory: %s (%d)", target_dir, GetLastError ()); nr_of_files = 0; report (R_STATUS, "Counting tests"); if (!EnumResourceNames (NULL, "TESTRES", EnumTestFileProc, (LPARAM)&nr_of_files)) report (R_FATAL, "Can't enumerate test files: %d", GetLastError ()); wine_tests = heap_alloc (nr_of_files * sizeof wine_tests[0] ); report (R_STATUS, "Extracting tests"); report (R_PROGRESS, 0, nr_of_files); nr_of_files = 0; if (!EnumResourceNames (NULL, "TESTRES", extract_only_proc, (LPARAM)target_dir)) report (R_FATAL, "Can't enumerate test files: %d", GetLastError ()); report (R_DELTA, 0, "Extracting: Done"); } static void usage (void) { fprintf (stderr, "Usage: winetest [OPTION]... [TESTS]\n\n" " --help print this message and exit\n" " --version print the build version and exit\n" " -c console mode, no GUI\n" " -d DIR Use DIR as temp directory (default: %%TEMP%%\\wct)\n" " -e preserve the environment\n" " -h print this message and exit\n" " -p shutdown when the tests are done\n" " -q quiet mode, no output at all\n" " -o FILE put report into FILE, do not submit\n" " -s FILE submit FILE, do not run tests\n" " -t TAG include TAG of characters [-.0-9a-zA-Z] in the report\n" " -x DIR Extract tests to DIR (default: .\\wct) and exit\n"); } int main( int argc, char *argv[] ) { char *logname = NULL, *outdir = NULL; const char *extract = NULL; const char *cp, *submit = NULL; int reset_env = 1; int poweroff = 0; int interactive = 1; int i; if (!LoadStringA( 0, IDS_BUILD_ID, build_id, sizeof(build_id) )) build_id[0] = 0; for (i = 1; i < argc && argv[i]; i++) { if (!strcmp(argv[i], "--help")) { usage (); exit (0); } else if (!strcmp(argv[i], "--version")) { printf("%-12.12s\n", build_id[0] ? build_id : "unknown"); exit (0); } else if ((argv[i][0] != '-' && argv[i][0] != '/') || argv[i][2]) { if (nb_filters == sizeof(filters)/sizeof(filters[0])) { report (R_ERROR, "Too many test filters specified"); exit (2); } filters[nb_filters++] = argv[i]; } else switch (argv[i][1]) { case 'c': report (R_TEXTMODE); interactive = 0; break; case 'e': reset_env = 0; break; case 'h': case '?': usage (); exit (0); case 'p': poweroff = 1; break; case 'q': report (R_QUIET); interactive = 0; break; case 's': if (!(submit = argv[++i])) { usage(); exit( 2 ); } if (tag) report (R_WARNING, "ignoring tag for submission"); send_file (submit); break; case 'o': if (!(logname = argv[++i])) { usage(); exit( 2 ); } break; case 't': if (!(tag = argv[++i])) { usage(); exit( 2 ); } if (strlen (tag) > MAXTAGLEN) report (R_FATAL, "tag is too long (maximum %d characters)", MAXTAGLEN); cp = findbadtagchar (tag); if (cp) { report (R_ERROR, "invalid char in tag: %c", *cp); usage (); exit (2); } break; case 'x': report (R_TEXTMODE); if (!(extract = argv[++i])) extract = ".\\wct"; extract_only (extract); break; case 'd': outdir = argv[++i]; break; default: report (R_ERROR, "invalid option: -%c", argv[i][1]); usage (); exit (2); } } if (!submit && !extract) { report (R_STATUS, "Starting up"); if (!running_on_visible_desktop ()) report (R_FATAL, "Tests must be run on a visible desktop"); SetConsoleCtrlHandler(ctrl_handler, TRUE); if (reset_env) { SetEnvironmentVariableA( "WINETEST_PLATFORM", running_under_wine () ? "wine" : "windows" ); SetEnvironmentVariableA( "WINETEST_DEBUG", "1" ); SetEnvironmentVariableA( "WINETEST_INTERACTIVE", "0" ); SetEnvironmentVariableA( "WINETEST_REPORT_SUCCESS", "0" ); } if (!nb_filters) /* don't submit results when filtering */ { while (!tag) { if (!interactive) report (R_FATAL, "Please specify a tag (-t option) if " "running noninteractive!"); if (guiAskTag () == IDABORT) exit (1); } report (R_TAG); if (!build_id[0]) report( R_WARNING, "You won't be able to submit results without a valid build id.\n" "To submit results, winetest needs to be built from a git checkout." ); } if (!logname) { logname = run_tests (NULL, outdir); if (build_id[0] && !nb_filters && report (R_ASK, MB_YESNO, "Do you want to submit the test results?") == IDYES) if (!send_file (logname) && !DeleteFileA(logname)) report (R_WARNING, "Can't remove logfile: %u", GetLastError()); } else run_tests (logname, outdir); report (R_STATUS, "Finished"); } if (poweroff) { HANDLE hToken; TOKEN_PRIVILEGES npr; /* enable the shutdown privilege for the current process */ if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken)) { LookupPrivilegeValueA(0, SE_SHUTDOWN_NAME, &npr.Privileges[0].Luid); npr.PrivilegeCount = 1; npr.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED; AdjustTokenPrivileges(hToken, FALSE, &npr, 0, 0, 0); CloseHandle(hToken); } ExitWindowsEx(EWX_SHUTDOWN | EWX_POWEROFF | EWX_FORCEIFHUNG, SHTDN_REASON_MAJOR_OTHER | SHTDN_REASON_MINOR_OTHER); } exit (0); }