/* * Copyright (C) 2008 Tobias Brunner * Copyright (C) 2007 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. */ #define _GNU_SOURCE #include #include #include #include #include #include #include #include #include "dumm.h" #define PERME (S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) #define GUEST_DIR "guests" #define TEMPLATE_DIR "templates" typedef struct private_dumm_t private_dumm_t; struct private_dumm_t { /** public dumm interface */ dumm_t public; /** working dir */ char *dir; /** directory of guests */ char *guest_dir; /** directory of loaded template */ char *template; /** list of managed guests */ linked_list_t *guests; /** list of managed bridges */ linked_list_t *bridges; }; /** * Implementation of dumm_t.create_guest. */ static guest_t* create_guest(private_dumm_t *this, char *name, char *kernel, char *master, char *args) { guest_t *guest; guest = guest_create(this->guest_dir, name, kernel, master, args); if (guest) { this->guests->insert_last(this->guests, guest); } return guest; } /** * Implementation of dumm_t.create_guest_enumerator. */ static enumerator_t* create_guest_enumerator(private_dumm_t *this) { return this->guests->create_enumerator(this->guests); } /** * Implementation of dumm_t.delete_guest. */ static void delete_guest(private_dumm_t *this, guest_t *guest) { if (this->guests->remove(this->guests, guest, NULL)) { char buf[512]; int len; len = snprintf(buf, sizeof(buf), "rm -Rf %s/%s", this->guest_dir, guest->get_name(guest)); guest->destroy(guest); if (len > 8 && len < 512) { ignore_result(system(buf)); } } } /** * Implementation of dumm_t.create_bridge. */ static bridge_t* create_bridge(private_dumm_t *this, char *name) { bridge_t *bridge; bridge = bridge_create(name); if (bridge) { this->bridges->insert_last(this->bridges, bridge); } return bridge; } /** * Implementation of dumm_t.create_bridge_enumerator. */ static enumerator_t* create_bridge_enumerator(private_dumm_t *this) { return this->bridges->create_enumerator(this->bridges); } /** * Implementation of dumm_t.delete_bridge. */ static void delete_bridge(private_dumm_t *this, bridge_t *bridge) { if (this->bridges->remove(this->bridges, bridge, NULL)) { bridge->destroy(bridge); } } /** * disable the currently enabled template */ static void clear_template(private_dumm_t *this) { enumerator_t *enumerator; guest_t *guest; free(this->template); this->template = NULL; enumerator = this->guests->create_enumerator(this->guests); while (enumerator->enumerate(enumerator, (void**)&guest)) { guest->load_template(guest, NULL); } enumerator->destroy(enumerator); } /** * Implementation of dumm_t.load_template. */ static bool load_template(private_dumm_t *this, char *dir) { enumerator_t *enumerator; guest_t *guest; clear_template(this); if (dir == NULL) { return TRUE; } if (strlen(dir) > PATH_MAX) { DBG1("template directory string '%s' is too long", dir); return FALSE; } if (asprintf(&this->template, "%s/%s", TEMPLATE_DIR, dir) < 0) { this->template = NULL; return FALSE; } if (access(this->template, F_OK) != 0) { /* does not exist, create template */ if (!mkdir_p(this->template, PERME)) { DBG1("creating template directory '%s' failed: %m", this->template); return FALSE; } } enumerator = this->guests->create_enumerator(this->guests); while (enumerator->enumerate(enumerator, (void**)&guest)) { if (!guest->load_template(guest, this->template)) { enumerator->destroy(enumerator); clear_template(this); return FALSE; } } enumerator->destroy(enumerator); return TRUE; } /** * Template directory enumerator */ typedef struct { /** implements enumerator_t */ enumerator_t public; /** directory enumerator */ enumerator_t *inner; } template_enumerator_t; /** * Implementation of template_enumerator_t.enumerate */ static bool template_enumerate(template_enumerator_t *this, char **template) { struct stat st; char *rel; while (this->inner->enumerate(this->inner, &rel, NULL, &st)) { if (S_ISDIR(st.st_mode) && *rel != '.') { *template = rel; return TRUE; } } return FALSE; } /** * Implementation of template_enumerator_t.destroy */ static void template_enumerator_destroy(template_enumerator_t *this) { this->inner->destroy(this->inner); free(this); } /** * Implementation of dumm_t.create_template_enumerator */ static enumerator_t* create_template_enumerator(private_dumm_t *this) { template_enumerator_t *enumerator; enumerator = malloc_thing(template_enumerator_t); enumerator->public.enumerate = (void*)template_enumerate; enumerator->public.destroy = (void*)template_enumerator_destroy; enumerator->inner = enumerator_create_directory(TEMPLATE_DIR); return &enumerator->public; } /** * Implementation of dumm_t.destroy */ static void destroy(private_dumm_t *this) { enumerator_t *enumerator; guest_t *guest; this->bridges->destroy_offset(this->bridges, offsetof(bridge_t, destroy)); enumerator = this->guests->create_enumerator(this->guests); while (enumerator->enumerate(enumerator, (void**)&guest)) { guest->stop(guest, NULL); } enumerator->destroy(enumerator); while (this->guests->remove_last(this->guests, (void**)&guest) == SUCCESS) { guest->destroy(guest); } this->guests->destroy(this->guests); free(this->guest_dir); free(this->template); free(this->dir); free(this); } /** * load all guests in our working dir */ static void load_guests(private_dumm_t *this) { DIR *dir; struct dirent *ent; guest_t *guest; dir = opendir(this->guest_dir); if (dir == NULL) { return; } while ((ent = readdir(dir))) { if (*ent->d_name == '.') { /* skip ".", ".." and hidden files (such as ".svn") */ continue; } guest = guest_load(this->guest_dir, ent->d_name); if (guest) { this->guests->insert_last(this->guests, guest); } else { DBG1("loading guest in directory '%s' failed, skipped", ent->d_name); } } closedir(dir); } /** * create a dumm instance */ dumm_t *dumm_create(char *dir) { char cwd[PATH_MAX]; private_dumm_t *this = malloc_thing(private_dumm_t); this->public.create_guest = (guest_t*(*)(dumm_t*,char*,char*,char*,char*))create_guest; this->public.create_guest_enumerator = (enumerator_t*(*)(dumm_t*))create_guest_enumerator; this->public.delete_guest = (void(*)(dumm_t*,guest_t*))delete_guest; this->public.create_bridge = (bridge_t*(*)(dumm_t*, char *name))create_bridge; this->public.create_bridge_enumerator = (enumerator_t*(*)(dumm_t*))create_bridge_enumerator; this->public.delete_bridge = (void(*)(dumm_t*,bridge_t*))delete_bridge; this->public.load_template = (bool(*)(dumm_t*, char *name))load_template; this->public.create_template_enumerator = (enumerator_t*(*)(dumm_t*))create_template_enumerator; this->public.destroy = (void(*)(dumm_t*))destroy; if (dir && *dir == '/') { this->dir = strdup(dir); } else { if (getcwd(cwd, sizeof(cwd)) == NULL) { free(this); return NULL; } if (dir) { if (asprintf(&this->dir, "%s/%s", cwd, dir) < 0) { this->dir = NULL; } } else { this->dir = strdup(cwd); } } this->template = NULL; if (asprintf(&this->guest_dir, "%s/%s", this->dir, GUEST_DIR) < 0) { this->guest_dir = NULL; } this->guests = linked_list_create(); this->bridges = linked_list_create(); if (this->dir == NULL || this->guest_dir == NULL || (mkdir(this->guest_dir, PERME) < 0 && errno != EEXIST)) { DBG1("creating guest directory '%s' failed: %m", this->guest_dir); destroy(this); return NULL; } load_guests(this); return &this->public; }