mstask: Implement ITask::DeleteTrigger().
[wine.git] / dlls / msi / handle.c
blob1002f9d969bd6d772e8a4fd6a774fe5173f9a246
1 /*
2 * Implementation of the Microsoft Installer (msi.dll)
4 * Copyright 2002-2004 Mike McCormack for CodeWeavers
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
21 #define COBJMACROS
23 #include <stdarg.h>
25 #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"
33 #include "msipriv.h"
34 #include "winemsi.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(msi);
38 static CRITICAL_SECTION MSI_handle_cs;
39 static CRITICAL_SECTION_DEBUG MSI_handle_cs_debug =
41 0, 0, &MSI_handle_cs,
42 { &MSI_handle_cs_debug.ProcessLocksList,
43 &MSI_handle_cs_debug.ProcessLocksList },
44 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_handle_cs") }
46 static CRITICAL_SECTION MSI_handle_cs = { &MSI_handle_cs_debug, -1, 0, 0, 0, 0 };
48 static CRITICAL_SECTION MSI_object_cs;
49 static CRITICAL_SECTION_DEBUG MSI_object_cs_debug =
51 0, 0, &MSI_object_cs,
52 { &MSI_object_cs_debug.ProcessLocksList,
53 &MSI_object_cs_debug.ProcessLocksList },
54 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_object_cs") }
56 static CRITICAL_SECTION MSI_object_cs = { &MSI_object_cs_debug, -1, 0, 0, 0, 0 };
58 typedef struct msi_handle_info_t
60 BOOL remote;
61 union {
62 MSIOBJECTHDR *obj;
63 MSIHANDLE rem;
64 } u;
65 DWORD dwThreadId;
66 } msi_handle_info;
68 static msi_handle_info *msihandletable = NULL;
69 static unsigned int msihandletable_size = 0;
71 void msi_free_handle_table(void)
73 msi_free( msihandletable );
74 msihandletable = NULL;
75 msihandletable_size = 0;
76 DeleteCriticalSection(&MSI_handle_cs);
77 DeleteCriticalSection(&MSI_object_cs);
80 static MSIHANDLE alloc_handle_table_entry(void)
82 UINT i;
84 /* find a slot */
85 for(i=0; i<msihandletable_size; i++)
86 if( !msihandletable[i].u.obj && !msihandletable[i].u.rem )
87 break;
88 if( i==msihandletable_size )
90 msi_handle_info *p;
91 int newsize;
92 if (msihandletable_size == 0)
94 newsize = 256;
95 p = msi_alloc_zero(newsize*sizeof(msi_handle_info));
97 else
99 newsize = msihandletable_size * 2;
100 p = msi_realloc_zero(msihandletable,
101 newsize*sizeof(msi_handle_info));
103 if (!p)
104 return 0;
105 msihandletable = p;
106 msihandletable_size = newsize;
108 return i + 1;
111 MSIHANDLE alloc_msihandle( MSIOBJECTHDR *obj )
113 msi_handle_info *entry;
114 MSIHANDLE ret;
116 EnterCriticalSection( &MSI_handle_cs );
118 ret = alloc_handle_table_entry();
119 if (ret)
121 entry = &msihandletable[ ret - 1 ];
122 msiobj_addref( obj );
123 entry->u.obj = obj;
124 entry->dwThreadId = GetCurrentThreadId();
125 entry->remote = FALSE;
128 LeaveCriticalSection( &MSI_handle_cs );
130 TRACE("%p -> %d\n", obj, ret );
132 return ret;
135 MSIHANDLE alloc_msi_remote_handle(MSIHANDLE remote)
137 msi_handle_info *entry;
138 MSIHANDLE ret;
140 EnterCriticalSection( &MSI_handle_cs );
142 ret = alloc_handle_table_entry();
143 if (ret)
145 entry = &msihandletable[ ret - 1 ];
146 entry->u.rem = remote;
147 entry->dwThreadId = GetCurrentThreadId();
148 entry->remote = TRUE;
151 LeaveCriticalSection( &MSI_handle_cs );
153 TRACE("%d -> %d\n", remote, ret);
155 return ret;
158 void *msihandle2msiinfo(MSIHANDLE handle, UINT type)
160 MSIOBJECTHDR *ret = NULL;
162 EnterCriticalSection( &MSI_handle_cs );
163 handle--;
164 if( handle >= msihandletable_size )
165 goto out;
166 if( msihandletable[handle].remote)
167 goto out;
168 if( !msihandletable[handle].u.obj )
169 goto out;
170 if( msihandletable[handle].u.obj->magic != MSIHANDLE_MAGIC )
171 goto out;
172 if( type && (msihandletable[handle].u.obj->type != type) )
173 goto out;
174 ret = msihandletable[handle].u.obj;
175 msiobj_addref( ret );
177 out:
178 LeaveCriticalSection( &MSI_handle_cs );
180 return ret;
183 MSIHANDLE msi_get_remote( MSIHANDLE handle )
185 MSIHANDLE ret = 0;
187 EnterCriticalSection( &MSI_handle_cs );
188 handle--;
189 if( handle>=msihandletable_size )
190 goto out;
191 if( !msihandletable[handle].remote)
192 goto out;
193 ret = msihandletable[handle].u.rem;
195 out:
196 LeaveCriticalSection( &MSI_handle_cs );
198 return ret;
201 void *alloc_msiobject(UINT type, UINT size, msihandledestructor destroy )
203 MSIOBJECTHDR *info;
205 info = msi_alloc_zero( size );
206 if( info )
208 info->magic = MSIHANDLE_MAGIC;
209 info->type = type;
210 info->refcount = 1;
211 info->destructor = destroy;
214 return info;
217 void msiobj_addref( MSIOBJECTHDR *info )
219 if( !info )
220 return;
222 if( info->magic != MSIHANDLE_MAGIC )
224 ERR("Invalid handle!\n");
225 return;
228 InterlockedIncrement(&info->refcount);
231 void msiobj_lock( MSIOBJECTHDR *info )
233 EnterCriticalSection( &MSI_object_cs );
236 void msiobj_unlock( MSIOBJECTHDR *info )
238 LeaveCriticalSection( &MSI_object_cs );
241 int msiobj_release( MSIOBJECTHDR *info )
243 int ret;
245 if( !info )
246 return -1;
248 if( info->magic != MSIHANDLE_MAGIC )
250 ERR("Invalid handle!\n");
251 return -1;
254 ret = InterlockedDecrement( &info->refcount );
255 if( ret==0 )
257 if( info->destructor )
258 info->destructor( info );
259 msi_free( info );
260 TRACE("object %p destroyed\n", info);
263 return ret;
266 /***********************************************************
267 * MsiCloseHandle [MSI.@]
269 UINT WINAPI MsiCloseHandle(MSIHANDLE handle)
271 MSIOBJECTHDR *info = NULL;
272 UINT ret = ERROR_INVALID_HANDLE;
274 TRACE("%x\n",handle);
276 if (!handle)
277 return ERROR_SUCCESS;
279 EnterCriticalSection( &MSI_handle_cs );
281 handle--;
282 if (handle >= msihandletable_size)
283 goto out;
285 if (msihandletable[handle].remote)
287 remote_CloseHandle( msihandletable[handle].u.rem );
289 else
291 info = msihandletable[handle].u.obj;
292 if( !info )
293 goto out;
295 if( info->magic != MSIHANDLE_MAGIC )
297 ERR("Invalid handle!\n");
298 goto out;
302 msihandletable[handle].u.obj = NULL;
303 msihandletable[handle].remote = 0;
304 msihandletable[handle].dwThreadId = 0;
306 ret = ERROR_SUCCESS;
308 TRACE("handle %x destroyed\n", handle+1);
309 out:
310 LeaveCriticalSection( &MSI_handle_cs );
311 if( info )
312 msiobj_release( info );
314 return ret;
317 /***********************************************************
318 * MsiCloseAllHandles [MSI.@]
320 * Closes all handles owned by the current thread
322 * RETURNS:
323 * The number of handles closed
325 UINT WINAPI MsiCloseAllHandles(void)
327 UINT i, n=0;
329 TRACE("\n");
331 EnterCriticalSection( &MSI_handle_cs );
332 for(i=0; i<msihandletable_size; i++)
334 if(msihandletable[i].dwThreadId == GetCurrentThreadId())
336 LeaveCriticalSection( &MSI_handle_cs );
337 MsiCloseHandle( i+1 );
338 EnterCriticalSection( &MSI_handle_cs );
339 n++;
342 LeaveCriticalSection( &MSI_handle_cs );
344 return n;
347 UINT __cdecl remote_CloseHandle(MSIHANDLE handle)
349 return MsiCloseHandle(handle);