/* * setupapi query functions * * Copyright 2006 James Hawkins * * 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 "windef.h" #include "winbase.h" #include "winuser.h" #include "winreg.h" #include "setupapi.h" #include "advpub.h" #include "winnls.h" #include "wine/debug.h" #include "wine/unicode.h" #include "setupapi_private.h" WINE_DEFAULT_DEBUG_CHANNEL(setupapi); #ifdef __i386__ static const WCHAR source_disks_names_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','x','8','6',0}; static const WCHAR source_disks_files_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','x','8','6',0}; #elif defined(__x86_64) static const WCHAR source_disks_names_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','a','m','d','6','4',0}; static const WCHAR source_disks_files_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','a','m','d','6','4',0}; #else /* FIXME: other platforms */ static const WCHAR source_disks_names_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s',0}; static const WCHAR source_disks_files_platform[] = {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s',0}; #endif static const WCHAR source_disks_names[] = {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s',0}; static const WCHAR source_disks_files[] = {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s',0}; /* fills the PSP_INF_INFORMATION struct fill_info is TRUE * always returns the required size of the information */ static BOOL fill_inf_info(HINF inf, PSP_INF_INFORMATION buffer, DWORD size, DWORD *required) { LPCWSTR filename = PARSER_get_inf_filename(inf); DWORD total_size = FIELD_OFFSET(SP_INF_INFORMATION, VersionData) + (lstrlenW(filename) + 1) * sizeof(WCHAR); if (required) *required = total_size; /* FIXME: we need to parse the INF file to find the correct version info */ if (buffer) { if (size < total_size) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } buffer->InfStyle = INF_STYLE_WIN4; buffer->InfCount = 1; /* put the filename in buffer->VersionData */ lstrcpyW((LPWSTR)&buffer->VersionData[0], filename); } return TRUE; } static HINF search_for_inf(LPCVOID InfSpec, DWORD SearchControl) { HINF hInf = INVALID_HANDLE_VALUE; WCHAR inf_path[MAX_PATH]; static const WCHAR infW[] = {'\\','i','n','f','\\',0}; static const WCHAR system32W[] = {'\\','s','y','s','t','e','m','3','2','\\',0}; if (SearchControl == INFINFO_REVERSE_DEFAULT_SEARCH) { GetWindowsDirectoryW(inf_path, MAX_PATH); lstrcatW(inf_path, system32W); lstrcatW(inf_path, InfSpec); hInf = SetupOpenInfFileW(inf_path, NULL, INF_STYLE_OLDNT | INF_STYLE_WIN4, NULL); if (hInf != INVALID_HANDLE_VALUE) return hInf; GetWindowsDirectoryW(inf_path, MAX_PATH); lstrcpyW(inf_path, infW); lstrcatW(inf_path, InfSpec); return SetupOpenInfFileW(inf_path, NULL, INF_STYLE_OLDNT | INF_STYLE_WIN4, NULL); } return INVALID_HANDLE_VALUE; } /*********************************************************************** * SetupGetInfInformationA (SETUPAPI.@) * */ BOOL WINAPI SetupGetInfInformationA(LPCVOID InfSpec, DWORD SearchControl, PSP_INF_INFORMATION ReturnBuffer, DWORD ReturnBufferSize, PDWORD RequiredSize) { LPWSTR inf = (LPWSTR)InfSpec; DWORD len; BOOL ret; if (InfSpec && SearchControl >= INFINFO_INF_NAME_IS_ABSOLUTE) { len = lstrlenA(InfSpec) + 1; inf = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, InfSpec, -1, inf, len); } ret = SetupGetInfInformationW(inf, SearchControl, ReturnBuffer, ReturnBufferSize, RequiredSize); if (SearchControl >= INFINFO_INF_NAME_IS_ABSOLUTE) HeapFree(GetProcessHeap(), 0, inf); return ret; } /*********************************************************************** * SetupGetInfInformationW (SETUPAPI.@) * * BUGS * Only handles the case when InfSpec is an INF handle. */ BOOL WINAPI SetupGetInfInformationW(LPCVOID InfSpec, DWORD SearchControl, PSP_INF_INFORMATION ReturnBuffer, DWORD ReturnBufferSize, PDWORD RequiredSize) { HINF inf; BOOL ret; DWORD infSize; TRACE("(%p, %d, %p, %d, %p)\n", InfSpec, SearchControl, ReturnBuffer, ReturnBufferSize, RequiredSize); if (!InfSpec) { if (SearchControl == INFINFO_INF_SPEC_IS_HINF) SetLastError(ERROR_INVALID_HANDLE); else SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } switch (SearchControl) { case INFINFO_INF_SPEC_IS_HINF: inf = (HINF)InfSpec; break; case INFINFO_INF_NAME_IS_ABSOLUTE: case INFINFO_DEFAULT_SEARCH: inf = SetupOpenInfFileW(InfSpec, NULL, INF_STYLE_OLDNT | INF_STYLE_WIN4, NULL); break; case INFINFO_REVERSE_DEFAULT_SEARCH: inf = search_for_inf(InfSpec, SearchControl); break; case INFINFO_INF_PATH_LIST_SEARCH: FIXME("Unhandled search control: %d\n", SearchControl); if (RequiredSize) *RequiredSize = 0; return FALSE; default: SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (inf == INVALID_HANDLE_VALUE) { SetLastError(ERROR_FILE_NOT_FOUND); return FALSE; } ret = fill_inf_info(inf, ReturnBuffer, ReturnBufferSize, &infSize); if (!ReturnBuffer && (ReturnBufferSize >= infSize)) { SetLastError(ERROR_INVALID_PARAMETER); ret = FALSE; } if (RequiredSize) *RequiredSize = infSize; if (SearchControl >= INFINFO_INF_NAME_IS_ABSOLUTE) SetupCloseInfFile(inf); return ret; } /*********************************************************************** * SetupQueryInfFileInformationA (SETUPAPI.@) */ BOOL WINAPI SetupQueryInfFileInformationA(PSP_INF_INFORMATION InfInformation, UINT InfIndex, PSTR ReturnBuffer, DWORD ReturnBufferSize, PDWORD RequiredSize) { LPWSTR filenameW; DWORD size; BOOL ret; ret = SetupQueryInfFileInformationW(InfInformation, InfIndex, NULL, 0, &size); if (!ret) return FALSE; filenameW = HeapAlloc(GetProcessHeap(), 0, size * sizeof(WCHAR)); ret = SetupQueryInfFileInformationW(InfInformation, InfIndex, filenameW, size, &size); if (!ret) { HeapFree(GetProcessHeap(), 0, filenameW); return FALSE; } if (RequiredSize) *RequiredSize = size; if (!ReturnBuffer) { HeapFree(GetProcessHeap(), 0, filenameW); if (ReturnBufferSize) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } return TRUE; } if (size > ReturnBufferSize) { HeapFree(GetProcessHeap(), 0, filenameW); SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } WideCharToMultiByte(CP_ACP, 0, filenameW, -1, ReturnBuffer, size, NULL, NULL); HeapFree(GetProcessHeap(), 0, filenameW); return ret; } /*********************************************************************** * SetupQueryInfFileInformationW (SETUPAPI.@) */ BOOL WINAPI SetupQueryInfFileInformationW(PSP_INF_INFORMATION InfInformation, UINT InfIndex, PWSTR ReturnBuffer, DWORD ReturnBufferSize, PDWORD RequiredSize) { DWORD len; LPWSTR ptr; TRACE("(%p, %u, %p, %d, %p) Stub!\n", InfInformation, InfIndex, ReturnBuffer, ReturnBufferSize, RequiredSize); if (!InfInformation) { SetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (InfIndex != 0) FIXME("Appended INF files are not handled\n"); ptr = (LPWSTR)&InfInformation->VersionData[0]; len = lstrlenW(ptr); if (RequiredSize) *RequiredSize = len + 1; if (!ReturnBuffer) return TRUE; if (ReturnBufferSize < len) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } lstrcpyW(ReturnBuffer, ptr); return TRUE; } /*********************************************************************** * SetupGetSourceFileLocationA (SETUPAPI.@) */ BOOL WINAPI SetupGetSourceFileLocationA( HINF hinf, PINFCONTEXT context, PCSTR filename, PUINT source_id, PSTR buffer, DWORD buffer_size, PDWORD required_size ) { BOOL ret = FALSE; WCHAR *filenameW = NULL, *bufferW = NULL; DWORD required; INT size; TRACE("%p, %p, %s, %p, %p, 0x%08x, %p\n", hinf, context, debugstr_a(filename), source_id, buffer, buffer_size, required_size); if (filename && *filename && !(filenameW = strdupAtoW( filename ))) return FALSE; if (!SetupGetSourceFileLocationW( hinf, context, filenameW, source_id, NULL, 0, &required )) goto done; if (!(bufferW = HeapAlloc( GetProcessHeap(), 0, required * sizeof(WCHAR) ))) goto done; if (!SetupGetSourceFileLocationW( hinf, context, filenameW, source_id, bufferW, required, NULL )) goto done; size = WideCharToMultiByte( CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL ); if (required_size) *required_size = size; if (buffer) { if (buffer_size >= size) WideCharToMultiByte( CP_ACP, 0, bufferW, -1, buffer, buffer_size, NULL, NULL ); else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); goto done; } } ret = TRUE; done: HeapFree( GetProcessHeap(), 0, filenameW ); HeapFree( GetProcessHeap(), 0, bufferW ); return ret; } static LPWSTR get_source_id( HINF hinf, PINFCONTEXT context, PCWSTR filename ) { DWORD size; LPWSTR source_id; if (!SetupFindFirstLineW( hinf, source_disks_files_platform, filename, context ) && !SetupFindFirstLineW( hinf, source_disks_files, filename, context )) return NULL; if (!SetupGetStringFieldW( context, 1, NULL, 0, &size )) return NULL; if (!(source_id = HeapAlloc( GetProcessHeap(), 0, size * sizeof(WCHAR) ))) return NULL; if (!SetupGetStringFieldW( context, 1, source_id, size, NULL )) { HeapFree( GetProcessHeap(), 0, source_id ); return NULL; } if (!SetupFindFirstLineW( hinf, source_disks_names_platform, source_id, context ) && !SetupFindFirstLineW( hinf, source_disks_names, source_id, context )) { HeapFree( GetProcessHeap(), 0, source_id ); return NULL; } return source_id; } /*********************************************************************** * SetupGetSourceFileLocationW (SETUPAPI.@) */ BOOL WINAPI SetupGetSourceFileLocationW( HINF hinf, PINFCONTEXT context, PCWSTR filename, PUINT source_id, PWSTR buffer, DWORD buffer_size, PDWORD required_size ) { INFCONTEXT ctx; WCHAR *end, *source_id_str; TRACE("%p, %p, %s, %p, %p, 0x%08x, %p\n", hinf, context, debugstr_w(filename), source_id, buffer, buffer_size, required_size); if (!context) context = &ctx; if (!(source_id_str = get_source_id( hinf, context, filename ))) return FALSE; *source_id = strtolW( source_id_str, &end, 10 ); if (end == source_id_str || *end) { HeapFree( GetProcessHeap(), 0, source_id_str ); return FALSE; } HeapFree( GetProcessHeap(), 0, source_id_str ); if (SetupGetStringFieldW( context, 4, buffer, buffer_size, required_size )) return TRUE; if (required_size) *required_size = 1; if (buffer) { if (buffer_size >= 1) buffer[0] = 0; else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); return FALSE; } } return TRUE; } /*********************************************************************** * SetupGetSourceInfoA (SETUPAPI.@) */ BOOL WINAPI SetupGetSourceInfoA( HINF hinf, UINT source_id, UINT info, PSTR buffer, DWORD buffer_size, LPDWORD required_size ) { BOOL ret = FALSE; WCHAR *bufferW = NULL; DWORD required; INT size; TRACE("%p, %d, %d, %p, %d, %p\n", hinf, source_id, info, buffer, buffer_size, required_size); if (!SetupGetSourceInfoW( hinf, source_id, info, NULL, 0, &required )) return FALSE; if (!(bufferW = HeapAlloc( GetProcessHeap(), 0, required * sizeof(WCHAR) ))) return FALSE; if (!SetupGetSourceInfoW( hinf, source_id, info, bufferW, required, NULL )) goto done; size = WideCharToMultiByte( CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL ); if (required_size) *required_size = size; if (buffer) { if (buffer_size >= size) WideCharToMultiByte( CP_ACP, 0, bufferW, -1, buffer, buffer_size, NULL, NULL ); else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); goto done; } } ret = TRUE; done: HeapFree( GetProcessHeap(), 0, bufferW ); return ret; } /*********************************************************************** * SetupGetSourceInfoW (SETUPAPI.@) */ BOOL WINAPI SetupGetSourceInfoW( HINF hinf, UINT source_id, UINT info, PWSTR buffer, DWORD buffer_size, LPDWORD required_size ) { INFCONTEXT ctx; WCHAR source_id_str[11]; static const WCHAR fmt[] = {'%','d',0}; DWORD index; TRACE("%p, %d, %d, %p, %d, %p\n", hinf, source_id, info, buffer, buffer_size, required_size); sprintfW( source_id_str, fmt, source_id ); if (!SetupFindFirstLineW( hinf, source_disks_names_platform, source_id_str, &ctx ) && !SetupFindFirstLineW( hinf, source_disks_names, source_id_str, &ctx )) return FALSE; switch (info) { case SRCINFO_PATH: index = 4; break; case SRCINFO_TAGFILE: index = 2; break; case SRCINFO_DESCRIPTION: index = 1; break; default: WARN("unknown info level: %d\n", info); return FALSE; } if (SetupGetStringFieldW( &ctx, index, buffer, buffer_size, required_size )) return TRUE; if (required_size) *required_size = 1; if (buffer) { if (buffer_size >= 1) buffer[0] = 0; else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); return FALSE; } } return TRUE; } /*********************************************************************** * SetupGetTargetPathA (SETUPAPI.@) */ BOOL WINAPI SetupGetTargetPathA( HINF hinf, PINFCONTEXT context, PCSTR section, PSTR buffer, DWORD buffer_size, PDWORD required_size ) { BOOL ret = FALSE; WCHAR *sectionW = NULL, *bufferW = NULL; DWORD required; INT size; TRACE("%p, %p, %s, %p, 0x%08x, %p\n", hinf, context, debugstr_a(section), buffer, buffer_size, required_size); if (section && !(sectionW = strdupAtoW( section ))) return FALSE; if (!SetupGetTargetPathW( hinf, context, sectionW, NULL, 0, &required )) goto done; if (!(bufferW = HeapAlloc( GetProcessHeap(), 0, required * sizeof(WCHAR) ))) goto done; if (!SetupGetTargetPathW( hinf, context, sectionW, bufferW, required, NULL )) goto done; size = WideCharToMultiByte( CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL ); if (required_size) *required_size = size; if (buffer) { if (buffer_size >= size) WideCharToMultiByte( CP_ACP, 0, bufferW, -1, buffer, buffer_size, NULL, NULL ); else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); goto done; } } ret = TRUE; done: HeapFree( GetProcessHeap(), 0, sectionW ); HeapFree( GetProcessHeap(), 0, bufferW ); return ret; } /*********************************************************************** * SetupGetTargetPathW (SETUPAPI.@) */ BOOL WINAPI SetupGetTargetPathW( HINF hinf, PINFCONTEXT context, PCWSTR section, PWSTR buffer, DWORD buffer_size, PDWORD required_size ) { static const WCHAR destination_dirs[] = {'D','e','s','t','i','n','a','t','i','o','n','D','i','r','s',0}; static const WCHAR default_dest_dir[] = {'D','e','f','a','u','l','t','D','e','s','t','D','i','r',0}; INFCONTEXT ctx; WCHAR *dir; unsigned int size; TRACE("%p, %p, %s, %p, 0x%08x, %p\n", hinf, context, debugstr_w(section), buffer, buffer_size, required_size); if (context && !SetupFindFirstLineW( hinf, destination_dirs, NULL, context )) return FALSE; else if (section && !SetupFindFirstLineW( hinf, section, NULL, &ctx )) return FALSE; else if (!SetupFindFirstLineW( hinf, destination_dirs, default_dest_dir, &ctx )) return FALSE; if (!(dir = PARSER_get_dest_dir( context ? context : &ctx ))) return FALSE; size = strlenW( dir ) + 1; if (required_size) *required_size = size; if (buffer) { if (buffer_size >= size) lstrcpyW( buffer, dir ); else { SetLastError( ERROR_INSUFFICIENT_BUFFER ); HeapFree( GetProcessHeap(), 0, dir ); return FALSE; } } HeapFree( GetProcessHeap(), 0, dir ); return TRUE; } /*********************************************************************** * SetupQueryInfOriginalFileInformationA (SETUPAPI.@) */ BOOL WINAPI SetupQueryInfOriginalFileInformationA( PSP_INF_INFORMATION InfInformation, UINT InfIndex, PSP_ALTPLATFORM_INFO AlternativePlatformInfo, PSP_ORIGINAL_FILE_INFO_A OriginalFileInfo) { BOOL ret; SP_ORIGINAL_FILE_INFO_W OriginalFileInfoW; TRACE("(%p, %d, %p, %p)\n", InfInformation, InfIndex, AlternativePlatformInfo, OriginalFileInfo); if (OriginalFileInfo->cbSize != sizeof(*OriginalFileInfo)) { WARN("incorrect OriginalFileInfo->cbSize of %d\n", OriginalFileInfo->cbSize); SetLastError( ERROR_INVALID_USER_BUFFER ); return FALSE; } OriginalFileInfoW.cbSize = sizeof(OriginalFileInfoW); ret = SetupQueryInfOriginalFileInformationW(InfInformation, InfIndex, AlternativePlatformInfo, &OriginalFileInfoW); if (ret) { WideCharToMultiByte(CP_ACP, 0, OriginalFileInfoW.OriginalInfName, -1, OriginalFileInfo->OriginalInfName, MAX_PATH, NULL, NULL); WideCharToMultiByte(CP_ACP, 0, OriginalFileInfoW.OriginalCatalogName, -1, OriginalFileInfo->OriginalCatalogName, MAX_PATH, NULL, NULL); } return ret; } /*********************************************************************** * SetupQueryInfOriginalFileInformationW (SETUPAPI.@) */ BOOL WINAPI SetupQueryInfOriginalFileInformationW( PSP_INF_INFORMATION InfInformation, UINT InfIndex, PSP_ALTPLATFORM_INFO AlternativePlatformInfo, PSP_ORIGINAL_FILE_INFO_W OriginalFileInfo) { LPCWSTR inf_name; LPCWSTR inf_path; HINF hinf; static const WCHAR wszVersion[] = { 'V','e','r','s','i','o','n',0 }; static const WCHAR wszCatalogFile[] = { 'C','a','t','a','l','o','g','F','i','l','e',0 }; FIXME("(%p, %d, %p, %p): semi-stub\n", InfInformation, InfIndex, AlternativePlatformInfo, OriginalFileInfo); if (OriginalFileInfo->cbSize != sizeof(*OriginalFileInfo)) { WARN("incorrect OriginalFileInfo->cbSize of %d\n", OriginalFileInfo->cbSize); SetLastError(ERROR_INVALID_USER_BUFFER); return FALSE; } inf_path = (LPWSTR)&InfInformation->VersionData[0]; /* FIXME: we should get OriginalCatalogName from CatalogFile line in * the original inf file and cache it, but that would require building a * .pnf file. */ hinf = SetupOpenInfFileW(inf_path, NULL, INF_STYLE_WIN4, NULL); if (hinf == INVALID_HANDLE_VALUE) return FALSE; if (!SetupGetLineTextW(NULL, hinf, wszVersion, wszCatalogFile, OriginalFileInfo->OriginalCatalogName, sizeof(OriginalFileInfo->OriginalCatalogName)/sizeof(OriginalFileInfo->OriginalCatalogName[0]), NULL)) { OriginalFileInfo->OriginalCatalogName[0] = '\0'; } SetupCloseInfFile(hinf); /* FIXME: not quite correct as we just return the same file name as * destination (copied) inf file, not the source (original) inf file. * to fix it properly would require building a .pnf file */ /* file name is stored in VersionData field of InfInformation */ inf_name = strrchrW(inf_path, '\\'); if (inf_name) inf_name++; else inf_name = inf_path; strcpyW(OriginalFileInfo->OriginalInfName, inf_name); return TRUE; }