/* * lock_manager.c: Implements the internal lock manager API * * Copyright (C) 2010-2011 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . * */ #include #include "lock_manager.h" #include "lock_driver_nop.h" #include "virterror_internal.h" #include "logging.h" #include "util.h" #include "memory.h" #include "uuid.h" #if HAVE_DLFCN_H # include #endif #include #include #include "configmake.h" #define VIR_FROM_THIS VIR_FROM_LOCKING #define CHECK_DRIVER(field, errret) \ if (!driver->field) { \ virReportError(VIR_ERR_INTERNAL_ERROR, \ _("Missing '%s' field in lock manager driver"), \ #field); \ return errret; \ } #define CHECK_MANAGER(field, errret) \ if (!lock->driver->field) { \ virReportError(VIR_ERR_INTERNAL_ERROR, \ _("Missing '%s' field in lock manager driver"), \ #field); \ return errret; \ } struct _virLockManagerPlugin { char *name; virLockDriverPtr driver; void *handle; int refs; }; #define DEFAULT_LOCK_MANAGER_PLUGIN_DIR LIBDIR "/libvirt/lock-driver" static const char *virLockManagerPluginDir = DEFAULT_LOCK_MANAGER_PLUGIN_DIR; void virLockManagerSetPluginDir(const char *dir) { if (dir) virLockManagerPluginDir = dir; } static void virLockManagerLogParams(size_t nparams, virLockManagerParamPtr params) { int i; char uuidstr[VIR_UUID_STRING_BUFLEN]; for (i = 0 ; i < nparams ; i++) { switch (params[i].type) { case VIR_LOCK_MANAGER_PARAM_TYPE_INT: VIR_DEBUG(" key=%s type=int value=%d", params[i].key, params[i].value.i); break; case VIR_LOCK_MANAGER_PARAM_TYPE_UINT: VIR_DEBUG(" key=%s type=uint value=%u", params[i].key, params[i].value.ui); break; case VIR_LOCK_MANAGER_PARAM_TYPE_LONG: VIR_DEBUG(" key=%s type=long value=%lld", params[i].key, params[i].value.l); break; case VIR_LOCK_MANAGER_PARAM_TYPE_ULONG: VIR_DEBUG(" key=%s type=ulong value=%llu", params[i].key, params[i].value.ul); break; case VIR_LOCK_MANAGER_PARAM_TYPE_DOUBLE: VIR_DEBUG(" key=%s type=double value=%lf", params[i].key, params[i].value.d); break; case VIR_LOCK_MANAGER_PARAM_TYPE_STRING: VIR_DEBUG(" key=%s type=string value=%s", params[i].key, params[i].value.str); break; case VIR_LOCK_MANAGER_PARAM_TYPE_CSTRING: VIR_DEBUG(" key=%s type=cstring value=%s", params[i].key, params[i].value.cstr); break; case VIR_LOCK_MANAGER_PARAM_TYPE_UUID: virUUIDFormat(params[i].value.uuid, uuidstr); VIR_DEBUG(" key=%s type=uuid value=%s", params[i].key, uuidstr); break; } } } /** * virLockManagerPluginNew: * @name: the name of the plugin * @flag: optional plugin flags * * Attempt to load the plugin $(libdir)/libvirt/lock-driver/@name.so * The plugin driver entry point will be resolved & invoked to obtain * the lock manager driver. * * Even if the loading of the plugin succeeded, this may still * return NULL if the plugin impl decided that we (libvirtd) * are too old to support a feature it requires * * Returns a plugin object, or NULL if loading failed. */ #if HAVE_DLFCN_H virLockManagerPluginPtr virLockManagerPluginNew(const char *name, const char *configFile, unsigned int flags) { void *handle = NULL; virLockDriverPtr driver; virLockManagerPluginPtr plugin = NULL; const char *moddir = getenv("LIBVIRT_LOCK_MANAGER_PLUGIN_DIR"); char *modfile = NULL; if (STREQ(name, "nop")) { driver = &virLockDriverNop; } else { if (moddir == NULL) moddir = virLockManagerPluginDir; VIR_DEBUG("Module load %s from %s", name, moddir); if (virAsprintf(&modfile, "%s/%s.so", moddir, name) < 0) { virReportOOMError(); return NULL; } if (access(modfile, R_OK) < 0) { virReportSystemError(errno, _("Plugin %s not accessible"), modfile); goto cleanup; } handle = dlopen(modfile, RTLD_NOW | RTLD_LOCAL); if (!handle) { virReportError(VIR_ERR_SYSTEM_ERROR, _("Failed to load plugin %s: %s"), modfile, dlerror()); goto cleanup; } if (!(driver = dlsym(handle, "virLockDriverImpl"))) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing plugin initialization symbol 'virLockDriverImpl'")); goto cleanup; } } if (driver->drvInit(VIR_LOCK_MANAGER_VERSION, configFile, flags) < 0) goto cleanup; if (VIR_ALLOC(plugin) < 0) { virReportOOMError(); goto cleanup; } plugin->driver = driver; plugin->handle = handle; plugin->refs = 1; if (!(plugin->name = strdup(name))) { virReportOOMError(); goto cleanup; } VIR_FREE(modfile); return plugin; cleanup: VIR_FREE(plugin); VIR_FREE(modfile); if (handle) dlclose(handle); return NULL; } #else /* !HAVE_DLFCN_H */ virLockManagerPluginPtr virLockManagerPluginNew(const char *name ATTRIBUTE_UNUSED, const char *configFile ATTRIBUTE_UNUSED, unsigned int flags_unused ATTRIBUTE_UNUSED) { virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("this platform is missing dlopen")); return NULL; } #endif /* !HAVE_DLFCN_H */ /** * virLockManagerPluginRef: * @plugin: the plugin implementation to ref * * Acquires an additional reference on the plugin. */ void virLockManagerPluginRef(virLockManagerPluginPtr plugin) { plugin->refs++; } /** * virLockManagerPluginUnref: * @plugin: the plugin implementation to unref * * Releases a reference on the plugin. When the last reference * is released, it will attempt to unload the plugin from memory. * The plugin may refuse to allow unloading if this would * result in an unsafe scenario. * */ #if HAVE_DLFCN_H void virLockManagerPluginUnref(virLockManagerPluginPtr plugin) { if (!plugin) return; plugin->refs--; if (plugin->refs > 0) return; if (plugin->driver->drvDeinit() >= 0) { if (plugin->handle) dlclose(plugin->handle); } else { VIR_WARN("Unable to unload lock maanger plugin from memory"); return; } VIR_FREE(plugin->name); VIR_FREE(plugin); } #else /* !HAVE_DLFCN_H */ void virLockManagerPluginUnref(virLockManagerPluginPtr plugin ATTRIBUTE_UNUSED) { } #endif /* !HAVE_DLFCN_H */ const char *virLockManagerPluginGetName(virLockManagerPluginPtr plugin) { VIR_DEBUG("plugin=%p", plugin); return plugin->name; } bool virLockManagerPluginUsesState(virLockManagerPluginPtr plugin) { VIR_DEBUG("plugin=%p", plugin); return plugin->driver->flags & VIR_LOCK_MANAGER_USES_STATE; } virLockDriverPtr virLockManagerPluginGetDriver(virLockManagerPluginPtr plugin) { VIR_DEBUG("plugin=%p", plugin); return plugin->driver; } /** * virLockManagerNew: * @driver: the lock manager implementation to use * @type: the type of process to be supervised * @flags: optional flags, currently unused * * Create a new context to supervise a process, usually * a virtual machine. * * Returns a new lock manager context */ virLockManagerPtr virLockManagerNew(virLockDriverPtr driver, unsigned int type, size_t nparams, virLockManagerParamPtr params, unsigned int flags) { virLockManagerPtr lock; VIR_DEBUG("driver=%p type=%u nparams=%zu params=%p flags=%x", driver, type, nparams, params, flags); virLockManagerLogParams(nparams, params); CHECK_DRIVER(drvNew, NULL); if (VIR_ALLOC(lock) < 0) { virReportOOMError(); return NULL; } lock->driver = driver; if (driver->drvNew(lock, type, nparams, params, flags) < 0) { VIR_FREE(lock); return NULL; } return lock; } int virLockManagerAddResource(virLockManagerPtr lock, unsigned int type, const char *name, size_t nparams, virLockManagerParamPtr params, unsigned int flags) { VIR_DEBUG("lock=%p type=%u name=%s nparams=%zu params=%p flags=%x", lock, type, name, nparams, params, flags); virLockManagerLogParams(nparams, params); CHECK_MANAGER(drvAddResource, -1); return lock->driver->drvAddResource(lock, type, name, nparams, params, flags); } int virLockManagerAcquire(virLockManagerPtr lock, const char *state, unsigned int flags, virDomainLockFailureAction action, int *fd) { VIR_DEBUG("lock=%p state='%s' flags=%x action=%d fd=%p", lock, NULLSTR(state), flags, action, fd); CHECK_MANAGER(drvAcquire, -1); if (fd) *fd = -1; return lock->driver->drvAcquire(lock, state, flags, action, fd); } int virLockManagerRelease(virLockManagerPtr lock, char **state, unsigned int flags) { VIR_DEBUG("lock=%p state=%p flags=%x", lock, state, flags); CHECK_MANAGER(drvRelease, -1); return lock->driver->drvRelease(lock, state, flags); } int virLockManagerInquire(virLockManagerPtr lock, char **state, unsigned int flags) { VIR_DEBUG("lock=%p state=%p flags=%x", lock, state, flags); CHECK_MANAGER(drvInquire, -1); return lock->driver->drvInquire(lock, state, flags); } int virLockManagerFree(virLockManagerPtr lock) { VIR_DEBUG("lock=%p", lock); if (!lock) return 0; CHECK_MANAGER(drvFree, -1); lock->driver->drvFree(lock); VIR_FREE(lock); return 0; }