blob: cd6a37b2a4c1641b15d74a6649db7cb624df2070 [file] [log] [blame]
/*
* BRLTTY - A background process providing access to the console screen (when in
* text mode) for a blind person using a refreshable braille display.
*
* Copyright (C) 1995-2023 by The BRLTTY Developers.
*
* BRLTTY comes with ABSOLUTELY NO WARRANTY.
*
* This is free software, placed under the terms of the
* GNU Lesser General Public License, as published by the Free Software
* Foundation; either version 2.1 of the License, or (at your option) any
* later version. Please see the file LICENSE-LGPL for details.
*
* Web Page: http://brltty.app/
*
* This software is maintained by Dave Mielke <dave@mielke.cc>.
*/
#include "prologue.h"
#include <string.h>
#include <errno.h>
#include <sys/msg.h>
#include "log.h"
#include "msg_queue.h"
#include "async_event.h"
#include "thread.h"
typedef struct {
MessageType type;
char content[];
} Message;
#define MESSAGE(name, length) \
unsigned char name##_bytes[sizeof(MessageType) + (length)]; \
Message *name = (void *)name##_bytes;
int
sendMessage (int queue, MessageType type, const void *content, size_t length, int flags) {
MESSAGE(message, length);
if (!content) {
length = 0;
} else if (length) {
memcpy(message->content, content, length);
}
message->type = type;
if (msgsnd(queue, message, length, flags) != -1) return 1;
logSystemError("msgsnd");
return 0;
}
ssize_t
receiveMessage (int queue, MessageType *type, void *buffer, size_t size, int flags) {
MESSAGE(message, size);
if (!buffer) size = 0;
ssize_t length = msgrcv(queue, message, size, *type, flags);
if (length != -1) {
*type = message->type;
if (length) memcpy(buffer, message->content, length);
} else if (errno != EIDRM) {
logSystemError("msgrcv");
}
return length;
}
typedef struct {
AsyncEvent *event;
pthread_t thread;
MessageHandler *handler;
void *data;
int queue;
MessageType type;
size_t size;
} MessageReceiverArgument;
ASYNC_EVENT_CALLBACK(handleReceivedMessage) {
MessageReceiverArgument *mra = parameters->eventData;
MessageHandlerParameters *mhp = parameters->signalData;
if (mhp) {
mra->handler(mhp);
free(mhp);
} else {
void *result;
pthread_join(mra->thread, &result);
asyncDiscardEvent(mra->event);
free(mra);
}
}
THREAD_FUNCTION(messageReceiverThread) {
MessageReceiverArgument *mra = argument;
char buffer[mra->size];
while (1) {
MessageType type = mra->type;
ssize_t length = receiveMessage(mra->queue, &type, buffer, mra->size, 0);
if (length != -1) {
MessageHandlerParameters *mhp;
if ((mhp = malloc(sizeof(*mhp) + length))) {
memset(mhp, 0, sizeof(*mhp));
mhp->data = mra->data;
mhp->type = type;
mhp->length = length;
memcpy(mhp->content, buffer, mhp->length);
if (asyncSignalEvent(mra->event, mhp)) continue;
free(mhp);
} else {
logMallocError();
}
}
break;
}
asyncSignalEvent(mra->event, NULL);
return NULL;
}
int
startMessageReceiver (const char *name, int queue, MessageType type, size_t size, MessageHandler *handler, void *data) {
MessageReceiverArgument *mra;
if ((mra = malloc(sizeof(*mra)))) {
memset(mra, 0, sizeof(*mra));
mra->handler = handler;
mra->data = data;
mra->queue = queue;
mra->type = type;
mra->size = size;
if ((mra->event = asyncNewEvent(handleReceivedMessage, mra))) {
int threadCreationError = createThread(name, &mra->thread, NULL, messageReceiverThread, mra);
if (!threadCreationError) {
logMessage(LOG_DEBUG, "message receiver started: %s", name);
return 1;
}
asyncDiscardEvent(mra->event);
}
free(mra);
} else {
logMallocError();
}
logMessage(LOG_WARNING, "message receiver not started: %s", name);
return 0;
}