Fixed alignment.
[wine.git] / dlls / version / info.c
blob616f48707cf1792f5209ba149bca1ffc94395a4a
1 /*
2 * Implementation of VERSION.DLL - Version Info access
3 *
4 * Copyright 1996,1997 Marcus Meissner
5 * Copyright 1997 David Cuthbert
6 * Copyright 1999 Ulrich Weigand
7 */
9 #include <stdlib.h>
10 #include <string.h>
12 #include "winreg.h"
13 #include "winver.h"
14 #include "wine/winestring.h"
15 #include "winerror.h"
16 #include "heap.h"
17 #include "crtdll.h"
18 #include "debugtools.h"
20 DEFAULT_DEBUG_CHANNEL(ver)
23 /******************************************************************************
25 * This function will print via dprintf[_]ver to stddeb debug info regarding
26 * the file info structure vffi.
27 * 15-Feb-1998 Dimitrie Paun (dimi@cs.toronto.edu)
28 * Added this function to clean up the code.
30 *****************************************************************************/
31 static void print_vffi_debug(VS_FIXEDFILEINFO *vffi)
33 TRACE(" structversion=%u.%u, fileversion=%u.%u.%u.%u, productversion=%u.%u.%u.%u, flagmask=0x%lx, flags=%s%s%s%s%s%s\n",
34 HIWORD(vffi->dwStrucVersion),LOWORD(vffi->dwStrucVersion),
35 HIWORD(vffi->dwFileVersionMS),LOWORD(vffi->dwFileVersionMS),
36 HIWORD(vffi->dwFileVersionLS),LOWORD(vffi->dwFileVersionLS),
37 HIWORD(vffi->dwProductVersionMS),LOWORD(vffi->dwProductVersionMS),
38 HIWORD(vffi->dwProductVersionLS),LOWORD(vffi->dwProductVersionLS),
39 vffi->dwFileFlagsMask,
40 (vffi->dwFileFlags & VS_FF_DEBUG) ? "DEBUG," : "",
41 (vffi->dwFileFlags & VS_FF_PRERELEASE) ? "PRERELEASE," : "",
42 (vffi->dwFileFlags & VS_FF_PATCHED) ? "PATCHED," : "",
43 (vffi->dwFileFlags & VS_FF_PRIVATEBUILD) ? "PRIVATEBUILD," : "",
44 (vffi->dwFileFlags & VS_FF_INFOINFERRED) ? "INFOINFERRED," : "",
45 (vffi->dwFileFlags & VS_FF_SPECIALBUILD) ? "SPECIALBUILD," : ""
48 TRACE("(");
49 DPRINTF(" OS=0x%x.0x%x ",
50 HIWORD(vffi->dwFileOS),
51 LOWORD(vffi->dwFileOS)
53 switch (vffi->dwFileOS&0xFFFF0000) {
54 case VOS_DOS:DPRINTF("DOS,");break;
55 case VOS_OS216:DPRINTF("OS/2-16,");break;
56 case VOS_OS232:DPRINTF("OS/2-32,");break;
57 case VOS_NT:DPRINTF("NT,");break;
58 case VOS_UNKNOWN:
59 default:
60 DPRINTF("UNKNOWN(0x%lx),",vffi->dwFileOS&0xFFFF0000);break;
62 switch (LOWORD(vffi->dwFileOS)) {
63 case VOS__BASE:DPRINTF("BASE");break;
64 case VOS__WINDOWS16:DPRINTF("WIN16");break;
65 case VOS__WINDOWS32:DPRINTF("WIN32");break;
66 case VOS__PM16:DPRINTF("PM16");break;
67 case VOS__PM32:DPRINTF("PM32");break;
68 default:DPRINTF("UNKNOWN(0x%x)",LOWORD(vffi->dwFileOS));break;
70 DPRINTF(")\n");
72 switch (vffi->dwFileType) {
73 default:
74 case VFT_UNKNOWN:
75 TRACE("filetype=Unknown(0x%lx)",vffi->dwFileType);
76 break;
77 case VFT_APP:TRACE("filetype=APP,");break;
78 case VFT_DLL:TRACE("filetype=DLL,");break;
79 case VFT_DRV:
80 TRACE("filetype=DRV,");
81 switch(vffi->dwFileSubtype) {
82 default:
83 case VFT2_UNKNOWN:
84 DPRINTF("UNKNOWN(0x%lx)",vffi->dwFileSubtype);
85 break;
86 case VFT2_DRV_PRINTER:
87 DPRINTF("PRINTER");
88 break;
89 case VFT2_DRV_KEYBOARD:
90 DPRINTF("KEYBOARD");
91 break;
92 case VFT2_DRV_LANGUAGE:
93 DPRINTF("LANGUAGE");
94 break;
95 case VFT2_DRV_DISPLAY:
96 DPRINTF("DISPLAY");
97 break;
98 case VFT2_DRV_MOUSE:
99 DPRINTF("MOUSE");
100 break;
101 case VFT2_DRV_NETWORK:
102 DPRINTF("NETWORK");
103 break;
104 case VFT2_DRV_SYSTEM:
105 DPRINTF("SYSTEM");
106 break;
107 case VFT2_DRV_INSTALLABLE:
108 DPRINTF("INSTALLABLE");
109 break;
110 case VFT2_DRV_SOUND:
111 DPRINTF("SOUND");
112 break;
113 case VFT2_DRV_COMM:
114 DPRINTF("COMM");
115 break;
116 case VFT2_DRV_INPUTMETHOD:
117 DPRINTF("INPUTMETHOD");
118 break;
120 break;
121 case VFT_FONT:
122 TRACE("filetype=FONT.");
123 switch (vffi->dwFileSubtype) {
124 default:
125 DPRINTF("UNKNOWN(0x%lx)",vffi->dwFileSubtype);
126 break;
127 case VFT2_FONT_RASTER:DPRINTF("RASTER");break;
128 case VFT2_FONT_VECTOR:DPRINTF("VECTOR");break;
129 case VFT2_FONT_TRUETYPE:DPRINTF("TRUETYPE");break;
131 break;
132 case VFT_VXD:TRACE("filetype=VXD");break;
133 case VFT_STATIC_LIB:TRACE("filetype=STATIC_LIB");break;
135 DPRINTF("\n");
136 TRACE(" filedata=0x%lx.0x%lx\n",
137 vffi->dwFileDateMS,vffi->dwFileDateLS);
141 /***********************************************************************
142 * Version Info Structure
145 typedef struct
147 WORD wLength;
148 WORD wValueLength;
149 CHAR szKey[1];
150 #if 0 /* variable length structure */
151 /* DWORD aligned */
152 BYTE Value[];
153 /* DWORD aligned */
154 VS_VERSION_INFO_STRUCT16 Children[];
155 #endif
156 } VS_VERSION_INFO_STRUCT16;
158 typedef struct
160 WORD wLength;
161 WORD wValueLength;
162 WORD bText;
163 WCHAR szKey[1];
164 #if 0 /* variable length structure */
165 /* DWORD aligned */
166 BYTE Value[];
167 /* DWORD aligned */
168 VS_VERSION_INFO_STRUCT32 Children[];
169 #endif
170 } VS_VERSION_INFO_STRUCT32;
172 #define VersionInfoIs16( ver ) \
173 ( ((VS_VERSION_INFO_STRUCT16 *)ver)->szKey[0] >= ' ' )
175 #define VersionInfo16_Value( ver ) \
176 (LPBYTE)( (DWORD)((ver)->szKey) + (((lstrlenA((ver)->szKey)+1) + 3) & ~3 ))
177 #define VersionInfo32_Value( ver ) \
178 (LPBYTE)( (DWORD)((ver)->szKey) + ((2*(lstrlenW((ver)->szKey)+1) + 3) & ~3 ))
180 #define VersionInfo16_Children( ver ) \
181 (VS_VERSION_INFO_STRUCT16 *)( VersionInfo16_Value( ver ) + \
182 ( ( (ver)->wValueLength + 3 ) & ~3 ) )
183 #define VersionInfo32_Children( ver ) \
184 (VS_VERSION_INFO_STRUCT32 *)( VersionInfo32_Value( ver ) + \
185 ( ( (ver)->wValueLength * \
186 ((ver)->bText? 2 : 1) + 3 ) & ~3 ) )
188 #define VersionInfo16_Next( ver ) \
189 (VS_VERSION_INFO_STRUCT16 *)( (LPBYTE)ver + (((ver)->wLength + 3) & ~3) )
190 #define VersionInfo32_Next( ver ) \
191 (VS_VERSION_INFO_STRUCT32 *)( (LPBYTE)ver + (((ver)->wLength + 3) & ~3) )
193 /***********************************************************************
194 * ConvertVersionInfo32To16 [internal]
196 void ConvertVersionInfo32To16( VS_VERSION_INFO_STRUCT32 *info32,
197 VS_VERSION_INFO_STRUCT16 *info16 )
199 /* Copy data onto local stack to prevent overwrites */
200 WORD wLength = info32->wLength;
201 WORD wValueLength = info32->wValueLength;
202 WORD bText = info32->bText;
203 LPBYTE lpValue = VersionInfo32_Value( info32 );
204 VS_VERSION_INFO_STRUCT32 *child32 = VersionInfo32_Children( info32 );
205 VS_VERSION_INFO_STRUCT16 *child16;
207 TRACE("Converting %p to %p\n", info32, info16 );
208 TRACE("wLength %d, wValueLength %d, bText %d, value %p, child %p\n",
209 wLength, wValueLength, bText, lpValue, child32 );
211 /* Convert key */
212 lstrcpyWtoA( info16->szKey, info32->szKey );
214 TRACE("Copied key from %p to %p: %s\n", info32->szKey, info16->szKey,
215 debugstr_a(info16->szKey) );
217 /* Convert value */
218 if ( wValueLength == 0 )
220 info16->wValueLength = 0;
221 TRACE("No value present\n" );
223 else if ( bText )
225 info16->wValueLength = lstrlenW( (LPCWSTR)lpValue ) + 1;
226 lstrcpyWtoA( VersionInfo16_Value( info16 ), (LPCWSTR)lpValue );
228 TRACE("Copied value from %p to %p: %s\n", lpValue,
229 VersionInfo16_Value( info16 ),
230 debugstr_a(VersionInfo16_Value( info16 )) );
232 else
234 info16->wValueLength = wValueLength;
235 memmove( VersionInfo16_Value( info16 ), lpValue, wValueLength );
237 TRACE("Copied value from %p to %p: %d bytes\n", lpValue,
238 VersionInfo16_Value( info16 ), wValueLength );
241 /* Convert children */
242 child16 = VersionInfo16_Children( info16 );
243 while ( (DWORD)child32 < (DWORD)info32 + wLength )
245 VS_VERSION_INFO_STRUCT32 *nextChild = VersionInfo32_Next( child32 );
247 ConvertVersionInfo32To16( child32, child16 );
249 child16 = VersionInfo16_Next( child16 );
250 child32 = nextChild;
253 /* Fixup length */
254 info16->wLength = (DWORD)child16 - (DWORD)info16;
256 TRACE("Finished, length is %d (%p - %p)\n",
257 info16->wLength, info16, child16 );
261 /***********************************************************************
262 * GetFileVersionInfoSize32A [VERSION.2]
264 DWORD WINAPI GetFileVersionInfoSizeA( LPCSTR filename, LPDWORD handle )
266 VS_FIXEDFILEINFO *vffi;
267 DWORD len, ret, offset;
268 BYTE buf[144];
270 TRACE("(%s,%p)\n", debugstr_a(filename), handle );
272 len = GetFileResourceSize( filename,
273 MAKEINTRESOURCEA(VS_FILE_INFO),
274 MAKEINTRESOURCEA(VS_VERSION_INFO),
275 &offset );
276 if (!len) return 0;
278 ret = GetFileResource( filename,
279 MAKEINTRESOURCEA(VS_FILE_INFO),
280 MAKEINTRESOURCEA(VS_VERSION_INFO),
281 offset, sizeof( buf ), buf );
282 if (!ret) return 0;
284 if ( handle ) *handle = offset;
286 if ( VersionInfoIs16( buf ) )
287 vffi = (VS_FIXEDFILEINFO *)VersionInfo16_Value( (VS_VERSION_INFO_STRUCT16 *)buf );
288 else
289 vffi = (VS_FIXEDFILEINFO *)VersionInfo32_Value( (VS_VERSION_INFO_STRUCT32 *)buf );
291 if ( vffi->dwSignature != VS_FFI_SIGNATURE )
293 WARN("vffi->dwSignature is 0x%08lx, but not 0x%08lx!\n",
294 vffi->dwSignature, VS_FFI_SIGNATURE );
295 return 0;
298 if ( ((VS_VERSION_INFO_STRUCT16 *)buf)->wLength < len )
299 len = ((VS_VERSION_INFO_STRUCT16 *)buf)->wLength;
301 if ( TRACE_ON(ver) )
302 print_vffi_debug( vffi );
304 return len;
307 /***********************************************************************
308 * GetFileVersionInfoSize32W [VERSION.3]
310 DWORD WINAPI GetFileVersionInfoSizeW( LPCWSTR filename, LPDWORD handle )
312 LPSTR fn = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
313 DWORD ret = GetFileVersionInfoSizeA( fn, handle );
314 HeapFree( GetProcessHeap(), 0, fn );
315 return ret;
318 /***********************************************************************
319 * GetFileVersionInfo32A [VERSION.1]
321 DWORD WINAPI GetFileVersionInfoA( LPCSTR filename, DWORD handle,
322 DWORD datasize, LPVOID data )
324 TRACE("(%s,%ld,size=%ld,data=%p)\n",
325 debugstr_a(filename), handle, datasize, data );
327 if ( !GetFileResource( filename, MAKEINTRESOURCEA(VS_FILE_INFO),
328 MAKEINTRESOURCEA(VS_VERSION_INFO),
329 handle, datasize, data ) )
330 return FALSE;
332 if ( datasize >= sizeof(VS_VERSION_INFO_STRUCT16)
333 && datasize >= ((VS_VERSION_INFO_STRUCT16 *)data)->wLength
334 && !VersionInfoIs16( data ) )
336 /* convert resource from PE format to NE format */
337 ConvertVersionInfo32To16( (VS_VERSION_INFO_STRUCT32 *)data,
338 (VS_VERSION_INFO_STRUCT16 *)data );
341 return TRUE;
344 /***********************************************************************
345 * GetFileVersionInfo32W [VERSION.4]
347 DWORD WINAPI GetFileVersionInfoW( LPCWSTR filename, DWORD handle,
348 DWORD datasize, LPVOID data )
350 LPSTR fn = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
351 DWORD retv = TRUE;
353 TRACE("(%s,%ld,size=%ld,data=%p)\n",
354 debugstr_a(fn), handle, datasize, data );
356 if ( !GetFileResource( fn, MAKEINTRESOURCEA(VS_FILE_INFO),
357 MAKEINTRESOURCEA(VS_VERSION_INFO),
358 handle, datasize, data ) )
359 retv = FALSE;
361 else if ( datasize >= sizeof(VS_VERSION_INFO_STRUCT16)
362 && datasize >= ((VS_VERSION_INFO_STRUCT16 *)data)->wLength
363 && VersionInfoIs16( data ) )
365 ERR("Cannot access NE resource in %s\n", debugstr_a(fn) );
366 retv = FALSE;
369 HeapFree( GetProcessHeap(), 0, fn );
370 return retv;
374 /***********************************************************************
375 * VersionInfo16_FindChild [internal]
377 VS_VERSION_INFO_STRUCT16 *VersionInfo16_FindChild( VS_VERSION_INFO_STRUCT16 *info,
378 LPCSTR szKey, UINT cbKey )
380 VS_VERSION_INFO_STRUCT16 *child = VersionInfo16_Children( info );
382 while ( (DWORD)child < (DWORD)info + info->wLength )
384 if ( !lstrncmpiA( child->szKey, szKey, cbKey ) )
385 return child;
387 if (!(child->wLength)) return NULL;
388 child = VersionInfo16_Next( child );
391 return NULL;
394 /***********************************************************************
395 * VersionInfo32_FindChild [internal]
397 VS_VERSION_INFO_STRUCT32 *VersionInfo32_FindChild( VS_VERSION_INFO_STRUCT32 *info,
398 LPCWSTR szKey, UINT cbKey )
400 VS_VERSION_INFO_STRUCT32 *child = VersionInfo32_Children( info );
402 while ( (DWORD)child < (DWORD)info + info->wLength )
404 if ( !CRTDLL__wcsnicmp( child->szKey, szKey, cbKey ) )
405 return child;
407 child = VersionInfo32_Next( child );
410 return NULL;
413 /***********************************************************************
414 * VerQueryValue32A [VERSION.12]
416 DWORD WINAPI VerQueryValueA( LPVOID pBlock, LPCSTR lpSubBlock,
417 LPVOID *lplpBuffer, UINT *puLen )
419 VS_VERSION_INFO_STRUCT16 *info = (VS_VERSION_INFO_STRUCT16 *)pBlock;
420 if ( !VersionInfoIs16( info ) )
422 ERR("called on PE resource!\n" );
423 return FALSE;
426 TRACE("(%p,%s,%p,%p)\n",
427 pBlock, debugstr_a(lpSubBlock), lplpBuffer, puLen );
429 while ( *lpSubBlock )
431 /* Find next path component */
432 LPCSTR lpNextSlash;
433 for ( lpNextSlash = lpSubBlock; *lpNextSlash; lpNextSlash++ )
434 if ( *lpNextSlash == '\\' )
435 break;
437 /* Skip empty components */
438 if ( lpNextSlash == lpSubBlock )
440 lpSubBlock++;
441 continue;
444 /* We have a non-empty component: search info for key */
445 info = VersionInfo16_FindChild( info, lpSubBlock, lpNextSlash-lpSubBlock );
446 if ( !info ) return FALSE;
448 /* Skip path component */
449 lpSubBlock = lpNextSlash;
452 /* Return value */
453 *lplpBuffer = VersionInfo16_Value( info );
454 *puLen = info->wValueLength;
456 return TRUE;
459 /***********************************************************************
460 * VerQueryValue32W [VERSION.13]
462 DWORD WINAPI VerQueryValueW( LPVOID pBlock, LPCWSTR lpSubBlock,
463 LPVOID *lplpBuffer, UINT *puLen )
465 VS_VERSION_INFO_STRUCT32 *info = (VS_VERSION_INFO_STRUCT32 *)pBlock;
466 if ( VersionInfoIs16( info ) )
468 ERR("called on NE resource!\n" );
469 return FALSE;
472 TRACE("(%p,%s,%p,%p)\n",
473 pBlock, debugstr_w(lpSubBlock), lplpBuffer, puLen );
475 while ( *lpSubBlock )
477 /* Find next path component */
478 LPCWSTR lpNextSlash;
479 for ( lpNextSlash = lpSubBlock; *lpNextSlash; lpNextSlash++ )
480 if ( *lpNextSlash == '\\' )
481 break;
483 /* Skip empty components */
484 if ( lpNextSlash == lpSubBlock )
486 lpSubBlock++;
487 continue;
490 /* We have a non-empty component: search info for key */
491 info = VersionInfo32_FindChild( info, lpSubBlock, lpNextSlash-lpSubBlock );
492 if ( !info ) return FALSE;
494 /* Skip path component */
495 lpSubBlock = lpNextSlash;
498 /* Return value */
499 *lplpBuffer = VersionInfo32_Value( info );
500 *puLen = info->wValueLength;
502 return TRUE;
505 extern LPCSTR WINE_GetLanguageName( UINT langid );
507 /***********************************************************************
508 * VerLanguageName32A [VERSION.9]
510 DWORD WINAPI VerLanguageNameA( UINT wLang, LPSTR szLang, UINT nSize )
512 char buffer[80];
513 LPCSTR name;
514 DWORD result;
516 TRACE("(%d,%p,%d)\n", wLang, szLang, nSize );
519 * First, check \System\CurrentControlSet\control\Nls\Locale\<langid>
520 * from the registry.
523 sprintf( buffer,
524 "\\System\\CurrentControlSet\\control\\Nls\\Locale\\%08x",
525 wLang );
527 result = RegQueryValueA( HKEY_LOCAL_MACHINE, buffer, szLang, (LPDWORD)&nSize );
528 if ( result == ERROR_SUCCESS || result == ERROR_MORE_DATA )
529 return nSize;
532 * If that fails, use the internal table
533 * (actually, Windows stores the names in a string table resource ...)
536 name = WINE_GetLanguageName( wLang );
537 lstrcpynA( szLang, name, nSize );
538 return lstrlenA( name );
541 /***********************************************************************
542 * VerLanguageName32W [VERSION.10]
544 DWORD WINAPI VerLanguageNameW( UINT wLang, LPWSTR szLang, UINT nSize )
546 char buffer[80];
547 LPWSTR keyname;
548 LPCSTR name;
549 DWORD result;
551 TRACE("(%d,%p,%d)\n", wLang, szLang, nSize );
554 * First, check \System\CurrentControlSet\control\Nls\Locale\<langid>
555 * from the registry.
558 sprintf( buffer,
559 "\\System\\CurrentControlSet\\control\\Nls\\Locale\\%08x",
560 wLang );
562 keyname = HEAP_strdupAtoW( GetProcessHeap(), 0, buffer );
563 result = RegQueryValueW( HKEY_LOCAL_MACHINE, keyname, szLang, (LPDWORD)&nSize );
564 HeapFree( GetProcessHeap(), 0, keyname );
566 if ( result == ERROR_SUCCESS || result == ERROR_MORE_DATA )
567 return nSize;
570 * If that fails, use the internal table
571 * (actually, Windows stores the names in a string table resource ...)
574 name = WINE_GetLanguageName( wLang );
575 lstrcpynAtoW( szLang, name, nSize );
576 return lstrlenA( name );