/* * Ouroboros - Copyright (C) 2016 - 2017 * * The IPC Resource Manager - Flows * * Dimitri Staessens <dimitri.staessens@ugent.be> * Sander Vrijders <sander.vrijders@ugent.be> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., http://www.fsf.org/about/contact/. */ #define _POSIX_C_SOURCE 199309L #include "config.h" #define OUROBOROS_PREFIX "irm_flow" #include <ouroboros/errno.h> #include <ouroboros/logs.h> #include <ouroboros/time_utils.h> #include "irm_flow.h" #include <stdlib.h> #include <stdbool.h> #include <assert.h> struct irm_flow * irm_flow_create(pid_t n_api, pid_t n_1_api, int port_id, qoscube_t qc) { struct irm_flow * f = malloc(sizeof(*f)); if (f == NULL) return NULL; if (pthread_cond_init(&f->state_cond, NULL)) { free(f); return NULL; } if (pthread_mutex_init(&f->state_lock, NULL)) { pthread_cond_destroy(&f->state_cond); free(f); return NULL; } f->n_api = n_api; f->n_1_api = n_1_api; f->port_id = port_id; f->qc = qc; f->n_rb = shm_rbuff_create(n_api, port_id); if (f->n_rb == NULL) { log_err("Could not create ringbuffer for AP-I %d.", n_api); pthread_mutex_destroy(&f->state_lock); pthread_cond_destroy(&f->state_cond); free(f); return NULL; } f->n_1_rb = shm_rbuff_create(n_1_api, port_id); if (f->n_1_rb == NULL) { log_err("Could not create ringbuffer for AP-I %d.", n_1_api); shm_rbuff_destroy(f->n_rb); pthread_mutex_destroy(&f->state_lock); pthread_cond_destroy(&f->state_cond); free(f); return NULL; } f->state = FLOW_ALLOC_PENDING; if (clock_gettime(CLOCK_MONOTONIC, &f->t0) < 0) log_warn("Failed to set timestamp."); return f; } void irm_flow_destroy(struct irm_flow * f) { assert(f); pthread_mutex_lock(&f->state_lock); if (f->state == FLOW_DESTROY) { pthread_mutex_unlock(&f->state_lock); return; } if (f->state == FLOW_ALLOC_PENDING) f->state = FLOW_DESTROY; else f->state = FLOW_NULL; pthread_cond_signal(&f->state_cond); while (f->state != FLOW_NULL) pthread_cond_wait(&f->state_cond, &f->state_lock); pthread_mutex_unlock(&f->state_lock); pthread_cond_destroy(&f->state_cond); pthread_mutex_destroy(&f->state_lock); shm_rbuff_destroy(f->n_rb); shm_rbuff_destroy(f->n_1_rb); free(f); } enum flow_state irm_flow_get_state(struct irm_flow * f) { enum flow_state state; assert(f); pthread_mutex_lock(&f->state_lock); state = f->state; pthread_mutex_unlock(&f->state_lock); return state; } void irm_flow_set_state(struct irm_flow * f, enum flow_state state) { assert(f); assert(state != FLOW_DESTROY); pthread_mutex_lock(&f->state_lock); f->state = state; pthread_cond_broadcast(&f->state_cond); pthread_mutex_unlock(&f->state_lock); } int irm_flow_wait_state(struct irm_flow * f, enum flow_state state, struct timespec * timeo) { int ret = 0; int s; struct timespec dl; assert(f); assert(state != FLOW_NULL); assert(state != FLOW_DESTROY); assert(state != FLOW_DEALLOC_PENDING); if (timeo != NULL) { clock_gettime(PTHREAD_COND_CLOCK, &dl); ts_add(&dl, timeo, &dl); } pthread_mutex_lock(&f->state_lock); assert(f->state != FLOW_NULL); while (!(f->state == state || f->state == FLOW_DESTROY || f->state == FLOW_DEALLOC_PENDING) && ret != -ETIMEDOUT) { if (timeo == NULL) ret = -pthread_cond_wait(&f->state_cond, &f->state_lock); else ret = -pthread_cond_timedwait(&f->state_cond, &f->state_lock, &dl); } if (f->state == FLOW_DESTROY || f->state == FLOW_DEALLOC_PENDING || ret == -ETIMEDOUT) { f->state = FLOW_NULL; pthread_cond_broadcast(&f->state_cond); } s = f->state; pthread_mutex_unlock(&f->state_lock); return ret ? ret : s; }