usp10: Free default_language items when freeing script cache.
[wine.git] / dlls / msi / handle.c
blob80c6873981626b52b64eb6cd836c2e3b6d189f45
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"
32 #include "msipriv.h"
34 WINE_DEFAULT_DEBUG_CHANNEL(msi);
36 static CRITICAL_SECTION MSI_handle_cs;
37 static CRITICAL_SECTION_DEBUG MSI_handle_cs_debug =
39 0, 0, &MSI_handle_cs,
40 { &MSI_handle_cs_debug.ProcessLocksList,
41 &MSI_handle_cs_debug.ProcessLocksList },
42 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_handle_cs") }
44 static CRITICAL_SECTION MSI_handle_cs = { &MSI_handle_cs_debug, -1, 0, 0, 0, 0 };
46 static CRITICAL_SECTION MSI_object_cs;
47 static CRITICAL_SECTION_DEBUG MSI_object_cs_debug =
49 0, 0, &MSI_object_cs,
50 { &MSI_object_cs_debug.ProcessLocksList,
51 &MSI_object_cs_debug.ProcessLocksList },
52 0, 0, { (DWORD_PTR)(__FILE__ ": MSI_object_cs") }
54 static CRITICAL_SECTION MSI_object_cs = { &MSI_object_cs_debug, -1, 0, 0, 0, 0 };
56 typedef struct msi_handle_info_t
58 BOOL remote;
59 union {
60 MSIOBJECTHDR *obj;
61 IUnknown *unk;
62 } u;
63 DWORD dwThreadId;
64 } msi_handle_info;
66 static msi_handle_info *msihandletable = NULL;
67 static unsigned int msihandletable_size = 0;
69 void msi_free_handle_table(void)
71 msi_free( msihandletable );
72 msihandletable = NULL;
73 msihandletable_size = 0;
74 DeleteCriticalSection(&MSI_handle_cs);
75 DeleteCriticalSection(&MSI_object_cs);
78 static MSIHANDLE alloc_handle_table_entry(void)
80 UINT i;
82 /* find a slot */
83 for(i=0; i<msihandletable_size; i++)
84 if( !msihandletable[i].u.obj && !msihandletable[i].u.unk )
85 break;
86 if( i==msihandletable_size )
88 msi_handle_info *p;
89 int newsize;
90 if (msihandletable_size == 0)
92 newsize = 256;
93 p = msi_alloc_zero(newsize*sizeof(msi_handle_info));
95 else
97 newsize = msihandletable_size * 2;
98 p = msi_realloc_zero(msihandletable,
99 newsize*sizeof(msi_handle_info));
101 if (!p)
102 return 0;
103 msihandletable = p;
104 msihandletable_size = newsize;
106 return i + 1;
109 MSIHANDLE alloc_msihandle( MSIOBJECTHDR *obj )
111 msi_handle_info *entry;
112 MSIHANDLE ret;
114 EnterCriticalSection( &MSI_handle_cs );
116 ret = alloc_handle_table_entry();
117 if (ret)
119 entry = &msihandletable[ ret - 1 ];
120 msiobj_addref( obj );
121 entry->u.obj = obj;
122 entry->dwThreadId = GetCurrentThreadId();
123 entry->remote = FALSE;
126 LeaveCriticalSection( &MSI_handle_cs );
128 TRACE("%p -> %d\n", obj, ret );
130 return ret;
133 MSIHANDLE alloc_msi_remote_handle( IUnknown *unk )
135 msi_handle_info *entry;
136 MSIHANDLE ret;
138 EnterCriticalSection( &MSI_handle_cs );
140 ret = alloc_handle_table_entry();
141 if (ret)
143 entry = &msihandletable[ ret - 1 ];
144 IUnknown_AddRef( unk );
145 entry->u.unk = unk;
146 entry->dwThreadId = GetCurrentThreadId();
147 entry->remote = TRUE;
150 LeaveCriticalSection( &MSI_handle_cs );
152 TRACE("%p -> %d\n", unk, ret);
154 return ret;
157 void *msihandle2msiinfo(MSIHANDLE handle, UINT type)
159 MSIOBJECTHDR *ret = NULL;
161 EnterCriticalSection( &MSI_handle_cs );
162 handle--;
163 if( handle >= msihandletable_size )
164 goto out;
165 if( msihandletable[handle].remote)
166 goto out;
167 if( !msihandletable[handle].u.obj )
168 goto out;
169 if( msihandletable[handle].u.obj->magic != MSIHANDLE_MAGIC )
170 goto out;
171 if( type && (msihandletable[handle].u.obj->type != type) )
172 goto out;
173 ret = msihandletable[handle].u.obj;
174 msiobj_addref( ret );
176 out:
177 LeaveCriticalSection( &MSI_handle_cs );
179 return ret;
182 IUnknown *msi_get_remote( MSIHANDLE handle )
184 IUnknown *unk = NULL;
186 EnterCriticalSection( &MSI_handle_cs );
187 handle--;
188 if( handle>=msihandletable_size )
189 goto out;
190 if( !msihandletable[handle].remote)
191 goto out;
192 unk = msihandletable[handle].u.unk;
193 if( unk )
194 IUnknown_AddRef( unk );
196 out:
197 LeaveCriticalSection( &MSI_handle_cs );
199 return unk;
202 void *alloc_msiobject(UINT type, UINT size, msihandledestructor destroy )
204 MSIOBJECTHDR *info;
206 info = msi_alloc_zero( size );
207 if( info )
209 info->magic = MSIHANDLE_MAGIC;
210 info->type = type;
211 info->refcount = 1;
212 info->destructor = destroy;
215 return info;
218 void msiobj_addref( MSIOBJECTHDR *info )
220 if( !info )
221 return;
223 if( info->magic != MSIHANDLE_MAGIC )
225 ERR("Invalid handle!\n");
226 return;
229 InterlockedIncrement(&info->refcount);
232 void msiobj_lock( MSIOBJECTHDR *info )
234 EnterCriticalSection( &MSI_object_cs );
237 void msiobj_unlock( MSIOBJECTHDR *info )
239 LeaveCriticalSection( &MSI_object_cs );
242 int msiobj_release( MSIOBJECTHDR *info )
244 int ret;
246 if( !info )
247 return -1;
249 if( info->magic != MSIHANDLE_MAGIC )
251 ERR("Invalid handle!\n");
252 return -1;
255 ret = InterlockedDecrement( &info->refcount );
256 if( ret==0 )
258 if( info->destructor )
259 info->destructor( info );
260 msi_free( info );
261 TRACE("object %p destroyed\n", info);
264 return ret;
267 /***********************************************************
268 * MsiCloseHandle [MSI.@]
270 UINT WINAPI MsiCloseHandle(MSIHANDLE handle)
272 MSIOBJECTHDR *info = NULL;
273 UINT ret = ERROR_INVALID_HANDLE;
275 TRACE("%x\n",handle);
277 if (!handle)
278 return ERROR_SUCCESS;
280 EnterCriticalSection( &MSI_handle_cs );
282 handle--;
283 if (handle >= msihandletable_size)
284 goto out;
286 if (msihandletable[handle].remote)
288 IUnknown_Release( msihandletable[handle].u.unk );
290 else
292 info = msihandletable[handle].u.obj;
293 if( !info )
294 goto out;
296 if( info->magic != MSIHANDLE_MAGIC )
298 ERR("Invalid handle!\n");
299 goto out;
303 msihandletable[handle].u.obj = NULL;
304 msihandletable[handle].remote = 0;
305 msihandletable[handle].dwThreadId = 0;
307 ret = ERROR_SUCCESS;
309 TRACE("handle %x destroyed\n", handle+1);
310 out:
311 LeaveCriticalSection( &MSI_handle_cs );
312 if( info )
313 msiobj_release( info );
315 return ret;
318 /***********************************************************
319 * MsiCloseAllHandles [MSI.@]
321 * Closes all handles owned by the current thread
323 * RETURNS:
324 * The number of handles closed
326 UINT WINAPI MsiCloseAllHandles(void)
328 UINT i, n=0;
330 TRACE("\n");
332 EnterCriticalSection( &MSI_handle_cs );
333 for(i=0; i<msihandletable_size; i++)
335 if(msihandletable[i].dwThreadId == GetCurrentThreadId())
337 LeaveCriticalSection( &MSI_handle_cs );
338 MsiCloseHandle( i+1 );
339 EnterCriticalSection( &MSI_handle_cs );
340 n++;
343 LeaveCriticalSection( &MSI_handle_cs );
345 return n;