Commit c8bb3359 authored by Hans Leidekker's avatar Hans Leidekker Committed by Alexandre Julliard

msi: Check supported languages in the AppSearch action.

parent 52f3d5be
...@@ -608,6 +608,65 @@ static void ACTION_ExpandAnyPath(MSIPACKAGE *package, WCHAR *src, WCHAR *dst, ...@@ -608,6 +608,65 @@ static void ACTION_ExpandAnyPath(MSIPACKAGE *package, WCHAR *src, WCHAR *dst,
msi_free(deformatted); msi_free(deformatted);
} }
static LANGID *parse_languages( const WCHAR *languages, DWORD *num_ids )
{
UINT i, count = 1;
WCHAR *str = strdupW( languages ), *p, *q;
LANGID *ret;
if (!str) return NULL;
for (p = q = str; (q = strchrW( q, ',' )); q++) count++;
if (!(ret = msi_alloc( count * sizeof(LANGID) )))
{
msi_free( str );
return NULL;
}
i = 0;
while (*p)
{
q = strchrW( p, ',' );
if (q) *q = 0;
ret[i] = atoiW( p );
if (!q) break;
p = q + 1;
i++;
}
msi_free( str );
*num_ids = count;
return ret;
}
static BOOL match_languages( const void *version, const WCHAR *languages )
{
struct lang
{
USHORT id;
USHORT codepage;
} *lang;
DWORD len, num_ids, i, j;
BOOL found = FALSE;
LANGID *ids;
if (!languages || !languages[0]) return TRUE;
if (!VerQueryValueW( version, szLangResource, (void **)&lang, &len )) return FALSE;
if (!(ids = parse_languages( languages, &num_ids ))) return FALSE;
for (i = 0; i < num_ids; i++)
{
found = FALSE;
for (j = 0; j < len / sizeof(struct lang); j++)
{
if (!ids[i] || ids[i] == lang[j].id) found = TRUE;
}
if (!found) goto done;
}
done:
msi_free( ids );
return found;
}
/* Sets *matches to whether the file (whose path is filePath) matches the /* Sets *matches to whether the file (whose path is filePath) matches the
* versions set in sig. * versions set in sig.
* Return ERROR_SUCCESS in case of success (whether or not the file matches), * Return ERROR_SUCCESS in case of success (whether or not the file matches),
...@@ -616,69 +675,55 @@ static void ACTION_ExpandAnyPath(MSIPACKAGE *package, WCHAR *src, WCHAR *dst, ...@@ -616,69 +675,55 @@ static void ACTION_ExpandAnyPath(MSIPACKAGE *package, WCHAR *src, WCHAR *dst,
static UINT ACTION_FileVersionMatches(const MSISIGNATURE *sig, LPCWSTR filePath, static UINT ACTION_FileVersionMatches(const MSISIGNATURE *sig, LPCWSTR filePath,
BOOL *matches) BOOL *matches)
{ {
UINT rc = ERROR_SUCCESS; UINT len;
void *version;
VS_FIXEDFILEINFO *info = NULL;
DWORD zero, size = GetFileVersionInfoSizeW( filePath, &zero );
*matches = FALSE; *matches = FALSE;
if (sig->Languages)
{
FIXME(": need to check version for languages %s\n",
debugstr_w(sig->Languages));
}
else
{
DWORD zero, size = GetFileVersionInfoSizeW(filePath, &zero);
if (size) if (!size) return ERROR_SUCCESS;
{ if (!(version = msi_alloc( size ))) return ERROR_OUTOFMEMORY;
LPVOID buf = msi_alloc( size);
if (buf) if (GetFileVersionInfoW( filePath, 0, size, version ))
{ VerQueryValueW( version, szBackSlash, (void **)&info, &len );
UINT versionLen;
LPVOID subBlock = NULL; if (info)
{
if (GetFileVersionInfoW(filePath, 0, size, buf)) TRACE("comparing file version %d.%d.%d.%d:\n",
VerQueryValueW(buf, szBackSlash, &subBlock, &versionLen); HIWORD(info->dwFileVersionMS),
if (subBlock) LOWORD(info->dwFileVersionMS),
{ HIWORD(info->dwFileVersionLS),
VS_FIXEDFILEINFO *info = subBlock; LOWORD(info->dwFileVersionLS));
if (info->dwFileVersionMS < sig->MinVersionMS
TRACE("Comparing file version %d.%d.%d.%d:\n", || (info->dwFileVersionMS == sig->MinVersionMS &&
HIWORD(info->dwFileVersionMS), info->dwFileVersionLS < sig->MinVersionLS))
LOWORD(info->dwFileVersionMS), {
HIWORD(info->dwFileVersionLS), TRACE("less than minimum version %d.%d.%d.%d\n",
LOWORD(info->dwFileVersionLS)); HIWORD(sig->MinVersionMS),
if (info->dwFileVersionMS < sig->MinVersionMS LOWORD(sig->MinVersionMS),
|| (info->dwFileVersionMS == sig->MinVersionMS && HIWORD(sig->MinVersionLS),
info->dwFileVersionLS < sig->MinVersionLS)) LOWORD(sig->MinVersionLS));
{ }
TRACE("Less than minimum version %d.%d.%d.%d\n", else if ((sig->MaxVersionMS || sig->MaxVersionLS) &&
HIWORD(sig->MinVersionMS), (info->dwFileVersionMS > sig->MaxVersionMS ||
LOWORD(sig->MinVersionMS), (info->dwFileVersionMS == sig->MaxVersionMS &&
HIWORD(sig->MinVersionLS), info->dwFileVersionLS > sig->MaxVersionLS)))
LOWORD(sig->MinVersionLS)); {
} TRACE("greater than maximum version %d.%d.%d.%d\n",
else if ((sig->MaxVersionMS || sig->MaxVersionLS) && HIWORD(sig->MaxVersionMS),
(info->dwFileVersionMS > sig->MaxVersionMS || LOWORD(sig->MaxVersionMS),
(info->dwFileVersionMS == sig->MaxVersionMS && HIWORD(sig->MaxVersionLS),
info->dwFileVersionLS > sig->MaxVersionLS))) LOWORD(sig->MaxVersionLS));
{ }
TRACE("Greater than maximum version %d.%d.%d.%d\n", else if (!match_languages( version, sig->Languages ))
HIWORD(sig->MaxVersionMS), {
LOWORD(sig->MaxVersionMS), TRACE("languages %s not supported\n", debugstr_w( sig->Languages ));
HIWORD(sig->MaxVersionLS),
LOWORD(sig->MaxVersionLS));
}
else
*matches = TRUE;
}
msi_free( buf);
}
else
rc = ERROR_OUTOFMEMORY;
} }
else *matches = TRUE;
} }
return rc; msi_free( version );
return ERROR_SUCCESS;
} }
/* Sets *matches to whether the file in findData matches that in sig. /* Sets *matches to whether the file in findData matches that in sig.
......
...@@ -3133,11 +3133,7 @@ static UINT get_file_version( const WCHAR *path, WCHAR *verbuf, DWORD *verlen, ...@@ -3133,11 +3133,7 @@ static UINT get_file_version( const WCHAR *path, WCHAR *verbuf, DWORD *verlen,
WCHAR *langbuf, DWORD *langlen ) WCHAR *langbuf, DWORD *langlen )
{ {
static const WCHAR szVersionResource[] = {'\\',0}; static const WCHAR szVersionResource[] = {'\\',0};
static const WCHAR szVersionFormat[] = { static const WCHAR szVersionFormat[] = {'%','d','.','%','d','.','%','d','.','%','d',0};
'%','d','.','%','d','.','%','d','.','%','d',0};
static const WCHAR szLangResource[] = {
'\\','V','a','r','F','i','l','e','I','n','f','o','\\',
'T','r','a','n','s','l','a','t','i','o','n',0};
static const WCHAR szLangFormat[] = {'%','d',0}; static const WCHAR szLangFormat[] = {'%','d',0};
UINT ret = ERROR_SUCCESS; UINT ret = ERROR_SUCCESS;
DWORD len, error; DWORD len, error;
......
...@@ -1172,6 +1172,7 @@ static const WCHAR szAppDataFolder[] = {'A','p','p','D','a','t','a','F','o','l', ...@@ -1172,6 +1172,7 @@ static const WCHAR szAppDataFolder[] = {'A','p','p','D','a','t','a','F','o','l',
static const WCHAR szRollbackDisabled[] = {'R','o','l','l','b','a','c','k','D','i','s','a','b','l','e','d',0}; static const WCHAR szRollbackDisabled[] = {'R','o','l','l','b','a','c','k','D','i','s','a','b','l','e','d',0};
static const WCHAR szName[] = {'N','a','m','e',0}; static const WCHAR szName[] = {'N','a','m','e',0};
static const WCHAR szData[] = {'D','a','t','a',0}; static const WCHAR szData[] = {'D','a','t','a',0};
static const WCHAR szLangResource[] = {'\\','V','a','r','F','i','l','e','I','n','f','o','\\','T','r','a','n','s','l','a','t','i','o','n',0};
/* memory allocation macro functions */ /* memory allocation macro functions */
static void *msi_alloc( size_t len ) __WINE_ALLOC_SIZE(1); static void *msi_alloc( size_t len ) __WINE_ALLOC_SIZE(1);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment