fence-virt/server/plugin.c
2012-10-26 18:15:31 -04:00

426 lines
8.3 KiB
C

/*
Copyright Red Hat, Inc. 2002-2004, 2009
The Magma Cluster API 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.
The Magma Cluster API 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, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
USA.
*/
/** @file
* Plugin loading routines
*/
#include <dlfcn.h>
#include <stdlib.h>
#include <errno.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdint.h>
#include <list.h>
#include <simpleconfig.h>
#include <static_map.h>
#include <server_plugin.h>
#include <malloc.h>
#include <string.h>
#include <dirent.h>
#include <debug.h>
typedef struct _plugin_list {
list_head();
const listener_plugin_t *listener;
const backend_plugin_t *backend;
plugin_type_t type;
} plugin_list_t;
static plugin_list_t *server_plugins = NULL;
int
plugin_reg_backend(const backend_plugin_t *plugin)
{
plugin_list_t *newplug;
if (plugin_find_backend(plugin->name)) {
errno = EEXIST;
return -1;
}
newplug = malloc(sizeof(*newplug));
if (!newplug)
return -1;
memset(newplug, 0, sizeof(*newplug));
newplug->backend = plugin;
newplug->type = PLUGIN_BACKEND;
list_insert(&server_plugins, newplug);
return 0;
}
int
plugin_reg_listener(const listener_plugin_t *plugin)
{
plugin_list_t *newplug;
if (plugin_find_listener(plugin->name)) {
errno = EEXIST;
return -1;
}
newplug = malloc(sizeof(*newplug));
if (!newplug)
return -1;
memset(newplug, 0, sizeof(*newplug));
newplug->listener = plugin;
newplug->type = PLUGIN_LISTENER;
list_insert(&server_plugins, newplug);
return 0;
}
void
plugin_dump(void)
{
plugin_list_t *p;
int x, y;
y = 0;
list_for(&server_plugins, p, x) {
if (p->type == PLUGIN_BACKEND) {
if (!y) {
y = 1;
printf("Available backends:\n");
}
printf(" %s %s\n",
p->backend->name, p->backend->version);
}
}
y = 0;
list_for(&server_plugins, p, x) {
if (p->type == PLUGIN_LISTENER) {
if (!y) {
y = 1;
printf("Available listeners:\n");
}
printf(" %s %s\n",
p->listener->name, p->listener->version);
}
}
}
const backend_plugin_t *
plugin_find_backend(const char *name)
{
plugin_list_t *p;
int x;
list_for(&server_plugins, p, x) {
if (p->type != PLUGIN_BACKEND)
continue;
if (!strcasecmp(name, p->backend->name))
return p->backend;
}
return NULL;
}
const listener_plugin_t *
plugin_find_listener(const char *name)
{
plugin_list_t *p;
int x;
list_for(&server_plugins, p, x) {
if (p->type != PLUGIN_LISTENER)
continue;
if (!strcasecmp(name, p->listener->name))
return p->listener;
}
return NULL;
}
static int
backend_plugin_load(void *handle, const char *libpath)
{
const backend_plugin_t *plug = NULL;
double (*modversion)(void);
backend_plugin_t *(*modinfo)(void);
modversion = dlsym(handle, BACKEND_VER_STR);
if (!modversion) {
dbg_printf(1, "Failed to map %s\n", BACKEND_VER_STR);
errno = EINVAL;
return -1;
}
if (modversion() != PLUGIN_VERSION_BACKEND) {
dbg_printf(1, "API version mismatch in %s: \n"
" %f expected; %f received.\n", libpath,
PLUGIN_VERSION_BACKEND, modversion());
errno = EINVAL;
return -1;
}
modinfo = dlsym(handle, BACKEND_INFO_STR);
if (!modinfo) {
dbg_printf(1, "Failed to map %s\n", BACKEND_INFO_STR);
errno = EINVAL;
return -1;
}
plug = modinfo();
if (plugin_reg_backend(plug) < 0) {
dbg_printf(1, "Failed to register %s %s\n", plug->name,
plug->version);
errno = EINVAL;
return -1;
} else {
dbg_printf(1, "Registered backend plugin %s %s\n",
plug->name, plug->version);
}
return 0;
}
static int
listener_plugin_load(void *handle, const char *libpath)
{
const listener_plugin_t *plug = NULL;
double (*modversion)(void);
listener_plugin_t *(*modinfo)(void);
modversion = dlsym(handle, LISTENER_VER_STR);
if (!modversion) {
dbg_printf(1, "Failed to map %s\n", LISTENER_VER_STR);
errno = EINVAL;
return -1;
}
if (modversion() != PLUGIN_VERSION_LISTENER) {
dbg_printf(1, "API version mismatch in %s: \n"
" %f expected; %f received.\n", libpath,
PLUGIN_VERSION_LISTENER, modversion());
dlclose(handle);
errno = EINVAL;
return -1;
}
modinfo = dlsym(handle, LISTENER_INFO_STR);
if (!modinfo) {
dbg_printf(1, "Failed to map %s\n", LISTENER_INFO_STR);
errno = EINVAL;
return -1;
}
plug = modinfo();
if (plugin_reg_listener(plug) < 0) {
dbg_printf(1, "Failed to register %s %s\n", plug->name,
plug->version);
errno = EINVAL;
return -1;
} else {
dbg_printf(1, "Registered listener plugin %s %s\n",
plug->name, plug->version);
}
return 0;
}
/**
* Load a cluster plugin .so file and map all the functions
* provided to entries in a backend_plugin_t structure.
*
* @param libpath Path to file.
* @return NULL on failure, or plugin-specific
* (const) backend_plugin_t * structure on
* success.
*/
int
plugin_load(const char *libpath)
{
void *handle = NULL;
struct stat sb;
errno = 0;
if (!libpath) {
errno = EINVAL;
return -1;
}
if (stat(libpath, &sb) != 0) {
return -1;
}
/*
If it's not owner-readable or it's a directory,
ignore/fail. Thus, a user may change the permission of
a plugin "u-r" and this would then prevent magma apps
from loading it.
*/
if (S_ISDIR(sb.st_mode)) {
errno = EISDIR;
return -1;
}
if (!(sb.st_mode & S_IRUSR)) {
dbg_printf(1, "Ignoring %s (User-readable bit not set)\n",
libpath);
errno = EPERM;
return -1;
}
dbg_printf(3, "Loading plugin from %s\n", libpath);
handle = dlopen(libpath, RTLD_NOW);
if (!handle) {
dbg_printf(3, "Could not dlopen %s: %s\n", libpath, dlerror());
errno = ELIBACC;
return -1;
}
if (!backend_plugin_load(handle, libpath) ||
!listener_plugin_load(handle, libpath))
return 0;
dbg_printf(3, "%s is not a valid plugin\n", libpath);
dlclose(handle);
errno = EINVAL;
return -1;
}
/**
Free up a null-terminated array of strings
*/
static void
free_dirnames(char **dirnames)
{
int x = 0;
for (; dirnames[x]; x++)
free(dirnames[x]);
free(dirnames);
}
static int
_compare(const void *a, const void *b)
{
return strcmp((const char *)a, (const char *)b);
}
/**
Read all entries in a directory and return them in a NULL-terminated,
sorted array.
*/
static int
read_dirnames_sorted(const char *directory, char ***dirnames)
{
DIR *dir;
struct dirent *entry;
char filename[1024];
int count = 0, x = 0;
dir = opendir(directory);
if (!dir)
return -1;
/* Count the number of plugins */
while ((entry = readdir(dir)) != NULL)
++count;
/* Malloc the entries */
*dirnames = malloc(sizeof(char *) * (count+1));
if (!*dirnames) {
#ifdef DEBUG
printf("%s: Failed to malloc %d bytes",
__FUNCTION__, (int)(sizeof(char *) * (count+1)));
#endif
closedir(dir);
errno = ENOMEM;
return -1;
}
memset(*dirnames, 0, sizeof(char *) * (count + 1));
rewinddir(dir);
/* Store the directory names. */
while ((entry = readdir(dir)) != NULL) {
snprintf(filename, sizeof(filename), "%s/%s", directory,
entry->d_name);
(*dirnames)[x] = strdup(filename);
if (!(*dirnames)[x]) {
#ifdef DEBUG
printf("Failed to duplicate %s\n",
filename);
#endif
free_dirnames(*dirnames);
closedir(dir);
errno = ENOMEM;
return -1;
}
++x;
}
closedir(dir);
/* Sort the directory names. */
qsort((*dirnames), count, sizeof(char *), _compare);
return 0;
}
/**
*/
int
plugin_search(const char *pathname)
{
int found = 0;
int fcount = 0;
char **filenames;
dbg_printf(1, "Searching for plugins in %s\n", pathname);
if (read_dirnames_sorted(pathname, &filenames) != 0) {
return -1;
}
for (fcount = 0; filenames[fcount]; fcount++) {
if (plugin_load(filenames[fcount]) == 0)
++found;
}
free_dirnames(filenames);
if (!found) {
dbg_printf(1, "No usable plugins found.\n");
errno = ELIBACC;
return -1;
}
return found;
}