853 lines
30 KiB
C
853 lines
30 KiB
C
/*
|
|
* Unit test for setupapi.dll install functions
|
|
*
|
|
* Copyright 2007 Misha Koshelev
|
|
*
|
|
* 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 <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winnls.h"
|
|
#include "winuser.h"
|
|
#include "winreg.h"
|
|
#include "winsvc.h"
|
|
#include "setupapi.h"
|
|
#include "shlobj.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
static const char inffile[] = "test.inf";
|
|
static const WCHAR inffileW[] = {'t','e','s','t','.','i','n','f',0};
|
|
static char CURR_DIR[MAX_PATH];
|
|
|
|
/* Notes on InstallHinfSectionA/W:
|
|
* - InstallHinfSectionW on Win98 and InstallHinfSectionA on WinXP seem to be stubs - they do not do anything
|
|
* and simply return without displaying any error message or setting last error. We test whether
|
|
* InstallHinfSectionA sets last error, and if it doesn't we set it to NULL and use the W version if available.
|
|
* - These functions do not return a value and do not always set last error to ERROR_SUCCESS when installation still
|
|
* occurs (e.g., unquoted inf file with spaces, registry keys are written but last error is 6). Also, on Win98 last error
|
|
* is set to ERROR_SUCCESS even if install fails (e.g., quoted inf file with spaces, no registry keys set, MessageBox with
|
|
* "Installation Error" displayed). Thus, we must use functional tests (e.g., is registry key created) to determine whether
|
|
* or not installation occurred.
|
|
* - On installation problems, a MessageBox() is displayed and a Beep() is issued. The MessageBox() is disabled with a
|
|
* CBT hook.
|
|
*/
|
|
|
|
static void (WINAPI *pInstallHinfSectionA)(HWND, HINSTANCE, LPCSTR, INT);
|
|
static void (WINAPI *pInstallHinfSectionW)(HWND, HINSTANCE, LPCWSTR, INT);
|
|
static BOOL (WINAPI *pSetupGetInfFileListA)(PCSTR, DWORD, PSTR, DWORD, PDWORD);
|
|
static BOOL (WINAPI *pSetupGetInfFileListW)(PCWSTR, DWORD, PWSTR, DWORD, PDWORD);
|
|
|
|
/*
|
|
* Helpers
|
|
*/
|
|
|
|
static void create_inf_file(LPCSTR filename, const char *data)
|
|
{
|
|
DWORD res;
|
|
BOOL ret;
|
|
HANDLE handle = CreateFileA(filename, GENERIC_WRITE, 0, NULL,
|
|
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
assert(handle != INVALID_HANDLE_VALUE);
|
|
ret = WriteFile(handle, data, strlen(data), &res, NULL);
|
|
assert(ret != 0);
|
|
CloseHandle(handle);
|
|
}
|
|
|
|
/* CBT hook to ensure a window (e.g., MessageBox) cannot be created */
|
|
static HHOOK hhook;
|
|
static LRESULT CALLBACK cbt_hook_proc(int nCode, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
return nCode == HCBT_CREATEWND ? 1: CallNextHookEx(hhook, nCode, wParam, lParam);
|
|
}
|
|
|
|
/*
|
|
* Tests
|
|
*/
|
|
|
|
static const char *cmdline_inf = "[Version]\n"
|
|
"Signature=\"$Chicago$\"\n"
|
|
"[DefaultInstall]\n"
|
|
"AddReg=Add.Settings\n"
|
|
"[Add.Settings]\n"
|
|
"HKCU,Software\\Wine\\setupapitest,,\n";
|
|
|
|
static void run_cmdline(LPCSTR section, int mode, LPCSTR path)
|
|
{
|
|
CHAR cmdline[MAX_PATH * 2];
|
|
|
|
sprintf(cmdline, "%s %d %s", section, mode, path);
|
|
if (pInstallHinfSectionA) pInstallHinfSectionA(NULL, NULL, cmdline, 0);
|
|
else
|
|
{
|
|
WCHAR cmdlinew[MAX_PATH * 2];
|
|
MultiByteToWideChar(CP_ACP, 0, cmdline, -1, cmdlinew, MAX_PATH*2);
|
|
pInstallHinfSectionW(NULL, NULL, cmdlinew, 0);
|
|
}
|
|
}
|
|
|
|
static void ok_registry(BOOL expectsuccess)
|
|
{
|
|
LONG ret;
|
|
|
|
/* Functional tests for success of install and clean up */
|
|
ret = RegDeleteKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest");
|
|
ok((expectsuccess && ret == ERROR_SUCCESS) ||
|
|
(!expectsuccess && ret == ERROR_FILE_NOT_FOUND),
|
|
"Expected registry key Software\\Wine\\setupapitest to %s, RegDeleteKey returned %d\n",
|
|
expectsuccess ? "exist" : "not exist",
|
|
ret);
|
|
}
|
|
|
|
/* Test command line processing */
|
|
static void test_cmdline(void)
|
|
{
|
|
static const char infwithspaces[] = "test file.inf";
|
|
char path[MAX_PATH];
|
|
BOOL ret;
|
|
|
|
create_inf_file(inffile, cmdline_inf);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
ok_registry(TRUE);
|
|
ret = DeleteFileA(inffile);
|
|
ok(ret, "Expected source inf to exist, last error was %d\n", GetLastError());
|
|
|
|
/* Test handling of spaces in path, unquoted and quoted */
|
|
create_inf_file(infwithspaces, cmdline_inf);
|
|
|
|
sprintf(path, "%s\\%s", CURR_DIR, infwithspaces);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
ok_registry(TRUE);
|
|
|
|
sprintf(path, "\"%s\\%s\"", CURR_DIR, infwithspaces);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
ok_registry(FALSE);
|
|
|
|
ret = DeleteFileA(infwithspaces);
|
|
ok(ret, "Expected source inf to exist, last error was %d\n", GetLastError());
|
|
}
|
|
|
|
static const char *cmdline_inf_reg = "[Version]\n"
|
|
"Signature=\"$Chicago$\"\n"
|
|
"[DefaultInstall]\n"
|
|
"DelReg=Del.Settings\n"
|
|
"[Del.Settings]\n"
|
|
"HKCU,Software\\Wine\\setupapitest\n";
|
|
|
|
static void test_registry(void)
|
|
{
|
|
HKEY key;
|
|
LONG res;
|
|
char path[MAX_PATH];
|
|
BOOL ret;
|
|
|
|
/* First create a registry structure we would like to be deleted */
|
|
ok(!RegCreateKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest\\setupapitest", &key),
|
|
"Expected RegCreateKeyA to succeed\n");
|
|
|
|
/* Doublecheck if the registry key is present */
|
|
ok(!RegOpenKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest\\setupapitest", &key),
|
|
"Expected registry key to exist\n");
|
|
|
|
create_inf_file(inffile, cmdline_inf_reg);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
|
|
/* Check if the registry key is recursively deleted */
|
|
res = RegOpenKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest", &key);
|
|
todo_wine
|
|
ok(res == ERROR_FILE_NOT_FOUND, "Didn't expect the registry key to exist\n");
|
|
/* Just in case */
|
|
if (res == ERROR_SUCCESS)
|
|
{
|
|
RegDeleteKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest\\setupapitest");
|
|
RegDeleteKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest");
|
|
}
|
|
ret = DeleteFileA(inffile);
|
|
ok(ret, "Expected source inf to exist, last error was %d\n", GetLastError());
|
|
}
|
|
|
|
static void test_install_svc_from(void)
|
|
{
|
|
char inf[2048];
|
|
char path[MAX_PATH];
|
|
HINF infhandle;
|
|
BOOL ret;
|
|
SC_HANDLE scm_handle, svc_handle;
|
|
|
|
/* Bail out if we are on win98 */
|
|
SetLastError(0xdeadbeef);
|
|
scm_handle = OpenSCManagerA(NULL, NULL, GENERIC_ALL);
|
|
|
|
if (!scm_handle && (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED))
|
|
{
|
|
win_skip("OpenSCManagerA is not implemented, we are most likely on win9x\n");
|
|
return;
|
|
}
|
|
CloseServiceHandle(scm_handle);
|
|
|
|
/* Basic inf file to satisfy SetupOpenInfFileA */
|
|
strcpy(inf, "[Version]\nSignature=\"$Chicago$\"\n");
|
|
create_inf_file(inffile, inf);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
|
|
/* Nothing but the Version section */
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_SECTION_NOT_FOUND,
|
|
"Expected ERROR_SECTION_NOT_FOUND, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Add the section */
|
|
strcat(inf, "[Winetest.Services]\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_SECTION_NOT_FOUND,
|
|
"Expected ERROR_SECTION_NOT_FOUND, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Add a reference */
|
|
strcat(inf, "AddService=Winetest,,Winetest.Service\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_BAD_SERVICE_INSTALLSECT,
|
|
"Expected ERROR_BAD_SERVICE_INSTALLSECT, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Add the section */
|
|
strcat(inf, "[Winetest.Service]\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_BAD_SERVICE_INSTALLSECT,
|
|
"Expected ERROR_BAD_SERVICE_INSTALLSECT, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Just the ServiceBinary */
|
|
strcat(inf, "ServiceBinary=%12%\\winetest.sys\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_BAD_SERVICE_INSTALLSECT,
|
|
"Expected ERROR_BAD_SERVICE_INSTALLSECT, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Add the ServiceType */
|
|
strcat(inf, "ServiceType=1\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_BAD_SERVICE_INSTALLSECT,
|
|
"Expected ERROR_BAD_SERVICE_INSTALLSECT, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* Add the StartType */
|
|
strcat(inf, "StartType=4\n");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
ok(!ret, "Expected failure\n");
|
|
ok(GetLastError() == ERROR_BAD_SERVICE_INSTALLSECT,
|
|
"Expected ERROR_BAD_SERVICE_INSTALLSECT, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* This should be it, the minimal entries to install a service */
|
|
strcat(inf, "ErrorControl=1");
|
|
create_inf_file(inffile, inf);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "Winetest.Services", 0);
|
|
if (!ret && GetLastError() == ERROR_ACCESS_DENIED)
|
|
{
|
|
skip("Not enough rights to install the service\n");
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
return;
|
|
}
|
|
ok(ret, "Expected success\n");
|
|
ok(GetLastError() == ERROR_SUCCESS,
|
|
"Expected ERROR_SUCCESS, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
scm_handle = OpenSCManagerA(NULL, NULL, GENERIC_ALL);
|
|
|
|
/* Open the service to see if it's really there */
|
|
svc_handle = OpenServiceA(scm_handle, "Winetest", DELETE);
|
|
ok(svc_handle != NULL, "Service was not created\n");
|
|
|
|
SetLastError(0xdeadbeef);
|
|
ret = DeleteService(svc_handle);
|
|
ok(ret, "Service could not be deleted : %d\n", GetLastError());
|
|
|
|
CloseServiceHandle(svc_handle);
|
|
CloseServiceHandle(scm_handle);
|
|
|
|
strcpy(inf, "[Version]\nSignature=\"$Chicago$\"\n");
|
|
strcat(inf, "[XSP.InstallPerVer]\n");
|
|
strcat(inf, "AddReg=AspEventlogMsg.Reg,Perf.Reg,AspVersions.Reg,FreeADO.Reg,IndexServer.Reg\n");
|
|
create_inf_file(inffile, inf);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
infhandle = SetupOpenInfFileA(path, NULL, INF_STYLE_WIN4, NULL);
|
|
|
|
SetLastError(0xdeadbeef);
|
|
ret = SetupInstallServicesFromInfSectionA(infhandle, "XSP.InstallPerVer", 0);
|
|
ok(ret, "Expected success\n");
|
|
ok(GetLastError() == ERROR_SUCCESS, "Expected ERROR_SUCCESS, got %08x\n", GetLastError());
|
|
SetupCloseInfFile(infhandle);
|
|
DeleteFileA(inffile);
|
|
|
|
/* TODO: Test the Flags */
|
|
}
|
|
|
|
static void test_driver_install(void)
|
|
{
|
|
HANDLE handle;
|
|
SC_HANDLE scm_handle, svc_handle;
|
|
BOOL ret;
|
|
char path[MAX_PATH], windir[MAX_PATH], driver[MAX_PATH];
|
|
DWORD attrs;
|
|
/* Minimal stuff needed */
|
|
static const char *inf =
|
|
"[Version]\n"
|
|
"Signature=\"$Chicago$\"\n"
|
|
"[DestinationDirs]\n"
|
|
"Winetest.DriverFiles=12\n"
|
|
"[DefaultInstall]\n"
|
|
"CopyFiles=Winetest.DriverFiles\n"
|
|
"[DefaultInstall.Services]\n"
|
|
"AddService=Winetest,,Winetest.Service\n"
|
|
"[Winetest.Service]\n"
|
|
"ServiceBinary=%12%\\winetest.sys\n"
|
|
"ServiceType=1\n"
|
|
"StartType=4\n"
|
|
"ErrorControl=1\n"
|
|
"[Winetest.DriverFiles]\n"
|
|
"winetest.sys";
|
|
|
|
/* Bail out if we are on win98 */
|
|
SetLastError(0xdeadbeef);
|
|
scm_handle = OpenSCManagerA(NULL, NULL, GENERIC_ALL);
|
|
|
|
if (!scm_handle && (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED))
|
|
{
|
|
win_skip("OpenSCManagerA is not implemented, we are most likely on win9x\n");
|
|
return;
|
|
}
|
|
else if (!scm_handle && (GetLastError() == ERROR_ACCESS_DENIED))
|
|
{
|
|
skip("Not enough rights to install the service\n");
|
|
return;
|
|
}
|
|
CloseServiceHandle(scm_handle);
|
|
|
|
/* Place where we expect the driver to be installed */
|
|
GetWindowsDirectoryA(windir, MAX_PATH);
|
|
lstrcpyA(driver, windir);
|
|
lstrcatA(driver, "\\system32\\drivers\\winetest.sys");
|
|
|
|
/* Create a dummy driver file */
|
|
handle = CreateFileA("winetest.sys", GENERIC_WRITE, 0, NULL,
|
|
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
CloseHandle(handle);
|
|
|
|
create_inf_file(inffile, inf);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
|
|
/* Driver should have been installed */
|
|
attrs = GetFileAttributesA(driver);
|
|
ok(attrs != INVALID_FILE_ATTRIBUTES, "Expected driver to exist\n");
|
|
|
|
scm_handle = OpenSCManagerA(NULL, NULL, GENERIC_ALL);
|
|
|
|
/* Open the service to see if it's really there */
|
|
svc_handle = OpenServiceA(scm_handle, "Winetest", DELETE);
|
|
ok(svc_handle != NULL, "Service was not created\n");
|
|
|
|
SetLastError(0xdeadbeef);
|
|
ret = DeleteService(svc_handle);
|
|
ok(ret, "Service could not be deleted : %d\n", GetLastError());
|
|
|
|
CloseServiceHandle(svc_handle);
|
|
CloseServiceHandle(scm_handle);
|
|
|
|
/* File cleanup */
|
|
DeleteFileA(inffile);
|
|
DeleteFileA("winetest.sys");
|
|
DeleteFileA(driver);
|
|
}
|
|
|
|
static void test_profile_items(void)
|
|
{
|
|
char path[MAX_PATH], commonprogs[MAX_PATH];
|
|
HMODULE hShell32;
|
|
BOOL (WINAPI *pSHGetFolderPathA)(HWND hwnd, int nFolder, HANDLE hToken, DWORD dwFlags, LPSTR pszPath);
|
|
|
|
static const char *inf =
|
|
"[Version]\n"
|
|
"Signature=\"$Chicago$\"\n"
|
|
"[DefaultInstall]\n"
|
|
"ProfileItems=TestItem,TestItem2,TestGroup\n"
|
|
"[TestItem]\n"
|
|
"Name=TestItem\n"
|
|
"CmdLine=11,,notepad.exe\n"
|
|
"[TestItem2]\n"
|
|
"Name=TestItem2\n"
|
|
"CmdLine=11,,notepad.exe\n"
|
|
"SubDir=TestDir\n"
|
|
"[TestGroup]\n"
|
|
"Name=TestGroup,4\n"
|
|
;
|
|
|
|
hShell32 = LoadLibraryA("shell32");
|
|
pSHGetFolderPathA = (void*)GetProcAddress(hShell32, "SHGetFolderPathA");
|
|
if (!pSHGetFolderPathA)
|
|
{
|
|
win_skip("SHGetFolderPathA is not available\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
if (S_OK != pSHGetFolderPathA(NULL, CSIDL_COMMON_PROGRAMS, NULL, SHGFP_TYPE_CURRENT, commonprogs))
|
|
{
|
|
skip("No common program files directory exists\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
snprintf(path, MAX_PATH, "%s\\TestDir", commonprogs);
|
|
if (!CreateDirectoryA(path, NULL) && GetLastError() == ERROR_ACCESS_DENIED)
|
|
{
|
|
skip("need admin rights\n");
|
|
return;
|
|
}
|
|
RemoveDirectoryA(path);
|
|
|
|
create_inf_file(inffile, inf);
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
|
|
snprintf(path, MAX_PATH, "%s\\TestItem.lnk", commonprogs);
|
|
if (INVALID_FILE_ATTRIBUTES == GetFileAttributesA(path))
|
|
{
|
|
win_skip("ProfileItems not implemented on this system\n");
|
|
}
|
|
else
|
|
{
|
|
snprintf(path, MAX_PATH, "%s\\TestDir", commonprogs);
|
|
ok(INVALID_FILE_ATTRIBUTES != GetFileAttributesA(path), "directory not created\n");
|
|
snprintf(path, MAX_PATH, "%s\\TestDir\\TestItem2.lnk", commonprogs);
|
|
ok(INVALID_FILE_ATTRIBUTES != GetFileAttributesA(path), "link not created\n");
|
|
snprintf(path, MAX_PATH, "%s\\TestGroup", commonprogs);
|
|
ok(INVALID_FILE_ATTRIBUTES != GetFileAttributesA(path), "group not created\n");
|
|
}
|
|
|
|
snprintf(path, MAX_PATH, "%s\\TestItem.lnk", commonprogs);
|
|
DeleteFileA(path);
|
|
snprintf(path, MAX_PATH, "%s\\TestDir\\TestItem2.lnk", commonprogs);
|
|
DeleteFileA(path);
|
|
snprintf(path, MAX_PATH, "%s\\TestItem2.lnk", commonprogs);
|
|
DeleteFileA(path);
|
|
snprintf(path, MAX_PATH, "%s\\TestDir", commonprogs);
|
|
RemoveDirectoryA(path);
|
|
snprintf(path, MAX_PATH, "%s\\TestGroup", commonprogs);
|
|
RemoveDirectoryA(path);
|
|
|
|
cleanup:
|
|
if (hShell32) FreeLibrary(hShell32);
|
|
DeleteFileA(inffile);
|
|
}
|
|
|
|
static void test_inffilelistA(void)
|
|
{
|
|
static const char inffile2[] = "test2.inf";
|
|
static const char *inf =
|
|
"[Version]\n"
|
|
"Signature=\"$Chicago$\"";
|
|
|
|
char buffer[MAX_PATH] = { 0 };
|
|
char dir[MAX_PATH], *p;
|
|
DWORD expected, outsize;
|
|
BOOL ret;
|
|
|
|
if(!pSetupGetInfFileListA)
|
|
{
|
|
win_skip("SetupGetInfFileListA not present\n");
|
|
return;
|
|
}
|
|
|
|
/* create a private directory, the temp directory may contain some
|
|
* inf files left over from old installations
|
|
*/
|
|
if (!GetTempFileNameA(CURR_DIR, "inftest", 1, dir))
|
|
{
|
|
win_skip("GetTempFileNameA failed with error %d\n", GetLastError());
|
|
return;
|
|
}
|
|
if (!CreateDirectoryA(dir, NULL ))
|
|
{
|
|
win_skip("CreateDirectoryA(%s) failed with error %d\n", dir, GetLastError());
|
|
return;
|
|
}
|
|
if (!SetCurrentDirectoryA(dir))
|
|
{
|
|
win_skip("SetCurrentDirectoryA failed with error %d\n", GetLastError());
|
|
RemoveDirectoryA(dir);
|
|
return;
|
|
}
|
|
|
|
create_inf_file(inffile, inf);
|
|
create_inf_file(inffile2, inf);
|
|
|
|
/* mixed style
|
|
*/
|
|
expected = 3 + strlen(inffile) + strlen(inffile2);
|
|
ret = pSetupGetInfFileListA(dir, INF_STYLE_OLDNT | INF_STYLE_WIN4, buffer,
|
|
MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListA to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenA(p) && (outsize > (p - buffer)); p+=lstrlenA(p) + 1)
|
|
ok(!lstrcmpA(p,inffile2) || !lstrcmpA(p,inffile),
|
|
"unexpected filename %s\n",p);
|
|
|
|
DeleteFileA(inffile);
|
|
DeleteFileA(inffile2);
|
|
SetCurrentDirectoryA(CURR_DIR);
|
|
RemoveDirectoryA(dir);
|
|
}
|
|
|
|
static void test_inffilelist(void)
|
|
{
|
|
static const char inffile2[] = "test2.inf";
|
|
static const WCHAR inffile2W[] = {'t','e','s','t','2','.','i','n','f',0};
|
|
static const char invalid_inf[] = "invalid.inf";
|
|
static const WCHAR invalid_infW[] = {'i','n','v','a','l','i','d','.','i','n','f',0};
|
|
static const char *inf =
|
|
"[Version]\n"
|
|
"Signature=\"$Chicago$\"";
|
|
static const char *inf2 =
|
|
"[Version]\n"
|
|
"Signature=\"$CHICAGO$\"";
|
|
static const char *infNT =
|
|
"[Version]\n"
|
|
"Signature=\"$WINDOWS NT$\"";
|
|
|
|
WCHAR *p, *ptr;
|
|
char dirA[MAX_PATH];
|
|
WCHAR dir[MAX_PATH] = { 0 };
|
|
WCHAR buffer[MAX_PATH] = { 0 };
|
|
DWORD expected, outsize;
|
|
BOOL ret;
|
|
|
|
if(!pSetupGetInfFileListW)
|
|
{
|
|
win_skip("SetupGetInfFileListW not present\n");
|
|
return;
|
|
}
|
|
|
|
/* NULL means %windir%\\inf
|
|
* get the value as reference
|
|
*/
|
|
expected = 0;
|
|
SetLastError(0xdeadbeef);
|
|
ret = pSetupGetInfFileListW(NULL, INF_STYLE_WIN4, NULL, 0, &expected);
|
|
if (!ret && GetLastError() == ERROR_CALL_NOT_IMPLEMENTED)
|
|
{
|
|
win_skip("SetupGetInfFileListW not implemented\n");
|
|
return;
|
|
}
|
|
ok(ret, "expected SetupGetInfFileListW to succeed! Error: %d\n", GetLastError());
|
|
ok(expected > 0, "expected required buffersize to be at least 1\n");
|
|
|
|
/* check if an empty string doesn't behaves like NULL */
|
|
outsize = 0;
|
|
SetLastError(0xdeadbeef);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, NULL, 0, &outsize);
|
|
ok(!ret, "expected SetupGetInfFileListW to fail!\n");
|
|
|
|
/* create a private directory, the temp directory may contain some
|
|
* inf files left over from old installations
|
|
*/
|
|
if (!GetTempFileNameA(CURR_DIR, "inftest", 1, dirA))
|
|
{
|
|
win_skip("GetTempFileNameA failed with error %d\n", GetLastError());
|
|
return;
|
|
}
|
|
if (!CreateDirectoryA(dirA, NULL ))
|
|
{
|
|
win_skip("CreateDirectoryA(%s) failed with error %d\n", dirA, GetLastError());
|
|
return;
|
|
}
|
|
if (!SetCurrentDirectoryA(dirA))
|
|
{
|
|
win_skip("SetCurrentDirectoryA failed with error %d\n", GetLastError());
|
|
RemoveDirectoryA(dirA);
|
|
return;
|
|
}
|
|
|
|
MultiByteToWideChar(CP_ACP, 0, dirA, -1, dir, MAX_PATH);
|
|
/* check a not existing directory
|
|
*/
|
|
ptr = dir + lstrlenW(dir);
|
|
MultiByteToWideChar(CP_ACP, 0, "\\not_existent", -1, ptr, MAX_PATH - lstrlenW(dir));
|
|
outsize = 0xffffffff;
|
|
SetLastError(0xdeadbeef);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, NULL, 0, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(outsize == 1, "expected required buffersize to be 1, got %d\n", outsize);
|
|
ok(ERROR_PATH_NOT_FOUND == GetLastError(),
|
|
"expected error ERROR_PATH_NOT_FOUND, got %d\n", GetLastError());
|
|
|
|
create_inf_file(inffile, inf);
|
|
create_inf_file(inffile2, inf);
|
|
create_inf_file(invalid_inf, "This content does not match the inf file format");
|
|
|
|
/* pass a filename instead of a directory
|
|
*/
|
|
*ptr = '\\';
|
|
MultiByteToWideChar(CP_ACP, 0, invalid_inf, -1, ptr+1, MAX_PATH - lstrlenW(dir));
|
|
outsize = 0xffffffff;
|
|
SetLastError(0xdeadbeef);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, NULL, 0, &outsize);
|
|
ok(!ret, "expected SetupGetInfFileListW to fail!\n");
|
|
ok(ERROR_DIRECTORY == GetLastError(),
|
|
"expected error ERROR_DIRECTORY, got %d\n", GetLastError());
|
|
|
|
/* make the filename look like directory
|
|
*/
|
|
dir[1 + lstrlenW(dir)] = 0;
|
|
dir[lstrlenW(dir)] = '\\';
|
|
SetLastError(0xdeadbeef);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, NULL, 0, &outsize);
|
|
ok(!ret, "expected SetupGetInfFileListW to fail!\n");
|
|
ok(ERROR_DIRECTORY == GetLastError(),
|
|
"expected error ERROR_DIRECTORY, got %d\n", GetLastError());
|
|
|
|
/* now check the buffer contents of a valid call
|
|
*/
|
|
*ptr = 0;
|
|
expected = 3 + strlen(inffile) + strlen(inffile2);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, buffer, MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenW(p) && (outsize > (p - buffer)); p+=lstrlenW(p) + 1)
|
|
ok(!lstrcmpW(p,inffile2W) || !lstrcmpW(p,inffileW),
|
|
"unexpected filename %s\n",wine_dbgstr_w(p));
|
|
|
|
/* upper case value
|
|
*/
|
|
create_inf_file(inffile2, inf2);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, buffer, MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenW(p) && (outsize > (p - buffer)); p+=lstrlenW(p) + 1)
|
|
ok(!lstrcmpW(p,inffile2W) || !lstrcmpW(p,inffileW),
|
|
"unexpected filename %s\n",wine_dbgstr_w(p));
|
|
|
|
/* signature Windows NT is also inf style win4
|
|
*/
|
|
create_inf_file(inffile2, infNT);
|
|
expected = 3 + strlen(inffile) + strlen(inffile2);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_WIN4, buffer, MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenW(p) && (outsize > (p - buffer)); p+=lstrlenW(p) + 1)
|
|
ok(!lstrcmpW(p,inffile2W) || !lstrcmpW(p,inffileW),
|
|
"unexpected filename %s\n",wine_dbgstr_w(p));
|
|
|
|
/* old style
|
|
*/
|
|
expected = 2 + strlen(invalid_inf);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_OLDNT, buffer, MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenW(p) && (outsize > (p - buffer)); p+=lstrlenW(p) + 1)
|
|
ok(!lstrcmpW(p,invalid_infW), "unexpected filename %s\n",wine_dbgstr_w(p));
|
|
|
|
/* mixed style
|
|
*/
|
|
expected = 4 + strlen(inffile) + strlen(inffile2) + strlen(invalid_inf);
|
|
ret = pSetupGetInfFileListW(dir, INF_STYLE_OLDNT | INF_STYLE_WIN4, buffer,
|
|
MAX_PATH, &outsize);
|
|
ok(ret, "expected SetupGetInfFileListW to succeed!\n");
|
|
ok(expected == outsize, "expected required buffersize to be %d, got %d\n",
|
|
expected, outsize);
|
|
for(p = buffer; lstrlenW(p) && (outsize > (p - buffer)); p+=lstrlenW(p) + 1)
|
|
ok(!lstrcmpW(p,inffile2W) || !lstrcmpW(p,inffileW) || !lstrcmpW(p,invalid_infW),
|
|
"unexpected filename %s\n",wine_dbgstr_w(p));
|
|
|
|
DeleteFileA(inffile);
|
|
DeleteFileA(inffile2);
|
|
DeleteFileA(invalid_inf);
|
|
SetCurrentDirectoryA(CURR_DIR);
|
|
RemoveDirectoryA(dirA);
|
|
}
|
|
|
|
static const char dirid_inf[] = "[Version]\n"
|
|
"Signature=\"$Chicago$\"\n"
|
|
"[DefaultInstall]\n"
|
|
"AddReg=Add.Settings\n"
|
|
"[Add.Settings]\n"
|
|
"HKCU,Software\\Wine\\setupapitest,dirid,,%%%i%%\n";
|
|
|
|
static void check_dirid(int dirid, LPCSTR expected)
|
|
{
|
|
char buffer[sizeof(dirid_inf)+11];
|
|
char path[MAX_PATH], actual[MAX_PATH];
|
|
LONG ret;
|
|
DWORD size, type;
|
|
HKEY key;
|
|
|
|
sprintf(buffer, dirid_inf, dirid);
|
|
|
|
create_inf_file(inffile, buffer);
|
|
|
|
sprintf(path, "%s\\%s", CURR_DIR, inffile);
|
|
run_cmdline("DefaultInstall", 128, path);
|
|
|
|
size = sizeof(actual);
|
|
actual[0] = '\0';
|
|
ret = RegOpenKeyA(HKEY_CURRENT_USER, "Software\\Wine\\setupapitest", &key);
|
|
if (ret == ERROR_SUCCESS)
|
|
{
|
|
ret = RegQueryValueExA(key, "dirid", NULL, &type, (BYTE*)&actual, &size);
|
|
RegCloseKey(key);
|
|
if (type != REG_SZ)
|
|
ret = ERROR_FILE_NOT_FOUND;
|
|
}
|
|
|
|
ok(ret == ERROR_SUCCESS, "Failed getting value for dirid %i, err=%d\n", dirid, ret);
|
|
ok(!strcmp(actual, expected), "Expected path for dirid %i was \"%s\", got \"%s\"\n", dirid, expected, actual);
|
|
|
|
ok_registry(TRUE);
|
|
ret = DeleteFileA(inffile);
|
|
ok(ret, "Expected source inf to exist, last error was %d\n", GetLastError());
|
|
}
|
|
|
|
/* Test dirid values */
|
|
static void test_dirid(void)
|
|
{
|
|
char expected[MAX_PATH];
|
|
|
|
check_dirid(DIRID_NULL, "");
|
|
|
|
GetWindowsDirectoryA(expected, MAX_PATH);
|
|
check_dirid(DIRID_WINDOWS, expected);
|
|
|
|
GetSystemDirectoryA(expected, MAX_PATH);
|
|
check_dirid(DIRID_SYSTEM, expected);
|
|
|
|
strcat(expected, "\\unknown");
|
|
check_dirid(40, expected);
|
|
}
|
|
|
|
START_TEST(install)
|
|
{
|
|
HMODULE hsetupapi = GetModuleHandleA("setupapi.dll");
|
|
char temp_path[MAX_PATH], prev_path[MAX_PATH];
|
|
DWORD len;
|
|
|
|
GetCurrentDirectoryA(MAX_PATH, prev_path);
|
|
GetTempPathA(MAX_PATH, temp_path);
|
|
SetCurrentDirectoryA(temp_path);
|
|
|
|
strcpy(CURR_DIR, temp_path);
|
|
len = strlen(CURR_DIR);
|
|
if(len && (CURR_DIR[len - 1] == '\\'))
|
|
CURR_DIR[len - 1] = 0;
|
|
|
|
pInstallHinfSectionA = (void *)GetProcAddress(hsetupapi, "InstallHinfSectionA");
|
|
pInstallHinfSectionW = (void *)GetProcAddress(hsetupapi, "InstallHinfSectionW");
|
|
pSetupGetInfFileListA = (void *)GetProcAddress(hsetupapi, "SetupGetInfFileListA");
|
|
pSetupGetInfFileListW = (void *)GetProcAddress(hsetupapi, "SetupGetInfFileListW");
|
|
|
|
if (pInstallHinfSectionA)
|
|
{
|
|
/* Check if pInstallHinfSectionA sets last error or is a stub (as on WinXP) */
|
|
static const char *minimal_inf = "[Version]\nSignature=\"$Chicago$\"\n";
|
|
char cmdline[MAX_PATH*2];
|
|
BOOL ret;
|
|
create_inf_file(inffile, minimal_inf);
|
|
sprintf(cmdline, "DefaultInstall 128 %s\\%s", CURR_DIR, inffile);
|
|
SetLastError(0xdeadbeef);
|
|
pInstallHinfSectionA(NULL, NULL, cmdline, 0);
|
|
if (GetLastError() == 0xdeadbeef)
|
|
{
|
|
skip("InstallHinfSectionA is broken (stub)\n");
|
|
pInstallHinfSectionA = NULL;
|
|
}
|
|
ret = DeleteFileA(inffile);
|
|
ok(ret, "Expected source inf to exist, last error was %d\n", GetLastError());
|
|
}
|
|
if (!pInstallHinfSectionW && !pInstallHinfSectionA)
|
|
win_skip("InstallHinfSectionA and InstallHinfSectionW are not available\n");
|
|
else
|
|
{
|
|
/* Set CBT hook to disallow MessageBox creation in current thread */
|
|
hhook = SetWindowsHookExA(WH_CBT, cbt_hook_proc, 0, GetCurrentThreadId());
|
|
assert(hhook != 0);
|
|
|
|
test_cmdline();
|
|
test_registry();
|
|
test_install_svc_from();
|
|
test_driver_install();
|
|
test_dirid();
|
|
|
|
UnhookWindowsHookEx(hhook);
|
|
|
|
/* We have to run this test after the CBT hook is disabled because
|
|
ProfileItems needs to create a window on Windows XP. */
|
|
test_profile_items();
|
|
}
|
|
|
|
test_inffilelist();
|
|
test_inffilelistA();
|
|
|
|
SetCurrentDirectoryA(prev_path);
|
|
}
|