Blob Blame History Raw
/*      -*- linux-c -*-
 *
 * Copyright (c) 2003 by Intel Corp.
 * (C) Copyright IBM Corp. 2003, 2005
 *
 * 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.  This
 * file and program are licensed under a BSD style license.  See
 * the Copying file included with the OpenHPI distribution for
 * full licensing terms.
 *
 * Authors:
 *     Louis Zhuang <louis.zhuang@linux.intel.com>
 *     Sean Dague <http://dague.net/sean>
 *     David Judkovics <djudkovi@us.ibm.com>
 *     Renier Morales <renier@openhpi.org>
 *     Anton Pak <anton.pak@pigeonpoint.com>
 */

#include <inttypes.h>
#include <stddef.h>
#include <string.h>

#include <glib.h>
#include <gmodule.h>

#include <config.h>

#include <oh_domain.h>
#include <oh_error.h>
#include <oh_plugin.h>
#include <oh_utils.h>

#include "conf.h"
#include "event.h"
#include "lock.h"
#include "sahpi_wrappers.h"

extern volatile int signal_stop;
/*
 * Structure containing global list of plugins (oh_plugin).
 */
struct oh_plugins oh_plugins = {
        .list = NULL,
#if !GLIB_CHECK_VERSION (2, 32, 0)
        .lock = G_STATIC_REC_MUTEX_INIT 
#endif
};
/*
 * Structure containing global table of handlers (oh_handler).
 */
struct oh_handlers oh_handlers = {
        .table = NULL,
        .list = NULL,
#if !GLIB_CHECK_VERSION (2, 32, 0)
        .lock = G_STATIC_REC_MUTEX_INIT
#endif
};

/**
 * oh_close_handlers
 *
 * When a client calls exit() this function
 * gets called to clean up the handlers
 * and any handler connections to the hardware.
 * Until clients use dynamic oHPI interface
 *
 * Returns: void
 **/
void oh_close_handlers()
{
        struct oh_handler *h = NULL;
        GSList *node = NULL;

        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        if (oh_handlers.list == NULL) {
                wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                return;
        }

        for (node = oh_handlers.list; node; node = node->next) {
                h = node->data;
                if (h && h->abi && h->abi->close) {
                        if (h->hnd) {
                            h->abi->close(h->hnd);
                        }
                }
        }
        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
}

static void __inc_plugin_refcount(struct oh_plugin *p)
{
        wrap_g_static_rec_mutex_lock(&p->refcount_lock);
        p->refcount++;
        wrap_g_static_rec_mutex_unlock(&p->refcount_lock);
}

static void __dec_plugin_refcount(struct oh_plugin *p)
{
        wrap_g_static_rec_mutex_lock(&p->refcount_lock);
        p->refcount--;
        wrap_g_static_rec_mutex_unlock(&p->refcount_lock);
}

static void __delete_plugin(struct oh_plugin *p)
{
        if (!p) return;

        g_free(p->name);
        wrap_g_static_rec_mutex_free_clear(&p->lock);
        wrap_g_static_rec_mutex_free_clear(&p->refcount_lock);
        g_free(p->abi);
        if (p->dl_handle) {
                g_module_close(p->dl_handle);
        }
        g_free(p);
}

/**
 * oh_get_plugin
 * @plugin_name: name of plugin. string.
 *
 * Lookup and get a reference for @plugin_name.
 * Locks out the plugin. Need to call oh_putback_plugin
 * to unlock it.
 *
 * Returns: oh_plugin reference or NULL if plugin_name was not found.
 **/
struct oh_plugin *oh_get_plugin(char *plugin_name)
{
        GSList *node = NULL;
        struct oh_plugin *plugin = NULL;

        if (!plugin_name) {
                CRIT("ERROR getting plugin. Invalid parameter.");
                return NULL;
        }

        wrap_g_static_rec_mutex_lock(&oh_plugins.lock);
        for (node = oh_plugins.list; node; node = node->next) {
                struct oh_plugin *p = (struct oh_plugin *)(node->data);
                if (strcmp(p->name, plugin_name) == 0) {
                        __inc_plugin_refcount(p);
                        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);
                        wrap_g_static_rec_mutex_lock(&p->lock);
                        plugin = p;
                        return plugin;
                }
        }
        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);

        return plugin;
}

/**
 * oh_release_plugin
 * @plugin: Pointer to plugin
 *
 * Decrements refcount on plugin and unlocks it.
 *
 * Returns: void
 **/
void oh_release_plugin(struct oh_plugin *plugin)
{
        if (!plugin) {
                CRIT("WARNING - NULL plugin parameter passed.");
                return;
        }

        __dec_plugin_refcount(plugin);
        if (plugin->refcount < 0)
                __delete_plugin(plugin);
        else
                wrap_g_static_rec_mutex_unlock(&plugin->lock);
}

/**
 * oh_getnext_plugin_name
 * @plugin_name: IN. If NULL is passed, the first plugin in the list is returned.
 * @next_plugin_name: OUT. Buffer to print the next plugin name to.
 * @size: IN. Size of the buffer at @next_plugin_name.
 *
 * Returns: 0 on Success.
 **/
int oh_getnext_plugin_name(char *plugin_name,
                           char *next_plugin_name,
                           unsigned int size)
{
        GSList *node = NULL;

        if (!next_plugin_name) {
                CRIT("ERROR. Invalid parameter.");
                return -1;
        }
        memset(next_plugin_name, '\0', size);

        if (!plugin_name) {
                wrap_g_static_rec_mutex_lock(&oh_plugins.lock);
                if (oh_plugins.list) {
                        struct oh_plugin *plugin = oh_plugins.list->data;
                        strncpy(next_plugin_name, plugin->name, size);
                        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);
                        return 0;
                } else {
                        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);
                        DBG("No plugins have been loaded yet.");
                        return -1;
                }
        } else {
                wrap_g_static_rec_mutex_lock(&oh_plugins.lock);
                for (node = oh_plugins.list; node; node = node->next) {
                        struct oh_plugin *p = node->data;
                        if (strcmp(p->name, plugin_name) == 0) {
                                if (node->next) {
                                        p = node->next->data;
                                        strncpy(next_plugin_name, p->name, size);
                                        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);
                                        return 0;
                                } else {
                                        break;
                                }
                        }
                }
                wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);
        }

        return -1;
}

/**
 * oh_load_plugin
 * @plugin_name: name of plugin to be loaded (e.g. "libdummy").
 *
 * Load plugin by name
 *
 * Returns: 0 on Success.
 **/
int oh_load_plugin(char *plugin_name)
{
        struct oh_plugin *plugin = NULL;
        int err;

        if (!plugin_name) {
                CRIT("ERROR. NULL plugin name passed.");
                return -1;
        }

        if (g_module_supported() == FALSE) {
                CRIT("ERROR. GModule is not supported. Cannot load plugins.");
                return -1;
        }

        plugin = oh_get_plugin(plugin_name);
        if (plugin) {
                oh_release_plugin(plugin);
                DBG("Plugin %s already loaded. Not loading twice.",
                    plugin_name);
                return 0;
        }

        plugin = g_new0(struct oh_plugin, 1);
        if (!plugin) {
                CRIT("Out of memory.");
                return -1;
        }
        plugin->name = g_strdup(plugin_name);
        plugin->handler_count = 0;
        plugin->refcount = 0;
        wrap_g_static_rec_mutex_init(&plugin->lock);
        wrap_g_static_rec_mutex_init(&plugin->refcount_lock);

#ifdef _WIN32
        plugin->dl_handle = g_module_open(plugin->name, G_MODULE_BIND_LOCAL);
#else
        {
            struct oh_global_param path_param = { .type = OPENHPI_PATH };
            gchar **plugin_search_dirs;
            size_t i;
            gchar *plugin_path;

            oh_get_global_param(&path_param);
            plugin_search_dirs = g_strsplit(path_param.u.path, ":", -1);
            for( i = 0; plugin_search_dirs[i] != 0; ++i) {
                plugin_path = g_module_build_path(plugin_search_dirs[i], plugin->name);
                plugin->dl_handle = g_module_open(plugin_path, G_MODULE_BIND_LOCAL);
                g_free(plugin_path);
                if (plugin->dl_handle) {
                    break;
                }
            }
            g_strfreev(plugin_search_dirs);
        }
#endif
        if (plugin->dl_handle == NULL) {
                CRIT("Can not open %s plugin: %s", plugin->name, g_module_error());
                goto cleanup_and_quit;
        }

        err = oh_load_plugin_functions(plugin, &plugin->abi);

        if (err < 0 || !plugin->abi || !plugin->abi->open) {
                CRIT("Can not get ABI");
                goto cleanup_and_quit;
        }
        wrap_g_static_rec_mutex_lock(&oh_plugins.lock);
        oh_plugins.list = g_slist_append(oh_plugins.list, plugin);
        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);

        return 0;
cleanup_and_quit:
        __delete_plugin(plugin);
        return -1;
}

/**
 * oh_unload_plugin
 * @plugin_name: String. Name of plugin to unload.
 *
 * Returns: 0 on Success.
 **/
int oh_unload_plugin(char *plugin_name)
{
        struct oh_plugin *plugin = NULL;

        if (!plugin_name) {
                CRIT("ERROR unloading plugin. NULL parameter passed.");
                return -1;
        }

        plugin = oh_get_plugin(plugin_name);
        if (!plugin) {
                CRIT("ERROR unloading plugin. Plugin not found.");
                return -2;
        }

        if (plugin->handler_count > 0) {
                oh_release_plugin(plugin);
                CRIT("ERROR unloading plugin. Handlers are still referencing it.");
                return -3;
        }

        wrap_g_static_rec_mutex_lock(&oh_plugins.lock);
        oh_plugins.list = g_slist_remove(oh_plugins.list, plugin);
        wrap_g_static_rec_mutex_unlock(&oh_plugins.lock);

        __dec_plugin_refcount(plugin);
        if (plugin->refcount < 1)
                __delete_plugin(plugin);
        else
                oh_release_plugin(plugin);

        return 0;
}

static void __inc_handler_refcount(struct oh_handler *h)
{
        wrap_g_static_rec_mutex_lock(&h->refcount_lock);
        h->refcount++;
        wrap_g_static_rec_mutex_unlock(&h->refcount_lock);
}

static void __dec_handler_refcount(struct oh_handler *h)
{
        wrap_g_static_rec_mutex_lock(&h->refcount_lock);
        h->refcount--;
        wrap_g_static_rec_mutex_unlock(&h->refcount_lock);
}

static void __delete_handler(struct oh_handler *h)
{
        struct oh_plugin *plugin = NULL;

        if (!h) return;

        /* Subtract one from the number of handlers using this plugin */
        plugin = oh_get_plugin(h->plugin_name);
        if (!plugin) {
                CRIT("BAD ERROR - Handler loaded, but plugin does not exist!");
        } else {
                plugin->handler_count--;
                oh_release_plugin(plugin);
        }

        /* Free the oh_handler members first, then the handler. */
        g_hash_table_destroy(h->config);

        wrap_g_static_rec_mutex_free_clear(&h->lock);
        wrap_g_static_rec_mutex_free_clear(&h->refcount_lock);
        g_free(h);
}

/**
 * oh_get_handler
 * @hid: id of handler being requested
 *
 * Returns: NULL if handler was not found.
 **/
struct oh_handler *oh_get_handler(unsigned int hid)
{
        GSList *node = NULL;
        struct oh_handler *handler = NULL;

        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        node = g_hash_table_lookup(oh_handlers.table, &hid);
        handler = node ? node->data : NULL;
        if (!handler) {
                wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                CRIT("Error - Handler %d was not found", hid);
                return NULL;
        }
        __inc_handler_refcount(handler);
        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
        wrap_g_static_rec_mutex_lock(&handler->lock);

        return handler;
}

/**
 * oh_release_handler
 * @handler: a handler, previously obtained (i.e. locked) with
 * oh_get_handler(), to be released (i.e. unlocked).
 *
 * Returns: void
 **/
void oh_release_handler(struct oh_handler *handler)
{
        if (!handler) {
                CRIT("Warning - NULL parameter passed.");
                return;
        }

        __dec_handler_refcount(handler);
        if (handler->refcount < 0)
                __delete_handler(handler);
        else
                wrap_g_static_rec_mutex_unlock(&handler->lock);
}

/**
 * oh_getnext_handler_id
 * @hid: If 0, will return the first handler id in the list.
 * Otherwise, indicates handler id previous to the one being requested.
 * @next_hid: Place where the next handler id after @hid
 * will be put.
 *
 * Returns: 0 on Success.
 **/
int oh_getnext_handler_id(unsigned int hid, unsigned int *next_hid)
{
        GSList *node = NULL;
        struct oh_handler *h = NULL;

        if (!next_hid) {
                CRIT("ERROR. Invalid parameter.");
                return -1;
        }
        *next_hid = 0;

        if (!hid) { /* Return first handler id in the list */
                wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
                if (oh_handlers.list) {
                        h = oh_handlers.list->data;
                        *next_hid = h->id;
                        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                        return 0;
                } else {
                        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                        CRIT("Warning - no handlers");
                        return -1;
                }
        } else { /* Return handler id coming after hid in the list */
                wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
                node = g_hash_table_lookup(oh_handlers.table, &hid);
                if (node && node->next && node->next->data) {
                        h = node->next->data;
                        *next_hid = h->id;
                        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                        return 0;
                }
                wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
        }

        return -1;
}

static void copy_hashed_new_config (gpointer key, gpointer value, gpointer newhash)
{
   g_hash_table_insert ( newhash, g_strdup(key), g_strdup(value) );
}
static struct oh_handler *new_handler(GHashTable *handler_config)
{       /* Return a new oh_handler instance */
        struct oh_plugin *plugin = NULL;
        struct oh_handler *handler = NULL;
	char *plugin_name = NULL;
        static unsigned int handler_id = 1;
        GHashTable *newconfig;

        if (!handler_config) {
                CRIT("ERROR creating new handler. Invalid parameter.");
                return NULL;
        }

	plugin_name = (char *)g_hash_table_lookup(handler_config, "plugin");
	if (!plugin_name) {
		CRIT("ERROR creating new handler. No plugin name received.");
		return NULL;
	}

        handler = g_new0(struct oh_handler, 1);

        plugin = oh_get_plugin(plugin_name);
        if(!plugin) { /* Attempt to load plugin here once */
		int rc = oh_load_plugin(plugin_name);
		if (rc) {
                	CRIT("Could not create handler. Plugin %s not loaded",
                    	    plugin_name);
                	goto cleanexit;
		}

		plugin = oh_get_plugin(plugin_name);
		if (!plugin) {
			CRIT("Tried but could not get a plugin to "
			    "create this handler.");
			goto cleanexit;
		}
        }

        /* Initialize handler */
        handler->abi = plugin->abi;
        plugin->handler_count++; /* Increment # of handlers using the plugin */
        oh_release_plugin(plugin);
        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        handler->id = handler_id++;
        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
        handler->plugin_name = g_strdup(g_hash_table_lookup(handler_config, "plugin"));
        //copy config in a new hash table
        newconfig = g_hash_table_new_full (
                                        g_str_hash, g_str_equal, g_free, g_free);
        g_hash_table_foreach(handler_config,copy_hashed_new_config,newconfig);
        handler->config = newconfig;
        handler->refcount = 0;
        wrap_g_static_rec_mutex_init(&handler->lock);
        wrap_g_static_rec_mutex_init(&handler->refcount_lock);

        return handler;
cleanexit:
        g_free(handler);
        return NULL;
}

/**
 * oh_create_handler
 * @handler_config: Hash table containing the configuration for a handler
 * read from the configuration file.
 * @hid: pointer where hid of newly created handler will be stored.
 *
 * Returns: SA_OK on success. If handler failed to open, then @hid will
 * contain a valid handler id, but SA_ERR_HPI_INTERNAL_ERROR will be
 * returned.
 **/
SaErrorT oh_create_handler (GHashTable *handler_config, unsigned int *hid)
{
        struct oh_handler *handler = NULL;

        if (!handler_config || !hid) {
                CRIT("ERROR creating handler. Invalid parameters.");
                return SA_ERR_HPI_INVALID_PARAMS;
        }

	*hid = 0;

        handler = new_handler(handler_config);
        if (!handler) return SA_ERR_HPI_ERROR;

        *hid = handler->id;
        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        oh_handlers.list = g_slist_append(oh_handlers.list, handler);
        g_hash_table_insert(oh_handlers.table,
                            &(handler->id),
                            g_slist_last(oh_handlers.list));

        handler->hnd = handler->abi->open(handler->config,
                                          handler->id,
                                          oh_process_q);
        if (!handler->hnd) {
                CRIT("A handler #%d on the %s plugin could not be opened.",
                    handler->id, handler->plugin_name);
		wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
		return SA_ERR_HPI_INTERNAL_ERROR;
        }

        // TODO reimplement to get timeout value from domain
        // set auto-extract timeout
        if (handler->abi->set_autoinsert_timeout) {
                struct oh_global_param param;
                SaErrorT rv;
                oh_get_global_param2(OPENHPI_AUTOINSERT_TIMEOUT, &param);
                SaHpiTimeoutT ai_timeout = param.u.ai_timeout;
                oh_get_global_param2(OPENHPI_AUTOINSERT_TIMEOUT_READONLY, &param);
                DBG("auto-insert timeout readonly=%d, auto-insert timeout to set=%" PRId64,
                       param.u.ai_timeout_readonly, (int64_t)ai_timeout);
                if (!param.u.ai_timeout_readonly && ai_timeout) {
                        rv = handler->abi->set_autoinsert_timeout(handler->hnd, ai_timeout);
                        if (rv != SA_OK) {
                                CRIT("Cannot propagate auto-insert timeout to handler.");
                        }
                 }
        }

        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);

        return SA_OK;
}

/**
 * oh_destroy_handler
 * @hid: Id of handler to destroy
 *
 * Returns: 0 on Success.
 **/
int oh_destroy_handler(unsigned int hid)
{
        struct oh_handler *handler = NULL;

        if (hid < 1) {
                CRIT("ERROR - Invalid handler 0 id passed.");
                return -1;
        }

        handler = oh_get_handler(hid);
        if (!handler) {
                CRIT("ERROR - Handler %d not found.", hid);
                return -1;
        }

        if (handler->abi && handler->abi->close) {
                if (handler->hnd) {
                        handler->abi->close(handler->hnd);
                }
        }

        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        g_hash_table_remove(oh_handlers.table, &handler->id);
        oh_handlers.list = g_slist_remove(oh_handlers.list, &(handler->id));
        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);

        __dec_handler_refcount(handler);
        if (handler->refcount < 1)
                __delete_handler(handler);
        else
                oh_release_handler(handler);

        return 0;
}

/**
 * oh_get_handler_info
 *
 * Returns: SA_OK on success.
 **/

// helper function to copy hash table
static void copy_hashed_config_info (gpointer key, gpointer value, gpointer newhash)
{
        g_hash_table_insert ( newhash, g_strdup(key), g_strdup(value) );
}

SaErrorT oh_get_handler_info(unsigned int hid, oHpiHandlerInfoT *info, GHashTable *conf_params)
{
        struct oh_handler *h = NULL;
        GSList *node = NULL;

        if ((hid == 0) || (!info) || (!conf_params)) {
               return SA_ERR_HPI_INVALID_PARAMS;
        }

        wrap_g_static_rec_mutex_lock(&oh_handlers.lock);
        node = g_hash_table_lookup(oh_handlers.table, &hid);
        h = node ? (struct oh_handler *)(node->data) : NULL;
        if (!h) {
               wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);
                return SA_ERR_HPI_NOT_PRESENT;
        }

        info->id = hid;
        strncpy((char*)info->plugin_name, (const char*)h->plugin_name, MAX_PLUGIN_NAME_LENGTH);
        oh_init_ep(&info->entity_root);
        const char * entity_root_str = (const char *)g_hash_table_lookup(h->config, "entity_root");
        if ( entity_root_str ) {
                oh_encode_entitypath(entity_root_str, &info->entity_root);
        }

        info->load_failed = (!h->hnd) ? 1 : 0;

        // copy h->config to the output hash table
        g_hash_table_foreach(h->config,copy_hashed_config_info,conf_params);

        //Don't transmit passwords in case the handler has a password in its config
        if (g_hash_table_lookup(conf_params,"password"))
           g_hash_table_replace(conf_params,
                                g_strdup("password"),
                                g_strdup("********"));

        wrap_g_static_rec_mutex_unlock(&oh_handlers.lock);

        return SA_OK;
}

/**
 * oh_discover_resources
 *
 * Returns: SA_OK on success.
 **/
SaErrorT oh_discovery(void)
{
        unsigned int hid = 0, next_hid;
        struct oh_handler *h = NULL;
        SaErrorT error = SA_ERR_HPI_ERROR;

        oh_getnext_handler_id(hid, &next_hid);
        while (next_hid) {
                hid = next_hid;

                if(signal_stop == TRUE){
                   error = SA_OK;
                   break;
                }

                SaErrorT cur_error;

                h = oh_get_handler(hid);
                if (!h) {
                        CRIT("No such handler %d", hid);
                        break;
                }

		if (h->abi->discover_resources && h->hnd) {
                        cur_error = h->abi->discover_resources(h->hnd);
                        if (cur_error == SA_OK && error) {
                                error = cur_error;
                        }
                }
                oh_release_handler(h);
                oh_getnext_handler_id(hid, &next_hid);
        }

        return error;
}

/**
 * oh_load_plugin_functions
 * @plugin: plugin structure.
 * @abi: oh_abi struct
 *
 * This function will load the symbol table from the plugin name and
 * assign the plugin functions to the abi struct.
 *
 * Return value: 0 on success, otherwise any negative value on failure.
 **/
int oh_load_plugin_functions(struct oh_plugin *plugin, struct oh_abi_v2 **abi)
{

        *abi = g_new0(struct oh_abi_v2, 1);


        if (!(*abi)) {
                CRIT("Out of Memory!");
                return -1;
        }

	g_module_symbol(plugin->dl_handle,
	                "oh_open",
	                (gpointer*)(&(*abi)->open));
	g_module_symbol(plugin->dl_handle,
	                "oh_close",
	                (gpointer*)(&(*abi)->close));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_event",
	                (gpointer*)(&(*abi)->get_event));
	g_module_symbol(plugin->dl_handle,
	                "oh_discover_resources",
	                (gpointer*)(&(*abi)->discover_resources));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_resource_tag",
	                (gpointer*)(&(*abi)->set_resource_tag));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_resource_severity",
	                (gpointer*)(&(*abi)->set_resource_severity));
	g_module_symbol(plugin->dl_handle,
	                "oh_resource_failed_remove",
	                (gpointer*)(&(*abi)->resource_failed_remove));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_el_info",
	                (gpointer*)(&(*abi)->get_el_info));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_el_caps",
	                (gpointer*)(&(*abi)->get_el_caps));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_el_time",
	                (gpointer*)(&(*abi)->set_el_time));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_el_entry",
	                (gpointer*)(&(*abi)->add_el_entry));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_el_entry",
	                (gpointer*)(&(*abi)->get_el_entry));
	g_module_symbol(plugin->dl_handle,
	                "oh_clear_el",
	                (gpointer*)(&(*abi)->clear_el));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_el_state",
	                (gpointer*)(&(*abi)->set_el_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_reset_el_overflow",
	                (gpointer*)(&(*abi)->reset_el_overflow));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_sensor_reading",
	                (gpointer*)(&(*abi)->get_sensor_reading));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_sensor_thresholds",
	                (gpointer*)(&(*abi)->get_sensor_thresholds));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_sensor_thresholds",
	                (gpointer*)(&(*abi)->set_sensor_thresholds));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_sensor_enable",
	                (gpointer*)(&(*abi)->get_sensor_enable));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_sensor_enable",
	                (gpointer*)(&(*abi)->set_sensor_enable));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_sensor_event_enables",
	                (gpointer*)(&(*abi)->get_sensor_event_enables));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_sensor_event_enables",
	                (gpointer*)(&(*abi)->set_sensor_event_enables));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_sensor_event_masks",
	                (gpointer*)(&(*abi)->get_sensor_event_masks));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_sensor_event_masks",
	                (gpointer*)(&(*abi)->set_sensor_event_masks));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_control_state",
	                (gpointer*)(&(*abi)->get_control_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_control_state",
	                (gpointer*)(&(*abi)->set_control_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_idr_info",
	                (gpointer*)(&(*abi)->get_idr_info));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_idr_area_header",
	                (gpointer*)(&(*abi)->get_idr_area_header));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_idr_area",
	                (gpointer*)(&(*abi)->add_idr_area));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_idr_area_id",
	                (gpointer*)(&(*abi)->add_idr_area_id));
	g_module_symbol(plugin->dl_handle,
	                "oh_del_idr_area",
	                (gpointer*)(&(*abi)->del_idr_area));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_idr_field",
	                (gpointer*)(&(*abi)->get_idr_field));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_idr_field",
	                (gpointer*)(&(*abi)->add_idr_field));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_idr_field_id",
	                (gpointer*)(&(*abi)->add_idr_field_id));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_idr_field",
	                (gpointer*)(&(*abi)->set_idr_field));
	g_module_symbol(plugin->dl_handle,
	                "oh_del_idr_field",
	                (gpointer*)(&(*abi)->del_idr_field));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_watchdog_info",
	                (gpointer*)(&(*abi)->get_watchdog_info));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_watchdog_info",
	                (gpointer*)(&(*abi)->set_watchdog_info));
	g_module_symbol(plugin->dl_handle,
	                "oh_reset_watchdog",
	                (gpointer*)(&(*abi)->reset_watchdog));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_next_announce",
	                (gpointer*)(&(*abi)->get_next_announce));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_announce",
	                (gpointer*)(&(*abi)->get_announce));
	g_module_symbol(plugin->dl_handle,
	                "oh_ack_announce",
	                (gpointer*)(&(*abi)->ack_announce));
	g_module_symbol(plugin->dl_handle,
	                "oh_add_announce",
	                (gpointer*)(&(*abi)->add_announce));
	g_module_symbol(plugin->dl_handle,
	                "oh_del_announce",
	                (gpointer*)(&(*abi)->del_announce));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_annunc_mode",
	                (gpointer*)(&(*abi)->get_annunc_mode));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_annunc_mode",
	                (gpointer*)(&(*abi)->set_annunc_mode));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_dimi_info",
	                (gpointer*)(&(*abi)->get_dimi_info));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_dimi_test",
	                (gpointer*)(&(*abi)->get_dimi_test));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_dimi_test_ready",
	                (gpointer*)(&(*abi)->get_dimi_test_ready));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_dimi_test",
	                (gpointer*)(&(*abi)->start_dimi_test));
	g_module_symbol(plugin->dl_handle,
	                "oh_cancel_dimi_test",
	                (gpointer*)(&(*abi)->cancel_dimi_test));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_dimi_test_status",
	                (gpointer*)(&(*abi)->get_dimi_test_status));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_dimi_test_results",
	                (gpointer*)(&(*abi)->get_dimi_test_results));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_spec",
	                (gpointer*)(&(*abi)->get_fumi_spec));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_service_impact",
	                (gpointer*)(&(*abi)->get_fumi_service_impact));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_fumi_source",
	                (gpointer*)(&(*abi)->set_fumi_source));
	g_module_symbol(plugin->dl_handle,
	                "oh_validate_fumi_source",
	                (gpointer*)(&(*abi)->validate_fumi_source));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_source",
	                (gpointer*)(&(*abi)->get_fumi_source));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_source_component",
	                (gpointer*)(&(*abi)->get_fumi_source_component));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_target",
	                (gpointer*)(&(*abi)->get_fumi_target));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_target_component",
	                (gpointer*)(&(*abi)->get_fumi_target_component));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_logical_target",
	                (gpointer*)(&(*abi)->get_fumi_logical_target));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_logical_target_component",
	                (gpointer*)(&(*abi)->get_fumi_logical_target_component));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_backup",
	                (gpointer*)(&(*abi)->start_fumi_backup));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_fumi_bank_order",
	                (gpointer*)(&(*abi)->set_fumi_bank_order));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_bank_copy",
	                (gpointer*)(&(*abi)->start_fumi_bank_copy));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_install",
	                (gpointer*)(&(*abi)->start_fumi_install));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_status",
	                (gpointer*)(&(*abi)->get_fumi_status));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_verify",
	                (gpointer*)(&(*abi)->start_fumi_verify));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_verify_main",
	                (gpointer*)(&(*abi)->start_fumi_verify_main));
	g_module_symbol(plugin->dl_handle,
	                "oh_cancel_fumi_upgrade",
	                (gpointer*)(&(*abi)->cancel_fumi_upgrade));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_fumi_autorollback_disable",
	                (gpointer*)(&(*abi)->get_fumi_autorollback_disable));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_fumi_autorollback_disable",
	                (gpointer*)(&(*abi)->set_fumi_autorollback_disable));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_rollback",
	                (gpointer*)(&(*abi)->start_fumi_rollback));
	g_module_symbol(plugin->dl_handle,
	                "oh_activate_fumi",
	                (gpointer*)(&(*abi)->activate_fumi));
	g_module_symbol(plugin->dl_handle,
	                "oh_start_fumi_activate",
	                (gpointer*)(&(*abi)->start_fumi_activate));
	g_module_symbol(plugin->dl_handle,
	                "oh_cleanup_fumi",
	                (gpointer*)(&(*abi)->cleanup_fumi));
	g_module_symbol(plugin->dl_handle,
	                "oh_hotswap_policy_cancel",
	                (gpointer*)(&(*abi)->hotswap_policy_cancel));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_hotswap_state",
	                (gpointer*)(&(*abi)->get_hotswap_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_autoinsert_timeout",
	                (gpointer*)(&(*abi)->set_autoinsert_timeout));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_hotswap_state",
	                (gpointer*)(&(*abi)->set_hotswap_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_request_hotswap_action",
	                (gpointer*)(&(*abi)->request_hotswap_action));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_autoextract_timeout",
	                (gpointer*)(&(*abi)->get_autoextract_timeout));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_autoextract_timeout",
	                (gpointer*)(&(*abi)->set_autoextract_timeout));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_power_state",
	                (gpointer*)(&(*abi)->get_power_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_power_state",
	                (gpointer*)(&(*abi)->set_power_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_indicator_state",
	                (gpointer*)(&(*abi)->get_indicator_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_indicator_state",
	                (gpointer*)(&(*abi)->set_indicator_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_control_parm",
	                (gpointer*)(&(*abi)->control_parm));
	g_module_symbol(plugin->dl_handle,
	                "oh_load_id_get",
	                (gpointer*)(&(*abi)->load_id_get));
	g_module_symbol(plugin->dl_handle,
	                "oh_load_id_set",
	                (gpointer*)(&(*abi)->load_id_set));
	g_module_symbol(plugin->dl_handle,
	                "oh_get_reset_state",
	                (gpointer*)(&(*abi)->get_reset_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_set_reset_state",
	                (gpointer*)(&(*abi)->set_reset_state));
	g_module_symbol(plugin->dl_handle,
	                "oh_inject_event",
	                (gpointer*)(&(*abi)->inject_event));

        return 0;

}