| /* |
| * Devinst tests |
| * |
| * Copyright 2006 Christian Gmeiner |
| * |
| * 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 "windef.h" |
| #include "winbase.h" |
| #include "wingdi.h" |
| #include "winuser.h" |
| #include "winreg.h" |
| #include "guiddef.h" |
| #include "setupapi.h" |
| |
| #include "wine/test.h" |
| |
| static BOOL is_wow64; |
| |
| /* function pointers */ |
| static HDEVINFO (WINAPI *pSetupDiCreateDeviceInfoList)(GUID*,HWND); |
| static HDEVINFO (WINAPI *pSetupDiCreateDeviceInfoListExW)(GUID*,HWND,PCWSTR,PVOID); |
| static BOOL (WINAPI *pSetupDiCreateDeviceInterfaceA)(HDEVINFO, PSP_DEVINFO_DATA, const GUID *, PCSTR, DWORD, PSP_DEVICE_INTERFACE_DATA); |
| static BOOL (WINAPI *pSetupDiCallClassInstaller)(DI_FUNCTION, HDEVINFO, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiDestroyDeviceInfoList)(HDEVINFO); |
| static BOOL (WINAPI *pSetupDiEnumDeviceInfo)(HDEVINFO, DWORD, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiEnumDeviceInterfaces)(HDEVINFO, PSP_DEVINFO_DATA, const GUID *, DWORD, PSP_DEVICE_INTERFACE_DATA); |
| static BOOL (WINAPI *pSetupDiGetINFClassA)(PCSTR, LPGUID, PSTR, DWORD, PDWORD); |
| static BOOL (WINAPI *pSetupDiInstallClassA)(HWND, PCSTR, DWORD, HSPFILEQ); |
| static HKEY (WINAPI *pSetupDiOpenClassRegKeyExA)(GUID*,REGSAM,DWORD,PCSTR,PVOID); |
| static HKEY (WINAPI *pSetupDiOpenDevRegKey)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, DWORD, DWORD, REGSAM); |
| static HKEY (WINAPI *pSetupDiCreateDevRegKeyW)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, DWORD, DWORD, HINF, PCWSTR); |
| static BOOL (WINAPI *pSetupDiCreateDeviceInfoA)(HDEVINFO, PCSTR, GUID *, PCSTR, HWND, DWORD, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiCreateDeviceInfoW)(HDEVINFO, PCWSTR, GUID *, PCWSTR, HWND, DWORD, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiGetDeviceInstanceIdA)(HDEVINFO, PSP_DEVINFO_DATA, PSTR, DWORD, PDWORD); |
| static BOOL (WINAPI *pSetupDiGetDeviceInterfaceDetailA)(HDEVINFO, PSP_DEVICE_INTERFACE_DATA, PSP_DEVICE_INTERFACE_DETAIL_DATA_A, DWORD, PDWORD, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiGetDeviceInterfaceDetailW)(HDEVINFO, PSP_DEVICE_INTERFACE_DATA, PSP_DEVICE_INTERFACE_DETAIL_DATA_W, DWORD, PDWORD, PSP_DEVINFO_DATA); |
| static BOOL (WINAPI *pSetupDiRegisterDeviceInfo)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, PSP_DETSIG_CMPPROC, PVOID, PSP_DEVINFO_DATA); |
| static HDEVINFO (WINAPI *pSetupDiGetClassDevsA)(CONST GUID *, LPCSTR, HWND, DWORD); |
| static HDEVINFO (WINAPI *pSetupDiGetClassDevsW)(CONST GUID *, LPCWSTR, HWND, DWORD); |
| static BOOL (WINAPI *pSetupDiSetDeviceRegistryPropertyA)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, const BYTE *, DWORD); |
| static BOOL (WINAPI *pSetupDiSetDeviceRegistryPropertyW)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, const BYTE *, DWORD); |
| static BOOL (WINAPI *pSetupDiGetDeviceRegistryPropertyA)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, PDWORD, PBYTE, DWORD, PDWORD); |
| static BOOL (WINAPI *pSetupDiGetDeviceRegistryPropertyW)(HDEVINFO, PSP_DEVINFO_DATA, DWORD, PDWORD, PBYTE, DWORD, PDWORD); |
| static BOOL (WINAPI *pIsWow64Process)(HANDLE, PBOOL); |
| |
| /* This is a unique guid for testing purposes */ |
| static GUID guid = {0x6a55b5a4, 0x3f65, 0x11db, {0xb7,0x04,0x00,0x11,0x95,0x5c,0x2b,0xdb}}; |
| |
| static void init_function_pointers(void) |
| { |
| HMODULE hSetupAPI = GetModuleHandleA("setupapi.dll"); |
| HMODULE hKernel32 = GetModuleHandleA("kernel32.dll"); |
| |
| pSetupDiCreateDeviceInfoA = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDeviceInfoA"); |
| pSetupDiCreateDeviceInfoW = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDeviceInfoW"); |
| pSetupDiCreateDeviceInfoList = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDeviceInfoList"); |
| pSetupDiCreateDeviceInfoListExW = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDeviceInfoListExW"); |
| pSetupDiCreateDeviceInterfaceA = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDeviceInterfaceA"); |
| pSetupDiDestroyDeviceInfoList = (void *)GetProcAddress(hSetupAPI, "SetupDiDestroyDeviceInfoList"); |
| pSetupDiCallClassInstaller = (void *)GetProcAddress(hSetupAPI, "SetupDiCallClassInstaller"); |
| pSetupDiEnumDeviceInfo = (void *)GetProcAddress(hSetupAPI, "SetupDiEnumDeviceInfo"); |
| pSetupDiEnumDeviceInterfaces = (void *)GetProcAddress(hSetupAPI, "SetupDiEnumDeviceInterfaces"); |
| pSetupDiGetDeviceInstanceIdA = (void *)GetProcAddress(hSetupAPI, "SetupDiGetDeviceInstanceIdA"); |
| pSetupDiGetDeviceInterfaceDetailA = (void *)GetProcAddress(hSetupAPI, "SetupDiGetDeviceInterfaceDetailA"); |
| pSetupDiGetDeviceInterfaceDetailW = (void *)GetProcAddress(hSetupAPI, "SetupDiGetDeviceInterfaceDetailW"); |
| pSetupDiInstallClassA = (void *)GetProcAddress(hSetupAPI, "SetupDiInstallClassA"); |
| pSetupDiOpenClassRegKeyExA = (void *)GetProcAddress(hSetupAPI, "SetupDiOpenClassRegKeyExA"); |
| pSetupDiOpenDevRegKey = (void *)GetProcAddress(hSetupAPI, "SetupDiOpenDevRegKey"); |
| pSetupDiCreateDevRegKeyW = (void *)GetProcAddress(hSetupAPI, "SetupDiCreateDevRegKeyW"); |
| pSetupDiRegisterDeviceInfo = (void *)GetProcAddress(hSetupAPI, "SetupDiRegisterDeviceInfo"); |
| pSetupDiGetClassDevsA = (void *)GetProcAddress(hSetupAPI, "SetupDiGetClassDevsA"); |
| pSetupDiGetClassDevsW = (void *)GetProcAddress(hSetupAPI, "SetupDiGetClassDevsW"); |
| pSetupDiGetINFClassA = (void *)GetProcAddress(hSetupAPI, "SetupDiGetINFClassA"); |
| pSetupDiSetDeviceRegistryPropertyA = (void *)GetProcAddress(hSetupAPI, "SetupDiSetDeviceRegistryPropertyA"); |
| pSetupDiSetDeviceRegistryPropertyW = (void *)GetProcAddress(hSetupAPI, "SetupDiSetDeviceRegistryPropertyW"); |
| pSetupDiGetDeviceRegistryPropertyA = (void *)GetProcAddress(hSetupAPI, "SetupDiGetDeviceRegistryPropertyA"); |
| pSetupDiGetDeviceRegistryPropertyW = (void *)GetProcAddress(hSetupAPI, "SetupDiGetDeviceRegistryPropertyW"); |
| pIsWow64Process = (void *)GetProcAddress(hKernel32, "IsWow64Process"); |
| } |
| |
| static void change_reg_permissions(const WCHAR *regkey) |
| { |
| HKEY hkey; |
| SID_IDENTIFIER_AUTHORITY ident = { SECURITY_WORLD_SID_AUTHORITY }; |
| SECURITY_DESCRIPTOR sd; |
| PSID EveryoneSid; |
| PACL pacl = NULL; |
| |
| RegOpenKeyExW(HKEY_LOCAL_MACHINE, regkey, 0, WRITE_DAC, &hkey); |
| |
| /* Initialize the 'Everyone' sid */ |
| AllocateAndInitializeSid(&ident, 1, SECURITY_WORLD_RID, 0, 0, 0, 0, 0, 0, 0, &EveryoneSid); |
| |
| pacl = HeapAlloc(GetProcessHeap(), 0, 256); |
| InitializeAcl(pacl, 256, ACL_REVISION); |
| |
| /* Add 'Full Control' for 'Everyone' */ |
| AddAccessAllowedAce(pacl, ACL_REVISION, KEY_ALL_ACCESS, EveryoneSid); |
| |
| InitializeSecurityDescriptor(&sd, SECURITY_DESCRIPTOR_REVISION); |
| |
| SetSecurityDescriptorDacl(&sd, TRUE, pacl, FALSE); |
| |
| /* Set the new security on the registry key */ |
| RegSetKeySecurity(hkey, DACL_SECURITY_INFORMATION, &sd); |
| |
| RegCloseKey(hkey); |
| |
| HeapFree(GetProcessHeap(), 0, pacl); |
| if (EveryoneSid) |
| FreeSid(EveryoneSid); |
| } |
| |
| static BOOL remove_device(void) |
| { |
| HDEVINFO set; |
| SP_DEVINFO_DATA devInfo = { sizeof(devInfo), { 0 } }; |
| BOOL ret, retval; |
| |
| SetLastError(0xdeadbeef); |
| set = pSetupDiGetClassDevsA(&guid, NULL, 0, 0); |
| ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsA failed: %08x\n", |
| GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiEnumDeviceInfo(set, 0, &devInfo); |
| ok(ret, "SetupDiEnumDeviceInfo failed: %08x\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = pSetupDiCallClassInstaller(DIF_REMOVE, set, &devInfo); |
| if(is_wow64) |
| todo_wine ok(!retval && GetLastError() == ERROR_IN_WOW64, |
| "SetupDiCallClassInstaller(DIF_REMOVE...) succeeded: %08x\n", GetLastError()); |
| else |
| todo_wine ok(retval, |
| "SetupDiCallClassInstaller(DIF_REMOVE...) failed: %08x\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiDestroyDeviceInfoList(set); |
| ok(ret, "SetupDiDestroyDeviceInfoList failed: %08x\n", GetLastError()); |
| |
| return retval; |
| } |
| |
| /* RegDeleteTreeW from dlls/advapi32/registry.c */ |
| static LSTATUS devinst_RegDeleteTreeW(HKEY hKey, LPCWSTR lpszSubKey) |
| { |
| LONG ret; |
| DWORD dwMaxSubkeyLen, dwMaxValueLen; |
| DWORD dwMaxLen, dwSize; |
| WCHAR szNameBuf[MAX_PATH], *lpszName = szNameBuf; |
| HKEY hSubKey = hKey; |
| |
| if(lpszSubKey) |
| { |
| ret = RegOpenKeyExW(hKey, lpszSubKey, 0, KEY_READ, &hSubKey); |
| if (ret) return ret; |
| } |
| |
| /* Get highest length for keys, values */ |
| ret = RegQueryInfoKeyW(hSubKey, NULL, NULL, NULL, NULL, |
| &dwMaxSubkeyLen, NULL, NULL, &dwMaxValueLen, NULL, NULL, NULL); |
| if (ret) goto cleanup; |
| |
| dwMaxSubkeyLen++; |
| dwMaxValueLen++; |
| dwMaxLen = max(dwMaxSubkeyLen, dwMaxValueLen); |
| if (dwMaxLen > sizeof(szNameBuf)/sizeof(WCHAR)) |
| { |
| /* Name too big: alloc a buffer for it */ |
| if (!(lpszName = HeapAlloc( GetProcessHeap(), 0, dwMaxLen*sizeof(WCHAR)))) |
| { |
| ret = ERROR_NOT_ENOUGH_MEMORY; |
| goto cleanup; |
| } |
| } |
| |
| |
| /* Recursively delete all the subkeys */ |
| while (TRUE) |
| { |
| dwSize = dwMaxLen; |
| if (RegEnumKeyExW(hSubKey, 0, lpszName, &dwSize, NULL, |
| NULL, NULL, NULL)) break; |
| |
| ret = devinst_RegDeleteTreeW(hSubKey, lpszName); |
| if (ret) goto cleanup; |
| } |
| |
| if (lpszSubKey) |
| ret = RegDeleteKeyW(hKey, lpszSubKey); |
| else |
| while (TRUE) |
| { |
| dwSize = dwMaxLen; |
| if (RegEnumValueW(hKey, 0, lpszName, &dwSize, |
| NULL, NULL, NULL, NULL)) break; |
| |
| ret = RegDeleteValueW(hKey, lpszName); |
| if (ret) goto cleanup; |
| } |
| |
| cleanup: |
| /* Free buffer if allocated */ |
| if (lpszName != szNameBuf) |
| HeapFree( GetProcessHeap(), 0, lpszName); |
| if(lpszSubKey) |
| RegCloseKey(hSubKey); |
| return ret; |
| } |
| |
| static void clean_devclass_key(void) |
| { |
| static const WCHAR devclass[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','D','e','v','i','c','e','C','l','a','s','s','e','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| HKEY key; |
| DWORD subkeys; |
| |
| /* Check if we have subkeys as Windows 2000 doesn't delete |
| * the keys under the DeviceClasses key after a SetupDiDestroyDeviceInfoList. |
| */ |
| RegOpenKeyW(HKEY_LOCAL_MACHINE, devclass, &key); |
| RegQueryInfoKey(key, NULL, NULL, NULL, &subkeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL); |
| if (subkeys > 0) |
| { |
| trace("We are most likely on Windows 2000\n"); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, devclass); |
| } |
| else |
| { |
| ok(!RegDeleteKeyW(HKEY_LOCAL_MACHINE, devclass), |
| "Couldn't delete deviceclass key\n"); |
| } |
| } |
| |
| static void test_SetupDiCreateDeviceInfoListEx(void) |
| { |
| HDEVINFO devlist; |
| BOOL ret; |
| DWORD error; |
| static CHAR notnull[] = "NotNull"; |
| static const WCHAR machine[] = { 'd','u','m','m','y',0 }; |
| static const WCHAR empty[] = { 0 }; |
| |
| SetLastError(0xdeadbeef); |
| /* create empty DeviceInfoList, but set Reserved to a value, which is not NULL */ |
| devlist = pSetupDiCreateDeviceInfoListExW(NULL, NULL, NULL, notnull); |
| |
| error = GetLastError(); |
| if (error == ERROR_CALL_NOT_IMPLEMENTED) |
| { |
| win_skip("SetupDiCreateDeviceInfoListExW is not implemented\n"); |
| return; |
| } |
| ok(devlist == INVALID_HANDLE_VALUE, "SetupDiCreateDeviceInfoListExW failed : %p %d (expected %p)\n", devlist, error, INVALID_HANDLE_VALUE); |
| ok(error == ERROR_INVALID_PARAMETER, "GetLastError returned wrong value : %d, (expected %d)\n", error, ERROR_INVALID_PARAMETER); |
| |
| SetLastError(0xdeadbeef); |
| /* create empty DeviceInfoList, but set MachineName to something */ |
| devlist = pSetupDiCreateDeviceInfoListExW(NULL, NULL, machine, NULL); |
| |
| error = GetLastError(); |
| ok(devlist == INVALID_HANDLE_VALUE, "SetupDiCreateDeviceInfoListExW failed : %p %d (expected %p)\n", devlist, error, INVALID_HANDLE_VALUE); |
| ok(error == ERROR_INVALID_MACHINENAME || error == ERROR_MACHINE_UNAVAILABLE, "GetLastError returned wrong value : %d, (expected %d or %d)\n", error, ERROR_INVALID_MACHINENAME, ERROR_MACHINE_UNAVAILABLE); |
| |
| /* create empty DeviceInfoList */ |
| devlist = pSetupDiCreateDeviceInfoListExW(NULL, NULL, NULL, NULL); |
| ok(devlist && devlist != INVALID_HANDLE_VALUE, "SetupDiCreateDeviceInfoListExW failed : %p %d (expected != %p)\n", devlist, error, INVALID_HANDLE_VALUE); |
| |
| /* destroy DeviceInfoList */ |
| ret = pSetupDiDestroyDeviceInfoList(devlist); |
| ok(ret, "SetupDiDestroyDeviceInfoList failed : %d\n", error); |
| |
| /* create empty DeviceInfoList with empty machine name */ |
| devlist = pSetupDiCreateDeviceInfoListExW(NULL, NULL, empty, NULL); |
| ok(devlist && devlist != INVALID_HANDLE_VALUE, "SetupDiCreateDeviceInfoListExW failed : %p %d (expected != %p)\n", devlist, error, INVALID_HANDLE_VALUE); |
| |
| /* destroy DeviceInfoList */ |
| ret = pSetupDiDestroyDeviceInfoList(devlist); |
| ok(ret, "SetupDiDestroyDeviceInfoList failed : %d\n", error); |
| } |
| |
| static void test_SetupDiOpenClassRegKeyExA(void) |
| { |
| static const CHAR guidString[] = "{6a55b5a4-3f65-11db-b704-0011955c2bdb}"; |
| HKEY hkey; |
| |
| /* Check return value for nonexistent key */ |
| hkey = pSetupDiOpenClassRegKeyExA(&guid, KEY_ALL_ACCESS, |
| DIOCR_INSTALLER, NULL, NULL); |
| ok(hkey == INVALID_HANDLE_VALUE, |
| "returned %p (expected INVALID_HANDLE_VALUE)\n", hkey); |
| |
| /* Test it for a key that exists */ |
| hkey = SetupDiOpenClassRegKey(NULL, KEY_ALL_ACCESS); |
| if (hkey != INVALID_HANDLE_VALUE) |
| { |
| HKEY classKey; |
| if (RegCreateKeyA(hkey, guidString, &classKey) == ERROR_SUCCESS) |
| { |
| RegCloseKey(classKey); |
| SetLastError(0xdeadbeef); |
| classKey = pSetupDiOpenClassRegKeyExA(&guid, KEY_ALL_ACCESS, |
| DIOCR_INSTALLER, NULL, NULL); |
| ok(classKey != INVALID_HANDLE_VALUE, |
| "opening class registry key failed with error %d\n", |
| GetLastError()); |
| if (classKey != INVALID_HANDLE_VALUE) |
| RegCloseKey(classKey); |
| RegDeleteKeyA(hkey, guidString); |
| } |
| else |
| trace("failed to create registry key for test\n"); |
| |
| RegCloseKey(hkey); |
| } |
| else |
| trace("failed to open classes key\n"); |
| } |
| |
| static void create_inf_file(LPCSTR filename) |
| { |
| DWORD dwNumberOfBytesWritten; |
| HANDLE hf = CreateFile(filename, GENERIC_WRITE, 0, NULL, |
| CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); |
| |
| static const char data[] = |
| "[Version]\n" |
| "Signature=\"$Chicago$\"\n" |
| "Class=Bogus\n" |
| "ClassGUID={6a55b5a4-3f65-11db-b704-0011955c2bdb}\n" |
| "[ClassInstall32]\n" |
| "AddReg=BogusClass.NT.AddReg\n" |
| "[BogusClass.NT.AddReg]\n" |
| "HKR,,,,\"Wine test devices\"\n"; |
| |
| WriteFile(hf, data, sizeof(data) - 1, &dwNumberOfBytesWritten, NULL); |
| CloseHandle(hf); |
| } |
| |
| static void get_temp_filename(LPSTR path) |
| { |
| static char curr[MAX_PATH] = { 0 }; |
| char temp[MAX_PATH]; |
| LPSTR ptr; |
| |
| if (!*curr) |
| GetCurrentDirectoryA(MAX_PATH, curr); |
| GetTempFileNameA(curr, "set", 0, temp); |
| ptr = strrchr(temp, '\\'); |
| |
| lstrcpyA(path, ptr + 1); |
| } |
| |
| static void testInstallClass(void) |
| { |
| static const WCHAR classKey[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','C','l','a','s','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| char tmpfile[MAX_PATH]; |
| BOOL ret; |
| |
| tmpfile[0] = '.'; |
| tmpfile[1] = '\\'; |
| get_temp_filename(tmpfile + 2); |
| create_inf_file(tmpfile + 2); |
| |
| ret = pSetupDiInstallClassA(NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| ret = pSetupDiInstallClassA(NULL, NULL, DI_NOVCP, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| ret = pSetupDiInstallClassA(NULL, tmpfile + 2, DI_NOVCP, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| ret = pSetupDiInstallClassA(NULL, tmpfile + 2, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND, |
| "Expected ERROR_FILE_NOT_FOUND, got %08x\n", GetLastError()); |
| /* The next call will succeed. Information is put into the registry but the |
| * location(s) is/are depending on the Windows version. |
| */ |
| ret = pSetupDiInstallClassA(NULL, tmpfile, 0, NULL); |
| ok(ret, "SetupDiInstallClassA failed: %08x\n", GetLastError()); |
| |
| ok(!RegDeleteKeyW(HKEY_LOCAL_MACHINE, classKey), |
| "Couldn't delete classkey\n"); |
| |
| DeleteFile(tmpfile); |
| } |
| |
| static void testCreateDeviceInfo(void) |
| { |
| BOOL ret; |
| HDEVINFO set; |
| HKEY key; |
| LONG res; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(NULL, NULL, NULL, NULL, NULL, 0, NULL); |
| ok(!ret, "Expected failure\n"); |
| ok(GetLastError() == ERROR_INVALID_DEVINST_NAME || |
| GetLastError() == ERROR_INVALID_PARAMETER /* NT4 */, |
| "Unexpected last error, got %08x\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(NULL, "Root\\LEGACY_BOGUS\\0000", NULL, |
| NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLEHANDLE, got %08x\n", GetLastError()); |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %08x\n", |
| GetLastError()); |
| if (set) |
| { |
| SP_DEVINFO_DATA devInfo = { 0 }; |
| DWORD i; |
| static GUID deadbeef = |
| {0xdeadbeef, 0xdead, 0xbeef, {0xde,0xad,0xbe,0xef,0xde,0xad,0xbe,0xef}}; |
| LONG res; |
| HKEY key; |
| static const WCHAR bogus0000[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S','\\','0','0','0','0',0}; |
| |
| /* So we know we have a clean start */ |
| res = RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus0000, &key); |
| ok(res != ERROR_SUCCESS, "Expected key to not exist\n"); |
| /* No GUID given */ |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, "Root\\LEGACY_BOGUS\\0000", NULL, |
| NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| /* Even though NT4 fails it still adds some stuff to the registry that |
| * can't be deleted via normal setupapi functions. As the registry is written |
| * by a different user (SYSTEM) we have to do some magic to get rid of the key |
| */ |
| if (!RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus0000, &key)) |
| { |
| trace("NT4 created a bogus key on failure, will be removed now\n"); |
| change_reg_permissions(bogus0000); |
| ok(!RegDeleteKeyW(HKEY_LOCAL_MACHINE, bogus0000), |
| "Could not delete LEGACY_BOGUS\\0000 key\n"); |
| } |
| /* We can't add device information to the set with a different GUID */ |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, "Root\\LEGACY_BOGUS\\0000", |
| &deadbeef, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_CLASS_MISMATCH, |
| "Expected ERROR_CLASS_MISMATCH, got %08x\n", GetLastError()); |
| if (!RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus0000, &key)) |
| { |
| trace("NT4 created a bogus key on failure, will be removed now\n"); |
| change_reg_permissions(bogus0000); |
| ok(!RegDeleteKeyW(HKEY_LOCAL_MACHINE, bogus0000), |
| "Could not delete LEGACY_BOGUS\\0000 key\n"); |
| } |
| /* Finally, with all three required parameters, this succeeds: */ |
| ret = pSetupDiCreateDeviceInfoA(set, "Root\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, NULL); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| /* This fails because the device ID already exists.. */ |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, "Root\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(!ret && GetLastError() == ERROR_DEVINST_ALREADY_EXISTS, |
| "Expected ERROR_DEVINST_ALREADY_EXISTS, got %08x\n", GetLastError()); |
| /* whereas this "fails" because cbSize is wrong.. */ |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, "LEGACY_BOGUS", &guid, NULL, NULL, |
| DICD_GENERATE_ID, &devInfo); |
| ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, |
| "Expected ERROR_INVALID_USER_BUFFER, got %08x\n", GetLastError()); |
| /* and this finally succeeds. */ |
| devInfo.cbSize = sizeof(devInfo); |
| ret = pSetupDiCreateDeviceInfoA(set, "LEGACY_BOGUS", &guid, NULL, NULL, |
| DICD_GENERATE_ID, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| /* There were three devices added, however - the second failure just |
| * resulted in the SP_DEVINFO_DATA not getting copied. |
| */ |
| SetLastError(0xdeadbeef); |
| i = 0; |
| while (pSetupDiEnumDeviceInfo(set, i, &devInfo)) |
| i++; |
| ok(i == 3, "Expected 3 devices, got %d\n", i); |
| ok(GetLastError() == ERROR_NO_MORE_ITEMS, |
| "SetupDiEnumDeviceInfo failed: %08x\n", GetLastError()); |
| pSetupDiDestroyDeviceInfoList(set); |
| } |
| |
| /* The bogus registry key shouldn't be there after this test. The only |
| * reasons this key would still be present: |
| * |
| * - We are running on Wine which has to be fixed |
| * - We have leftovers from old tests |
| */ |
| res = RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key); |
| todo_wine |
| ok(res == ERROR_FILE_NOT_FOUND, "Expected key to not exist\n"); |
| if (res == ERROR_SUCCESS) |
| { |
| DWORD subkeys; |
| |
| /* Check if we have subkeys */ |
| RegQueryInfoKey(key, NULL, NULL, NULL, &subkeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL); |
| if (subkeys > 0) |
| { |
| int i; |
| |
| /* Leftovers from old tests */ |
| trace("Going to remove %d devices\n", subkeys); |
| for (i = 0; i < subkeys; i++) |
| { |
| BOOL ret; |
| |
| ret = remove_device(); |
| ok(ret, "Expected a device to be removed\n"); |
| } |
| } |
| else |
| { |
| /* Wine doesn't delete the bogus key itself currently */ |
| trace("We are most likely on Wine\n"); |
| RegDeleteKeyW(HKEY_LOCAL_MACHINE, bogus); |
| } |
| } |
| } |
| |
| static void testGetDeviceInstanceId(void) |
| { |
| BOOL ret; |
| HDEVINFO set; |
| SP_DEVINFO_DATA devInfo = { 0 }; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(NULL, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(NULL, &devInfo, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %08x\n", |
| GetLastError()); |
| if (set) |
| { |
| char instanceID[MAX_PATH]; |
| DWORD size; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(set, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, NULL, 0, &size); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| devInfo.cbSize = sizeof(devInfo); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, NULL, 0, &size); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| ret = pSetupDiCreateDeviceInfoA(set, "Root\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, NULL, 0, &size); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %08x\n", GetLastError()); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, instanceID, |
| sizeof(instanceID), NULL); |
| ok(ret, "SetupDiGetDeviceInstanceIdA failed: %08x\n", GetLastError()); |
| ok(!lstrcmpA(instanceID, "ROOT\\LEGACY_BOGUS\\0000"), |
| "Unexpected instance ID %s\n", instanceID); |
| ret = pSetupDiCreateDeviceInfoA(set, "LEGACY_BOGUS", &guid, |
| NULL, NULL, DICD_GENERATE_ID, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| ret = pSetupDiGetDeviceInstanceIdA(set, &devInfo, instanceID, |
| sizeof(instanceID), NULL); |
| ok(ret, "SetupDiGetDeviceInstanceIdA failed: %08x\n", GetLastError()); |
| /* NT4 returns 'Root' and W2K and above 'ROOT' */ |
| ok(!lstrcmpiA(instanceID, "ROOT\\LEGACY_BOGUS\\0001"), |
| "Unexpected instance ID %s\n", instanceID); |
| pSetupDiDestroyDeviceInfoList(set); |
| } |
| } |
| |
| static void testRegisterDeviceInfo(void) |
| { |
| BOOL ret; |
| HDEVINFO set; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(NULL, NULL, 0, NULL, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError()); |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %d\n", GetLastError()); |
| if (set) |
| { |
| SP_DEVINFO_DATA devInfo = { 0 }; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(set, NULL, 0, NULL, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(set, &devInfo, 0, NULL, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| devInfo.cbSize = sizeof(devInfo); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(set, &devInfo, 0, NULL, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| ret = pSetupDiCreateDeviceInfoA(set, "USB\\BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(ret || GetLastError() == ERROR_DEVINST_ALREADY_EXISTS, |
| "SetupDiCreateDeviceInfoA failed: %d\n", GetLastError()); |
| if (ret) |
| { |
| /* If it already existed, registering it again will fail */ |
| ret = pSetupDiRegisterDeviceInfo(set, &devInfo, 0, NULL, NULL, |
| NULL); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %d\n", GetLastError()); |
| } |
| /* FIXME: On Win2K+ systems, this is now persisted to registry in |
| * HKLM\System\CCS\Enum\USB\Bogus\0000. |
| * FIXME: the key also becomes undeletable. How to get rid of it? |
| */ |
| pSetupDiDestroyDeviceInfoList(set); |
| } |
| } |
| |
| static void testCreateDeviceInterface(void) |
| { |
| BOOL ret; |
| HDEVINFO set; |
| HKEY key; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| static const WCHAR devclass[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','D','e','v','i','c','e','C','l','a','s','s','e','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| |
| if (!pSetupDiCreateDeviceInterfaceA || !pSetupDiEnumDeviceInterfaces) |
| { |
| win_skip("SetupDiCreateDeviceInterfaceA and/or SetupDiEnumDeviceInterfaces are not available\n"); |
| return; |
| } |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(NULL, NULL, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(NULL, NULL, &guid, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError()); |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %d\n", GetLastError()); |
| if (set) |
| { |
| SP_DEVINFO_DATA devInfo = { 0 }; |
| SP_DEVICE_INTERFACE_DATA interfaceData = { sizeof(interfaceData), |
| { 0 } }; |
| DWORD i; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(set, NULL, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, NULL, NULL, 0, |
| NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| devInfo.cbSize = sizeof(devInfo); |
| ret = pSetupDiCreateDeviceInfoA(set, "ROOT\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, NULL, NULL, 0, |
| NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, |
| "Expected ERROR_INVALID_USER_BUFFER, got %08x\n", GetLastError()); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, &guid, NULL, 0, |
| NULL); |
| ok(ret, "SetupDiCreateDeviceInterfaceA failed: %08x\n", GetLastError()); |
| /* Creating the same interface a second time succeeds */ |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, &guid, NULL, 0, |
| NULL); |
| ok(ret, "SetupDiCreateDeviceInterfaceA failed: %08x\n", GetLastError()); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, &guid, "Oogah", 0, |
| NULL); |
| ok(ret, "SetupDiCreateDeviceInterfaceA failed: %08x\n", GetLastError()); |
| ret = pSetupDiEnumDeviceInterfaces(set, &devInfo, &guid, 0, |
| &interfaceData); |
| ok(ret, "SetupDiEnumDeviceInterfaces failed: %d\n", GetLastError()); |
| i = 0; |
| while (pSetupDiEnumDeviceInterfaces(set, &devInfo, &guid, i, |
| &interfaceData)) |
| i++; |
| ok(i == 2, "expected 2 interfaces, got %d\n", i); |
| ok(GetLastError() == ERROR_NO_MORE_ITEMS, |
| "SetupDiEnumDeviceInterfaces failed: %08x\n", GetLastError()); |
| ret = pSetupDiDestroyDeviceInfoList(set); |
| ok(ret, "SetupDiDestroyDeviceInfoList failed: %08x\n", GetLastError()); |
| |
| /* Cleanup */ |
| /* FIXME: On Wine we still have the bogus entry in Enum\Root and |
| * subkeys, as well as the deviceclass key with subkeys. |
| * Only clean the deviceclass key once Wine if fixed. |
| */ |
| if (!RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key)) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, bogus); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, devclass); |
| } |
| else |
| { |
| clean_devclass_key(); |
| } |
| } |
| } |
| |
| static void testGetDeviceInterfaceDetail(void) |
| { |
| BOOL ret; |
| HDEVINFO set; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| static const WCHAR devclass[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','D','e','v','i','c','e','C','l','a','s','s','e','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| |
| if (!pSetupDiCreateDeviceInterfaceA || !pSetupDiGetDeviceInterfaceDetailA) |
| { |
| win_skip("SetupDiCreateDeviceInterfaceA and/or SetupDiGetDeviceInterfaceDetailA are not available\n"); |
| return; |
| } |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(NULL, NULL, NULL, 0, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError()); |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %d\n", GetLastError()); |
| if (set) |
| { |
| SP_DEVINFO_DATA devInfo = { sizeof(devInfo), { 0 } }; |
| SP_DEVICE_INTERFACE_DATA interfaceData = { sizeof(interfaceData), |
| { 0 } }; |
| DWORD size = 0; |
| HKEY key; |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, NULL, NULL, 0, NULL, |
| NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| ret = pSetupDiCreateDeviceInfoA(set, "ROOT\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, &guid, NULL, 0, |
| &interfaceData); |
| ok(ret, "SetupDiCreateDeviceInterfaceA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, NULL, |
| 0, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, NULL, |
| 100, NULL, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, |
| "Expected ERROR_INVALID_USER_BUFFER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, NULL, |
| 0, &size, NULL); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError()); |
| if (!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER) |
| { |
| static const char path[] = |
| "\\\\?\\root#legacy_bogus#0000#{6a55b5a4-3f65-11db-b704-0011955c2bdb}"; |
| static const char path_w2k[] = |
| "\\\\?\\root#legacy_bogus#0000#{6a55b5a4-3f65-11db-b704-0011955c2bdb}\\"; |
| LPBYTE buf = HeapAlloc(GetProcessHeap(), 0, size); |
| SP_DEVICE_INTERFACE_DETAIL_DATA_A *detail = |
| (SP_DEVICE_INTERFACE_DETAIL_DATA_A *)buf; |
| DWORD expectedsize = FIELD_OFFSET(SP_DEVICE_INTERFACE_DETAIL_DATA_W, DevicePath) + sizeof(WCHAR)*(1 + strlen(path)); |
| |
| detail->cbSize = 0; |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, detail, |
| size, &size, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, |
| "Expected ERROR_INVALID_USER_BUFFER, got %08x\n", GetLastError()); |
| detail->cbSize = size; |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, detail, |
| size, &size, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_USER_BUFFER, |
| "Expected ERROR_INVALID_USER_BUFFER, got %08x\n", GetLastError()); |
| detail->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA_A); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, detail, |
| size, &size, NULL); |
| ok(ret, "SetupDiGetDeviceInterfaceDetailA failed: %d\n", |
| GetLastError()); |
| ok(!lstrcmpiA(path, detail->DevicePath) || |
| !lstrcmpiA(path_w2k, detail->DevicePath), "Unexpected path %s\n", |
| detail->DevicePath); |
| /* Check SetupDiGetDeviceInterfaceDetailW */ |
| ret = pSetupDiGetDeviceInterfaceDetailW(set, &interfaceData, NULL, 0, &size, NULL); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got error code: %d\n", GetLastError()); |
| ok(expectedsize == size || |
| (expectedsize + sizeof(WCHAR)) == size /* W2K adds a backslash */, |
| "SetupDiGetDeviceInterfaceDetailW returned wrong reqsize, got %d\n", |
| size); |
| |
| HeapFree(GetProcessHeap(), 0, buf); |
| } |
| pSetupDiDestroyDeviceInfoList(set); |
| |
| /* Cleanup */ |
| /* FIXME: On Wine we still have the bogus entry in Enum\Root and |
| * subkeys, as well as the deviceclass key with subkeys. |
| * Only do the RegDeleteKey, once Wine is fixed. |
| */ |
| if (!RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key)) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, bogus); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, devclass); |
| } |
| else |
| { |
| clean_devclass_key(); |
| } |
| } |
| } |
| |
| static void testDevRegKey(void) |
| { |
| static const WCHAR classKey[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','C','l','a','s','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| BOOL ret; |
| HDEVINFO set; |
| HKEY key = NULL; |
| BOOL classKeyCreated; |
| |
| SetLastError(0xdeadbeef); |
| key = pSetupDiCreateDevRegKeyW(NULL, NULL, 0, 0, 0, NULL, NULL); |
| ok(key == INVALID_HANDLE_VALUE, |
| "Expected INVALID_HANDLE_VALUE, got %p\n", key); |
| ok(GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| |
| set = pSetupDiCreateDeviceInfoList(&guid, NULL); |
| ok(set != NULL, "SetupDiCreateDeviceInfoList failed: %d\n", GetLastError()); |
| if (set) |
| { |
| SP_DEVINFO_DATA devInfo = { sizeof(devInfo), { 0 } }; |
| LONG res; |
| |
| /* The device info key shouldn't be there */ |
| res = RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key); |
| ok(res != ERROR_SUCCESS, "Expected key to not exist\n"); |
| RegCloseKey(key); |
| /* Create the device information */ |
| ret = pSetupDiCreateDeviceInfoA(set, "ROOT\\LEGACY_BOGUS\\0000", &guid, |
| NULL, NULL, 0, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| /* The device info key should have been created */ |
| ok(!RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key), |
| "Expected registry key to exist\n"); |
| RegCloseKey(key); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(NULL, NULL, 0, 0, 0, 0); |
| ok(!key || key == INVALID_HANDLE_VALUE, |
| "Expected INVALID_HANDLE_VALUE or a NULL key (NT4)\n"); |
| ok(GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, NULL, 0, 0, 0, 0); |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, 0, 0, 0, 0); |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_INVALID_FLAGS, |
| "Expected ERROR_INVALID_FLAGS, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, 0, 0); |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_INVALID_FLAGS, |
| "Expected ERROR_INVALID_FLAGS, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_BOTH, 0); |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_INVALID_FLAGS, |
| "Expected ERROR_INVALID_FLAGS, got %d\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DRV, 0); |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_DEVINFO_NOT_REGISTERED, |
| "Expected ERROR_DEVINFO_NOT_REGISTERED, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(set, &devInfo, 0, NULL, NULL, NULL); |
| ok(ret, "SetupDiRegisterDeviceInfo failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DRV, 0); |
| /* The software key isn't created by default */ |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_KEY_DOES_NOT_EXIST, |
| "Expected ERROR_KEY_DOES_NOT_EXIST, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DEV, 0); |
| todo_wine |
| ok(key == INVALID_HANDLE_VALUE && |
| GetLastError() == ERROR_KEY_DOES_NOT_EXIST, |
| "Expected ERROR_KEY_DOES_NOT_EXIST, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| /* The class key shouldn't be there */ |
| res = RegOpenKeyW(HKEY_LOCAL_MACHINE, classKey, &key); |
| todo_wine |
| ok(res != ERROR_SUCCESS, "Expected key to not exist\n"); |
| RegCloseKey(key); |
| /* Create the device reg key */ |
| key = pSetupDiCreateDevRegKeyW(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DRV, NULL, NULL); |
| /* Vista and higher don't actually create the key */ |
| ok(key != INVALID_HANDLE_VALUE || GetLastError() == ERROR_KEY_DOES_NOT_EXIST, |
| "SetupDiCreateDevRegKey failed: %08x\n", GetLastError()); |
| if (key != INVALID_HANDLE_VALUE) |
| { |
| classKeyCreated = TRUE; |
| RegCloseKey(key); |
| /* The class key should have been created */ |
| ok(!RegOpenKeyW(HKEY_LOCAL_MACHINE, classKey, &key), |
| "Expected registry key to exist\n"); |
| RegCloseKey(key); |
| SetLastError(0xdeadbeef); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DRV, 0); |
| todo_wine |
| ok(key == INVALID_HANDLE_VALUE && |
| (GetLastError() == ERROR_INVALID_DATA || |
| GetLastError() == ERROR_ACCESS_DENIED), /* win2k3 */ |
| "Expected ERROR_INVALID_DATA or ERROR_ACCESS_DENIED, got %08x\n", GetLastError()); |
| key = pSetupDiOpenDevRegKey(set, &devInfo, DICS_FLAG_GLOBAL, 0, |
| DIREG_DRV, KEY_READ); |
| ok(key != INVALID_HANDLE_VALUE, "SetupDiOpenDevRegKey failed: %08x\n", |
| GetLastError()); |
| pSetupDiDestroyDeviceInfoList(set); |
| } |
| else |
| classKeyCreated = FALSE; |
| |
| /* Cleanup */ |
| ret = remove_device(); |
| if(!is_wow64) |
| todo_wine ok(ret, "Expected the device to be removed: %08x\n", GetLastError()); |
| |
| /* FIXME: Only do the RegDeleteKey, once Wine is fixed */ |
| if (!ret) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, bogus); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, classKey); |
| } |
| else if (classKeyCreated) |
| { |
| /* There should only be a class key entry, so a simple |
| * RegDeleteKey should work |
| * |
| * This could fail if it's the first time for this new test |
| * after running the old tests. |
| */ |
| ok(!RegDeleteKeyW(HKEY_LOCAL_MACHINE, classKey), |
| "Couldn't delete classkey\n"); |
| } |
| } |
| } |
| |
| static void testRegisterAndGetDetail(void) |
| { |
| HDEVINFO set; |
| BOOL ret; |
| SP_DEVINFO_DATA devInfo = { sizeof(SP_DEVINFO_DATA), { 0 } }; |
| SP_DEVICE_INTERFACE_DATA interfaceData = { sizeof(interfaceData), { 0 } }; |
| DWORD dwSize = 0; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| static const WCHAR devclass[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'C','o','n','t','r','o','l','\\','D','e','v','i','c','e','C','l','a','s','s','e','s','\\', |
| '{','6','a','5','5','b','5','a','4','-','3','f','6','5','-', |
| '1','1','d','b','-','b','7','0','4','-', |
| '0','0','1','1','9','5','5','c','2','b','d','b','}',0}; |
| |
| if (!pSetupDiCreateDeviceInterfaceA || !pSetupDiEnumDeviceInterfaces || |
| !pSetupDiGetDeviceInterfaceDetailA) |
| { |
| win_skip("Needed functions are not available\n"); |
| return; |
| } |
| |
| SetLastError(0xdeadbeef); |
| set = pSetupDiGetClassDevsA(&guid, NULL, 0, DIGCF_ALLCLASSES); |
| ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsA failed: %08x\n", |
| GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, "LEGACY_BOGUS", &guid, NULL, 0, |
| DICD_GENERATE_ID, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInterfaceA(set, &devInfo, &guid, NULL, 0, &interfaceData); |
| ok(ret, "SetupDiCreateDeviceInterfaceA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiRegisterDeviceInfo(set, &devInfo, 0, NULL, NULL, NULL); |
| ok(ret, "SetupDiRegisterDeviceInfo failed: %08x\n", GetLastError()); |
| |
| pSetupDiDestroyDeviceInfoList(set); |
| |
| SetLastError(0xdeadbeef); |
| set = pSetupDiGetClassDevsA(&guid, NULL, 0, DIGCF_DEVICEINTERFACE); |
| ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsA failed: %08x\n", |
| GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiEnumDeviceInterfaces(set, NULL, &guid, 0, &interfaceData); |
| ok(ret, "SetupDiEnumDeviceInterfaces failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, NULL, 0, &dwSize, NULL); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %08x\n", GetLastError()); |
| if (!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER) |
| { |
| static const char path[] = |
| "\\\\?\\root#legacy_bogus#0000#{6a55b5a4-3f65-11db-b704-0011955c2bdb}"; |
| static const char path_wow64[] = |
| "\\\\?\\root#legacy_bogus#0001#{6a55b5a4-3f65-11db-b704-0011955c2bdb}"; |
| static const char path_w2k[] = |
| "\\\\?\\root#legacy_bogus#0000#{6a55b5a4-3f65-11db-b704-0011955c2bdb}\\"; |
| PSP_DEVICE_INTERFACE_DETAIL_DATA_A detail = NULL; |
| |
| detail = HeapAlloc(GetProcessHeap(), 0, dwSize); |
| if (detail) |
| { |
| detail->cbSize = sizeof(*detail); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceInterfaceDetailA(set, &interfaceData, |
| detail, dwSize, &dwSize, NULL); |
| ok(ret, "SetupDiGetDeviceInterfaceDetailA failed: %08x\n", GetLastError()); |
| /* FIXME: This one only worked because old data wasn't removed properly. As soon |
| * as all the tests are cleaned up correctly this has to be (or should be) fixed |
| */ |
| if(is_wow64) |
| ok(!lstrcmpiA(path_wow64, detail->DevicePath), |
| "Unexpected path %s\n", detail->DevicePath); |
| else |
| todo_wine ok(!lstrcmpiA(path, detail->DevicePath) || |
| !lstrcmpiA(path_w2k, detail->DevicePath), |
| "Unexpected path %s\n", detail->DevicePath); |
| HeapFree(GetProcessHeap(), 0, detail); |
| } |
| } |
| |
| pSetupDiDestroyDeviceInfoList(set); |
| |
| /* Cleanup */ |
| ret = remove_device(); |
| if(!is_wow64) |
| todo_wine ok(ret, "Expected the device to be removed: %08x\n", GetLastError()); |
| |
| /* FIXME: Only do the RegDeleteKey, once Wine is fixed */ |
| if (!ret) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, bogus); |
| devinst_RegDeleteTreeW(HKEY_LOCAL_MACHINE, devclass); |
| } |
| else |
| { |
| clean_devclass_key(); |
| } |
| } |
| |
| static void testDeviceRegistryPropertyA(void) |
| { |
| HDEVINFO set; |
| SP_DEVINFO_DATA devInfo = { sizeof(SP_DEVINFO_DATA), { 0 } }; |
| CHAR devName[] = "LEGACY_BOGUS"; |
| CHAR friendlyName[] = "Bogus"; |
| CHAR buf[6] = ""; |
| DWORD buflen = 6; |
| DWORD size; |
| DWORD regType; |
| BOOL ret; |
| LONG res; |
| HKEY key; |
| static const CHAR bogus[] = |
| "System\\CurrentControlSet\\Enum\\Root\\LEGACY_BOGUS"; |
| |
| SetLastError(0xdeadbeef); |
| set = pSetupDiGetClassDevsA(&guid, NULL, 0, DIGCF_DEVICEINTERFACE); |
| ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsA failed: %08x\n", |
| GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoA(set, devName, &guid, NULL, NULL, |
| DICD_GENERATE_ID, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyA(NULL, NULL, -1, NULL, 0); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyA(set, NULL, -1, NULL, 0); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyA(set, &devInfo, -1, NULL, 0); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_REG_PROPERTY, |
| "Expected ERROR_INVALID_REG_PROPERTY, got %08x\n", GetLastError()); |
| /* GetLastError() returns nonsense in win2k3 */ |
| ret = pSetupDiSetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, 0); |
| todo_wine |
| ok(!ret, "Expected failure, got %d\n", ret); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| (PBYTE)friendlyName, buflen); |
| ok(ret, "SetupDiSetDeviceRegistryPropertyA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(NULL, NULL, -1, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, NULL, -1, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, -1, NULL, NULL, 0, NULL); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_REG_PROPERTY, |
| "Expected ERROR_INVALID_REG_PROPERTY, got %08x\n", GetLastError()); |
| /* GetLastError() returns nonsense in win2k3 */ |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, NULL, buflen, NULL); |
| ok(!ret, "Expected failure, got %d\n", ret); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, NULL, 0, &size); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %08x\n", GetLastError()); |
| ok(buflen == size, "Unexpected size: %d\n", size); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, (PBYTE)buf, buflen, NULL); |
| ok(ret, "SetupDiGetDeviceRegistryPropertyA failed: %08x\n", GetLastError()); |
| ok(!lstrcmpiA(friendlyName, buf), "Unexpected property\n"); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| ®Type, (PBYTE)buf, buflen, NULL); |
| ok(ret, "SetupDiGetDeviceRegistryPropertyA failed: %08x\n", GetLastError()); |
| ok(!lstrcmpiA(friendlyName, buf), "Unexpected value of property\n"); |
| ok(regType == REG_SZ, "Unexpected type of property: %d\n", regType); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, 0); |
| ok(ret, "SetupDiSetDeviceRegistryPropertyA failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyA(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, (PBYTE)buf, buflen, &size); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_DATA, |
| "Expected ERROR_INVALID_DATA, got %08x\n", GetLastError()); |
| pSetupDiDestroyDeviceInfoList(set); |
| |
| res = RegOpenKeyA(HKEY_LOCAL_MACHINE, bogus, &key); |
| if(!is_wow64) |
| todo_wine ok(res == ERROR_FILE_NOT_FOUND, "Expected key to not exist\n"); |
| /* FIXME: Remove when Wine is fixed */ |
| if (res == ERROR_SUCCESS) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| RegDeleteKeyA(HKEY_LOCAL_MACHINE, bogus); |
| } |
| } |
| |
| static void testDeviceRegistryPropertyW(void) |
| { |
| HDEVINFO set; |
| SP_DEVINFO_DATA devInfo = { sizeof(SP_DEVINFO_DATA), { 0 } }; |
| WCHAR devName[] = {'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| WCHAR friendlyName[] = {'B','o','g','u','s',0}; |
| WCHAR buf[6] = {0}; |
| DWORD buflen = 6 * sizeof(WCHAR); |
| DWORD size; |
| DWORD regType; |
| BOOL ret; |
| LONG res; |
| HKEY key; |
| static const WCHAR bogus[] = {'S','y','s','t','e','m','\\', |
| 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\', |
| 'E','n','u','m','\\','R','o','o','t','\\', |
| 'L','E','G','A','C','Y','_','B','O','G','U','S',0}; |
| |
| SetLastError(0xdeadbeef); |
| set = pSetupDiGetClassDevsW(&guid, NULL, 0, DIGCF_DEVICEINTERFACE); |
| ok(set != INVALID_HANDLE_VALUE, "SetupDiGetClassDevsW failed: %08x\n", |
| GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiCreateDeviceInfoW(set, devName, &guid, NULL, NULL, |
| DICD_GENERATE_ID, &devInfo); |
| ok(ret, "SetupDiCreateDeviceInfoW failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyW(NULL, NULL, -1, NULL, 0); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyW(set, NULL, -1, NULL, 0); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyW(set, &devInfo, -1, NULL, 0); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_REG_PROPERTY, |
| "Expected ERROR_INVALID_REG_PROPERTY, got %08x\n", GetLastError()); |
| /* GetLastError() returns nonsense in win2k3 */ |
| ret = pSetupDiSetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, 0); |
| todo_wine |
| ok(!ret, "Expected failure, got %d\n", ret); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| (PBYTE)friendlyName, buflen); |
| ok(ret, "SetupDiSetDeviceRegistryPropertyW failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(NULL, NULL, -1, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_HANDLE, |
| "Expected ERROR_INVALID_HANDLE, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, NULL, -1, NULL, NULL, 0, NULL); |
| ok(!ret && GetLastError() == ERROR_INVALID_PARAMETER, |
| "Expected ERROR_INVALID_PARAMETER, got %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, -1, NULL, NULL, 0, NULL); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_REG_PROPERTY, |
| "Expected ERROR_INVALID_REG_PROPERTY, got %08x\n", GetLastError()); |
| /* GetLastError() returns nonsense in win2k3 */ |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, NULL, buflen, NULL); |
| ok(!ret, "Expected failure, got %d\n", ret); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, NULL, 0, &size); |
| ok(!ret && GetLastError() == ERROR_INSUFFICIENT_BUFFER, |
| "Expected ERROR_INSUFFICIENT_BUFFER, got %08x\n", GetLastError()); |
| ok(buflen == size, "Unexpected size: %d\n", size); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, (PBYTE)buf, buflen, NULL); |
| ok(ret, "SetupDiGetDeviceRegistryPropertyW failed: %08x\n", GetLastError()); |
| ok(!lstrcmpiW(friendlyName, buf), "Unexpected property\n"); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| ®Type, (PBYTE)buf, buflen, NULL); |
| ok(ret, "SetupDiGetDeviceRegistryPropertyW failed: %08x\n", GetLastError()); |
| ok(!lstrcmpiW(friendlyName, buf), "Unexpected value of property\n"); |
| ok(regType == REG_SZ, "Unexpected type of property: %d\n", regType); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiSetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, 0); |
| ok(ret, "SetupDiSetDeviceRegistryPropertyW failed: %08x\n", GetLastError()); |
| SetLastError(0xdeadbeef); |
| ret = pSetupDiGetDeviceRegistryPropertyW(set, &devInfo, SPDRP_FRIENDLYNAME, |
| NULL, (PBYTE)buf, buflen, &size); |
| todo_wine |
| ok(!ret && GetLastError() == ERROR_INVALID_DATA, |
| "Expected ERROR_INVALID_DATA, got %08x\n", GetLastError()); |
| pSetupDiDestroyDeviceInfoList(set); |
| |
| res = RegOpenKeyW(HKEY_LOCAL_MACHINE, bogus, &key); |
| if(!is_wow64) |
| todo_wine ok(res == ERROR_FILE_NOT_FOUND, "Expected key to not exist\n"); |
| /* FIXME: Remove when Wine is fixed */ |
| if (res == ERROR_SUCCESS) |
| { |
| /* Wine doesn't delete the information currently */ |
| trace("We are most likely on Wine\n"); |
| RegDeleteKeyW(HKEY_LOCAL_MACHINE, bogus); |
| } |
| } |
| |
| static void testSetupDiGetINFClassA(void) |
| { |
| static const char inffile[] = "winetest.inf"; |
| static const char content[] = "[Version]\r\n\r\n"; |
| |
| char cn[MAX_PATH]; |
| char filename[MAX_PATH]; |
| DWORD count; |
| BOOL retval; |
| GUID guid; |
| HANDLE h; |
| |
| if(!pSetupDiGetINFClassA) |
| { |
| win_skip("SetupDiGetINFClassA not present\n"); |
| return; |
| } |
| |
| count = GetTempPathA(MAX_PATH, filename); |
| if(!count) |
| { |
| win_skip("GetTempPathA failed\n"); |
| return; |
| } |
| |
| strcat(filename, inffile); |
| DeleteFileA(filename); |
| |
| /* not existing file */ |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| if (ERROR_CALL_NOT_IMPLEMENTED == GetLastError()) |
| { |
| skip("SetupDiGetINFClassA is not implemented\n"); |
| return; |
| } |
| ok(ERROR_FILE_NOT_FOUND == GetLastError(), |
| "expected error ERROR_FILE_NOT_FOUND, got %u\n", GetLastError()); |
| |
| /* missing file wins against other invalid parameter */ |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, NULL, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_FILE_NOT_FOUND == GetLastError(), |
| "expected error ERROR_FILE_NOT_FOUND, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, NULL, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_FILE_NOT_FOUND == GetLastError(), |
| "expected error ERROR_FILE_NOT_FOUND, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, 0, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_FILE_NOT_FOUND == GetLastError(), |
| "expected error ERROR_FILE_NOT_FOUND, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, NULL); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_FILE_NOT_FOUND == GetLastError(), |
| "expected error ERROR_FILE_NOT_FOUND, got %u\n", GetLastError()); |
| |
| /* test file content */ |
| h = CreateFileA(filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, |
| FILE_ATTRIBUTE_NORMAL, NULL); |
| if(h == INVALID_HANDLE_VALUE) |
| { |
| win_skip("failed to create file %s (error %u)\n", filename, GetLastError()); |
| return; |
| } |
| CloseHandle( h); |
| |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| |
| h = CreateFileA(filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, |
| FILE_ATTRIBUTE_NORMAL, NULL); |
| if(h == INVALID_HANDLE_VALUE) |
| { |
| win_skip("failed to create file %s (error %u)\n", filename, GetLastError()); |
| return; |
| } |
| WriteFile( h, content, sizeof(content), &count, NULL); |
| CloseHandle( h); |
| |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| |
| WritePrivateProfileStringA("Version", "Signature", "\"$CHICAGO$\"", filename); |
| |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| |
| WritePrivateProfileStringA("Version", "Class", "WINE", filename); |
| |
| count = 0xdeadbeef; |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(retval, "expected SetupDiGetINFClassA to succeed! error %u\n", GetLastError()); |
| ok(count == 5, "expected count==5, got %u\n", count); |
| |
| count = 0xdeadbeef; |
| retval = SetupDiGetINFClassA(filename, &guid, cn, 5, &count); |
| ok(retval, "expected SetupDiGetINFClassA to succeed! error %u\n", GetLastError()); |
| ok(count == 5, "expected count==5, got %u\n", count); |
| |
| count = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, 4, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_INSUFFICIENT_BUFFER == GetLastError(), |
| "expected error ERROR_INSUFFICIENT_BUFFER, got %u\n", GetLastError()); |
| ok(count == 5, "expected count==5, got %u\n", count); |
| |
| /* invalid parameter */ |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(NULL, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_INVALID_PARAMETER == GetLastError(), |
| "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, NULL, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_INVALID_PARAMETER == GetLastError(), |
| "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, NULL, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_INVALID_PARAMETER == GetLastError(), |
| "expected error ERROR_INVALID_PARAMETER, got %u\n", GetLastError()); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, 0, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(ERROR_INSUFFICIENT_BUFFER == GetLastError() || |
| ERROR_INVALID_PARAMETER == GetLastError(), |
| "expected error ERROR_INSUFFICIENT_BUFFER or ERROR_INVALID_PARAMETER, " |
| "got %u\n", GetLastError()); |
| |
| DeleteFileA(filename); |
| |
| WritePrivateProfileStringA("Version", "Signature", "\"$CHICAGO$\"", filename); |
| WritePrivateProfileStringA("Version", "ClassGUID", "WINE", filename); |
| |
| SetLastError(0xdeadbeef); |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(!retval, "expected SetupDiGetINFClassA to fail!\n"); |
| ok(RPC_S_INVALID_STRING_UUID == GetLastError() || |
| ERROR_INVALID_PARAMETER == GetLastError(), |
| "expected error RPC_S_INVALID_STRING_UUID or ERROR_INVALID_PARAMETER, " |
| "got %u\n", GetLastError()); |
| |
| /* network adapter guid */ |
| WritePrivateProfileStringA("Version", "ClassGUID", |
| "{4d36e972-e325-11ce-bfc1-08002be10318}", filename); |
| |
| /* this test succeeds only if the guid is known to the system */ |
| count = 0xdeadbeef; |
| retval = SetupDiGetINFClassA(filename, &guid, cn, MAX_PATH, &count); |
| ok(retval, "expected SetupDiGetINFClassA to succeed! error %u\n", GetLastError()); |
| todo_wine |
| ok(count == 4, "expected count==4, got %u(%s)\n", count, cn); |
| |
| DeleteFileA(filename); |
| } |
| |
| START_TEST(devinst) |
| { |
| init_function_pointers(); |
| |
| if (pIsWow64Process) |
| pIsWow64Process(GetCurrentProcess(), &is_wow64); |
| |
| if (pSetupDiCreateDeviceInfoListExW) |
| test_SetupDiCreateDeviceInfoListEx(); |
| else |
| win_skip("SetupDiCreateDeviceInfoListExW is not available\n"); |
| |
| if (pSetupDiOpenClassRegKeyExA) |
| test_SetupDiOpenClassRegKeyExA(); |
| else |
| win_skip("SetupDiOpenClassRegKeyExA is not available\n"); |
| |
| testInstallClass(); |
| testCreateDeviceInfo(); |
| testGetDeviceInstanceId(); |
| testRegisterDeviceInfo(); |
| testCreateDeviceInterface(); |
| testGetDeviceInterfaceDetail(); |
| testDevRegKey(); |
| testRegisterAndGetDetail(); |
| testDeviceRegistryPropertyA(); |
| testDeviceRegistryPropertyW(); |
| testSetupDiGetINFClassA(); |
| } |