2 * setupapi query functions
4 * Copyright 2006 James Hawkins
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
31 #include "wine/debug.h"
32 #include "setupapi_private.h"
34 WINE_DEFAULT_DEBUG_CHANNEL(setupapi
);
37 static const WCHAR source_disks_names_platform
[] =
38 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','x','8','6',0};
39 static const WCHAR source_disks_files_platform
[] =
40 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','x','8','6',0};
41 #elif defined(__x86_64__)
42 static const WCHAR source_disks_names_platform
[] =
43 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','a','m','d','6','4',0};
44 static const WCHAR source_disks_files_platform
[] =
45 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','a','m','d','6','4',0};
46 #elif defined(__arm__)
47 static const WCHAR source_disks_names_platform
[] =
48 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','a','r','m',0};
49 static const WCHAR source_disks_files_platform
[] =
50 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','a','r','m',0};
51 #elif defined(__aarch64__)
52 static const WCHAR source_disks_names_platform
[] =
53 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s','.','a','r','m','6','4',0};
54 static const WCHAR source_disks_files_platform
[] =
55 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s','.','a','r','m','6','4',0};
56 #else /* FIXME: other platforms */
57 static const WCHAR source_disks_names_platform
[] =
58 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s',0};
59 static const WCHAR source_disks_files_platform
[] =
60 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s',0};
62 static const WCHAR source_disks_names
[] =
63 {'S','o','u','r','c','e','D','i','s','k','s','N','a','m','e','s',0};
64 static const WCHAR source_disks_files
[] =
65 {'S','o','u','r','c','e','D','i','s','k','s','F','i','l','e','s',0};
67 /* fills the PSP_INF_INFORMATION struct fill_info is TRUE
68 * always returns the required size of the information
70 static BOOL
fill_inf_info(HINF inf
, PSP_INF_INFORMATION buffer
, DWORD size
, DWORD
*required
)
72 LPCWSTR filename
= PARSER_get_inf_filename(inf
);
73 DWORD total_size
= FIELD_OFFSET(SP_INF_INFORMATION
, VersionData
)
74 + (lstrlenW(filename
) + 1) * sizeof(WCHAR
);
76 if (required
) *required
= total_size
;
78 /* FIXME: we need to parse the INF file to find the correct version info */
81 if (size
< total_size
)
83 SetLastError(ERROR_INSUFFICIENT_BUFFER
);
86 buffer
->InfStyle
= INF_STYLE_WIN4
;
88 /* put the filename in buffer->VersionData */
89 lstrcpyW((LPWSTR
)&buffer
->VersionData
[0], filename
);
94 static HINF
search_for_inf(LPCVOID InfSpec
, DWORD SearchControl
)
96 HINF hInf
= INVALID_HANDLE_VALUE
;
97 WCHAR inf_path
[MAX_PATH
];
99 static const WCHAR infW
[] = {'\\','i','n','f','\\',0};
100 static const WCHAR system32W
[] = {'\\','s','y','s','t','e','m','3','2','\\',0};
102 if (SearchControl
== INFINFO_REVERSE_DEFAULT_SEARCH
)
104 GetWindowsDirectoryW(inf_path
, MAX_PATH
);
105 lstrcatW(inf_path
, system32W
);
106 lstrcatW(inf_path
, InfSpec
);
108 hInf
= SetupOpenInfFileW(inf_path
, NULL
,
109 INF_STYLE_OLDNT
| INF_STYLE_WIN4
, NULL
);
110 if (hInf
!= INVALID_HANDLE_VALUE
)
113 GetWindowsDirectoryW(inf_path
, MAX_PATH
);
114 lstrcpyW(inf_path
, infW
);
115 lstrcatW(inf_path
, InfSpec
);
117 return SetupOpenInfFileW(inf_path
, NULL
,
118 INF_STYLE_OLDNT
| INF_STYLE_WIN4
, NULL
);
121 return INVALID_HANDLE_VALUE
;
124 /***********************************************************************
125 * SetupGetInfInformationA (SETUPAPI.@)
128 BOOL WINAPI
SetupGetInfInformationA(LPCVOID InfSpec
, DWORD SearchControl
,
129 PSP_INF_INFORMATION ReturnBuffer
,
130 DWORD ReturnBufferSize
, PDWORD RequiredSize
)
132 LPWSTR inf
= (LPWSTR
)InfSpec
;
136 if (InfSpec
&& SearchControl
>= INFINFO_INF_NAME_IS_ABSOLUTE
)
138 len
= MultiByteToWideChar(CP_ACP
, 0, InfSpec
, -1, NULL
, 0);
139 inf
= HeapAlloc(GetProcessHeap(), 0, len
* sizeof(WCHAR
));
142 SetLastError(ERROR_NOT_ENOUGH_MEMORY
);
145 MultiByteToWideChar(CP_ACP
, 0, InfSpec
, -1, inf
, len
);
148 ret
= SetupGetInfInformationW(inf
, SearchControl
, ReturnBuffer
,
149 ReturnBufferSize
, RequiredSize
);
151 if (SearchControl
>= INFINFO_INF_NAME_IS_ABSOLUTE
)
152 HeapFree(GetProcessHeap(), 0, inf
);
157 /***********************************************************************
158 * SetupGetInfInformationW (SETUPAPI.@)
161 * Only handles the case when InfSpec is an INF handle.
163 BOOL WINAPI
SetupGetInfInformationW(LPCVOID InfSpec
, DWORD SearchControl
,
164 PSP_INF_INFORMATION ReturnBuffer
,
165 DWORD ReturnBufferSize
, PDWORD RequiredSize
)
171 TRACE("(%p, %d, %p, %d, %p)\n", InfSpec
, SearchControl
, ReturnBuffer
,
172 ReturnBufferSize
, RequiredSize
);
176 if (SearchControl
== INFINFO_INF_SPEC_IS_HINF
)
177 SetLastError(ERROR_INVALID_HANDLE
);
179 SetLastError(ERROR_INVALID_PARAMETER
);
184 switch (SearchControl
)
186 case INFINFO_INF_SPEC_IS_HINF
:
189 case INFINFO_INF_NAME_IS_ABSOLUTE
:
190 case INFINFO_DEFAULT_SEARCH
:
191 inf
= SetupOpenInfFileW(InfSpec
, NULL
,
192 INF_STYLE_OLDNT
| INF_STYLE_WIN4
, NULL
);
194 case INFINFO_REVERSE_DEFAULT_SEARCH
:
195 inf
= search_for_inf(InfSpec
, SearchControl
);
197 case INFINFO_INF_PATH_LIST_SEARCH
:
198 FIXME("Unhandled search control: %d\n", SearchControl
);
205 SetLastError(ERROR_INVALID_PARAMETER
);
209 if (inf
== INVALID_HANDLE_VALUE
)
211 SetLastError(ERROR_FILE_NOT_FOUND
);
215 ret
= fill_inf_info(inf
, ReturnBuffer
, ReturnBufferSize
, &infSize
);
216 if (!ReturnBuffer
&& (ReturnBufferSize
>= infSize
))
218 SetLastError(ERROR_INVALID_PARAMETER
);
221 if (RequiredSize
) *RequiredSize
= infSize
;
223 if (SearchControl
>= INFINFO_INF_NAME_IS_ABSOLUTE
)
224 SetupCloseInfFile(inf
);
229 /***********************************************************************
230 * SetupQueryInfFileInformationA (SETUPAPI.@)
232 BOOL WINAPI
SetupQueryInfFileInformationA(PSP_INF_INFORMATION InfInformation
,
233 UINT InfIndex
, PSTR ReturnBuffer
,
234 DWORD ReturnBufferSize
, PDWORD RequiredSize
)
240 ret
= SetupQueryInfFileInformationW(InfInformation
, InfIndex
, NULL
, 0, &size
);
244 filenameW
= HeapAlloc(GetProcessHeap(), 0, size
* sizeof(WCHAR
));
246 ret
= SetupQueryInfFileInformationW(InfInformation
, InfIndex
,
247 filenameW
, size
, &size
);
250 HeapFree(GetProcessHeap(), 0, filenameW
);
255 *RequiredSize
= size
;
259 HeapFree(GetProcessHeap(), 0, filenameW
);
260 if (ReturnBufferSize
)
262 SetLastError(ERROR_INVALID_PARAMETER
);
269 if (size
> ReturnBufferSize
)
271 HeapFree(GetProcessHeap(), 0, filenameW
);
272 SetLastError(ERROR_INSUFFICIENT_BUFFER
);
276 WideCharToMultiByte(CP_ACP
, 0, filenameW
, -1, ReturnBuffer
, size
, NULL
, NULL
);
277 HeapFree(GetProcessHeap(), 0, filenameW
);
282 /***********************************************************************
283 * SetupQueryInfFileInformationW (SETUPAPI.@)
285 BOOL WINAPI
SetupQueryInfFileInformationW(PSP_INF_INFORMATION InfInformation
,
286 UINT InfIndex
, PWSTR ReturnBuffer
,
287 DWORD ReturnBufferSize
, PDWORD RequiredSize
)
292 TRACE("(%p, %u, %p, %d, %p) Stub!\n", InfInformation
, InfIndex
,
293 ReturnBuffer
, ReturnBufferSize
, RequiredSize
);
297 SetLastError(ERROR_INVALID_PARAMETER
);
302 FIXME("Appended INF files are not handled\n");
304 ptr
= (LPWSTR
)InfInformation
->VersionData
;
308 *RequiredSize
= len
+ 1;
313 if (ReturnBufferSize
< len
)
315 SetLastError(ERROR_INSUFFICIENT_BUFFER
);
319 lstrcpyW(ReturnBuffer
, ptr
);
323 /***********************************************************************
324 * SetupGetSourceFileLocationA (SETUPAPI.@)
327 BOOL WINAPI
SetupGetSourceFileLocationA( HINF hinf
, PINFCONTEXT context
, PCSTR filename
,
328 PUINT source_id
, PSTR buffer
, DWORD buffer_size
,
329 PDWORD required_size
)
332 WCHAR
*filenameW
= NULL
, *bufferW
= NULL
;
336 TRACE("%p, %p, %s, %p, %p, 0x%08x, %p\n", hinf
, context
, debugstr_a(filename
), source_id
,
337 buffer
, buffer_size
, required_size
);
339 if (filename
&& *filename
&& !(filenameW
= strdupAtoW( filename
)))
342 if (!SetupGetSourceFileLocationW( hinf
, context
, filenameW
, source_id
, NULL
, 0, &required
))
345 if (!(bufferW
= HeapAlloc( GetProcessHeap(), 0, required
* sizeof(WCHAR
) )))
348 if (!SetupGetSourceFileLocationW( hinf
, context
, filenameW
, source_id
, bufferW
, required
, NULL
))
351 size
= WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, NULL
, 0, NULL
, NULL
);
352 if (required_size
) *required_size
= size
;
356 if (buffer_size
>= size
)
357 WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, buffer
, buffer_size
, NULL
, NULL
);
360 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
367 HeapFree( GetProcessHeap(), 0, filenameW
);
368 HeapFree( GetProcessHeap(), 0, bufferW
);
372 static LPWSTR
get_source_id( HINF hinf
, PINFCONTEXT context
, PCWSTR filename
)
377 if (!SetupFindFirstLineW( hinf
, source_disks_files_platform
, filename
, context
) &&
378 !SetupFindFirstLineW( hinf
, source_disks_files
, filename
, context
))
381 if (!SetupGetStringFieldW( context
, 1, NULL
, 0, &size
))
384 if (!(source_id
= HeapAlloc( GetProcessHeap(), 0, size
* sizeof(WCHAR
) )))
387 if (!SetupGetStringFieldW( context
, 1, source_id
, size
, NULL
))
389 HeapFree( GetProcessHeap(), 0, source_id
);
393 if (!SetupFindFirstLineW( hinf
, source_disks_names_platform
, source_id
, context
) &&
394 !SetupFindFirstLineW( hinf
, source_disks_names
, source_id
, context
))
396 HeapFree( GetProcessHeap(), 0, source_id
);
402 /***********************************************************************
403 * SetupGetSourceFileLocationW (SETUPAPI.@)
406 BOOL WINAPI
SetupGetSourceFileLocationW( HINF hinf
, PINFCONTEXT context
, PCWSTR filename
,
407 PUINT source_id
, PWSTR buffer
, DWORD buffer_size
,
408 PDWORD required_size
)
411 WCHAR
*end
, *source_id_str
;
413 TRACE("%p, %p, %s, %p, %p, 0x%08x, %p\n", hinf
, context
, debugstr_w(filename
), source_id
,
414 buffer
, buffer_size
, required_size
);
416 if (!context
) context
= &ctx
;
418 if (!(source_id_str
= get_source_id( hinf
, context
, filename
)))
421 *source_id
= wcstol( source_id_str
, &end
, 10 );
422 if (end
== source_id_str
|| *end
)
424 HeapFree( GetProcessHeap(), 0, source_id_str
);
427 HeapFree( GetProcessHeap(), 0, source_id_str
);
429 if (SetupGetStringFieldW( context
, 4, buffer
, buffer_size
, required_size
))
432 if (required_size
) *required_size
= 1;
435 if (buffer_size
>= 1) buffer
[0] = 0;
438 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
445 /***********************************************************************
446 * SetupGetSourceInfoA (SETUPAPI.@)
449 BOOL WINAPI
SetupGetSourceInfoA( HINF hinf
, UINT source_id
, UINT info
,
450 PSTR buffer
, DWORD buffer_size
, LPDWORD required_size
)
453 WCHAR
*bufferW
= NULL
;
457 TRACE("%p, %d, %d, %p, %d, %p\n", hinf
, source_id
, info
, buffer
, buffer_size
,
460 if (!SetupGetSourceInfoW( hinf
, source_id
, info
, NULL
, 0, &required
))
463 if (!(bufferW
= HeapAlloc( GetProcessHeap(), 0, required
* sizeof(WCHAR
) )))
466 if (!SetupGetSourceInfoW( hinf
, source_id
, info
, bufferW
, required
, NULL
))
469 size
= WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, NULL
, 0, NULL
, NULL
);
470 if (required_size
) *required_size
= size
;
474 if (buffer_size
>= size
)
475 WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, buffer
, buffer_size
, NULL
, NULL
);
478 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
485 HeapFree( GetProcessHeap(), 0, bufferW
);
489 /***********************************************************************
490 * SetupGetSourceInfoW (SETUPAPI.@)
493 BOOL WINAPI
SetupGetSourceInfoW( HINF hinf
, UINT source_id
, UINT info
,
494 PWSTR buffer
, DWORD buffer_size
, LPDWORD required_size
)
497 WCHAR source_id_str
[11];
498 static const WCHAR fmt
[] = {'%','d',0};
501 TRACE("%p, %d, %d, %p, %d, %p\n", hinf
, source_id
, info
, buffer
, buffer_size
,
504 swprintf( source_id_str
, ARRAY_SIZE(source_id_str
), fmt
, source_id
);
506 if (!SetupFindFirstLineW( hinf
, source_disks_names_platform
, source_id_str
, &ctx
) &&
507 !SetupFindFirstLineW( hinf
, source_disks_names
, source_id_str
, &ctx
))
512 case SRCINFO_PATH
: index
= 4; break;
513 case SRCINFO_TAGFILE
: index
= 2; break;
514 case SRCINFO_DESCRIPTION
: index
= 1; break;
516 WARN("unknown info level: %d\n", info
);
520 if (SetupGetStringFieldW( &ctx
, index
, buffer
, buffer_size
, required_size
))
523 if (required_size
) *required_size
= 1;
526 if (buffer_size
>= 1) buffer
[0] = 0;
529 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
536 /***********************************************************************
537 * SetupGetTargetPathA (SETUPAPI.@)
540 BOOL WINAPI
SetupGetTargetPathA( HINF hinf
, PINFCONTEXT context
, PCSTR section
, PSTR buffer
,
541 DWORD buffer_size
, PDWORD required_size
)
544 WCHAR
*sectionW
= NULL
, *bufferW
= NULL
;
548 TRACE("%p, %p, %s, %p, 0x%08x, %p\n", hinf
, context
, debugstr_a(section
), buffer
,
549 buffer_size
, required_size
);
551 if (section
&& !(sectionW
= strdupAtoW( section
)))
554 if (!SetupGetTargetPathW( hinf
, context
, sectionW
, NULL
, 0, &required
))
557 if (!(bufferW
= HeapAlloc( GetProcessHeap(), 0, required
* sizeof(WCHAR
) )))
560 if (!SetupGetTargetPathW( hinf
, context
, sectionW
, bufferW
, required
, NULL
))
563 size
= WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, NULL
, 0, NULL
, NULL
);
564 if (required_size
) *required_size
= size
;
568 if (buffer_size
>= size
)
569 WideCharToMultiByte( CP_ACP
, 0, bufferW
, -1, buffer
, buffer_size
, NULL
, NULL
);
572 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
579 HeapFree( GetProcessHeap(), 0, sectionW
);
580 HeapFree( GetProcessHeap(), 0, bufferW
);
584 /***********************************************************************
585 * SetupGetTargetPathW (SETUPAPI.@)
588 BOOL WINAPI
SetupGetTargetPathW( HINF hinf
, PINFCONTEXT context
, PCWSTR section
, PWSTR buffer
,
589 DWORD buffer_size
, PDWORD required_size
)
591 static const WCHAR destination_dirs
[] =
592 {'D','e','s','t','i','n','a','t','i','o','n','D','i','r','s',0};
593 static const WCHAR default_dest_dir
[] =
594 {'D','e','f','a','u','l','t','D','e','s','t','D','i','r',0};
597 WCHAR
*dir
, systemdir
[MAX_PATH
];
601 TRACE("%p, %p, %s, %p, 0x%08x, %p\n", hinf
, context
, debugstr_w(section
), buffer
,
602 buffer_size
, required_size
);
604 if (context
) ret
= SetupFindFirstLineW( hinf
, destination_dirs
, NULL
, context
);
607 if (!(ret
= SetupFindFirstLineW( hinf
, destination_dirs
, section
, &ctx
)))
608 ret
= SetupFindFirstLineW( hinf
, destination_dirs
, default_dest_dir
, &ctx
);
610 if (!ret
|| !(dir
= PARSER_get_dest_dir( context
? context
: &ctx
)))
612 GetSystemDirectoryW( systemdir
, MAX_PATH
);
615 size
= lstrlenW( dir
) + 1;
616 if (required_size
) *required_size
= size
;
620 if (buffer_size
>= size
)
621 lstrcpyW( buffer
, dir
);
624 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
625 if (dir
!= systemdir
) HeapFree( GetProcessHeap(), 0, dir
);
629 if (dir
!= systemdir
) HeapFree( GetProcessHeap(), 0, dir
);
633 /***********************************************************************
634 * SetupQueryInfOriginalFileInformationA (SETUPAPI.@)
636 BOOL WINAPI
SetupQueryInfOriginalFileInformationA(
637 PSP_INF_INFORMATION InfInformation
, UINT InfIndex
,
638 PSP_ALTPLATFORM_INFO AlternativePlatformInfo
,
639 PSP_ORIGINAL_FILE_INFO_A OriginalFileInfo
)
642 SP_ORIGINAL_FILE_INFO_W OriginalFileInfoW
;
644 TRACE("(%p, %d, %p, %p)\n", InfInformation
, InfIndex
,
645 AlternativePlatformInfo
, OriginalFileInfo
);
647 if (OriginalFileInfo
->cbSize
!= sizeof(*OriginalFileInfo
))
649 WARN("incorrect OriginalFileInfo->cbSize of %d\n", OriginalFileInfo
->cbSize
);
650 SetLastError( ERROR_INVALID_USER_BUFFER
);
654 OriginalFileInfoW
.cbSize
= sizeof(OriginalFileInfoW
);
655 ret
= SetupQueryInfOriginalFileInformationW(InfInformation
, InfIndex
,
656 AlternativePlatformInfo
, &OriginalFileInfoW
);
659 WideCharToMultiByte(CP_ACP
, 0, OriginalFileInfoW
.OriginalInfName
, -1,
660 OriginalFileInfo
->OriginalInfName
, MAX_PATH
, NULL
, NULL
);
661 WideCharToMultiByte(CP_ACP
, 0, OriginalFileInfoW
.OriginalCatalogName
, -1,
662 OriginalFileInfo
->OriginalCatalogName
, MAX_PATH
, NULL
, NULL
);
668 /***********************************************************************
669 * SetupQueryInfOriginalFileInformationW (SETUPAPI.@)
671 BOOL WINAPI
SetupQueryInfOriginalFileInformationW(
672 PSP_INF_INFORMATION InfInformation
, UINT InfIndex
,
673 PSP_ALTPLATFORM_INFO AlternativePlatformInfo
,
674 PSP_ORIGINAL_FILE_INFO_W OriginalFileInfo
)
679 static const WCHAR wszVersion
[] = { 'V','e','r','s','i','o','n',0 };
680 static const WCHAR wszCatalogFile
[] = { 'C','a','t','a','l','o','g','F','i','l','e',0 };
682 FIXME("(%p, %d, %p, %p): semi-stub\n", InfInformation
, InfIndex
,
683 AlternativePlatformInfo
, OriginalFileInfo
);
685 if (OriginalFileInfo
->cbSize
!= sizeof(*OriginalFileInfo
))
687 WARN("incorrect OriginalFileInfo->cbSize of %d\n", OriginalFileInfo
->cbSize
);
688 SetLastError(ERROR_INVALID_USER_BUFFER
);
692 inf_path
= (LPWSTR
)InfInformation
->VersionData
;
694 /* FIXME: we should get OriginalCatalogName from CatalogFile line in
695 * the original inf file and cache it, but that would require building a
697 hinf
= SetupOpenInfFileW(inf_path
, NULL
, INF_STYLE_WIN4
, NULL
);
698 if (hinf
== INVALID_HANDLE_VALUE
) return FALSE
;
700 if (!SetupGetLineTextW(NULL
, hinf
, wszVersion
, wszCatalogFile
,
701 OriginalFileInfo
->OriginalCatalogName
,
702 ARRAY_SIZE(OriginalFileInfo
->OriginalCatalogName
), NULL
))
704 OriginalFileInfo
->OriginalCatalogName
[0] = '\0';
706 SetupCloseInfFile(hinf
);
708 /* FIXME: not quite correct as we just return the same file name as
709 * destination (copied) inf file, not the source (original) inf file.
710 * to fix it properly would require building a .pnf file */
711 /* file name is stored in VersionData field of InfInformation */
712 inf_name
= wcsrchr(inf_path
, '\\');
713 if (inf_name
) inf_name
++;
714 else inf_name
= inf_path
;
716 lstrcpyW(OriginalFileInfo
->OriginalInfName
, inf_name
);
721 /***********************************************************************
722 * SetupGetInfDriverStoreLocationW (SETUPAPI.@)
724 BOOL WINAPI
SetupGetInfDriverStoreLocationW(
725 PCWSTR FileName
, PSP_ALTPLATFORM_INFO AlternativePlatformInfo
,
726 PCWSTR LocaleName
, PWSTR ReturnBuffer
, DWORD ReturnBufferSize
,
729 FIXME("stub: %s %p %s %p %u %p\n", debugstr_w(FileName
), AlternativePlatformInfo
, debugstr_w(LocaleName
), ReturnBuffer
, ReturnBufferSize
, RequiredSize
);
731 SetLastError(ERROR_CALL_NOT_IMPLEMENTED
);