blob: 91629ef16d50dc3a2574850c764470202d1bee0e [file] [log] [blame]
Mike McCormack6386edc2003-08-13 01:27:48 +00001/*
2 * Implementation of the Microsoft Installer (msi.dll)
3 *
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +00004 * Copyright 2002-2004 Mike McCormack for CodeWeavers
Mike McCormack6386edc2003-08-13 01:27:48 +00005 *
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.
10 *
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.
15 *
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
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Mike McCormack6386edc2003-08-13 01:27:48 +000019 */
20
James Hawkins46158e02007-06-26 17:02:06 -070021#define COBJMACROS
22
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000023#include <stdarg.h>
24
Mike McCormack6386edc2003-08-13 01:27:48 +000025#include "windef.h"
26#include "winbase.h"
27#include "winreg.h"
28#include "shlwapi.h"
29#include "wine/debug.h"
30#include "msi.h"
31#include "msiquery.h"
32#include "msipriv.h"
33
34WINE_DEFAULT_DEBUG_CHANNEL(msi);
35
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000036static CRITICAL_SECTION MSI_handle_cs;
37static CRITICAL_SECTION_DEBUG MSI_handle_cs_debug =
Mike McCormack6386edc2003-08-13 01:27:48 +000038{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000039 0, 0, &MSI_handle_cs,
40 { &MSI_handle_cs_debug.ProcessLocksList,
41 &MSI_handle_cs_debug.ProcessLocksList },
Alexandre Julliard20a1a202005-09-09 10:19:44 +000042 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_handle_cs") }
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000043};
44static CRITICAL_SECTION MSI_handle_cs = { &MSI_handle_cs_debug, -1, 0, 0, 0, 0 };
45
Mike McCormack0f0b6c82004-12-27 19:29:33 +000046static CRITICAL_SECTION MSI_object_cs;
47static CRITICAL_SECTION_DEBUG MSI_object_cs_debug =
48{
49 0, 0, &MSI_object_cs,
50 { &MSI_object_cs_debug.ProcessLocksList,
51 &MSI_object_cs_debug.ProcessLocksList },
Alexandre Julliard20a1a202005-09-09 10:19:44 +000052 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_object_cs") }
Mike McCormack0f0b6c82004-12-27 19:29:33 +000053};
54static CRITICAL_SECTION MSI_object_cs = { &MSI_object_cs_debug, -1, 0, 0, 0, 0 };
55
Mike McCormack89640822005-01-06 20:43:04 +000056typedef struct msi_handle_info_t
57{
James Hawkins46158e02007-06-26 17:02:06 -070058 BOOL remote;
59 union {
60 MSIOBJECTHDR *obj;
61 IUnknown *unk;
62 } u;
Mike McCormack89640822005-01-06 20:43:04 +000063 DWORD dwThreadId;
64} msi_handle_info;
65
Dan Kegel29f08032006-08-28 07:51:30 -070066static msi_handle_info *msihandletable = NULL;
Alexandre Julliardbb8ba382007-06-27 14:14:49 +020067static unsigned int msihandletable_size = 0;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000068
Mike McCormack155a3252006-08-29 17:08:01 +090069void msi_free_handle_table(void)
70{
71 msi_free( msihandletable );
72 msihandletable = NULL;
73 msihandletable_size = 0;
74}
75
James Hawkins46158e02007-06-26 17:02:06 -070076static MSIHANDLE alloc_handle_table_entry(void)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +000077{
Mike McCormack6386edc2003-08-13 01:27:48 +000078 UINT i;
79
80 /* find a slot */
Dan Kegel29f08032006-08-28 07:51:30 -070081 for(i=0; i<msihandletable_size; i++)
James Hawkins46158e02007-06-26 17:02:06 -070082 if( !msihandletable[i].u.obj && !msihandletable[i].u.unk )
Mike McCormack6386edc2003-08-13 01:27:48 +000083 break;
Dan Kegel29f08032006-08-28 07:51:30 -070084 if( i==msihandletable_size )
85 {
86 msi_handle_info *p;
87 int newsize;
88 if (msihandletable_size == 0)
89 {
90 newsize = 256;
91 p = msi_alloc_zero(newsize*sizeof(msi_handle_info));
92 }
Mike McCormack155a3252006-08-29 17:08:01 +090093 else
94 {
Dan Kegel29f08032006-08-28 07:51:30 -070095 newsize = msihandletable_size * 2;
96 p = msi_realloc_zero(msihandletable,
97 newsize*sizeof(msi_handle_info));
Mike McCormack155a3252006-08-29 17:08:01 +090098 }
Dan Kegel29f08032006-08-28 07:51:30 -070099 if (!p)
James Hawkins46158e02007-06-26 17:02:06 -0700100 return 0;
Dan Kegel29f08032006-08-28 07:51:30 -0700101 msihandletable = p;
102 msihandletable_size = newsize;
103 }
James Hawkins46158e02007-06-26 17:02:06 -0700104 return i + 1;
105}
Mike McCormack6386edc2003-08-13 01:27:48 +0000106
James Hawkins46158e02007-06-26 17:02:06 -0700107MSIHANDLE alloc_msihandle( MSIOBJECTHDR *obj )
108{
109 msi_handle_info *entry;
110 MSIHANDLE ret;
111
112 EnterCriticalSection( &MSI_handle_cs );
113
114 ret = alloc_handle_table_entry();
115 if (ret)
116 {
117 entry = &msihandletable[ ret - 1 ];
118 msiobj_addref( obj );
119 entry->u.obj = obj;
120 entry->dwThreadId = GetCurrentThreadId();
121 entry->remote = FALSE;
122 }
Mike McCormack6386edc2003-08-13 01:27:48 +0000123
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000124 LeaveCriticalSection( &MSI_handle_cs );
James Hawkins46158e02007-06-26 17:02:06 -0700125
Michael Stefaniuc6bd893a2009-01-04 14:14:15 +0100126 TRACE("%p -> %d\n", obj, ret );
James Hawkins46158e02007-06-26 17:02:06 -0700127
128 return ret;
129}
130
131MSIHANDLE alloc_msi_remote_handle( IUnknown *unk )
132{
133 msi_handle_info *entry;
134 MSIHANDLE ret;
135
136 EnterCriticalSection( &MSI_handle_cs );
137
138 ret = alloc_handle_table_entry();
139 if (ret)
140 {
141 entry = &msihandletable[ ret - 1 ];
142 IUnknown_AddRef( unk );
143 entry->u.unk = unk;
144 entry->dwThreadId = GetCurrentThreadId();
145 entry->remote = TRUE;
146 }
147
148 LeaveCriticalSection( &MSI_handle_cs );
149
Michael Stefaniuc6bd893a2009-01-04 14:14:15 +0100150 TRACE("%p -> %d\n", unk, ret);
James Hawkins46158e02007-06-26 17:02:06 -0700151
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000152 return ret;
Mike McCormack6386edc2003-08-13 01:27:48 +0000153}
154
155void *msihandle2msiinfo(MSIHANDLE handle, UINT type)
156{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000157 MSIOBJECTHDR *ret = NULL;
158
159 EnterCriticalSection( &MSI_handle_cs );
Mike McCormack6386edc2003-08-13 01:27:48 +0000160 handle--;
James Hawkins46158e02007-06-26 17:02:06 -0700161 if( handle >= msihandletable_size )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000162 goto out;
James Hawkins46158e02007-06-26 17:02:06 -0700163 if( msihandletable[handle].remote)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000164 goto out;
James Hawkins46158e02007-06-26 17:02:06 -0700165 if( !msihandletable[handle].u.obj )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000166 goto out;
James Hawkins46158e02007-06-26 17:02:06 -0700167 if( msihandletable[handle].u.obj->magic != MSIHANDLE_MAGIC )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000168 goto out;
James Hawkins46158e02007-06-26 17:02:06 -0700169 if( type && (msihandletable[handle].u.obj->type != type) )
170 goto out;
171 ret = msihandletable[handle].u.obj;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000172 msiobj_addref( ret );
Mike McCormack3f2cedd2006-11-01 15:09:02 +0900173
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000174out:
175 LeaveCriticalSection( &MSI_handle_cs );
Mike McCormack6386edc2003-08-13 01:27:48 +0000176
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000177 return (void*) ret;
Mike McCormack6386edc2003-08-13 01:27:48 +0000178}
179
James Hawkins46158e02007-06-26 17:02:06 -0700180IUnknown *msi_get_remote( MSIHANDLE handle )
181{
182 IUnknown *unk = NULL;
183
184 EnterCriticalSection( &MSI_handle_cs );
185 handle--;
James Hawkins46158e02007-06-26 17:02:06 -0700186 if( handle>=msihandletable_size )
187 goto out;
188 if( !msihandletable[handle].remote)
189 goto out;
190 unk = msihandletable[handle].u.unk;
191 if( unk )
192 IUnknown_AddRef( unk );
193
194out:
195 LeaveCriticalSection( &MSI_handle_cs );
196
197 return unk;
198}
199
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000200void *alloc_msiobject(UINT type, UINT size, msihandledestructor destroy )
201{
202 MSIOBJECTHDR *info;
203
Mike McCormack8dc28d52005-09-20 11:57:19 +0000204 info = msi_alloc_zero( size );
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000205 if( info )
206 {
207 info->magic = MSIHANDLE_MAGIC;
208 info->type = type;
209 info->refcount = 1;
210 info->destructor = destroy;
211 }
212
213 return info;
214}
215
216void msiobj_addref( MSIOBJECTHDR *info )
217{
Aric Stewart401bd3f2004-06-28 20:34:35 +0000218 if( !info )
219 return;
220
Aric Stewart401bd3f2004-06-28 20:34:35 +0000221 if( info->magic != MSIHANDLE_MAGIC )
222 {
223 ERR("Invalid handle!\n");
224 return;
225 }
226
Mike McCormackb3a7f372005-01-20 20:34:29 +0000227 InterlockedIncrement(&info->refcount);
Aric Stewart401bd3f2004-06-28 20:34:35 +0000228}
229
Mike McCormack0f0b6c82004-12-27 19:29:33 +0000230void msiobj_lock( MSIOBJECTHDR *info )
231{
232 EnterCriticalSection( &MSI_object_cs );
233}
234
235void msiobj_unlock( MSIOBJECTHDR *info )
236{
237 LeaveCriticalSection( &MSI_object_cs );
238}
239
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000240int msiobj_release( MSIOBJECTHDR *info )
Mike McCormack6386edc2003-08-13 01:27:48 +0000241{
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000242 int ret;
Mike McCormack6386edc2003-08-13 01:27:48 +0000243
Mike McCormack6386edc2003-08-13 01:27:48 +0000244 if( !info )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000245 return -1;
Mike McCormack6386edc2003-08-13 01:27:48 +0000246
247 if( info->magic != MSIHANDLE_MAGIC )
248 {
249 ERR("Invalid handle!\n");
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000250 return -1;
Mike McCormack6386edc2003-08-13 01:27:48 +0000251 }
252
Mike McCormackb3a7f372005-01-20 20:34:29 +0000253 ret = InterlockedDecrement( &info->refcount );
254 if( ret==0 )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000255 {
Mike McCormackb3a7f372005-01-20 20:34:29 +0000256 if( info->destructor )
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000257 info->destructor( info );
Mike McCormack8dc28d52005-09-20 11:57:19 +0000258 msi_free( info );
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000259 TRACE("object %p destroyed\n", info);
260 }
261
262 return ret;
263}
264
Mike McCormack89640822005-01-06 20:43:04 +0000265/***********************************************************
266 * MsiCloseHandle [MSI.@]
267 */
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000268UINT WINAPI MsiCloseHandle(MSIHANDLE handle)
269{
James Hawkins46158e02007-06-26 17:02:06 -0700270 MSIOBJECTHDR *info = NULL;
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000271 UINT ret = ERROR_INVALID_HANDLE;
272
Michael Stefaniuc6bd893a2009-01-04 14:14:15 +0100273 TRACE("%x\n",handle);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000274
Mike McCormacka21f9e52006-06-11 09:51:42 +0900275 if (!handle)
276 return ERROR_SUCCESS;
277
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000278 EnterCriticalSection( &MSI_handle_cs );
279
Alexandre Julliardbb8ba382007-06-27 14:14:49 +0200280 handle--;
281 if (handle >= msihandletable_size)
282 goto out;
283
284 if (msihandletable[handle].remote)
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000285 {
Alexandre Julliardbb8ba382007-06-27 14:14:49 +0200286 IUnknown_Release( msihandletable[handle].u.unk );
James Hawkins46158e02007-06-26 17:02:06 -0700287 }
288 else
289 {
Alexandre Julliardbb8ba382007-06-27 14:14:49 +0200290 info = msihandletable[handle].u.obj;
James Hawkins46158e02007-06-26 17:02:06 -0700291 if( !info )
292 goto out;
293
294 if( info->magic != MSIHANDLE_MAGIC )
295 {
296 ERR("Invalid handle!\n");
297 goto out;
298 }
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000299 }
300
Alexandre Julliardbb8ba382007-06-27 14:14:49 +0200301 msihandletable[handle].u.obj = NULL;
302 msihandletable[handle].remote = 0;
303 msihandletable[handle].dwThreadId = 0;
James Hawkins46158e02007-06-26 17:02:06 -0700304
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000305 ret = ERROR_SUCCESS;
Mike McCormack6386edc2003-08-13 01:27:48 +0000306
Michael Stefaniuc6bd893a2009-01-04 14:14:15 +0100307 TRACE("handle %x destroyed\n", handle+1);
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000308out:
309 LeaveCriticalSection( &MSI_handle_cs );
310 if( info )
311 msiobj_release( info );
Mike McCormack6386edc2003-08-13 01:27:48 +0000312
Alexandre Julliarda7a6f5f2004-07-09 22:25:34 +0000313 return ret;
Mike McCormack6386edc2003-08-13 01:27:48 +0000314}
315
Mike McCormack89640822005-01-06 20:43:04 +0000316/***********************************************************
317 * MsiCloseAllHandles [MSI.@]
318 *
319 * Closes all handles owned by the current thread
320 *
321 * RETURNS:
322 * The number of handles closed
323 */
Mike McCormack6386edc2003-08-13 01:27:48 +0000324UINT WINAPI MsiCloseAllHandles(void)
325{
Mike McCormack89640822005-01-06 20:43:04 +0000326 UINT i, n=0;
Mike McCormack6386edc2003-08-13 01:27:48 +0000327
328 TRACE("\n");
329
Dan Kegel29f08032006-08-28 07:51:30 -0700330 EnterCriticalSection( &MSI_handle_cs );
331 for(i=0; i<msihandletable_size; i++)
Mike McCormack89640822005-01-06 20:43:04 +0000332 {
333 if(msihandletable[i].dwThreadId == GetCurrentThreadId())
334 {
Dan Kegel29f08032006-08-28 07:51:30 -0700335 LeaveCriticalSection( &MSI_handle_cs );
Mike McCormack89640822005-01-06 20:43:04 +0000336 MsiCloseHandle( i+1 );
Dan Kegel29f08032006-08-28 07:51:30 -0700337 EnterCriticalSection( &MSI_handle_cs );
Mike McCormack89640822005-01-06 20:43:04 +0000338 n++;
339 }
340 }
Dan Kegel29f08032006-08-28 07:51:30 -0700341 LeaveCriticalSection( &MSI_handle_cs );
Mike McCormack6386edc2003-08-13 01:27:48 +0000342
Mike McCormack89640822005-01-06 20:43:04 +0000343 return n;
Mike McCormack6386edc2003-08-13 01:27:48 +0000344}