2010-11-11 10:45:42 +01:00
|
|
|
/*
|
|
|
|
* Implementation of the Microsoft Installer (msi.dll)
|
|
|
|
*
|
|
|
|
* Copyright 2010 Hans Leidekker for CodeWeavers
|
|
|
|
*
|
|
|
|
* 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>
|
|
|
|
|
|
|
|
#define COBJMACROS
|
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
2011-01-24 15:22:11 +01:00
|
|
|
#include "winreg.h"
|
2010-11-11 10:45:42 +01:00
|
|
|
#include "wine/debug.h"
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
#include "msipriv.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msi);
|
|
|
|
|
2011-05-06 14:40:33 +02:00
|
|
|
static HRESULT (WINAPI *pCreateAssemblyCacheNet10)( IAssemblyCache **, DWORD );
|
2011-03-07 12:40:39 +01:00
|
|
|
static HRESULT (WINAPI *pCreateAssemblyCacheNet11)( IAssemblyCache **, DWORD );
|
|
|
|
static HRESULT (WINAPI *pCreateAssemblyCacheNet20)( IAssemblyCache **, DWORD );
|
2012-05-17 10:17:50 +02:00
|
|
|
static HRESULT (WINAPI *pCreateAssemblyCacheNet40)( IAssemblyCache **, DWORD );
|
2010-11-11 10:45:42 +01:00
|
|
|
static HRESULT (WINAPI *pCreateAssemblyCacheSxs)( IAssemblyCache **, DWORD );
|
|
|
|
static HRESULT (WINAPI *pLoadLibraryShim)( LPCWSTR, LPCWSTR, LPVOID, HMODULE * );
|
2011-03-07 12:40:39 +01:00
|
|
|
static HRESULT (WINAPI *pGetFileVersion)( LPCWSTR, LPWSTR, DWORD, DWORD * );
|
2015-04-10 12:59:01 +02:00
|
|
|
static HRESULT (WINAPI *pCreateAssemblyNameObject)( IAssemblyName **, LPCWSTR, DWORD, LPVOID );
|
|
|
|
static HRESULT (WINAPI *pCreateAssemblyEnum)( IAssemblyEnum **, IUnknown *, IAssemblyName *, DWORD, LPVOID );
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2012-05-17 10:17:50 +02:00
|
|
|
static HMODULE hfusion10, hfusion11, hfusion20, hfusion40, hmscoree, hsxs;
|
2011-05-06 14:39:58 +02:00
|
|
|
|
2010-11-11 10:45:42 +01:00
|
|
|
static BOOL init_function_pointers( void )
|
|
|
|
{
|
2012-06-12 09:24:58 +02:00
|
|
|
static const WCHAR szFusion[] = {'f','u','s','i','o','n','.','d','l','l',0};
|
|
|
|
static const WCHAR szMscoree[] = {'\\','m','s','c','o','r','e','e','.','d','l','l',0};
|
|
|
|
static const WCHAR szSxs[] = {'s','x','s','.','d','l','l',0};
|
2011-05-06 14:40:33 +02:00
|
|
|
static const WCHAR szVersion10[] = {'v','1','.','0','.','3','7','0','5',0};
|
2011-03-07 12:40:39 +01:00
|
|
|
static const WCHAR szVersion11[] = {'v','1','.','1','.','4','3','2','2',0};
|
|
|
|
static const WCHAR szVersion20[] = {'v','2','.','0','.','5','0','7','2','7',0};
|
2012-05-17 10:17:50 +02:00
|
|
|
static const WCHAR szVersion40[] = {'v','4','.','0','.','3','0','3','1','9',0};
|
2012-06-12 09:24:58 +02:00
|
|
|
WCHAR path[MAX_PATH];
|
|
|
|
DWORD len = GetSystemDirectoryW( path, MAX_PATH );
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2012-06-12 09:24:58 +02:00
|
|
|
if (!hsxs && !(hsxs = LoadLibraryW( szSxs ))) return FALSE;
|
2012-06-12 09:24:45 +02:00
|
|
|
if (!(pCreateAssemblyCacheSxs = (void *)GetProcAddress( hsxs, "CreateAssemblyCache" )))
|
|
|
|
{
|
|
|
|
FreeLibrary( hsxs );
|
|
|
|
hsxs = NULL;
|
|
|
|
return FALSE;
|
|
|
|
}
|
2012-06-12 09:24:58 +02:00
|
|
|
strcpyW( path + len, szMscoree );
|
|
|
|
if (hmscoree || !(hmscoree = LoadLibraryW( path ))) return TRUE;
|
2011-05-06 14:40:33 +02:00
|
|
|
pGetFileVersion = (void *)GetProcAddress( hmscoree, "GetFileVersion" ); /* missing from v1.0.3705 */
|
2012-06-12 09:24:45 +02:00
|
|
|
if (!(pLoadLibraryShim = (void *)GetProcAddress( hmscoree, "LoadLibraryShim" )))
|
|
|
|
{
|
|
|
|
FreeLibrary( hmscoree );
|
|
|
|
hmscoree = NULL;
|
|
|
|
return TRUE;
|
|
|
|
}
|
2011-05-06 14:40:33 +02:00
|
|
|
if (!pLoadLibraryShim( szFusion, szVersion10, NULL, &hfusion10 ))
|
|
|
|
pCreateAssemblyCacheNet10 = (void *)GetProcAddress( hfusion10, "CreateAssemblyCache" );
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
if (!pLoadLibraryShim( szFusion, szVersion11, NULL, &hfusion11 ))
|
|
|
|
pCreateAssemblyCacheNet11 = (void *)GetProcAddress( hfusion11, "CreateAssemblyCache" );
|
|
|
|
|
|
|
|
if (!pLoadLibraryShim( szFusion, szVersion20, NULL, &hfusion20 ))
|
|
|
|
pCreateAssemblyCacheNet20 = (void *)GetProcAddress( hfusion20, "CreateAssemblyCache" );
|
|
|
|
|
2012-05-17 10:17:50 +02:00
|
|
|
if (!pLoadLibraryShim( szFusion, szVersion40, NULL, &hfusion40 ))
|
2015-04-10 12:59:01 +02:00
|
|
|
{
|
2012-05-17 10:17:50 +02:00
|
|
|
pCreateAssemblyCacheNet40 = (void *)GetProcAddress( hfusion40, "CreateAssemblyCache" );
|
2015-04-10 12:59:01 +02:00
|
|
|
pCreateAssemblyNameObject = (void *)GetProcAddress( hfusion40, "CreateAssemblyNameObject" );
|
|
|
|
pCreateAssemblyEnum = (void *)GetProcAddress( hfusion40, "CreateAssemblyEnum" );
|
|
|
|
}
|
2010-11-11 10:45:42 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-05-06 14:39:58 +02:00
|
|
|
BOOL msi_init_assembly_caches( MSIPACKAGE *package )
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
|
|
|
if (!init_function_pointers()) return FALSE;
|
2011-03-07 12:40:39 +01:00
|
|
|
if (pCreateAssemblyCacheSxs( &package->cache_sxs, 0 ) != S_OK) return FALSE;
|
2012-05-17 10:17:29 +02:00
|
|
|
if (pCreateAssemblyCacheNet10) pCreateAssemblyCacheNet10( &package->cache_net[CLR_VERSION_V10], 0 );
|
2011-03-07 12:40:39 +01:00
|
|
|
if (pCreateAssemblyCacheNet11) pCreateAssemblyCacheNet11( &package->cache_net[CLR_VERSION_V11], 0 );
|
|
|
|
if (pCreateAssemblyCacheNet20) pCreateAssemblyCacheNet20( &package->cache_net[CLR_VERSION_V20], 0 );
|
2012-05-17 10:17:50 +02:00
|
|
|
if (pCreateAssemblyCacheNet40) pCreateAssemblyCacheNet40( &package->cache_net[CLR_VERSION_V40], 0 );
|
2012-06-12 09:24:45 +02:00
|
|
|
return TRUE;
|
2010-11-11 10:45:42 +01:00
|
|
|
}
|
|
|
|
|
2011-05-06 14:39:58 +02:00
|
|
|
void msi_destroy_assembly_caches( MSIPACKAGE *package )
|
|
|
|
{
|
|
|
|
UINT i;
|
|
|
|
|
2012-06-12 09:25:09 +02:00
|
|
|
if (package->cache_sxs)
|
|
|
|
{
|
|
|
|
IAssemblyCache_Release( package->cache_sxs );
|
|
|
|
package->cache_sxs = NULL;
|
|
|
|
}
|
2011-05-06 14:39:58 +02:00
|
|
|
for (i = 0; i < CLR_VERSION_MAX; i++)
|
|
|
|
{
|
|
|
|
if (package->cache_net[i])
|
|
|
|
{
|
|
|
|
IAssemblyCache_Release( package->cache_net[i] );
|
|
|
|
package->cache_net[i] = NULL;
|
|
|
|
}
|
|
|
|
}
|
2011-05-06 14:40:33 +02:00
|
|
|
pCreateAssemblyCacheNet10 = NULL;
|
2011-05-06 14:39:58 +02:00
|
|
|
pCreateAssemblyCacheNet11 = NULL;
|
|
|
|
pCreateAssemblyCacheNet20 = NULL;
|
2012-05-17 10:17:50 +02:00
|
|
|
pCreateAssemblyCacheNet40 = NULL;
|
2011-05-06 14:40:33 +02:00
|
|
|
FreeLibrary( hfusion10 );
|
2011-05-06 14:39:58 +02:00
|
|
|
FreeLibrary( hfusion11 );
|
|
|
|
FreeLibrary( hfusion20 );
|
2012-05-17 10:17:50 +02:00
|
|
|
FreeLibrary( hfusion40 );
|
2011-05-06 14:39:58 +02:00
|
|
|
FreeLibrary( hmscoree );
|
|
|
|
FreeLibrary( hsxs );
|
2012-06-12 09:24:45 +02:00
|
|
|
hfusion10 = NULL;
|
|
|
|
hfusion11 = NULL;
|
|
|
|
hfusion20 = NULL;
|
|
|
|
hfusion40 = NULL;
|
|
|
|
hmscoree = NULL;
|
|
|
|
hsxs = NULL;
|
2011-05-06 14:39:58 +02:00
|
|
|
}
|
|
|
|
|
2011-03-16 08:32:32 +01:00
|
|
|
static MSIRECORD *get_assembly_record( MSIPACKAGE *package, const WCHAR *comp )
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
|
|
|
static const WCHAR query[] = {
|
|
|
|
'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
|
|
|
|
'`','M','s','i','A','s','s','e','m','b','l','y','`',' ',
|
|
|
|
'W','H','E','R','E',' ','`','C','o','m','p','o','n','e','n','t','_','`',
|
|
|
|
' ','=',' ','\'','%','s','\'',0};
|
|
|
|
MSIQUERY *view;
|
|
|
|
MSIRECORD *rec;
|
|
|
|
UINT r;
|
|
|
|
|
|
|
|
r = MSI_OpenQuery( package->db, &view, query, comp );
|
|
|
|
if (r != ERROR_SUCCESS)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
r = MSI_ViewExecute( view, NULL );
|
|
|
|
if (r != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
msiobj_release( &view->hdr );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
r = MSI_ViewFetch( view, &rec );
|
|
|
|
if (r != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
msiobj_release( &view->hdr );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (!MSI_RecordGetString( rec, 4 ))
|
|
|
|
TRACE("component is a global assembly\n");
|
|
|
|
|
|
|
|
msiobj_release( &view->hdr );
|
|
|
|
return rec;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct assembly_name
|
|
|
|
{
|
2011-01-24 15:21:45 +01:00
|
|
|
UINT count;
|
|
|
|
UINT index;
|
|
|
|
WCHAR **attrs;
|
2010-11-11 10:45:42 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static UINT get_assembly_name_attribute( MSIRECORD *rec, LPVOID param )
|
|
|
|
{
|
2011-01-24 15:21:45 +01:00
|
|
|
static const WCHAR fmtW[] = {'%','s','=','"','%','s','"',0};
|
2010-11-11 10:45:42 +01:00
|
|
|
static const WCHAR nameW[] = {'n','a','m','e',0};
|
|
|
|
struct assembly_name *name = param;
|
|
|
|
const WCHAR *attr = MSI_RecordGetString( rec, 2 );
|
2011-01-24 15:21:45 +01:00
|
|
|
const WCHAR *value = MSI_RecordGetString( rec, 3 );
|
|
|
|
int len = strlenW( fmtW ) + strlenW( attr ) + strlenW( value );
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2011-01-24 15:21:45 +01:00
|
|
|
if (!(name->attrs[name->index] = msi_alloc( len * sizeof(WCHAR) )))
|
|
|
|
return ERROR_OUTOFMEMORY;
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2011-01-24 15:21:45 +01:00
|
|
|
if (!strcmpiW( attr, nameW )) strcpyW( name->attrs[name->index++], value );
|
|
|
|
else sprintfW( name->attrs[name->index++], fmtW, attr, value );
|
2010-11-11 10:45:42 +01:00
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static WCHAR *get_assembly_display_name( MSIDATABASE *db, const WCHAR *comp, MSIASSEMBLY *assembly )
|
|
|
|
{
|
2011-01-24 15:21:45 +01:00
|
|
|
static const WCHAR commaW[] = {',',0};
|
2010-11-11 10:45:42 +01:00
|
|
|
static const WCHAR queryW[] = {
|
|
|
|
'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',
|
|
|
|
'`','M','s','i','A','s','s','e','m','b','l','y','N','a','m','e','`',' ',
|
|
|
|
'W','H','E','R','E',' ','`','C','o','m','p','o','n','e','n','t','_','`',
|
|
|
|
' ','=',' ','\'','%','s','\'',0};
|
|
|
|
struct assembly_name name;
|
|
|
|
WCHAR *display_name = NULL;
|
|
|
|
MSIQUERY *view;
|
2011-01-24 15:21:45 +01:00
|
|
|
UINT i, r;
|
2010-11-11 10:45:42 +01:00
|
|
|
int len;
|
|
|
|
|
|
|
|
r = MSI_OpenQuery( db, &view, queryW, comp );
|
|
|
|
if (r != ERROR_SUCCESS)
|
|
|
|
return NULL;
|
|
|
|
|
2011-01-24 15:21:45 +01:00
|
|
|
name.count = 0;
|
|
|
|
name.index = 0;
|
|
|
|
name.attrs = NULL;
|
|
|
|
MSI_IterateRecords( view, &name.count, NULL, NULL );
|
|
|
|
if (!name.count) goto done;
|
|
|
|
|
|
|
|
name.attrs = msi_alloc( name.count * sizeof(WCHAR *) );
|
|
|
|
if (!name.attrs) goto done;
|
|
|
|
|
2010-11-11 10:45:42 +01:00
|
|
|
MSI_IterateRecords( view, NULL, get_assembly_name_attribute, &name );
|
|
|
|
|
2011-01-24 15:21:45 +01:00
|
|
|
len = 0;
|
|
|
|
for (i = 0; i < name.count; i++) len += strlenW( name.attrs[i] ) + 1;
|
|
|
|
|
|
|
|
display_name = msi_alloc( (len + 1) * sizeof(WCHAR) );
|
|
|
|
if (display_name)
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
2011-01-24 15:21:45 +01:00
|
|
|
display_name[0] = 0;
|
|
|
|
for (i = 0; i < name.count; i++)
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
2011-01-24 15:21:45 +01:00
|
|
|
strcatW( display_name, name.attrs[i] );
|
|
|
|
if (i < name.count - 1) strcatW( display_name, commaW );
|
2010-11-11 10:45:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
2011-01-24 15:21:45 +01:00
|
|
|
msiobj_release( &view->hdr );
|
2011-07-07 09:22:59 +02:00
|
|
|
if (name.attrs)
|
|
|
|
{
|
|
|
|
for (i = 0; i < name.count; i++) msi_free( name.attrs[i] );
|
|
|
|
msi_free( name.attrs );
|
|
|
|
}
|
2010-11-11 10:45:42 +01:00
|
|
|
return display_name;
|
|
|
|
}
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
static BOOL is_assembly_installed( IAssemblyCache *cache, const WCHAR *display_name )
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
|
|
|
HRESULT hr;
|
2011-03-07 12:40:39 +01:00
|
|
|
ASSEMBLY_INFO info;
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2012-06-12 09:24:45 +02:00
|
|
|
if (!cache) return FALSE;
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
memset( &info, 0, sizeof(info) );
|
|
|
|
info.cbAssemblyInfo = sizeof(info);
|
2012-03-30 10:22:05 +02:00
|
|
|
hr = IAssemblyCache_QueryAssemblyInfo( cache, 0, display_name, &info );
|
2014-10-08 06:27:49 +02:00
|
|
|
if (hr == S_OK /* sxs version */ || hr == E_NOT_SUFFICIENT_BUFFER)
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
2012-03-30 10:22:05 +02:00
|
|
|
return (info.dwAssemblyFlags == ASSEMBLYINFO_FLAG_INSTALLED);
|
2010-11-11 10:45:42 +01:00
|
|
|
}
|
2012-03-30 10:22:05 +02:00
|
|
|
TRACE("QueryAssemblyInfo returned 0x%08x\n", hr);
|
|
|
|
return FALSE;
|
2011-03-07 12:40:39 +01:00
|
|
|
}
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2015-04-10 12:59:01 +02:00
|
|
|
WCHAR *msi_get_assembly_path( MSIPACKAGE *package, const WCHAR *displayname )
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
ASSEMBLY_INFO info;
|
|
|
|
IAssemblyCache *cache = package->cache_net[CLR_VERSION_V40];
|
|
|
|
|
|
|
|
if (!cache) return NULL;
|
|
|
|
|
|
|
|
memset( &info, 0, sizeof(info) );
|
|
|
|
info.cbAssemblyInfo = sizeof(info);
|
|
|
|
hr = IAssemblyCache_QueryAssemblyInfo( cache, 0, displayname, &info );
|
|
|
|
if (hr != E_NOT_SUFFICIENT_BUFFER) return NULL;
|
|
|
|
|
|
|
|
if (!(info.pszCurrentAssemblyPathBuf = msi_alloc( info.cchBuf * sizeof(WCHAR) ))) return NULL;
|
|
|
|
|
|
|
|
hr = IAssemblyCache_QueryAssemblyInfo( cache, 0, displayname, &info );
|
|
|
|
if (FAILED( hr ))
|
|
|
|
{
|
|
|
|
msi_free( info.pszCurrentAssemblyPathBuf );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
TRACE("returning %s\n", debugstr_w(info.pszCurrentAssemblyPathBuf));
|
|
|
|
return info.pszCurrentAssemblyPathBuf;
|
|
|
|
}
|
|
|
|
|
|
|
|
IAssemblyEnum *msi_create_assembly_enum( MSIPACKAGE *package, const WCHAR *displayname )
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
IAssemblyName *name;
|
|
|
|
IAssemblyEnum *ret;
|
|
|
|
WCHAR *str;
|
|
|
|
UINT len = 0;
|
|
|
|
|
|
|
|
if (!pCreateAssemblyNameObject || !pCreateAssemblyEnum) return NULL;
|
|
|
|
|
|
|
|
hr = pCreateAssemblyNameObject( &name, displayname, CANOF_PARSE_DISPLAY_NAME, NULL );
|
|
|
|
if (FAILED( hr )) return NULL;
|
|
|
|
|
|
|
|
hr = IAssemblyName_GetName( name, &len, NULL );
|
|
|
|
if (hr != E_NOT_SUFFICIENT_BUFFER || !(str = msi_alloc( len * sizeof(WCHAR) )))
|
|
|
|
{
|
|
|
|
IAssemblyName_Release( name );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = IAssemblyName_GetName( name, &len, str );
|
|
|
|
IAssemblyName_Release( name );
|
|
|
|
if (FAILED( hr ))
|
|
|
|
{
|
|
|
|
msi_free( str );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = pCreateAssemblyNameObject( &name, str, 0, NULL );
|
|
|
|
msi_free( str );
|
|
|
|
if (FAILED( hr )) return NULL;
|
|
|
|
|
|
|
|
hr = pCreateAssemblyEnum( &ret, NULL, name, ASM_CACHE_GAC, NULL );
|
|
|
|
IAssemblyName_Release( name );
|
|
|
|
if (FAILED( hr )) return NULL;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-05-06 14:40:33 +02:00
|
|
|
static const WCHAR clr_version_v10[] = {'v','1','.','0','.','3','7','0','5',0};
|
2011-03-07 12:40:39 +01:00
|
|
|
static const WCHAR clr_version_v11[] = {'v','1','.','1','.','4','3','2','2',0};
|
|
|
|
static const WCHAR clr_version_v20[] = {'v','2','.','0','.','5','0','7','2','7',0};
|
2012-05-17 10:17:50 +02:00
|
|
|
static const WCHAR clr_version_v40[] = {'v','4','.','0','.','3','0','3','1','9',0};
|
2011-03-07 12:40:39 +01:00
|
|
|
static const WCHAR clr_version_unknown[] = {'u','n','k','n','o','w','n',0};
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
static const WCHAR *clr_version[] =
|
|
|
|
{
|
2011-05-06 14:40:33 +02:00
|
|
|
clr_version_v10,
|
2011-03-07 12:40:39 +01:00
|
|
|
clr_version_v11,
|
2012-05-17 10:17:50 +02:00
|
|
|
clr_version_v20,
|
|
|
|
clr_version_v40
|
2011-03-07 12:40:39 +01:00
|
|
|
};
|
2010-11-11 10:45:42 +01:00
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
static const WCHAR *get_clr_version_str( enum clr_version version )
|
|
|
|
{
|
2018-08-20 21:15:43 +02:00
|
|
|
if (version >= ARRAY_SIZE( clr_version )) return clr_version_unknown;
|
2011-03-07 12:40:39 +01:00
|
|
|
return clr_version[version];
|
2010-11-11 10:45:42 +01:00
|
|
|
}
|
|
|
|
|
2011-05-06 14:39:58 +02:00
|
|
|
/* assembly caches must be initialized */
|
|
|
|
MSIASSEMBLY *msi_load_assembly( MSIPACKAGE *package, MSICOMPONENT *comp )
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
|
|
|
MSIRECORD *rec;
|
|
|
|
MSIASSEMBLY *a;
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
if (!(rec = get_assembly_record( package, comp->Component ))) return NULL;
|
2010-11-11 10:45:42 +01:00
|
|
|
if (!(a = msi_alloc_zero( sizeof(MSIASSEMBLY) )))
|
|
|
|
{
|
|
|
|
msiobj_release( &rec->hdr );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
a->feature = strdupW( MSI_RecordGetString( rec, 2 ) );
|
|
|
|
TRACE("feature %s\n", debugstr_w(a->feature));
|
|
|
|
|
|
|
|
a->manifest = strdupW( MSI_RecordGetString( rec, 3 ) );
|
|
|
|
TRACE("manifest %s\n", debugstr_w(a->manifest));
|
|
|
|
|
|
|
|
a->application = strdupW( MSI_RecordGetString( rec, 4 ) );
|
|
|
|
TRACE("application %s\n", debugstr_w(a->application));
|
|
|
|
|
|
|
|
a->attributes = MSI_RecordGetInteger( rec, 5 );
|
|
|
|
TRACE("attributes %u\n", a->attributes);
|
|
|
|
|
|
|
|
if (!(a->display_name = get_assembly_display_name( package->db, comp->Component, a )))
|
|
|
|
{
|
|
|
|
WARN("can't get display name\n");
|
|
|
|
msiobj_release( &rec->hdr );
|
2011-01-18 09:16:28 +01:00
|
|
|
msi_free( a->feature );
|
|
|
|
msi_free( a->manifest );
|
|
|
|
msi_free( a->application );
|
2010-11-11 10:45:42 +01:00
|
|
|
msi_free( a );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
TRACE("display name %s\n", debugstr_w(a->display_name));
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
if (a->application)
|
|
|
|
{
|
2011-03-16 09:22:59 +01:00
|
|
|
/* We can't check the manifest here because the target path may still change.
|
|
|
|
So we assume that the assembly is not installed and lean on the InstallFiles
|
|
|
|
action to determine which files need to be installed.
|
|
|
|
*/
|
|
|
|
a->installed = FALSE;
|
2011-03-07 12:40:39 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (a->attributes == msidbAssemblyAttributesWin32)
|
|
|
|
a->installed = is_assembly_installed( package->cache_sxs, a->display_name );
|
|
|
|
else
|
|
|
|
{
|
|
|
|
UINT i;
|
|
|
|
for (i = 0; i < CLR_VERSION_MAX; i++)
|
|
|
|
{
|
|
|
|
a->clr_version[i] = is_assembly_installed( package->cache_net[i], a->display_name );
|
|
|
|
if (a->clr_version[i])
|
|
|
|
{
|
|
|
|
TRACE("runtime version %s\n", debugstr_w(get_clr_version_str( i )));
|
|
|
|
a->installed = TRUE;
|
2012-05-17 10:17:16 +02:00
|
|
|
break;
|
2011-03-07 12:40:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-11-11 10:45:42 +01:00
|
|
|
TRACE("assembly is %s\n", a->installed ? "installed" : "not installed");
|
|
|
|
msiobj_release( &rec->hdr );
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
static enum clr_version get_clr_version( const WCHAR *filename )
|
|
|
|
{
|
|
|
|
DWORD len;
|
|
|
|
HRESULT hr;
|
|
|
|
enum clr_version version = CLR_VERSION_V11;
|
|
|
|
WCHAR *strW;
|
|
|
|
|
2011-05-06 14:40:33 +02:00
|
|
|
if (!pGetFileVersion) return CLR_VERSION_V10;
|
|
|
|
|
2011-03-07 12:40:39 +01:00
|
|
|
hr = pGetFileVersion( filename, NULL, 0, &len );
|
2014-10-08 06:27:49 +02:00
|
|
|
if (hr != E_NOT_SUFFICIENT_BUFFER) return CLR_VERSION_V11;
|
2011-03-07 12:40:39 +01:00
|
|
|
if ((strW = msi_alloc( len * sizeof(WCHAR) )))
|
|
|
|
{
|
|
|
|
hr = pGetFileVersion( filename, strW, len, &len );
|
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
UINT i;
|
|
|
|
for (i = 0; i < CLR_VERSION_MAX; i++)
|
|
|
|
if (!strcmpW( strW, clr_version[i] )) version = i;
|
|
|
|
}
|
|
|
|
msi_free( strW );
|
|
|
|
}
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
2011-05-06 14:39:58 +02:00
|
|
|
UINT msi_install_assembly( MSIPACKAGE *package, MSICOMPONENT *comp )
|
2010-11-11 10:45:42 +01:00
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
const WCHAR *manifest;
|
|
|
|
IAssemblyCache *cache;
|
|
|
|
MSIASSEMBLY *assembly = comp->assembly;
|
|
|
|
MSIFEATURE *feature = NULL;
|
|
|
|
|
|
|
|
if (comp->assembly->feature)
|
2011-05-06 14:39:17 +02:00
|
|
|
feature = msi_get_loaded_feature( package, comp->assembly->feature );
|
2010-11-11 10:45:42 +01:00
|
|
|
|
|
|
|
if (assembly->application)
|
|
|
|
{
|
|
|
|
if (feature) feature->Action = INSTALLSTATE_LOCAL;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
if (assembly->attributes == msidbAssemblyAttributesWin32)
|
|
|
|
{
|
|
|
|
if (!assembly->manifest)
|
|
|
|
{
|
|
|
|
WARN("no manifest\n");
|
|
|
|
return ERROR_FUNCTION_FAILED;
|
|
|
|
}
|
2011-05-06 14:39:17 +02:00
|
|
|
manifest = msi_get_loaded_file( package, assembly->manifest )->TargetPath;
|
2010-11-11 10:45:42 +01:00
|
|
|
cache = package->cache_sxs;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-05-06 14:39:17 +02:00
|
|
|
manifest = msi_get_loaded_file( package, comp->KeyPath )->TargetPath;
|
2011-03-07 12:40:39 +01:00
|
|
|
cache = package->cache_net[get_clr_version( manifest )];
|
2012-06-12 09:24:45 +02:00
|
|
|
if (!cache) return ERROR_SUCCESS;
|
2010-11-11 10:45:42 +01:00
|
|
|
}
|
|
|
|
TRACE("installing assembly %s\n", debugstr_w(manifest));
|
|
|
|
|
|
|
|
hr = IAssemblyCache_InstallAssembly( cache, 0, manifest, NULL );
|
|
|
|
if (hr != S_OK)
|
|
|
|
{
|
|
|
|
ERR("Failed to install assembly %s (0x%08x)\n", debugstr_w(manifest), hr);
|
|
|
|
return ERROR_FUNCTION_FAILED;
|
|
|
|
}
|
|
|
|
if (feature) feature->Action = INSTALLSTATE_LOCAL;
|
|
|
|
assembly->installed = TRUE;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2012-04-30 09:35:25 +02:00
|
|
|
UINT msi_uninstall_assembly( MSIPACKAGE *package, MSICOMPONENT *comp )
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
IAssemblyCache *cache;
|
|
|
|
MSIASSEMBLY *assembly = comp->assembly;
|
|
|
|
MSIFEATURE *feature = NULL;
|
|
|
|
|
|
|
|
if (comp->assembly->feature)
|
|
|
|
feature = msi_get_loaded_feature( package, comp->assembly->feature );
|
|
|
|
|
|
|
|
if (assembly->application)
|
|
|
|
{
|
|
|
|
if (feature) feature->Action = INSTALLSTATE_ABSENT;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
TRACE("removing %s\n", debugstr_w(assembly->display_name));
|
|
|
|
|
|
|
|
if (assembly->attributes == msidbAssemblyAttributesWin32)
|
|
|
|
{
|
|
|
|
cache = package->cache_sxs;
|
|
|
|
hr = IAssemblyCache_UninstallAssembly( cache, 0, assembly->display_name, NULL, NULL );
|
|
|
|
if (FAILED( hr )) WARN("failed to uninstall assembly 0x%08x\n", hr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < CLR_VERSION_MAX; i++)
|
|
|
|
{
|
|
|
|
if (!assembly->clr_version[i]) continue;
|
|
|
|
cache = package->cache_net[i];
|
2012-06-12 09:24:45 +02:00
|
|
|
if (cache)
|
|
|
|
{
|
|
|
|
hr = IAssemblyCache_UninstallAssembly( cache, 0, assembly->display_name, NULL, NULL );
|
|
|
|
if (FAILED( hr )) WARN("failed to uninstall assembly 0x%08x\n", hr);
|
|
|
|
}
|
2012-04-30 09:35:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (feature) feature->Action = INSTALLSTATE_ABSENT;
|
|
|
|
assembly->installed = FALSE;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2011-01-24 15:22:11 +01:00
|
|
|
static WCHAR *build_local_assembly_path( const WCHAR *filename )
|
|
|
|
{
|
|
|
|
UINT i;
|
|
|
|
WCHAR *ret;
|
|
|
|
|
|
|
|
if (!(ret = msi_alloc( (strlenW( filename ) + 1) * sizeof(WCHAR) )))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (i = 0; filename[i]; i++)
|
|
|
|
{
|
|
|
|
if (filename[i] == '\\' || filename[i] == '/') ret[i] = '|';
|
|
|
|
else ret[i] = filename[i];
|
|
|
|
}
|
|
|
|
ret[i] = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static LONG open_assemblies_key( UINT context, BOOL win32, HKEY *hkey )
|
|
|
|
{
|
|
|
|
static const WCHAR path_win32[] =
|
|
|
|
{'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'I','n','s','t','a','l','l','e','r','\\','W','i','n','3','2','A','s','s','e','m','b','l','i','e','s','\\',0};
|
|
|
|
static const WCHAR path_dotnet[] =
|
|
|
|
{'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'I','n','s','t','a','l','l','e','r','\\','A','s','s','e','m','b','l','i','e','s','\\',0};
|
|
|
|
static const WCHAR classes_path_win32[] =
|
|
|
|
{'I','n','s','t','a','l','l','e','r','\\','W','i','n','3','2','A','s','s','e','m','b','l','i','e','s','\\',0};
|
|
|
|
static const WCHAR classes_path_dotnet[] =
|
|
|
|
{'I','n','s','t','a','l','l','e','r','\\','A','s','s','e','m','b','l','i','e','s','\\',0};
|
|
|
|
HKEY root;
|
|
|
|
const WCHAR *path;
|
|
|
|
|
|
|
|
if (context == MSIINSTALLCONTEXT_MACHINE)
|
|
|
|
{
|
|
|
|
root = HKEY_CLASSES_ROOT;
|
|
|
|
if (win32) path = classes_path_win32;
|
|
|
|
else path = classes_path_dotnet;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
root = HKEY_CURRENT_USER;
|
|
|
|
if (win32) path = path_win32;
|
|
|
|
else path = path_dotnet;
|
|
|
|
}
|
|
|
|
return RegCreateKeyW( root, path, hkey );
|
|
|
|
}
|
|
|
|
|
|
|
|
static LONG open_local_assembly_key( UINT context, BOOL win32, const WCHAR *filename, HKEY *hkey )
|
|
|
|
{
|
|
|
|
LONG res;
|
|
|
|
HKEY root;
|
|
|
|
WCHAR *path;
|
|
|
|
|
|
|
|
if (!(path = build_local_assembly_path( filename )))
|
|
|
|
return ERROR_OUTOFMEMORY;
|
|
|
|
|
|
|
|
if ((res = open_assemblies_key( context, win32, &root )))
|
|
|
|
{
|
|
|
|
msi_free( path );
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
res = RegCreateKeyW( root, path, hkey );
|
|
|
|
RegCloseKey( root );
|
|
|
|
msi_free( path );
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static LONG delete_local_assembly_key( UINT context, BOOL win32, const WCHAR *filename )
|
|
|
|
{
|
|
|
|
LONG res;
|
|
|
|
HKEY root;
|
|
|
|
WCHAR *path;
|
|
|
|
|
|
|
|
if (!(path = build_local_assembly_path( filename )))
|
|
|
|
return ERROR_OUTOFMEMORY;
|
|
|
|
|
|
|
|
if ((res = open_assemblies_key( context, win32, &root )))
|
|
|
|
{
|
|
|
|
msi_free( path );
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
res = RegDeleteKeyW( root, path );
|
|
|
|
RegCloseKey( root );
|
|
|
|
msi_free( path );
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static LONG open_global_assembly_key( UINT context, BOOL win32, HKEY *hkey )
|
|
|
|
{
|
|
|
|
static const WCHAR path_win32[] =
|
|
|
|
{'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'I','n','s','t','a','l','l','e','r','\\','W','i','n','3','2','A','s','s','e','m','b','l','i','e','s','\\',
|
|
|
|
'G','l','o','b','a','l',0};
|
|
|
|
static const WCHAR path_dotnet[] =
|
|
|
|
{'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'I','n','s','t','a','l','l','e','r','\\','A','s','s','e','m','b','l','i','e','s','\\',
|
|
|
|
'G','l','o','b','a','l',0};
|
|
|
|
static const WCHAR classes_path_win32[] =
|
|
|
|
{'I','n','s','t','a','l','l','e','r','\\','W','i','n','3','2','A','s','s','e','m','b','l','i','e','s','\\',
|
|
|
|
'G','l','o','b','a','l',0};
|
|
|
|
static const WCHAR classes_path_dotnet[] =
|
|
|
|
{'I','n','s','t','a','l','l','e','r','\\','A','s','s','e','m','b','l','i','e','s','\\','G','l','o','b','a','l',0};
|
|
|
|
HKEY root;
|
|
|
|
const WCHAR *path;
|
|
|
|
|
|
|
|
if (context == MSIINSTALLCONTEXT_MACHINE)
|
|
|
|
{
|
|
|
|
root = HKEY_CLASSES_ROOT;
|
|
|
|
if (win32) path = classes_path_win32;
|
|
|
|
else path = classes_path_dotnet;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
root = HKEY_CURRENT_USER;
|
|
|
|
if (win32) path = path_win32;
|
|
|
|
else path = path_dotnet;
|
|
|
|
}
|
|
|
|
return RegCreateKeyW( root, path, hkey );
|
|
|
|
}
|
|
|
|
|
2010-11-11 10:45:42 +01:00
|
|
|
UINT ACTION_MsiPublishAssemblies( MSIPACKAGE *package )
|
|
|
|
{
|
|
|
|
MSICOMPONENT *comp;
|
|
|
|
|
2018-07-30 04:16:20 +02:00
|
|
|
if (package->script == SCRIPT_NONE)
|
|
|
|
return msi_schedule_action(package, SCRIPT_INSTALL, szMsiPublishAssemblies);
|
|
|
|
|
2010-11-11 10:45:42 +01:00
|
|
|
LIST_FOR_EACH_ENTRY(comp, &package->components, MSICOMPONENT, entry)
|
|
|
|
{
|
2011-01-24 15:22:11 +01:00
|
|
|
LONG res;
|
|
|
|
HKEY hkey;
|
|
|
|
GUID guid;
|
|
|
|
DWORD size;
|
|
|
|
WCHAR buffer[43];
|
|
|
|
MSIRECORD *uirow;
|
|
|
|
MSIASSEMBLY *assembly = comp->assembly;
|
|
|
|
BOOL win32;
|
|
|
|
|
|
|
|
if (!assembly || !comp->ComponentId) continue;
|
|
|
|
|
2011-05-25 10:39:11 +02:00
|
|
|
comp->Action = msi_get_component_action( package, comp );
|
|
|
|
if (comp->Action != INSTALLSTATE_LOCAL)
|
2011-01-24 15:22:11 +01:00
|
|
|
{
|
2011-05-25 10:39:11 +02:00
|
|
|
TRACE("component not scheduled for installation %s\n", debugstr_w(comp->Component));
|
2011-01-24 15:22:11 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
TRACE("publishing %s\n", debugstr_w(comp->Component));
|
|
|
|
|
|
|
|
CLSIDFromString( package->ProductCode, &guid );
|
|
|
|
encode_base85_guid( &guid, buffer );
|
|
|
|
buffer[20] = '>';
|
|
|
|
CLSIDFromString( comp->ComponentId, &guid );
|
|
|
|
encode_base85_guid( &guid, buffer + 21 );
|
|
|
|
buffer[42] = 0;
|
|
|
|
|
|
|
|
win32 = assembly->attributes & msidbAssemblyAttributesWin32;
|
|
|
|
if (assembly->application)
|
|
|
|
{
|
2011-05-06 14:39:17 +02:00
|
|
|
MSIFILE *file = msi_get_loaded_file( package, assembly->application );
|
2018-09-04 02:17:56 +02:00
|
|
|
if (!file)
|
|
|
|
{
|
|
|
|
WARN("no matching file %s for local assembly\n", debugstr_w(assembly->application));
|
|
|
|
continue;
|
|
|
|
}
|
2011-01-24 15:22:11 +01:00
|
|
|
if ((res = open_local_assembly_key( package->Context, win32, file->TargetPath, &hkey )))
|
|
|
|
{
|
|
|
|
WARN("failed to open local assembly key %d\n", res);
|
|
|
|
return ERROR_FUNCTION_FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((res = open_global_assembly_key( package->Context, win32, &hkey )))
|
|
|
|
{
|
|
|
|
WARN("failed to open global assembly key %d\n", res);
|
|
|
|
return ERROR_FUNCTION_FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
size = sizeof(buffer);
|
|
|
|
if ((res = RegSetValueExW( hkey, assembly->display_name, 0, REG_MULTI_SZ, (const BYTE *)buffer, size )))
|
|
|
|
{
|
|
|
|
WARN("failed to set assembly value %d\n", res);
|
|
|
|
}
|
|
|
|
RegCloseKey( hkey );
|
|
|
|
|
|
|
|
uirow = MSI_CreateRecord( 2 );
|
|
|
|
MSI_RecordSetStringW( uirow, 2, assembly->display_name );
|
2017-07-07 16:03:15 +02:00
|
|
|
MSI_ProcessMessage(package, INSTALLMESSAGE_ACTIONDATA, uirow);
|
2011-01-24 15:22:11 +01:00
|
|
|
msiobj_release( &uirow->hdr );
|
|
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
UINT ACTION_MsiUnpublishAssemblies( MSIPACKAGE *package )
|
|
|
|
{
|
|
|
|
MSICOMPONENT *comp;
|
|
|
|
|
2018-07-30 04:16:20 +02:00
|
|
|
if (package->script == SCRIPT_NONE)
|
|
|
|
return msi_schedule_action(package, SCRIPT_INSTALL, szMsiUnpublishAssemblies);
|
|
|
|
|
2011-01-24 15:22:11 +01:00
|
|
|
LIST_FOR_EACH_ENTRY(comp, &package->components, MSICOMPONENT, entry)
|
|
|
|
{
|
|
|
|
LONG res;
|
|
|
|
MSIRECORD *uirow;
|
|
|
|
MSIASSEMBLY *assembly = comp->assembly;
|
|
|
|
BOOL win32;
|
|
|
|
|
|
|
|
if (!assembly || !comp->ComponentId) continue;
|
|
|
|
|
2011-05-25 10:39:11 +02:00
|
|
|
comp->Action = msi_get_component_action( package, comp );
|
|
|
|
if (comp->Action != INSTALLSTATE_ABSENT)
|
2011-01-24 15:22:11 +01:00
|
|
|
{
|
2011-05-25 10:39:11 +02:00
|
|
|
TRACE("component not scheduled for removal %s\n", debugstr_w(comp->Component));
|
2010-11-11 10:45:42 +01:00
|
|
|
continue;
|
2011-01-24 15:22:11 +01:00
|
|
|
}
|
|
|
|
TRACE("unpublishing %s\n", debugstr_w(comp->Component));
|
|
|
|
|
|
|
|
win32 = assembly->attributes & msidbAssemblyAttributesWin32;
|
|
|
|
if (assembly->application)
|
|
|
|
{
|
2011-05-06 14:39:17 +02:00
|
|
|
MSIFILE *file = msi_get_loaded_file( package, assembly->application );
|
2018-09-04 02:17:56 +02:00
|
|
|
if (!file)
|
|
|
|
{
|
|
|
|
WARN("no matching file %s for local assembly\n", debugstr_w(assembly->application));
|
|
|
|
continue;
|
|
|
|
}
|
2011-01-24 15:22:11 +01:00
|
|
|
if ((res = delete_local_assembly_key( package->Context, win32, file->TargetPath )))
|
|
|
|
WARN("failed to delete local assembly key %d\n", res);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
HKEY hkey;
|
|
|
|
if ((res = open_global_assembly_key( package->Context, win32, &hkey )))
|
|
|
|
WARN("failed to delete global assembly key %d\n", res);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((res = RegDeleteValueW( hkey, assembly->display_name )))
|
|
|
|
WARN("failed to delete global assembly value %d\n", res);
|
|
|
|
RegCloseKey( hkey );
|
|
|
|
}
|
|
|
|
}
|
2010-11-11 10:45:42 +01:00
|
|
|
|
|
|
|
uirow = MSI_CreateRecord( 2 );
|
2011-01-24 15:22:11 +01:00
|
|
|
MSI_RecordSetStringW( uirow, 2, assembly->display_name );
|
2017-07-07 16:03:15 +02:00
|
|
|
MSI_ProcessMessage(package, INSTALLMESSAGE_ACTIONDATA, uirow);
|
2010-11-11 10:45:42 +01:00
|
|
|
msiobj_release( &uirow->hdr );
|
|
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|