summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlon Levy <alevy@redhat.com>2011-09-08 02:16:24 +0300
committerAlon Levy <alevy@redhat.com>2011-10-31 17:35:54 +0200
commitbd8771adbcf3ff34d14333cf874191e8d105f612 (patch)
tree5c867e871cd4aea5acb4a92aba9883372aa8e7b1
parentedb91ccc09e4fd4cad0940d88f3455651eb7c367 (diff)
downloadspice-bd8771adbcf3ff34d14333cf874191e8d105f612.tar.gz
spice-bd8771adbcf3ff34d14333cf874191e8d105f612.tar.xz
spice-bd8771adbcf3ff34d14333cf874191e8d105f612.zip
[0.8 branch] server: add main_dispatcher
add main_dispatcher, a message passing mechanism for sending messages to the main thread. The main thread is the thread that implements SpiceCoreInterface, which is assumed to be a single thread. Similar to the async operation of red_worker, a socket pair is created and used to pass messages. The messages are a fixed size to ease parsing. A single message is defined to pass a channel_event. RHBZ: 746950 FDBZ: 41858 This patch is 0.8 branch only, for the master branch there should be a better approach to share code with red_dispatcher and ready the way for later adding more threads. cherry-pick from 0.8 80caf07e09efe14c67f89a3c01501a6a39681167 Conflicts: server/reds.c
-rw-r--r--server/Makefile.am2
-rw-r--r--server/main_dispatcher.c145
-rw-r--r--server/main_dispatcher.h9
-rw-r--r--server/reds.c4
4 files changed, 159 insertions, 1 deletions
diff --git a/server/Makefile.am b/server/Makefile.am
index a7cdd840..418d707c 100644
--- a/server/Makefile.am
+++ b/server/Makefile.am
@@ -80,6 +80,8 @@ libspice_server_la_SOURCES = \
red_common.h \
red_dispatcher.c \
red_dispatcher.h \
+ main_dispatcher.c \
+ main_dispatcher.h \
red_memslots.c \
red_memslots.h \
red_parse_qxl.c \
diff --git a/server/main_dispatcher.c b/server/main_dispatcher.c
new file mode 100644
index 00000000..73856bfe
--- /dev/null
+++ b/server/main_dispatcher.c
@@ -0,0 +1,145 @@
+#include <unistd.h>
+#include <string.h>
+#include <errno.h>
+#include <pthread.h>
+#include <assert.h>
+
+#include "red_common.h"
+#include "main_dispatcher.h"
+
+/*
+ * Main Dispatcher
+ * ===============
+ *
+ * Communication channel between any non main thread and the main thread.
+ *
+ * The main thread is that from which spice_server_init is called.
+ *
+ * Messages are single sized, sent from the non-main thread to the main-thread.
+ * No acknowledge is sent back. This prevents a possible deadlock with the main
+ * thread already waiting on a response for the existing red_dispatcher used
+ * by the worker thread.
+ *
+ * All events have three functions:
+ * main_dispatcher_<event_name> - non static, public function
+ * main_dispatcher_self_<event_name> - handler for main thread
+ * main_dispatcher_handle_<event_name> - handler for callback from main thread
+ * seperate from self because it may send an ack or do other work in the future.
+ */
+
+typedef struct {
+ SpiceCoreInterface *core;
+ int main_fd;
+ int other_fd;
+ pthread_t self;
+ pthread_mutex_t lock;
+} MainDispatcher;
+
+MainDispatcher main_dispatcher;
+
+enum {
+ MAIN_DISPATCHER_CHANNEL_EVENT = 0,
+
+ MAIN_DISPATCHER_NUM_MESSAGES
+};
+
+typedef struct MainDispatcherMessage {
+ uint32_t type;
+ union {
+ struct {
+ int event;
+ SpiceChannelEventInfo *info;
+ } channel_event;
+ } data;
+} MainDispatcherMessage;
+
+/* channel_event - calls core->channel_event, must be done in main thread */
+static void main_dispatcher_self_handle_channel_event(
+ int event,
+ SpiceChannelEventInfo *info)
+{
+ main_dispatcher.core->channel_event(event, info);
+}
+
+static void main_dispatcher_handle_channel_event(MainDispatcherMessage *msg)
+{
+ main_dispatcher_self_handle_channel_event(msg->data.channel_event.event,
+ msg->data.channel_event.info);
+}
+
+void main_dispatcher_channel_event(int event, SpiceChannelEventInfo *info)
+{
+ MainDispatcherMessage msg;
+ ssize_t written = 0;
+ ssize_t ret;
+
+ if (pthread_self() == main_dispatcher.self) {
+ main_dispatcher_self_handle_channel_event(event, info);
+ return;
+ }
+ msg.type = MAIN_DISPATCHER_CHANNEL_EVENT;
+ msg.data.channel_event.event = event;
+ msg.data.channel_event.info = info;
+ pthread_mutex_lock(&main_dispatcher.lock);
+ while (written < sizeof(msg)) {
+ ret = write(main_dispatcher.other_fd, &msg + written,
+ sizeof(msg) - written);
+ if (ret == -1) {
+ assert(errno == EINTR);
+ continue;
+ }
+ written += ret;
+ }
+ pthread_mutex_unlock(&main_dispatcher.lock);
+}
+
+
+static void main_dispatcher_handle_read(int fd, int event, void *opaque)
+{
+ int ret;
+ MainDispatcher *md = opaque;
+ MainDispatcherMessage msg;
+ int read_size = 0;
+
+ while (read_size < sizeof(msg)) {
+ /* blocks until sizeof(msg) is read */
+ ret = read(md->main_fd, &msg + read_size, sizeof(msg) - read_size);
+ if (ret == -1) {
+ if (errno != EINTR) {
+ red_printf("error reading from main dispatcher: %d\n", errno);
+ /* TODO: close channel? */
+ return;
+ }
+ continue;
+ }
+ read_size += ret;
+ }
+ switch (msg.type) {
+ case MAIN_DISPATCHER_CHANNEL_EVENT:
+ main_dispatcher_handle_channel_event(&msg);
+ break;
+ default:
+ red_printf("error: unhandled main dispatcher message type %d\n",
+ msg.type);
+ }
+}
+
+void main_dispatcher_init(SpiceCoreInterface *core)
+{
+ int channels[2];
+
+ memset(&main_dispatcher, 0, sizeof(main_dispatcher));
+ main_dispatcher.core = core;
+
+ if (socketpair(AF_LOCAL, SOCK_STREAM, 0, channels) == -1) {
+ red_error("socketpair failed %s", strerror(errno));
+ return;
+ }
+ pthread_mutex_init(&main_dispatcher.lock, NULL);
+ main_dispatcher.main_fd = channels[0];
+ main_dispatcher.other_fd = channels[1];
+ main_dispatcher.self = pthread_self();
+
+ core->watch_add(main_dispatcher.main_fd, SPICE_WATCH_EVENT_READ,
+ main_dispatcher_handle_read, &main_dispatcher);
+}
diff --git a/server/main_dispatcher.h b/server/main_dispatcher.h
new file mode 100644
index 00000000..2c201c75
--- /dev/null
+++ b/server/main_dispatcher.h
@@ -0,0 +1,9 @@
+#ifndef MAIN_DISPATCHER_H
+#define MAIN_DISPATCHER_H
+
+#include <spice.h>
+
+void main_dispatcher_channel_event(int event, SpiceChannelEventInfo *info);
+void main_dispatcher_init(SpiceCoreInterface *core);
+
+#endif //MAIN_DISPATCHER_H
diff --git a/server/reds.c b/server/reds.c
index 90779ffb..65c98aef 100644
--- a/server/reds.c
+++ b/server/reds.c
@@ -56,6 +56,7 @@
#include "main_channel.h"
#include "red_common.h"
#include "red_dispatcher.h"
+#include "main_dispatcher.h"
#include "snd_worker.h"
#include <spice/stats.h>
#include "stat.h"
@@ -322,7 +323,7 @@ static void reds_stream_channel_event(RedsStream *s, int event)
{
if (core->base.minor_version < 3 || core->channel_event == NULL)
return;
- core->channel_event(event, &s->info);
+ main_dispatcher_channel_event(event, &s->info);
}
static ssize_t stream_write_cb(RedsStream *s, const void *buf, size_t size)
@@ -3519,6 +3520,7 @@ static int do_spice_init(SpiceCoreInterface *core_interface)
init_vd_agent_resources();
ring_init(&reds->clients);
reds->num_clients = 0;
+ main_dispatcher_init(core);
if (!(reds->mig_timer = core->timer_add(migrate_timout, NULL))) {
red_error("migration timer create failed");