/* * Copyright (C) 2008 Martin Willi * Hochschule fuer Technik Rapperswil * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. See . * * 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. */ #include "ha_plugin.h" #include "ha_ike.h" #include "ha_child.h" #include "ha_socket.h" #include "ha_tunnel.h" #include "ha_dispatcher.h" #include "ha_segments.h" #include "ha_ctl.h" #include "ha_cache.h" #include "ha_attribute.h" #include #include #include typedef struct private_ha_plugin_t private_ha_plugin_t; /** * private data of ha plugin */ struct private_ha_plugin_t { /** * implements plugin interface */ ha_plugin_t public; /** * Communication socket */ ha_socket_t *socket; /** * Tunnel securing sync messages. */ ha_tunnel_t *tunnel; /** * IKE_SA synchronization */ ha_ike_t *ike; /** * CHILD_SA synchronization */ ha_child_t *child; /** * Dispatcher to process incoming messages */ ha_dispatcher_t *dispatcher; /** * Active/Passive segment management */ ha_segments_t *segments; /** * Interface to control segments at kernel level */ ha_kernel_t *kernel; /** * Segment control interface via FIFO */ ha_ctl_t *ctl; /** * Message cache for resynchronization */ ha_cache_t *cache; /** * Attribute provider */ ha_attribute_t *attr; }; METHOD(plugin_t, get_name, char*, private_ha_plugin_t *this) { return "ha"; } /** * Register listener */ static bool plugin_cb(private_ha_plugin_t *this, plugin_feature_t *feature, bool reg, void *cb_data) { if (reg) { charon->bus->add_listener(charon->bus, &this->segments->listener); charon->bus->add_listener(charon->bus, &this->ike->listener); charon->bus->add_listener(charon->bus, &this->child->listener); hydra->attributes->add_provider(hydra->attributes, &this->attr->provider); } else { hydra->attributes->remove_provider(hydra->attributes, &this->attr->provider); charon->bus->remove_listener(charon->bus, &this->segments->listener); charon->bus->remove_listener(charon->bus, &this->ike->listener); charon->bus->remove_listener(charon->bus, &this->child->listener); } return TRUE; } METHOD(plugin_t, get_features, int, private_ha_plugin_t *this, plugin_feature_t *features[]) { static plugin_feature_t f[] = { PLUGIN_CALLBACK((plugin_feature_callback_t)plugin_cb, NULL), PLUGIN_PROVIDE(CUSTOM, "ha"), }; *features = f; return countof(f); } METHOD(plugin_t, destroy, void, private_ha_plugin_t *this) { DESTROY_IF(this->ctl); this->ike->destroy(this->ike); this->child->destroy(this->child); this->dispatcher->destroy(this->dispatcher); this->attr->destroy(this->attr); this->cache->destroy(this->cache); this->segments->destroy(this->segments); this->kernel->destroy(this->kernel); this->socket->destroy(this->socket); DESTROY_IF(this->tunnel); free(this); } /** * Plugin constructor */ plugin_t *ha_plugin_create() { private_ha_plugin_t *this; char *local, *remote, *secret; u_int count; bool fifo, monitor, resync; local = lib->settings->get_str(lib->settings, "%s.plugins.ha.local", NULL, charon->name); remote = lib->settings->get_str(lib->settings, "%s.plugins.ha.remote", NULL, charon->name); secret = lib->settings->get_str(lib->settings, "%s.plugins.ha.secret", NULL, charon->name); fifo = lib->settings->get_bool(lib->settings, "%s.plugins.ha.fifo_interface", TRUE, charon->name); monitor = lib->settings->get_bool(lib->settings, "%s.plugins.ha.monitor", TRUE, charon->name); resync = lib->settings->get_bool(lib->settings, "%s.plugins.ha.resync", TRUE, charon->name); count = min(SEGMENTS_MAX, lib->settings->get_int(lib->settings, "%s.plugins.ha.segment_count", 1, charon->name)); if (!local || !remote) { DBG1(DBG_CFG, "HA config misses local/remote address"); return NULL; } if (!lib->caps->keep(lib->caps, CAP_CHOWN)) { /* required to chown(2) control socket, ha_kernel also needs it at * runtime */ DBG1(DBG_CFG, "ha plugin requires CAP_CHOWN capability"); return NULL; } INIT(this, .public = { .plugin = { .get_name = _get_name, .get_features = _get_features, .destroy = _destroy, }, }, ); if (secret) { this->tunnel = ha_tunnel_create(local, remote, secret); } this->socket = ha_socket_create(local, remote); if (!this->socket) { DESTROY_IF(this->tunnel); free(this); return NULL; } this->kernel = ha_kernel_create(count); this->segments = ha_segments_create(this->socket, this->kernel, this->tunnel, count, strcmp(local, remote) > 0, monitor); this->cache = ha_cache_create(this->kernel, this->socket, resync, count); if (fifo) { this->ctl = ha_ctl_create(this->segments, this->cache); } this->attr = ha_attribute_create(this->kernel, this->segments); this->dispatcher = ha_dispatcher_create(this->socket, this->segments, this->cache, this->kernel, this->attr); this->ike = ha_ike_create(this->socket, this->tunnel, this->cache); this->child = ha_child_create(this->socket, this->tunnel, this->segments, this->kernel); return &this->public.plugin; }