/* * Ouroboros - Copyright (C) 2016 - 2017 * * The IPC Resource Manager - Registry * * 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 200809L #include "config.h" #define OUROBOROS_PREFIX "registry" #include <ouroboros/errno.h> #include <ouroboros/logs.h> #include <ouroboros/irm.h> #include <ouroboros/time_utils.h> #include "registry.h" #include "utils.h" #include <stdlib.h> #include <stdbool.h> #include <string.h> #include <signal.h> #include <unistd.h> #include <limits.h> #include <assert.h> struct reg_dif { struct list_head next; char * dif_name; enum ipcp_type type; }; static struct reg_entry * reg_entry_create(void) { struct reg_entry * e = malloc(sizeof(*e)); if (e == NULL) return NULL; e->name = NULL; e->state = REG_NAME_NULL; return e; } static int reg_entry_init(struct reg_entry * e, char * name) { pthread_condattr_t cattr; if (e == NULL || name == NULL) return -1; list_head_init(&e->next); list_head_init(&e->difs); list_head_init(&e->reg_apns); list_head_init(&e->reg_apis); e->name = name; if (pthread_condattr_init(&cattr)) return -1; #ifndef __APPLE__ pthread_condattr_setclock(&cattr, PTHREAD_COND_CLOCK); #endif if (pthread_cond_init(&e->state_cond, &cattr)) return -1; if (pthread_mutex_init(&e->state_lock, NULL)) return -1; e->state = REG_NAME_IDLE; return 0; } static void cancel_reg_entry_destroy(void * o) { struct reg_entry * e; struct list_head * p; struct list_head * h; e = (struct reg_entry *) o; pthread_mutex_unlock(&e->state_lock); pthread_cond_destroy(&e->state_cond); pthread_mutex_destroy(&e->state_lock); if (e->name != NULL) free(e->name); list_for_each_safe(p, h, &e->reg_apis) { struct pid_el * pe = list_entry(p, struct pid_el, next); list_del(&pe->next); free(pe); } list_for_each_safe(p, h, &e->reg_apns) { struct str_el * a = list_entry(p, struct str_el, next); list_del(&a->next); free(a->str); free(a); } list_for_each_safe(p, h, &e->difs) { struct reg_dif * d = list_entry(p, struct reg_dif, next); list_del(&d->next); free(d->dif_name); free(d); } free(e); } static void reg_entry_destroy(struct reg_entry * e) { if (e == NULL) return; pthread_mutex_lock(&e->state_lock); if (e->state == REG_NAME_DESTROY) { pthread_mutex_unlock(&e->state_lock); return; } if (e->state != REG_NAME_FLOW_ACCEPT) e->state = REG_NAME_NULL; else e->state = REG_NAME_DESTROY; pthread_cond_broadcast(&e->state_cond); pthread_cleanup_push(cancel_reg_entry_destroy, e); while (e->state != REG_NAME_NULL) pthread_cond_wait(&e->state_cond, &e->state_lock); pthread_cleanup_pop(true); } static bool reg_entry_is_local_in_dif(struct reg_entry * e, const char * dif_name) { struct list_head * p = NULL; list_for_each(p, &e->difs) { struct reg_dif * d = list_entry(p, struct reg_dif, next); if (!strcmp(dif_name, d->dif_name)) return true; } return false; } static int reg_entry_add_local_in_dif(struct reg_entry * e, const char * dif_name, enum ipcp_type type) { struct reg_dif * rdn; /* already registered. Is ok */ if (reg_entry_is_local_in_dif(e, dif_name)) return 0; rdn = malloc(sizeof(*rdn)); if (rdn == NULL) return -1; rdn->dif_name = strdup(dif_name); if (rdn->dif_name == NULL) { free(rdn); return -1; } rdn->type = type; list_add(&rdn->next, &e->difs); return 0; } static void reg_entry_del_local_from_dif(struct reg_entry * e, const char * dif_name) { struct list_head * p = NULL; struct list_head * h = NULL; list_for_each_safe(p, h, &e->difs) { struct reg_dif * d = list_entry(p, struct reg_dif, next); if (!strcmp(dif_name, d->dif_name)) { list_del(&d->next); free(d->dif_name); free(d); } } } static bool reg_entry_has_apn(struct reg_entry * e, const char * apn) { struct list_head * p; list_for_each(p, &e->reg_apns) { struct str_el * e = list_entry(p, struct str_el, next); if (!strcmp(e->str, apn)) return true; } return false; } int reg_entry_add_apn(struct reg_entry * e, struct apn_entry * a) { struct str_el * n; if (reg_entry_has_apn(e, a->apn)) { log_warn("AP %s already accepting flows for %s.", a->apn, e->name); return 0; } if (!(a->flags & BIND_AP_AUTO)) { log_dbg("AP %s cannot be auto-instantiated.", a->apn); return -EINVAL; } n = malloc(sizeof(*n)); if (n == NULL) return -ENOMEM; n->str = strdup(a->apn); if (n->str == NULL) { free(n); return -ENOMEM; } list_add(&n->next, &e->reg_apns); pthread_mutex_lock(&e->state_lock); if (e->state == REG_NAME_IDLE) e->state = REG_NAME_AUTO_ACCEPT; pthread_mutex_unlock(&e->state_lock); return 0; } void reg_entry_del_apn(struct reg_entry * e, const char * apn) { struct list_head * p = NULL; struct list_head * h = NULL; list_for_each_safe(p, h, &e->reg_apns) { struct str_el * e = list_entry(p, struct str_el, next); if (!wildcard_match(apn, e->str)) { list_del(&e->next); free(e->str); free(e); } } pthread_mutex_lock(&e->state_lock); if (e->state == REG_NAME_AUTO_ACCEPT && list_is_empty(&e->reg_apns)) { e->state = REG_NAME_IDLE; pthread_cond_broadcast(&e->state_cond); } pthread_mutex_unlock(&e->state_lock); } char * reg_entry_get_apn(struct reg_entry * e) { if (!list_is_empty(&e->reg_apis) || list_is_empty(&e->reg_apns)) return NULL; return list_first_entry(&e->reg_apns, struct str_el, next)->str; } static bool reg_entry_has_api(struct reg_entry * e, pid_t api) { struct list_head * p; list_for_each(p, &e->reg_apns) { struct pid_el * e = list_entry(p, struct pid_el, next); if (e->pid == api) return true; } return false; } int reg_entry_add_api(struct reg_entry * e, pid_t api) { struct pid_el * i; assert(e); if (reg_entry_has_api(e, api)) { log_dbg("Instance already registered with this name."); return -EPERM; } pthread_mutex_lock(&e->state_lock); if (e->state == REG_NAME_NULL) { pthread_mutex_unlock(&e->state_lock); log_dbg("Tried to add instance in NULL state."); return -EPERM; } i = malloc(sizeof(*i)); if (i == NULL) { pthread_mutex_unlock(&e->state_lock); return -ENOMEM; } i->pid = api; list_add(&i->next, &e->reg_apis); if (e->state == REG_NAME_IDLE || e->state == REG_NAME_AUTO_ACCEPT || e->state == REG_NAME_AUTO_EXEC) { e->state = REG_NAME_FLOW_ACCEPT; pthread_cond_broadcast(&e->state_cond); } pthread_mutex_unlock(&e->state_lock); return 0; } static void reg_entry_check_state(struct reg_entry * e) { assert(e); if (e->state == REG_NAME_DESTROY) { e->state = REG_NAME_NULL; pthread_cond_broadcast(&e->state_cond); return; } if (list_is_empty(&e->reg_apis)) { if (!list_is_empty(&e->reg_apns)) e->state = REG_NAME_AUTO_ACCEPT; else e->state = REG_NAME_IDLE; } else { e->state = REG_NAME_FLOW_ACCEPT; } pthread_cond_broadcast(&e->state_cond); } void reg_entry_del_pid_el(struct reg_entry * e, struct pid_el * p) { assert(e); assert(p); list_del(&p->next); free(p); reg_entry_check_state(e); } void reg_entry_del_api(struct reg_entry * e, pid_t api) { struct list_head * p; struct list_head * h; assert(e); if (e == NULL) return; list_for_each_safe(p, h, &e->reg_apis) { struct pid_el * a = list_entry(p, struct pid_el, next); if (a->pid == api) { list_del(&a->next); free(a); } } reg_entry_check_state(e); } pid_t reg_entry_get_api(struct reg_entry * e) { if (e == NULL) return -1; if (list_is_empty(&e->reg_apis)) return -1; return list_first_entry(&e->reg_apis, struct pid_el, next)->pid; } enum reg_name_state reg_entry_get_state(struct reg_entry * e) { enum reg_name_state state; assert(e); pthread_mutex_lock(&e->state_lock); state = e->state; pthread_mutex_unlock(&e->state_lock); return state; } int reg_entry_set_state(struct reg_entry * e, enum reg_name_state state) { assert(state != REG_NAME_DESTROY); pthread_mutex_lock(&e->state_lock); e->state = state; pthread_cond_broadcast(&e->state_cond); pthread_mutex_unlock(&e->state_lock); return 0; } int reg_entry_leave_state(struct reg_entry * e, enum reg_name_state state, struct timespec * timeout) { struct timespec abstime; int ret = 0; assert(e); assert(state != REG_NAME_DESTROY); if (timeout != NULL) { clock_gettime(PTHREAD_COND_CLOCK, &abstime); ts_add(&abstime, timeout, &abstime); } pthread_mutex_lock(&e->state_lock); pthread_cleanup_push((void *)(void *) pthread_mutex_unlock, &e->state_lock); while (e->state == state && ret != -ETIMEDOUT) if (timeout) ret = -pthread_cond_timedwait(&e->state_cond, &e->state_lock, &abstime); else ret = -pthread_cond_wait(&e->state_cond, &e->state_lock); if (e->state == REG_NAME_DESTROY) { ret = -1; e->state = REG_NAME_NULL; pthread_cond_broadcast(&e->state_cond); } pthread_cleanup_pop(true); return ret; } int reg_entry_wait_state(struct reg_entry * e, enum reg_name_state state, struct timespec * timeout) { struct timespec abstime; int ret = 0; assert(e); assert(state != REG_NAME_DESTROY); if (timeout != NULL) { clock_gettime(PTHREAD_COND_CLOCK, &abstime); ts_add(&abstime, timeout, &abstime); } pthread_mutex_lock(&e->state_lock); while (e->state != state && e->state != REG_NAME_DESTROY && ret != -ETIMEDOUT) if (timeout) ret = -pthread_cond_timedwait(&e->state_cond, &e->state_lock, &abstime); else ret = -pthread_cond_wait(&e->state_cond, &e->state_lock); if (e->state == REG_NAME_DESTROY) { ret = -1; e->state = REG_NAME_NULL; pthread_cond_broadcast(&e->state_cond); } pthread_mutex_unlock(&e->state_lock); return ret; } struct reg_entry * registry_get_entry(struct list_head * registry, const char * name) { struct list_head * p = NULL; assert(registry); list_for_each(p, registry) { struct reg_entry * e = list_entry(p, struct reg_entry, next); if (!wildcard_match(name, e->name)) return e; } return NULL; } struct reg_entry * registry_get_entry_by_hash(struct list_head * registry, enum hash_algo algo, const uint8_t * hash, size_t len) { struct list_head * p = NULL; uint8_t * thash; thash = malloc(len); if (thash == NULL) return NULL; assert(registry); list_for_each(p, registry) { struct reg_entry * e = list_entry(p, struct reg_entry, next); str_hash(algo, thash, e->name); if (memcmp(thash, hash, len) == 0) { free(thash); return e; } } free(thash); return NULL; } struct reg_entry * registry_add_name(struct list_head * registry, const char * name) { struct reg_entry * e = NULL; assert(registry); assert(name); if (registry_has_name(registry, name)) { log_dbg("Name %s already registered.", name); return NULL; } e = reg_entry_create(); if (e == NULL) { log_dbg("Could not create registry entry."); return NULL; } if (reg_entry_init(e, strdup(name))) { reg_entry_destroy(e); log_dbg("Could not initialize registry entry."); return NULL; } list_add(&e->next, registry); return e; } void registry_del_name(struct list_head * registry, const char * name) { struct reg_entry * e = registry_get_entry(registry, name); if (e == NULL) return; list_del(&e->next); reg_entry_destroy(e); return; } void registry_del_api(struct list_head * registry, pid_t api) { struct list_head * p; assert(registry); assert(api > 0); list_for_each(p, registry) { struct reg_entry * e = list_entry(p, struct reg_entry, next); pthread_mutex_lock(&e->state_lock); assert(e); reg_entry_del_api(e, api); pthread_mutex_unlock(&e->state_lock); } return; } int registry_add_name_to_dif(struct list_head * registry, const char * name, const char * dif_name, enum ipcp_type type) { struct reg_entry * re = registry_get_entry(registry, name); if (re == NULL) return -1; return reg_entry_add_local_in_dif(re, dif_name, type); } void registry_del_name_from_dif(struct list_head * registry, const char * name, const char * dif_name) { struct reg_entry * re = registry_get_entry(registry, name); if (re == NULL) return; reg_entry_del_local_from_dif(re, dif_name); } void registry_destroy(struct list_head * registry) { struct list_head * p = NULL; struct list_head * h = NULL; assert(registry); list_for_each_safe(p, h, registry) { struct reg_entry * e = list_entry(p, struct reg_entry, next); list_del(&e->next); reg_entry_set_state(e, REG_NAME_NULL); reg_entry_destroy(e); } }