418 lines
13 KiB
C
418 lines
13 KiB
C
/*
|
|
* Copyright 2018 Fabian Maurer
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#define COBJMACROS
|
|
#include <stdio.h>
|
|
|
|
#include "windows.h"
|
|
#include "ole2.h"
|
|
#include "mscoree.h"
|
|
#include "corerror.h"
|
|
#include "shlwapi.h"
|
|
#include "shlobj.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
#include "initguid.h"
|
|
#include "interfaces.h"
|
|
|
|
HMODULE hmscoree;
|
|
|
|
DEFINE_GUID(IID_ITest2, 0x50adb433, 0xf6c5, 0x3b30, 0x92,0x0a, 0x55,0x57,0x11,0x86,0x75,0x09);
|
|
|
|
typedef enum _run_type
|
|
{
|
|
run_type_current_working_directory = 0,
|
|
run_type_exe_directory,
|
|
run_type_system32,
|
|
} run_type;
|
|
|
|
static BOOL write_resource_file(const char *path_tmp, const char *name_res, const char *name_file, char *path_file)
|
|
{
|
|
HRSRC rsrc;
|
|
void *rsrc_data;
|
|
DWORD rsrc_size;
|
|
BOOL ret;
|
|
HANDLE hfile;
|
|
|
|
rsrc = FindResourceA(GetModuleHandleA(NULL), name_res, (LPCSTR)RT_RCDATA);
|
|
if (!rsrc) return FALSE;
|
|
|
|
rsrc_data = LockResource(LoadResource(GetModuleHandleA(NULL), rsrc));
|
|
if (!rsrc_data) return FALSE;
|
|
|
|
rsrc_size = SizeofResource(GetModuleHandleA(NULL), rsrc);
|
|
if (!rsrc_size) return FALSE;
|
|
|
|
strcpy(path_file, path_tmp);
|
|
PathAppendA(path_file, name_file);
|
|
hfile = CreateFileA(path_file, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
|
|
if (hfile == INVALID_HANDLE_VALUE) return FALSE;
|
|
|
|
ret = WriteFile(hfile, rsrc_data, rsrc_size, &rsrc_size, NULL);
|
|
|
|
CloseHandle(hfile);
|
|
return ret;
|
|
}
|
|
|
|
static BOOL compile_cs_to_dll(char *source_path, char *dest_path)
|
|
{
|
|
const char *path_csc = "C:\\windows\\Microsoft.NET\\Framework\\v2.0.50727\\csc.exe";
|
|
char cmdline[2 * MAX_PATH + 74];
|
|
char path_temp[MAX_PATH];
|
|
PROCESS_INFORMATION pi;
|
|
STARTUPINFOA si = { 0 };
|
|
BOOL ret;
|
|
|
|
if (!PathFileExistsA(path_csc))
|
|
{
|
|
skip("Can't find csc.exe\n");
|
|
return FALSE;
|
|
}
|
|
|
|
GetTempPathA(MAX_PATH, path_temp);
|
|
PathAppendA(path_temp, "comtest.dll");
|
|
|
|
sprintf(cmdline, "%s /t:library /out:\"%s\" \"%s\"", path_csc, path_temp, source_path);
|
|
|
|
si.cb = sizeof(si);
|
|
ret = CreateProcessA(path_csc, cmdline, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi);
|
|
ok(ret, "Could not create process: %u\n", GetLastError());
|
|
|
|
WaitForSingleObject(pi.hProcess, 5000);
|
|
CloseHandle(pi.hThread);
|
|
CloseHandle(pi.hProcess);
|
|
|
|
ret = PathFileExistsA(path_temp);
|
|
ok(ret, "Compilation failed\n");
|
|
|
|
ret = MoveFileA(path_temp, dest_path);
|
|
ok(ret, "Moving temporary file failed\n");
|
|
return ret;
|
|
}
|
|
|
|
static void run_test(BOOL expect_success)
|
|
{
|
|
typedef HRESULT (WINAPI *_DllGetClassObject)(REFCLSID rclsid, REFIID riid, LPVOID *ppv);
|
|
ITest *test = NULL;
|
|
HRESULT hr;
|
|
_DllGetClassObject getClassObject;
|
|
IClassFactory *classFactory = NULL;
|
|
HRESULT result_expected = expect_success ? S_OK : HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
|
|
hr = CoCreateInstance(&CLSID_Test, NULL, CLSCTX_INPROC_SERVER, &IID_ITest, (void**)&test);
|
|
todo_wine_if(!expect_success)
|
|
ok(hr == result_expected, "Expected %x, got %x\n", result_expected, hr);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
int i = 0;
|
|
hr = ITest_Func(test, &i);
|
|
ok(hr == S_OK, "Got %x\n", hr);
|
|
ok(i == 42, "Expected 42, got %d\n", i);
|
|
ITest_Release(test);
|
|
}
|
|
|
|
getClassObject = (_DllGetClassObject)GetProcAddress(hmscoree, "DllGetClassObject");
|
|
hr = getClassObject(&CLSID_Test, &IID_IClassFactory, (void **)&classFactory);
|
|
todo_wine_if(!expect_success)
|
|
ok(hr == result_expected, "Expected %x, got %x\n", result_expected, hr);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
ITest *test2 = NULL;
|
|
hr = IClassFactory_CreateInstance(classFactory, NULL, &IID_ITest, (void **)&test2);
|
|
todo_wine_if(!expect_success)
|
|
ok(hr == S_OK, "Got %x\n", hr);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
int i = 0;
|
|
hr = ITest_Func(test2, &i);
|
|
ok(hr == S_OK, "Got %x\n", hr);
|
|
ok(i == 42, "Expected 42, got %d\n", i);
|
|
ITest_Release(test2);
|
|
}
|
|
IClassFactory_Release(classFactory);
|
|
}
|
|
|
|
}
|
|
|
|
static void run_registry_test(run_type run)
|
|
{
|
|
char buffer[256];
|
|
ITest *test = NULL;
|
|
HRESULT hr, result_expected;
|
|
IUnknown *unk = NULL;
|
|
HKEY hkey;
|
|
DWORD ret;
|
|
int i = 0;
|
|
|
|
if (run == run_type_exe_directory) result_expected = S_OK;
|
|
else result_expected = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
|
|
sprintf(buffer, "CLSID\\%s", wine_dbgstr_guid(&CLSID_Test), "");
|
|
ret = RegCreateKeyA( HKEY_CLASSES_ROOT, buffer, &hkey );
|
|
if (ret == ERROR_ACCESS_DENIED && !IsUserAnAdmin())
|
|
{
|
|
win_skip("cannot run the registry tests due to user not being admin\n");
|
|
RegCloseKey(hkey);
|
|
return;
|
|
}
|
|
ok(ret == ERROR_SUCCESS, "RegCreateKeyA returned %x\n", ret);
|
|
|
|
ret = RegSetKeyValueA(hkey, "InprocServer32", NULL, REG_SZ, "mscoree.dll", 11);
|
|
ok(ret == ERROR_SUCCESS, "RegSetKeyValueA returned %x\n", ret);
|
|
ret = RegSetKeyValueA(hkey, "InprocServer32", "Assembly", REG_SZ, "comtest, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null", 74);
|
|
ok(ret == ERROR_SUCCESS, "RegSetKeyValueA returned %x\n", ret);
|
|
ret = RegSetKeyValueA(hkey, "InprocServer32", "Class", REG_SZ, "DLL.Test", 8);
|
|
ok(ret == ERROR_SUCCESS, "RegSetKeyValueA returned %x\n", ret);
|
|
ret = RegSetKeyValueA(hkey, "InprocServer32", "CodeBase", REG_SZ, "file:///U:/invalid/path/to/comtest.dll", 41);
|
|
ok(ret == ERROR_SUCCESS, "RegSetKeyValueA returned %x\n", ret);
|
|
|
|
hr = CoCreateInstance(&CLSID_Test, NULL, CLSCTX_INPROC_SERVER, &IID_ITest, (void**)&test);
|
|
todo_wine_if(result_expected != S_OK)
|
|
ok(hr == result_expected, "Expected %x, got %x\n", result_expected, hr);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
hr = ITest_Func(test, &i);
|
|
ok(hr == S_OK, "Got %x\n", hr);
|
|
ok(i == 42, "Expected 42, got %d\n", i);
|
|
hr = ITest_QueryInterface(test, &IID_ITest2, (void**)&unk);
|
|
ok(hr == S_OK, "ITest_QueryInterface returned %x\n", hr);
|
|
if (hr == S_OK) IUnknown_Release(unk);
|
|
ITest_Release(test);
|
|
}
|
|
|
|
RegDeleteKeyValueA(hkey, "InprocServer32", "CodeBase");
|
|
RegDeleteKeyValueA(hkey, "InprocServer32", "Class");
|
|
RegDeleteKeyValueA(hkey, "InprocServer32", "Assembly");
|
|
RegDeleteKeyValueA(hkey, "InprocServer32", NULL);
|
|
RegDeleteKeyA(hkey, "InprocServer32");
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
static void get_dll_path_for_run(char *path_dll, UINT path_dll_size, run_type run)
|
|
{
|
|
char path_tmp[MAX_PATH];
|
|
|
|
GetTempPathA(MAX_PATH, path_tmp);
|
|
|
|
switch (run)
|
|
{
|
|
case run_type_current_working_directory:
|
|
strcpy(path_dll, path_tmp);
|
|
PathAppendA(path_dll, "comtest.dll");
|
|
break;
|
|
case run_type_exe_directory:
|
|
GetModuleFileNameA(NULL, path_dll, path_dll_size);
|
|
PathRemoveFileSpecA(path_dll);
|
|
PathAppendA(path_dll, "comtest.dll");
|
|
break;
|
|
case run_type_system32:
|
|
GetSystemDirectoryA(path_dll, path_dll_size);
|
|
PathAppendA(path_dll, "comtest.dll");
|
|
break;
|
|
}
|
|
}
|
|
static void prepare_and_run_test(const char *dll_source, run_type run)
|
|
{
|
|
char path_tmp[MAX_PATH];
|
|
char path_tmp_manifest[MAX_PATH];
|
|
char path_dll[MAX_PATH];
|
|
char path_dll_source[MAX_PATH];
|
|
char path_manifest_dll[MAX_PATH];
|
|
char path_manifest_exe[MAX_PATH];
|
|
BOOL success;
|
|
ACTCTXA context = {0};
|
|
ULONG_PTR cookie;
|
|
HANDLE handle_context = 0;
|
|
|
|
path_manifest_exe[0] = path_manifest_dll[0] = path_dll_source[0] = 0;
|
|
|
|
GetTempPathA(MAX_PATH, path_tmp);
|
|
GetTempPathA(MAX_PATH, path_tmp_manifest);
|
|
PathAppendA(path_tmp_manifest, "manifests");
|
|
|
|
CreateDirectoryA(path_tmp_manifest, NULL);
|
|
|
|
if (run == run_type_system32)
|
|
{
|
|
if (!IsUserAnAdmin())
|
|
{
|
|
skip("Can't test dll in system32 due to user not being admin.\n");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!write_resource_file(path_tmp, dll_source, "comtest.cs", path_dll_source))
|
|
{
|
|
ok(0, "run: %d, Failed to create file for testing\n", run);
|
|
goto cleanup;
|
|
}
|
|
|
|
get_dll_path_for_run(path_dll, sizeof(path_dll), run);
|
|
|
|
if (!compile_cs_to_dll(path_dll_source, path_dll))
|
|
goto cleanup;
|
|
|
|
if (!write_resource_file(path_tmp_manifest, "comtest_exe.manifest", "exe.manifest", path_manifest_exe))
|
|
{
|
|
ok(0, "run: %d, Failed to create file for testing\n", run);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (!write_resource_file(path_tmp_manifest, "comtest_dll.manifest", "comtest.manifest", path_manifest_dll))
|
|
{
|
|
ok(0, "run: %d, Failed to create file for testing\n", run);
|
|
goto cleanup;
|
|
}
|
|
|
|
context.cbSize = sizeof(ACTCTXA);
|
|
context.lpSource = path_manifest_exe;
|
|
context.lpAssemblyDirectory = path_tmp_manifest;
|
|
context.dwFlags = ACTCTX_FLAG_ASSEMBLY_DIRECTORY_VALID;
|
|
|
|
handle_context = CreateActCtxA(&context);
|
|
ok(handle_context != NULL && handle_context != INVALID_HANDLE_VALUE, "run: %d, CreateActCtxA failed: %d\n", run, GetLastError());
|
|
|
|
if (handle_context == NULL || handle_context == INVALID_HANDLE_VALUE)
|
|
{
|
|
ok(0, "run: %d, Failed to create activation context\n", run);
|
|
goto cleanup;
|
|
}
|
|
|
|
success = ActivateActCtx(handle_context, &cookie);
|
|
ok(success, "run: %d, ActivateActCtx failed: %d\n", run, GetLastError());
|
|
|
|
if (run == run_type_current_working_directory)
|
|
SetCurrentDirectoryA(path_tmp);
|
|
|
|
run_test(run == run_type_exe_directory);
|
|
run_registry_test(run);
|
|
|
|
cleanup:
|
|
if (handle_context != NULL && handle_context != INVALID_HANDLE_VALUE)
|
|
{
|
|
success = DeactivateActCtx(0, cookie);
|
|
ok(success, "run: %d, DeactivateActCtx failed: %d\n", run, GetLastError());
|
|
ReleaseActCtx(handle_context);
|
|
}
|
|
if (*path_manifest_exe)
|
|
{
|
|
success = DeleteFileA(path_manifest_exe);
|
|
ok(success, "run: %d, DeleteFileA failed: %d\n", run, GetLastError());
|
|
}
|
|
if(*path_manifest_dll)
|
|
{
|
|
success = DeleteFileA(path_manifest_dll);
|
|
ok(success, "run: %d, DeleteFileA failed: %d\n", run, GetLastError());
|
|
}
|
|
if(*path_dll_source)
|
|
{
|
|
success = DeleteFileA(path_dll_source);
|
|
ok(success, "run: %d, DeleteFileA failed: %d\n", run, GetLastError());
|
|
}
|
|
RemoveDirectoryA(path_tmp_manifest);
|
|
/* dll cleanup is handled by the parent, because it might still be used by the child */
|
|
}
|
|
|
|
|
|
static void cleanup_test(run_type run)
|
|
{
|
|
char path_dll[MAX_PATH];
|
|
BOOL success;
|
|
|
|
get_dll_path_for_run(path_dll, sizeof(path_dll), run);
|
|
|
|
if (!PathFileExistsA(path_dll))
|
|
return;
|
|
|
|
success = DeleteFileA(path_dll);
|
|
if (!success)
|
|
{
|
|
Sleep(500);
|
|
success = DeleteFileA(path_dll);
|
|
}
|
|
ok(success, "DeleteFileA failed: %d\n", GetLastError());
|
|
}
|
|
|
|
static void run_child_process(const char *dll_source, run_type run)
|
|
{
|
|
char cmdline[MAX_PATH];
|
|
char exe[MAX_PATH];
|
|
char **argv;
|
|
PROCESS_INFORMATION pi;
|
|
STARTUPINFOA si = { 0 };
|
|
BOOL ret;
|
|
|
|
winetest_get_mainargs(&argv);
|
|
|
|
if (strstr(argv[0], ".exe"))
|
|
sprintf(exe, "%s", argv[0]);
|
|
else
|
|
sprintf(exe, "%s.exe", argv[0]);
|
|
sprintf(cmdline, "\"%s\" %s %s %d", argv[0], argv[1], dll_source, run);
|
|
|
|
si.cb = sizeof(si);
|
|
ret = CreateProcessA(exe, cmdline, NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi);
|
|
ok(ret, "Could not create process: %u\n", GetLastError());
|
|
|
|
wait_child_process(pi.hProcess);
|
|
|
|
CloseHandle(pi.hThread);
|
|
CloseHandle(pi.hProcess);
|
|
|
|
/* Cleanup dll, because it might still have been used by the child */
|
|
cleanup_test(run);
|
|
}
|
|
|
|
START_TEST(comtest)
|
|
{
|
|
int argc;
|
|
char **argv;
|
|
|
|
CoInitialize(NULL);
|
|
|
|
hmscoree = LoadLibraryA("mscoree.dll");
|
|
if (!hmscoree)
|
|
{
|
|
skip(".NET or mono not available\n");
|
|
return;
|
|
}
|
|
|
|
argc = winetest_get_mainargs(&argv);
|
|
if (argc > 2)
|
|
{
|
|
const char *dll_source = argv[2];
|
|
run_type run = atoi(argv[3]);
|
|
prepare_and_run_test(dll_source, run);
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
run_child_process("comtest.cs", run_type_current_working_directory);
|
|
run_child_process("comtest.cs", run_type_exe_directory);
|
|
run_child_process("comtest.cs", run_type_system32);
|
|
|
|
cleanup:
|
|
FreeLibrary(hmscoree);
|
|
CoUninitialize();
|
|
}
|