/**
|
* \file
|
* Runtime support for managed Semaphore on Unix
|
*
|
* Author:
|
* Ludovic Henry (luhenry@microsoft.com)
|
*
|
* Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
*/
|
|
#include "w32semaphore.h"
|
|
#include "w32error.h"
|
#include "w32handle-namespace.h"
|
#include "mono/utils/mono-logger-internals.h"
|
#include "mono/metadata/w32handle.h"
|
|
#define MAX_PATH 260
|
|
typedef struct {
|
guint32 val;
|
gint32 max;
|
} MonoW32HandleSemaphore;
|
|
struct MonoW32HandleNamedSemaphore {
|
MonoW32HandleSemaphore s;
|
MonoW32HandleNamespace sharedns;
|
};
|
|
static void sem_handle_signal (MonoW32Handle *handle_data)
|
{
|
MonoW32HandleSemaphore *sem_handle;
|
|
sem_handle = (MonoW32HandleSemaphore*) handle_data->specific;
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: signalling %s handle %p",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle_data);
|
|
/* No idea why max is signed, but thats the spec :-( */
|
if (sem_handle->val + 1 > (guint32)sem_handle->max) {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: %s handle %p val %d count %d max %d, max value would be exceeded",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle_data, sem_handle->val, 1, sem_handle->max);
|
} else {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: %s handle %p val %d count %d max %d",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle_data, sem_handle->val, 1, sem_handle->max);
|
|
sem_handle->val += 1;
|
mono_w32handle_set_signal_state (handle_data, TRUE, TRUE);
|
}
|
}
|
|
static gboolean sem_handle_own (MonoW32Handle *handle_data, gboolean *abandoned)
|
{
|
MonoW32HandleSemaphore *sem_handle;
|
|
*abandoned = FALSE;
|
|
sem_handle = (MonoW32HandleSemaphore*) handle_data->specific;
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: owning %s handle %p",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle_data);
|
|
sem_handle->val--;
|
|
if (sem_handle->val == 0)
|
mono_w32handle_set_signal_state (handle_data, FALSE, FALSE);
|
|
return TRUE;
|
}
|
|
static void sema_details (MonoW32Handle *handle_data)
|
{
|
MonoW32HandleSemaphore *sem = (MonoW32HandleSemaphore *)handle_data->specific;
|
g_print ("val: %5u, max: %5d", sem->val, sem->max);
|
}
|
|
static void namedsema_details (MonoW32Handle *handle_data)
|
{
|
MonoW32HandleNamedSemaphore *namedsem = (MonoW32HandleNamedSemaphore *)handle_data->specific;
|
g_print ("val: %5u, max: %5d, name: \"%s\"", namedsem->s.val, namedsem->s.max, namedsem->sharedns.name);
|
}
|
|
static const gchar* sema_typename (void)
|
{
|
return "Semaphore";
|
}
|
|
static gsize sema_typesize (void)
|
{
|
return sizeof (MonoW32HandleSemaphore);
|
}
|
|
static const gchar* namedsema_typename (void)
|
{
|
return "N.Semaphore";
|
}
|
|
static gsize namedsema_typesize (void)
|
{
|
return sizeof (MonoW32HandleNamedSemaphore);
|
}
|
|
void
|
mono_w32semaphore_init (void)
|
{
|
static MonoW32HandleOps sem_ops = {
|
NULL, /* close */
|
sem_handle_signal, /* signal */
|
sem_handle_own, /* own */
|
NULL, /* is_owned */
|
NULL, /* special_wait */
|
NULL, /* prewait */
|
sema_details, /* details */
|
sema_typename, /* typename */
|
sema_typesize, /* typesize */
|
};
|
|
static MonoW32HandleOps namedsem_ops = {
|
NULL, /* close */
|
sem_handle_signal, /* signal */
|
sem_handle_own, /* own */
|
NULL, /* is_owned */
|
NULL, /* special_wait */
|
NULL, /* prewait */
|
namedsema_details, /* details */
|
namedsema_typename, /* typename */
|
namedsema_typesize, /* typesize */
|
};
|
|
mono_w32handle_register_ops (MONO_W32TYPE_SEM, &sem_ops);
|
mono_w32handle_register_ops (MONO_W32TYPE_NAMEDSEM, &namedsem_ops);
|
|
mono_w32handle_register_capabilities (MONO_W32TYPE_SEM,
|
(MonoW32HandleCapability)(MONO_W32HANDLE_CAP_WAIT | MONO_W32HANDLE_CAP_SIGNAL));
|
mono_w32handle_register_capabilities (MONO_W32TYPE_NAMEDSEM,
|
(MonoW32HandleCapability)(MONO_W32HANDLE_CAP_WAIT | MONO_W32HANDLE_CAP_SIGNAL));
|
}
|
|
static gpointer
|
sem_handle_create (MonoW32HandleSemaphore *sem_handle, MonoW32Type type, gint32 initial, gint32 max)
|
{
|
MonoW32Handle *handle_data;
|
gpointer handle;
|
|
sem_handle->val = initial;
|
sem_handle->max = max;
|
|
handle = mono_w32handle_new (type, sem_handle);
|
if (handle == INVALID_HANDLE_VALUE) {
|
g_warning ("%s: error creating %s handle",
|
__func__, mono_w32handle_get_typename (type));
|
mono_w32error_set_last (ERROR_GEN_FAILURE);
|
return NULL;
|
}
|
|
if (!mono_w32handle_lookup_and_ref (handle, &handle_data))
|
g_error ("%s: unkown handle %p", __func__, handle);
|
|
if (handle_data->type != type)
|
g_error ("%s: unknown semaphore handle %p", __func__, handle);
|
|
mono_w32handle_lock (handle_data);
|
|
if (initial != 0)
|
mono_w32handle_set_signal_state (handle_data, TRUE, FALSE);
|
|
mono_w32handle_unlock (handle_data);
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: created %s handle %p",
|
__func__, mono_w32handle_get_typename (type), handle);
|
|
mono_w32handle_unref (handle_data);
|
|
return handle;
|
}
|
|
static gpointer
|
sem_create (gint32 initial, gint32 max)
|
{
|
MonoW32HandleSemaphore sem_handle;
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: creating %s handle, initial %d max %d",
|
__func__, mono_w32handle_get_typename (MONO_W32TYPE_SEM), initial, max);
|
return sem_handle_create (&sem_handle, MONO_W32TYPE_SEM, initial, max);
|
}
|
|
static gpointer
|
namedsem_create (gint32 initial, gint32 max, const gunichar2 *name)
|
{
|
gpointer handle;
|
gchar *utf8_name;
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: creating %s handle, initial %d max %d name \"%s\"",
|
__func__, mono_w32handle_get_typename (MONO_W32TYPE_NAMEDSEM), initial, max, (const char*)name);
|
|
/* w32 seems to guarantee that opening named objects can't race each other */
|
mono_w32handle_namespace_lock ();
|
|
glong utf8_len = 0;
|
utf8_name = g_utf16_to_utf8 (name, -1, NULL, &utf8_len, NULL);
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: Creating named sem name [%s] initial %d max %d", __func__, utf8_name, initial, max);
|
|
handle = mono_w32handle_namespace_search_handle (MONO_W32TYPE_NAMEDSEM, utf8_name);
|
if (handle == INVALID_HANDLE_VALUE) {
|
/* The name has already been used for a different object. */
|
handle = NULL;
|
mono_w32error_set_last (ERROR_INVALID_HANDLE);
|
} else if (handle) {
|
/* Not an error, but this is how the caller is informed that the semaphore wasn't freshly created */
|
mono_w32error_set_last (ERROR_ALREADY_EXISTS);
|
|
/* mono_w32handle_namespace_search_handle already adds a ref to the handle */
|
} else {
|
/* A new named semaphore */
|
MonoW32HandleNamedSemaphore namedsem_handle;
|
|
size_t len = utf8_len < MAX_PATH ? utf8_len : MAX_PATH;
|
memcpy (&namedsem_handle.sharedns.name [0], utf8_name, len);
|
namedsem_handle.sharedns.name [len] = '\0';
|
|
handle = sem_handle_create ((MonoW32HandleSemaphore*) &namedsem_handle, MONO_W32TYPE_NAMEDSEM, initial, max);
|
}
|
|
g_free (utf8_name);
|
|
mono_w32handle_namespace_unlock ();
|
|
return handle;
|
}
|
|
gpointer
|
ves_icall_System_Threading_Semaphore_CreateSemaphore_internal (gint32 initialCount, gint32 maximumCount, MonoString *name, gint32 *error)
|
{
|
gpointer sem;
|
|
if (maximumCount <= 0) {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: maximumCount <= 0", __func__);
|
|
*error = ERROR_INVALID_PARAMETER;
|
return NULL;
|
}
|
|
if (initialCount > maximumCount || initialCount < 0) {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: initialCount > maximumCount or < 0", __func__);
|
|
*error = ERROR_INVALID_PARAMETER;
|
return NULL;
|
}
|
|
/* Need to blow away any old errors here, because code tests
|
* for ERROR_ALREADY_EXISTS on success (!) to see if a
|
* semaphore was freshly created
|
*/
|
mono_w32error_set_last (ERROR_SUCCESS);
|
|
if (!name)
|
sem = sem_create (initialCount, maximumCount);
|
else
|
sem = namedsem_create (initialCount, maximumCount, mono_string_chars (name));
|
|
*error = mono_w32error_get_last ();
|
|
return sem;
|
}
|
|
MonoBoolean
|
ves_icall_System_Threading_Semaphore_ReleaseSemaphore_internal (gpointer handle, gint32 releaseCount, gint32 *prevcount)
|
{
|
MonoW32Handle *handle_data;
|
MonoW32HandleSemaphore *sem_handle;
|
MonoBoolean ret;
|
|
if (!mono_w32handle_lookup_and_ref (handle, &handle_data)) {
|
g_warning ("%s: unkown handle %p", __func__, handle);
|
mono_w32error_set_last (ERROR_INVALID_HANDLE);
|
return FALSE;
|
}
|
|
if (handle_data->type != MONO_W32TYPE_SEM && handle_data->type != MONO_W32TYPE_NAMEDSEM) {
|
g_warning ("%s: unknown sem handle %p", __func__, handle);
|
mono_w32error_set_last (ERROR_INVALID_HANDLE);
|
mono_w32handle_unref (handle_data);
|
return FALSE;
|
}
|
|
sem_handle = (MonoW32HandleSemaphore*) handle_data->specific;
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: releasing %s handle %p",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle);
|
|
mono_w32handle_lock (handle_data);
|
|
/* Do this before checking for count overflow, because overflowing
|
* max is a listed technique for finding the current value */
|
if (prevcount)
|
*prevcount = sem_handle->val;
|
|
/* No idea why max is signed, but thats the spec :-( */
|
if (sem_handle->val + releaseCount > (guint32)sem_handle->max) {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: %s handle %p val %d count %d max %d, max value would be exceeded",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle, sem_handle->val, releaseCount, sem_handle->max);
|
|
ret = FALSE;
|
} else {
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: %s handle %p val %d count %d max %d",
|
__func__, mono_w32handle_get_typename (handle_data->type), handle, sem_handle->val, releaseCount, sem_handle->max);
|
|
sem_handle->val += releaseCount;
|
mono_w32handle_set_signal_state (handle_data, TRUE, TRUE);
|
|
ret = TRUE;
|
}
|
|
mono_w32handle_unlock (handle_data);
|
mono_w32handle_unref (handle_data);
|
|
return ret;
|
}
|
|
gpointer
|
ves_icall_System_Threading_Semaphore_OpenSemaphore_internal (MonoString *name, gint32 rights, gint32 *error)
|
{
|
gpointer handle;
|
gchar *utf8_name;
|
|
*error = ERROR_SUCCESS;
|
|
/* w32 seems to guarantee that opening named objects can't race each other */
|
mono_w32handle_namespace_lock ();
|
|
utf8_name = g_utf16_to_utf8 (mono_string_chars (name), -1, NULL, NULL, NULL);
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: Opening named sem [%s]", __func__, utf8_name);
|
|
handle = mono_w32handle_namespace_search_handle (MONO_W32TYPE_NAMEDSEM, utf8_name);
|
if (handle == INVALID_HANDLE_VALUE) {
|
/* The name has already been used for a different object. */
|
*error = ERROR_INVALID_HANDLE;
|
goto cleanup;
|
} else if (!handle) {
|
/* This name doesn't exist */
|
*error = ERROR_FILE_NOT_FOUND;
|
goto cleanup;
|
}
|
|
mono_trace (G_LOG_LEVEL_DEBUG, MONO_TRACE_IO_LAYER_SEMAPHORE, "%s: returning named sem handle %p", __func__, handle);
|
|
cleanup:
|
g_free (utf8_name);
|
|
mono_w32handle_namespace_unlock ();
|
|
return handle;
|
}
|
|
MonoW32HandleNamespace*
|
mono_w32semaphore_get_namespace (MonoW32HandleNamedSemaphore *semaphore)
|
{
|
return &semaphore->sharedns;
|
}
|