2 * handles-private.h: Internal operations on handles
5 * Dick Porter (dick@ximian.com)
7 * (C) 2002-2006 Novell, Inc.
10 #ifndef _WAPI_HANDLES_PRIVATE_H_
11 #define _WAPI_HANDLES_PRIVATE_H_
17 #include <sys/types.h>
19 #include <mono/io-layer/wapi-private.h>
20 #include <mono/io-layer/collection.h>
21 #include <mono/io-layer/shared.h>
22 #include <mono/utils/atomic.h>
24 #define _WAPI_PRIVATE_MAX_SLOTS (1024 * 16)
25 #define _WAPI_PRIVATE_HANDLES(x) (_wapi_private_handles [x / _WAPI_HANDLE_INITIAL_COUNT][x % _WAPI_HANDLE_INITIAL_COUNT])
26 #define _WAPI_PRIVATE_HAVE_SLOT(x) ((GPOINTER_TO_UINT (x) / _WAPI_PRIVATE_MAX_SLOTS) < _WAPI_PRIVATE_MAX_SLOTS && \
27 _wapi_private_handles [GPOINTER_TO_UINT (x) / _WAPI_HANDLE_INITIAL_COUNT] != NULL)
28 #define _WAPI_PRIVATE_VALID_SLOT(x) (((x) / _WAPI_HANDLE_INITIAL_COUNT) < _WAPI_PRIVATE_MAX_SLOTS)
32 extern struct _WapiHandleUnshared *_wapi_private_handles [];
33 extern struct _WapiHandleSharedLayout *_wapi_shared_layout;
34 extern struct _WapiFileShareLayout *_wapi_fileshare_layout;
36 extern guint32 _wapi_fd_reserve;
37 extern mono_mutex_t *_wapi_global_signal_mutex;
38 extern pthread_cond_t *_wapi_global_signal_cond;
39 extern int _wapi_sem_id;
40 extern gboolean _wapi_has_shut_down;
42 extern pid_t _wapi_getpid (void);
43 extern gpointer _wapi_handle_new (WapiHandleType type,
44 gpointer handle_specific);
45 extern gpointer _wapi_handle_new_fd (WapiHandleType type, int fd,
46 gpointer handle_specific);
47 extern gpointer _wapi_handle_new_from_offset (WapiHandleType type,
50 extern gboolean _wapi_lookup_handle (gpointer handle, WapiHandleType type,
51 gpointer *handle_specific);
52 extern gpointer _wapi_search_handle (WapiHandleType type,
53 gboolean (*check)(gpointer, gpointer),
55 gpointer *handle_specific,
56 gboolean search_shared);
57 extern gint32 _wapi_search_handle_namespace (WapiHandleType type,
59 extern void _wapi_handle_ref (gpointer handle);
60 extern void _wapi_handle_unref (gpointer handle);
61 extern void _wapi_handle_register_capabilities (WapiHandleType type,
62 WapiHandleCapability caps);
63 extern gboolean _wapi_handle_test_capabilities (gpointer handle,
64 WapiHandleCapability caps);
65 extern void _wapi_handle_ops_close (gpointer handle, gpointer data);
66 extern void _wapi_handle_ops_signal (gpointer handle);
67 extern gboolean _wapi_handle_ops_own (gpointer handle);
68 extern gboolean _wapi_handle_ops_isowned (gpointer handle);
69 extern guint32 _wapi_handle_ops_special_wait (gpointer handle,
72 extern void _wapi_handle_ops_prewait (gpointer handle);
74 extern gboolean _wapi_handle_count_signalled_handles (guint32 numhandles,
79 extern void _wapi_handle_unlock_handles (guint32 numhandles,
81 extern int _wapi_handle_timedwait_signal (guint32 timeout, gboolean poll, gboolean *alerted);
82 extern int _wapi_handle_timedwait_signal_handle (gpointer handle, guint32 timeout, gboolean alertable, gboolean poll, gboolean *alerted);
83 extern gboolean _wapi_handle_get_or_set_share (guint64 device, guint64 inode,
84 guint32 new_sharemode,
86 guint32 *old_sharemode,
88 struct _WapiFileShare **info);
89 extern void _wapi_handle_check_share (struct _WapiFileShare *share_info,
91 extern void _wapi_handle_dump (void);
92 extern void _wapi_handle_update_refs (void);
93 extern void _wapi_handle_foreach (WapiHandleType type,
94 gboolean (*on_each)(gpointer test, gpointer user),
96 void _wapi_free_share_info (_WapiFileShare *share_info);
98 /* This is OK to use for atomic writes of individual struct members, as they
101 #define WAPI_SHARED_HANDLE_DATA(handle) _wapi_shared_layout->handles[_WAPI_PRIVATE_HANDLES(GPOINTER_TO_UINT((handle))).u.shared.offset]
103 #define WAPI_SHARED_HANDLE_TYPED_DATA(handle, type) _wapi_shared_layout->handles[_WAPI_PRIVATE_HANDLES(GPOINTER_TO_UINT((handle))).u.shared.offset].u.type
105 static inline WapiHandleType _wapi_handle_type (gpointer handle)
107 guint32 idx = GPOINTER_TO_UINT(handle);
109 if (!_WAPI_PRIVATE_VALID_SLOT (idx) || !_WAPI_PRIVATE_HAVE_SLOT (idx)) {
110 return(WAPI_HANDLE_UNUSED); /* An impossible type */
113 return(_WAPI_PRIVATE_HANDLES(idx).type);
116 static inline void _wapi_handle_set_signal_state (gpointer handle,
120 guint32 idx = GPOINTER_TO_UINT(handle);
121 struct _WapiHandleUnshared *handle_data;
124 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
128 g_assert (!_WAPI_SHARED_HANDLE(_wapi_handle_type (handle)));
130 handle_data = &_WAPI_PRIVATE_HANDLES(idx);
133 g_message ("%s: setting state of %p to %s (broadcast %s)", __func__,
134 handle, state?"TRUE":"FALSE", broadcast?"TRUE":"FALSE");
138 /* Tell everyone blocking on a single handle */
140 /* The condition the global signal cond is waiting on is the signalling of
141 * _any_ handle. So lock it before setting the signalled state.
143 thr_ret = mono_os_mutex_lock (_wapi_global_signal_mutex);
145 g_warning ("Bad call to mono_os_mutex_lock result %d for global signal mutex", thr_ret);
146 g_assert (thr_ret == 0);
148 /* This function _must_ be called with
149 * handle->signal_mutex locked
151 handle_data->signalled=state;
153 if (broadcast == TRUE) {
154 thr_ret = pthread_cond_broadcast (&handle_data->signal_cond);
156 g_warning ("Bad call to pthread_cond_broadcast result %d for handle %p", thr_ret, handle);
157 g_assert (thr_ret == 0);
159 thr_ret = pthread_cond_signal (&handle_data->signal_cond);
161 g_warning ("Bad call to pthread_cond_signal result %d for handle %p", thr_ret, handle);
162 g_assert (thr_ret == 0);
165 /* Tell everyone blocking on multiple handles that something
168 thr_ret = pthread_cond_broadcast (_wapi_global_signal_cond);
170 g_warning ("Bad call to pthread_cond_broadcast result %d for handle %p", thr_ret, handle);
171 g_assert (thr_ret == 0);
173 thr_ret = mono_os_mutex_unlock (_wapi_global_signal_mutex);
175 g_warning ("Bad call to mono_os_mutex_unlock result %d for global signal mutex", thr_ret);
176 g_assert (thr_ret == 0);
178 handle_data->signalled=state;
182 static inline void _wapi_shared_handle_set_signal_state (gpointer handle,
185 guint32 idx = GPOINTER_TO_UINT(handle);
186 struct _WapiHandleUnshared *handle_data;
187 struct _WapiHandle_shared_ref *ref;
188 struct _WapiHandleShared *shared_data;
190 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
194 g_assert (_WAPI_SHARED_HANDLE(_wapi_handle_type (handle)));
196 handle_data = &_WAPI_PRIVATE_HANDLES(idx);
198 ref = &handle_data->u.shared;
199 shared_data = &_wapi_shared_layout->handles[ref->offset];
200 shared_data->signalled = state;
203 g_message ("%s: signalled shared handle offset 0x%x", __func__,
208 static inline gboolean _wapi_handle_issignalled (gpointer handle)
210 guint32 idx = GPOINTER_TO_UINT(handle);
212 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
216 if (_WAPI_SHARED_HANDLE(_wapi_handle_type (handle))) {
217 return(WAPI_SHARED_HANDLE_DATA(handle).signalled);
219 return(_WAPI_PRIVATE_HANDLES(idx).signalled);
223 static inline int _wapi_handle_lock_signal_mutex (void)
226 g_message ("%s: lock global signal mutex", __func__);
229 return(mono_os_mutex_lock (_wapi_global_signal_mutex));
232 /* the parameter makes it easier to call from a pthread cleanup handler */
233 static inline int _wapi_handle_unlock_signal_mutex (void *unused)
236 g_message ("%s: unlock global signal mutex", __func__);
239 return(mono_os_mutex_unlock (_wapi_global_signal_mutex));
242 static inline int _wapi_handle_lock_handle (gpointer handle)
244 guint32 idx = GPOINTER_TO_UINT(handle);
247 g_message ("%s: locking handle %p", __func__, handle);
250 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
254 _wapi_handle_ref (handle);
256 if (_WAPI_SHARED_HANDLE (_wapi_handle_type (handle))) {
260 return(mono_os_mutex_lock (&_WAPI_PRIVATE_HANDLES(idx).signal_mutex));
263 static inline int _wapi_handle_trylock_handle (gpointer handle)
265 guint32 idx = GPOINTER_TO_UINT(handle);
269 g_message ("%s: locking handle %p", __func__, handle);
272 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
276 _wapi_handle_ref (handle);
278 if (_WAPI_SHARED_HANDLE (_wapi_handle_type (handle))) {
282 ret = mono_os_mutex_trylock (&_WAPI_PRIVATE_HANDLES(idx).signal_mutex);
284 _wapi_handle_unref (handle);
290 static inline int _wapi_handle_unlock_handle (gpointer handle)
292 guint32 idx = GPOINTER_TO_UINT(handle);
296 g_message ("%s: unlocking handle %p", __func__, handle);
299 if (!_WAPI_PRIVATE_VALID_SLOT (idx)) {
303 if (_WAPI_SHARED_HANDLE (_wapi_handle_type (handle))) {
304 _wapi_handle_unref (handle);
308 ret = mono_os_mutex_unlock (&_WAPI_PRIVATE_HANDLES(idx).signal_mutex);
310 _wapi_handle_unref (handle);
315 static inline void _wapi_handle_spin (guint32 ms)
317 struct timespec sleepytime;
319 g_assert (ms < 1000);
321 sleepytime.tv_sec = 0;
322 sleepytime.tv_nsec = ms * 1000000;
324 nanosleep (&sleepytime, NULL);
327 static inline int _wapi_handle_lock_shared_handles (void)
329 return(_wapi_shm_sem_lock (_WAPI_SHARED_SEM_SHARED_HANDLES));
332 static inline int _wapi_handle_trylock_shared_handles (void)
334 return(_wapi_shm_sem_trylock (_WAPI_SHARED_SEM_SHARED_HANDLES));
337 static inline int _wapi_handle_unlock_shared_handles (void)
339 return(_wapi_shm_sem_unlock (_WAPI_SHARED_SEM_SHARED_HANDLES));
342 static inline int _wapi_namespace_lock (void)
344 return(_wapi_shm_sem_lock (_WAPI_SHARED_SEM_NAMESPACE));
347 /* This signature makes it easier to use in pthread cleanup handlers */
348 static inline int _wapi_namespace_unlock (gpointer data G_GNUC_UNUSED)
350 return(_wapi_shm_sem_unlock (_WAPI_SHARED_SEM_NAMESPACE));
353 static inline void _wapi_handle_share_release (struct _WapiFileShare *info)
357 g_assert (info->handle_refs > 0);
359 /* Prevent new entries racing with us */
360 thr_ret = _wapi_shm_sem_lock (_WAPI_SHARED_SEM_FILESHARE);
361 g_assert(thr_ret == 0);
363 if (InterlockedDecrement ((gint32 *)&info->handle_refs) == 0) {
364 _wapi_free_share_info (info);
367 thr_ret = _wapi_shm_sem_unlock (_WAPI_SHARED_SEM_FILESHARE);
370 #endif /* _WAPI_HANDLES_PRIVATE_H_ */