diff options
Diffstat (limited to 'src/kiro-server.c')
-rw-r--r-- | src/kiro-server.c | 88 |
1 files changed, 61 insertions, 27 deletions
diff --git a/src/kiro-server.c b/src/kiro-server.c index 276388b..9a76007 100644 --- a/src/kiro-server.c +++ b/src/kiro-server.c @@ -50,13 +50,14 @@ struct _KiroServerPrivate { /* 'Real' private structures */ /* (Not accessible by properties) */ - struct rdma_event_channel *ec; // Main Event Channel - struct rdma_cm_id *base; // Base-Listening-Connection - struct kiro_connection *client; // Connection to the client - pthread_t event_listener; // Pointer to the completion-listener thread of this connection - pthread_mutex_t mtx; // Mutex to signal the listener-thread termination - void *mem; // Pointer to the server buffer - size_t mem_size; // Server Buffer Size in bytes + struct rdma_event_channel *ec; // Main Event Channel + struct rdma_cm_id *base; // Base-Listening-Connection + struct kiro_connection *client; // Connection to the client + pthread_t event_listener; // Pointer to the completion-listener thread of this connection + pthread_mutex_t mtx; // Mutex to signal the listener-thread termination + int close_signal; // Integer flag used to signal to the listener-thread that the server is going to shut down + void *mem; // Pointer to the server buffer + size_t mem_size; // Server Buffer Size in bytes }; @@ -75,9 +76,9 @@ static void kiro_server_finalize (GObject *object) { KiroServer *self = KIRO_SERVER(object); - KiroServerPrivate *priv = KIRO_SERVER_GET_PRIVATE(self); - pthread_mutex_unlock(&(priv->mtx)); - pthread_join(priv->event_listener, NULL); + + //Clean up the server + kiro_server_stop(self); } @@ -201,11 +202,14 @@ void * event_loop (void *self) KiroServerPrivate *priv = KIRO_SERVER_GET_PRIVATE((KiroServer *)self); struct rdma_cm_event *active_event; - int stop = 0; - - while(0 == stop) { + while(0 == priv->close_signal) { if(0 <= rdma_get_cm_event(priv->ec, &active_event)) { + //Lock mutex to signal that this thread is currently handling an event + //and disable cancellation to prevent undefined states during shutdown + pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL); + pthread_mutex_lock(&(priv->mtx)); + struct rdma_cm_event *ev = malloc(sizeof(*active_event)); if(!ev) @@ -219,7 +223,15 @@ void * event_loop (void *self) if (ev->event == RDMA_CM_EVENT_CONNECT_REQUEST) { - + + if (0 != priv->close_signal) + { + //Main thread has signalled shutdown! + //Don't connect this client any more + //Sorry mate! + rdma_reject(ev->id, NULL, 0); + } + /* priv->client = (struct kiro_connection *)calloc(1, sizeof(struct kiro_connection)); if(!(priv->client)) @@ -242,16 +254,14 @@ void * event_loop (void *self) else if(ev->event == RDMA_CM_EVENT_DISCONNECTED) { printf("Got disconnect request.\n"); - //pthread_mutex_unlock(&(priv->mtx)); kiro_destroy_connection(&(ev->id)); printf("Connection closed successfully\n"); } free(ev); } - // Mutex will be freed as a signal to stop request - if(0 == pthread_mutex_trylock(&(priv->mtx))) - stop = 1; + pthread_mutex_unlock(&(priv->mtx)); + pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL); } printf("Closing Event Listener Thread\n"); @@ -332,14 +342,7 @@ int kiro_server_start (KiroServer *self, char *address, char *port, void* mem, s priv->mem = mem; priv->mem_size = mem_size; - priv->ec = rdma_create_event_channel(); - int oldflags = fcntl (priv->ec->fd, F_GETFL, 0); - /* Only change the FD Mode if we were able to get its flags */ - if (oldflags >= 0) { - oldflags |= O_NONBLOCK; - /* Store modified flag word in the descriptor. */ - fcntl (priv->ec->fd, F_SETFL, oldflags); - } + priv->ec = rdma_create_event_channel(); if(rdma_migrate_id(priv->base, priv->ec)) { printf("Was unable to migrate connection to new Event Channel.\n"); @@ -348,7 +351,6 @@ int kiro_server_start (KiroServer *self, char *address, char *port, void* mem, s } pthread_mutex_init(&(priv->mtx), NULL); - pthread_mutex_lock(&(priv->mtx)); pthread_create(&(priv->event_listener), NULL, event_loop, self); printf("Enpoint listening.\n"); @@ -358,6 +360,38 @@ int kiro_server_start (KiroServer *self, char *address, char *port, void* mem, s } +void +kiro_server_stop (KiroServer *self) +{ + if(!self) + return; + + KiroServerPrivate *priv = KIRO_SERVER_GET_PRIVATE (self); + + if(!priv->base) + return; + + //Shut down the listener-thread + priv->close_signal = 1; + pthread_mutex_lock(&(priv->mtx)); + pthread_cancel(priv->event_listener); + pthread_join(priv->event_listener, NULL); + printf("Event Listener Thread stopped.\n"); + priv->close_signal = 0; + + /* + * FOR ALL PRIV->CLIENT : DISCONNECT + */ + + rdma_destroy_ep(priv->base); + priv->base = NULL; + rdma_destroy_event_channel(priv->ec); + priv->ec = NULL; + + printf("Server stopped successfully.\n"); +} + + |