107 lines
2.5 KiB
C
107 lines
2.5 KiB
C
|
/* Copyright 2002, Red Hat Inc. */
|
||
|
|
||
|
#include <mqueue.h>
|
||
|
#include <unistd.h>
|
||
|
#include <errno.h>
|
||
|
#include <sys/ipc.h>
|
||
|
#include <sys/sem.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "internals.h"
|
||
|
#include <sys/lock.h>
|
||
|
|
||
|
#include "mqlocal.h"
|
||
|
|
||
|
static void *mq_notify_process (void *);
|
||
|
|
||
|
void
|
||
|
__cleanup_mq_notify (struct libc_mq *info)
|
||
|
{
|
||
|
struct sembuf sb4 = {4, 1, 0};
|
||
|
/* kill notification thread and allow other processes to set a notification */
|
||
|
pthread_cancel ((pthread_t)info->th);
|
||
|
semop (info->semid, &sb4, 1);
|
||
|
}
|
||
|
|
||
|
static void *
|
||
|
mq_notify_process (void *arg)
|
||
|
{
|
||
|
struct libc_mq *info = (struct libc_mq *)arg;
|
||
|
struct sembuf sb3[2] = {{3, 0, 0}, {5, 0, 0}};
|
||
|
struct sembuf sb4 = {4, 1, 0};
|
||
|
int rc;
|
||
|
|
||
|
/* wait until queue is empty */
|
||
|
while (!(rc = semop (info->semid, sb3, 1)) && errno == EINTR)
|
||
|
/* empty */ ;
|
||
|
|
||
|
if (!rc)
|
||
|
{
|
||
|
/* now wait until there are 0 readers and the queue has something in it */
|
||
|
sb3[0].sem_op = -1;
|
||
|
while (!(rc = semop (info->semid, sb3, 2)) && errno == EINTR)
|
||
|
/* empty */ ;
|
||
|
/* restore value since we have not actually performed a read */
|
||
|
sb3[0].sem_op = 1;
|
||
|
semop (info->semid, sb3, 1);
|
||
|
/* perform desired notification - either run function in this thread or pass signal */
|
||
|
if (!rc)
|
||
|
{
|
||
|
if (info->sigevent->sigev_notify == SIGEV_SIGNAL)
|
||
|
raise (info->sigevent->sigev_signo);
|
||
|
else if (info->sigevent->sigev_notify == SIGEV_THREAD)
|
||
|
info->sigevent->sigev_notify_function (info->sigevent->sigev_value);
|
||
|
/* allow other processes to now mq_notify */
|
||
|
semop (info->semid, &sb4, 1);
|
||
|
}
|
||
|
}
|
||
|
pthread_exit (NULL);
|
||
|
}
|
||
|
|
||
|
int
|
||
|
mq_notify (mqd_t msgid, const struct sigevent *notification)
|
||
|
{
|
||
|
struct libc_mq *info;
|
||
|
struct sembuf sb4 = {4, -1, IPC_NOWAIT};
|
||
|
int rc;
|
||
|
pthread_attr_t *attr = NULL;
|
||
|
|
||
|
info = __find_mq (msgid);
|
||
|
|
||
|
if (info == NULL)
|
||
|
{
|
||
|
errno = EBADF;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* get notification lock */
|
||
|
rc = semop (info->semid, &sb4, 1);
|
||
|
|
||
|
if (rc == -1)
|
||
|
{
|
||
|
errno = EBUSY;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
/* to get the notification running we use a pthread - if the user has requested
|
||
|
an action in a pthread, we use the user's attributes when setting up the thread */
|
||
|
info->sigevent = (struct sigevent *)notification;
|
||
|
if (info->sigevent->sigev_notify == SIGEV_THREAD)
|
||
|
attr = (pthread_attr_t *)info->sigevent->sigev_notify_attributes;
|
||
|
rc = pthread_create ((pthread_t *)&info->th, attr, mq_notify_process, (void *)info);
|
||
|
|
||
|
if (rc != 0)
|
||
|
rc = -1;
|
||
|
else
|
||
|
info->cleanup_notify = &__cleanup_mq_notify;
|
||
|
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|