/* * 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 <stdio.h> #include <assert.h> #include <windows.h> #include <mshtml.h> #include "winetest.h" #include "resource.h" /* Don't submit the results if more than SKIP_LIMIT tests have been skipped */ #define SKIP_LIMIT 10 /* Don't submit the results if more than FAILURES_LIMIT tests have failed */ #define FAILURES_LIMIT 50 struct wine_test { char *name; int subtest_count; char **subtests; char *exename; char *maindllpath; }; char *tag = NULL; char *description = NULL; char *url = NULL; char *email = NULL; BOOL aborting = FALSE; static struct wine_test *wine_tests; static int nr_of_files, nr_of_tests, nr_of_skips; static int nr_native_dlls; static const char whitespace[] = " \t\r\n"; static const char testexe[] = "_test.exe"; static char build_id[64]; static BOOL is_wow64; static int failures; /* filters for running only specific tests */ static char *filters[64]; static unsigned int nb_filters = 0; static BOOL exclude_tests = FALSE; /* Needed to check for .NET dlls */ static HMODULE hmscoree; static HRESULT (WINAPI *pLoadLibraryShim)(LPCWSTR, LPCWSTR, LPVOID, HMODULE *); /* For SxS DLLs e.g. msvcr90 */ static HANDLE (WINAPI *pCreateActCtxA)(PACTCTXA); static BOOL (WINAPI *pActivateActCtx)(HANDLE, ULONG_PTR *); static BOOL (WINAPI *pDeactivateActCtx)(DWORD, ULONG_PTR); static void (WINAPI *pReleaseActCtx)(HANDLE); /* 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 exclude_tests; for (i = 0; i < nb_filters; i++) { if (!strncmp( dllname, filters[i], len )) { if (!filters[i][len]) return exclude_tests; if (filters[i][len] != ':') continue; if (testname && !strcmp( testname, &filters[i][len+1] )) return exclude_tests; if (!testname && !exclude_tests) return FALSE; } } return !exclude_tests; } 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 const 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 BOOL running_under_wine (void) { HMODULE module = GetModuleHandleA("ntdll.dll"); if (!module) return FALSE; return (GetProcAddress(module, "wine_server_call") != NULL); } static BOOL check_mount_mgr(void) { HANDLE handle = CreateFileA( "\\\\.\\MountPointManager", GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, 0 ); if (handle == INVALID_HANDLE_VALUE) return FALSE; CloseHandle( handle ); return TRUE; } static BOOL check_wow64_registry(void) { char buffer[MAX_PATH]; DWORD type, size = MAX_PATH; HKEY hkey; BOOL ret; if (!is_wow64) return TRUE; if (RegOpenKeyA( HKEY_LOCAL_MACHINE, "Software\\Microsoft\\Windows\\CurrentVersion", &hkey )) return FALSE; ret = !RegQueryValueExA( hkey, "ProgramFilesDir (x86)", NULL, &type, (BYTE *)buffer, &size ); RegCloseKey( hkey ); return ret; } static BOOL check_display_driver(void) { HWND hwnd = CreateWindowA( "STATIC", "", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, 0, 0, GetModuleHandleA(0), 0 ); if (!hwnd) return FALSE; DestroyWindow( hwnd ); return TRUE; } static BOOL running_on_visible_desktop (void) { HWND desktop; HMODULE huser32 = GetModuleHandleA("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); } static int running_as_admin (void) { PSID administrators = NULL; SID_IDENTIFIER_AUTHORITY nt_authority = { SECURITY_NT_AUTHORITY }; HANDLE token; DWORD groups_size; PTOKEN_GROUPS groups; DWORD group_index; /* Create a well-known SID for the Administrators group. */ if (! AllocateAndInitializeSid(&nt_authority, 2, SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &administrators)) return -1; /* Get the process token */ if (! OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &token)) { FreeSid(administrators); return -1; } /* Get the group info from the token */ groups_size = 0; GetTokenInformation(token, TokenGroups, NULL, 0, &groups_size); groups = heap_alloc(groups_size); if (groups == NULL) { CloseHandle(token); FreeSid(administrators); return -1; } if (! GetTokenInformation(token, TokenGroups, groups, groups_size, &groups_size)) { heap_free(groups); CloseHandle(token); FreeSid(administrators); return -1; } CloseHandle(token); /* Now check if the token groups include the Administrators group */ for (group_index = 0; group_index < groups->GroupCount; group_index++) { if (EqualSid(groups->Groups[group_index].Sid, administrators)) { heap_free(groups); FreeSid(administrators); return 1; } } /* If we end up here we didn't find the Administrators group */ heap_free(groups); FreeSid(administrators); return 0; } static int running_elevated (void) { HANDLE token; TOKEN_ELEVATION elevation_info; DWORD size; /* Get the process token */ if (! OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &token)) return -1; /* Get the elevation info from the token */ if (! GetTokenInformation(token, TokenElevation, &elevation_info, sizeof(TOKEN_ELEVATION), &size)) { CloseHandle(token); return -1; } CloseHandle(token); return elevation_info.TokenIsElevated; } /* check for native dll when running under wine */ static BOOL is_native_dll( HMODULE module ) { static const char fakedll_signature[] = "Wine placeholder DLL"; const IMAGE_DOS_HEADER *dos; if (!running_under_wine()) return FALSE; if (!((ULONG_PTR)module & 1)) return FALSE; /* not loaded as datafile */ /* builtin dlls can't be loaded as datafile, so we must have native or fake dll */ 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 FALSE; return TRUE; } static void print_version (void) { #ifdef __i386__ static const char platform[] = "i386"; #elif defined(__x86_64__) static const char platform[] = "x86_64"; #elif defined(__powerpc__) static const char platform[] = "powerpc"; #elif defined(__arm__) static const char platform[] = "arm"; #elif defined(__aarch64__) static const char platform[] = "arm64"; #else # error CPU unknown #endif OSVERSIONINFOEXA ver; BOOL ext; int is_win2k3_r2, is_admin, is_elevated; const char *(CDECL *wine_get_build_id)(void); void (CDECL *wine_get_host_version)( const char **sysname, const char **release ); BOOL (WINAPI *pGetProductInfo)(DWORD, DWORD, DWORD, DWORD, DWORD *); ver.dwOSVersionInfoSize = sizeof(ver); if (!(ext = GetVersionExA ((OSVERSIONINFOA *) &ver))) { ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFOA); if (!GetVersionExA ((OSVERSIONINFOA *) &ver)) report (R_FATAL, "Can't get OS version."); } xprintf (" Platform=%s%s\n", platform, is_wow64 ? " (WOW64)" : ""); xprintf (" bRunningUnderWine=%d\n", running_under_wine ()); xprintf (" bRunningOnVisibleDesktop=%d\n", running_on_visible_desktop ()); is_admin = running_as_admin (); if (0 <= is_admin) { xprintf (" Account=%s", is_admin ? "admin" : "non-admin"); is_elevated = running_elevated (); if (0 <= is_elevated) xprintf(", %s", is_elevated ? "elevated" : "not elevated"); xprintf ("\n"); } xprintf (" Submitter=%s\n", email ); if (description) xprintf (" Description=%s\n", description ); if (url) xprintf (" URL=%s\n", url ); 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); pGetProductInfo = (void *)GetProcAddress(GetModuleHandleA("kernel32.dll"),"GetProductInfo"); if (pGetProductInfo && !running_under_wine()) { DWORD prodtype = 0; pGetProductInfo(ver.dwMajorVersion, ver.dwMinorVersion, ver.wServicePackMajor, ver.wServicePackMinor, &prodtype); xprintf(" dwProductInfo=%u\n", prodtype); } } static void print_language(void) { HMODULE hkernel32; LANGID (WINAPI *pGetUserDefaultUILanguage)(void); LANGID (WINAPI *pGetThreadUILanguage)(void); xprintf (" SystemDefaultLCID=%x\n", GetSystemDefaultLCID()); xprintf (" UserDefaultLCID=%x\n", GetUserDefaultLCID()); xprintf (" ThreadLocale=%x\n", GetThreadLocale()); hkernel32 = GetModuleHandleA("kernel32.dll"); pGetUserDefaultUILanguage = (void*)GetProcAddress(hkernel32, "GetUserDefaultUILanguage"); pGetThreadUILanguage = (void*)GetProcAddress(hkernel32, "GetThreadUILanguage"); if (pGetUserDefaultUILanguage) xprintf (" UserDefaultUILanguage=%x\n", pGetUserDefaultUILanguage()); if (pGetThreadUILanguage) xprintf (" ThreadUILanguage=%x\n", pGetThreadUILanguage()); } static inline BOOL 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_DATAA 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 = FindFirstFileA (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 (!DeleteFileA(path)) report (R_WARNING, "Can't delete file %s: error %d", path, GetLastError ()); } while (FindNextFileA(hFind, &wfd)); FindClose (hFind); if (!RemoveDirectoryA(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, len = strlen(test); if (len > 4 && !strcmp( test + len - 4, ".exe" )) { len = sprintf(buffer, "programs/%s", test) - 4; buffer[len] = 0; } else len = sprintf(buffer, "dlls/%s", test); for (i = 0; special_dirs[i][0]; i++) { if (strcmp(test, special_dirs[i][0]) == 0) { strcpy( buffer, special_dirs[i][1] ); len = strlen(buffer); break; } } sprintf(buffer + len, "/tests/%s.c", subtest); return buffer; } static void* extract_rcdata (LPCSTR name, LPCSTR type, DWORD* size) { HRSRC rsrc; HGLOBAL hdl; LPVOID addr; if (!(rsrc = FindResourceA(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, LPSTR 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 )) DispatchMessageA( &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) { STARTUPINFOA si; PROCESS_INFORMATION pi; DWORD wait, status; GetStartupInfoA (&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, LPSTR 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; } 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); CloseHandle( subfile ); 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*)); } 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) { const char* file = get_test_source_file(test->name, subtest); if (test_filtered_out( test->name, subtest )) { report (R_STEP, "Skipping: %s:%s", test->name, subtest); xprintf ("%s:%s skipped %s -\n", test->name, subtest, file); nr_of_skips++; } else { int status; DWORD start = GetTickCount(); char *cmd = strmake (NULL, "%s %s", test->exename, subtest); report (R_STEP, "Running: %s:%s", test->name, 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) in %ds\n", test->name, subtest, status, (GetTickCount()-start)/1000); if (status) failures++; } if (failures) report (R_STATUS, "Running tests - %u failures", failures); } static BOOL CALLBACK EnumTestFileProc (HMODULE hModule, LPCSTR lpszType, LPSTR 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, LPCSTR lpszType, LPSTR 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; HANDLE actctx; ULONG_PTR cookie; if (aborting) return TRUE; /* Check if the main dll is present on this system */ CharLowerA(lpszName); strcpy(dllname, lpszName); *strstr(dllname, testexe) = 0; if (test_filtered_out( lpszName, NULL )) { nr_of_skips++; xprintf (" %s=skipped\n", dllname); return TRUE; } extract_test (&wine_tests[nr_of_files], tempdir, lpszName); if (pCreateActCtxA != NULL && pActivateActCtx != NULL && pDeactivateActCtx != NULL && pReleaseActCtx != NULL) { ACTCTXA actctxinfo; memset(&actctxinfo, 0, sizeof(ACTCTXA)); actctxinfo.cbSize = sizeof(ACTCTXA); actctxinfo.dwFlags = ACTCTX_FLAG_RESOURCE_NAME_VALID; actctxinfo.lpSource = wine_tests[nr_of_files].exename; actctxinfo.lpResourceName = (LPSTR)CREATEPROCESS_MANIFEST_RESOURCE_ID; actctx = pCreateActCtxA(&actctxinfo); if (actctx != INVALID_HANDLE_VALUE && ! pActivateActCtx(actctx, &cookie)) { pReleaseActCtx(actctx); actctx = INVALID_HANDLE_VALUE; } } else actctx = INVALID_HANDLE_VALUE; 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 && pLoadLibraryShim) { MultiByteToWideChar(CP_ACP, 0, dllname, -1, dllnameW, MAX_PATH); if (SUCCEEDED( pLoadLibraryShim(dllnameW, NULL, NULL, &dll) ) && dll) { get_dll_path(dll, &wine_tests[nr_of_files].maindllpath, filename); FreeLibrary(dll); dll = LoadLibraryExA(filename, NULL, LOAD_LIBRARY_AS_DATAFILE); } else dll = 0; } if (!dll) { xprintf (" %s=dll is missing\n", dllname); if (actctx != INVALID_HANDLE_VALUE) { pDeactivateActCtx(0, cookie); pReleaseActCtx(actctx); } return TRUE; } if (is_native_dll(dll)) { FreeLibrary(dll); xprintf (" %s=load error Configured as native\n", dllname); nr_native_dlls++; if (actctx != INVALID_HANDLE_VALUE) { pDeactivateActCtx(0, cookie); pReleaseActCtx(actctx); } 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); } if (actctx != INVALID_HANDLE_VALUE) { pDeactivateActCtx(0, cookie); pReleaseActCtx(actctx); } 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; HMODULE kernel32; /* 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 (); print_language (); xprintf ("Dll info:\n" ); report (R_STATUS, "Counting tests"); if (!EnumResourceNamesA (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"); kernel32 = GetModuleHandleA("kernel32.dll"); pCreateActCtxA = (void *)GetProcAddress(kernel32, "CreateActCtxA"); pActivateActCtx = (void *)GetProcAddress(kernel32, "ActivateActCtx"); pDeactivateActCtx = (void *)GetProcAddress(kernel32, "DeactivateActCtx"); pReleaseActCtx = (void *)GetProcAddress(kernel32, "ReleaseActCtx"); report (R_STATUS, "Extracting tests"); report (R_PROGRESS, 0, nr_of_files); nr_of_files = 0; nr_of_tests = 0; nr_of_skips = 0; if (!EnumResourceNamesA (NULL, "TESTRES", extract_test_proc, (LPARAM)tempdir)) report (R_FATAL, "Can't enumerate test files: %d", GetLastError ()); FreeLibrary(hmscoree); if (aborting) return logname; xprintf ("Test output:\n" ); report (R_DELTA, 0, "Extracting: Done"); if (nr_native_dlls) report( R_WARNING, "Some dlls are configured as native, you won't be able to submit results." ); 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 (aborting) break; 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++) { if (aborting) break; 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 - %u failures", failures); 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, LPCSTR lpszType, LPSTR 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 (!EnumResourceNamesA(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 (!EnumResourceNamesA(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" " -i INFO an optional description of the test platform\n" " -m MAIL an email address to enable developers to contact you\n" " -n exclude the specified tests\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" " -u URL include TestBot URL in the report\n" " -x DIR Extract tests to DIR (default: .\\wct) and exit\n"); } int main( int argc, char *argv[] ) { BOOL (WINAPI *pIsWow64Process)(HANDLE hProcess, PBOOL Wow64Process); 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; pIsWow64Process = (void *)GetProcAddress(GetModuleHandleA("kernel32.dll"),"IsWow64Process"); if (!pIsWow64Process || !pIsWow64Process( GetCurrentProcess(), &is_wow64 )) is_wow64 = FALSE; 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 'i': if (!(description = argv[++i])) { usage(); exit( 2 ); } break; case 'm': if (!(email = argv[++i])) { usage(); exit( 2 ); } break; case 'n': exclude_tests = TRUE; break; 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 'u': if (!(url = argv[++i])) { 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) { int is_win9x = (GetVersion() & 0x80000000) != 0; report (R_STATUS, "Starting up"); if (is_win9x) report (R_WARNING, "Running on win9x is not supported. You won't be able to submit results."); if (!running_on_visible_desktop ()) report (R_FATAL, "Tests must be run on a visible desktop"); if (running_under_wine()) { if (!check_mount_mgr()) report (R_FATAL, "Mount manager not running, most likely your WINEPREFIX wasn't created correctly."); if (!check_wow64_registry()) report (R_FATAL, "WoW64 keys missing, most likely your WINEPREFIX wasn't created correctly."); if (!check_display_driver()) report (R_FATAL, "Unable to create a window, the display driver is not working."); } 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" ); } while (!tag) { if (!interactive) report (R_FATAL, "Please specify a tag (-t option) if " "running noninteractive!"); if (guiAskTag () == IDABORT) exit (1); } report (R_TAG); while (!email) { if (!interactive) report (R_FATAL, "Please specify an email address (-m option) to enable developers\n" " to contact you about your report if necessary."); if (guiAskEmail () == IDABORT) exit (1); } 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 (aborting) { DeleteFileA(logname); exit (0); } if (failures > FAILURES_LIMIT) report( R_WARNING, "%d tests failed, there's probably something broken with your setup.\n" "You need to address this before submitting results.", failures ); if (build_id[0] && nr_of_skips <= SKIP_LIMIT && failures <= FAILURES_LIMIT && !nr_native_dlls && !is_win9x && 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 - %u failures", failures); } if (poweroff) { HANDLE hToken; TOKEN_PRIVILEGES npr; /* enable the shutdown privilege for the current process */ if (OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken)) { LookupPrivilegeValueA(0, "SeShutdownPrivilege", &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); }