Blob Blame History Raw
/*      -*- linux-c -*-
 *
 * (C) Copyright IBM Corp. 2003-2006
 * Copyright (c) 2003 by Intel Corp.
 *
 * 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:
 *     Sean Dague <http://dague.net/sean>
 *     Louis Zhuang <louis.zhuang@linux.intel.com>
 *     David Judkovics <djudkovi@us.ibm.com>
 *     Thomas Kangieser <Thomas.Kanngieser@fci.com>
 *     Renier Morales <renier@openhpi.org>
 *     Bryan Sutula <sutula@users.sourceforge.net>
 */

#include <inttypes.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <limits.h>

#ifndef _WIN32
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <libgen.h>
#include <errno.h>
#endif // _WIN32
 
#include <config.h>
#include <oh_plugin.h>
#include <oh_error.h>

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

/*
 * Global Parameters
 */
static const char *known_globals[] = {
        "OPENHPI_LOG_ON_SEV",
        "OPENHPI_EVT_QUEUE_LIMIT",
        "OPENHPI_DEL_SIZE_LIMIT",
        "OPENHPI_DEL_SAVE",
        "OPENHPI_DAT_SIZE_LIMIT",
        "OPENHPI_DAT_USER_LIMIT",
        "OPENHPI_DAT_SAVE",
        "OPENHPI_PATH",
        "OPENHPI_VARPATH",
        "OPENHPI_CONF",
        // The parameters below are not accessible via oHpiGlobalParamSet/oHpiGlobalParamGet
        "OPENHPI_UNCONFIGURED",
        "OPENHPI_AUTOINSERT_TIMEOUT",
        "OPENHPI_AUTOINSERT_TIMEOUT_READONLY",
        NULL
};

static struct {
        SaHpiSeverityT log_on_sev;
        SaHpiUint32T evt_queue_limit;
        SaHpiUint32T del_size_limit;
        SaHpiBoolT del_save;
        SaHpiUint32T dat_size_limit;
        SaHpiUint32T dat_user_limit;
        SaHpiBoolT dat_save;
        char path[OH_PATH_PARAM_MAX_LENGTH];
        char varpath[OH_PATH_PARAM_MAX_LENGTH];
        char conf[OH_PATH_PARAM_MAX_LENGTH];
        // The parameters below are not accessible via oHpiGlobalParamSet/oHpiGlobalParamGet
        SaHpiBoolT unconfigured;
        SaHpiTimeoutT ai_timeout;
        SaHpiBoolT ai_timeout_readonly;
        unsigned char read_env;
        GStaticRecMutex lock;
} global_params = { /* Defaults for global params are set here */
        .log_on_sev = SAHPI_MINOR,
        .evt_queue_limit = 10000,
        .del_size_limit = 10000, /* 0 is unlimited size */
        .del_save = SAHPI_FALSE,
        .dat_size_limit = 0, /* Unlimited size */
        .dat_user_limit = 0, /* Unlimited size */
        .dat_save = SAHPI_FALSE,
        .path = OH_PLUGIN_PATH,
        .varpath = VARPATH,
        .conf = OH_DEFAULT_CONF,
        .unconfigured = SAHPI_FALSE,
        .ai_timeout = 0,
        .ai_timeout_readonly = SAHPI_TRUE,
        .read_env = 0,
        .lock = G_STATIC_REC_MUTEX_INIT
};

/*
 *  List of handler configs (parameter tables).  This list is
 *  populated during config file parse, and used to build the handler_table
 */
static GSList *handler_configs = NULL;

/*******************************************************************************
 *  In order to use the glib lexical parser we need to define token
 *  types which we want to switch on
 ******************************************************************************/

enum {
        HPI_CONF_TOKEN_HANDLER = G_TOKEN_LAST
} hpiConfType;

struct tokens {
        gchar *name;
        guint token;
};

static struct tokens oh_conf_tokens[] = {
        {
                .name = "handler",
                .token = HPI_CONF_TOKEN_HANDLER
        }

};

/*******************************************************************************
 * In order to initialize the lexical scanner, you need the following config.
 * This config was figured out by reading the glib sources, and lots of
 * trial and error (documentation for this isn't very good).
 *
 * G_TOKEN_STRING will be created when anything starts with a-zA-z_/.
 * due to cset_identifier_first and identifier2string values below.
 * Therefor, if you want 0 to be scanned as a string, you need to quote
 * it (e.g. "0")
 *
 *******************************************************************************/

static GScannerConfig oh_scanner_config =
        {
                (
                        " \t\n"
                        )                       /* cset_skip_characters */,
                (
                        G_CSET_a_2_z
                        "_/."
                        G_CSET_A_2_Z
                        )                       /* cset_identifier_first */,
                (
                        G_CSET_a_2_z
                        "_-0123456789/."
                        G_CSET_A_2_Z
                        )                       /* cset_identifier_nth */,
                ( "#\n" )               /* cpair_comment_single */,
                FALSE                   /* case_sensitive */,
                TRUE                    /* skip_comment_multi */,
                TRUE                    /* skip_comment_single */,
                TRUE                    /* scan_comment_multi */,
                TRUE                    /* scan_identifier */,
                TRUE                    /* scan_identifier_1char */,
                TRUE                    /* scan_identifier_NULL */,
                TRUE                    /* scan_symbols */,
                TRUE                    /* scan_binary */,
                TRUE                    /* scan_octal */,
                TRUE                    /* scan_float */,
                TRUE                    /* scan_hex */,
                TRUE                    /* scan_hex_dollar */,
                TRUE                    /* scan_string_sq */,
                TRUE                    /* scan_string_dq */,
                TRUE                    /* numbers_2_int */,
                FALSE                   /* int_2_float */,
                TRUE                    /* identifier_2_string */,
                TRUE                    /* char_2_token */,
                TRUE                    /* symbol_2_token */,
                FALSE                   /* scope_0_fallback */,
                TRUE                    /* store_int64 */,
        };

static void process_global_param(const char *name, char *value)
{
        wrap_g_static_rec_mutex_lock((void *)&global_params.lock);

        if (!strcmp("OPENHPI_LOG_ON_SEV", name)) {
                SaHpiTextBufferT buffer;
                strncpy((char *)buffer.Data, value, SAHPI_MAX_TEXT_BUFFER_LENGTH);
                oh_encode_severity(&buffer, &global_params.log_on_sev);
        } else if (!strcmp("OPENHPI_EVT_QUEUE_LIMIT", name)) {
                global_params.evt_queue_limit = atoi(value);
        } else if (!strcmp("OPENHPI_DEL_SIZE_LIMIT", name)) {
                global_params.del_size_limit = atoi(value);
        } else if (!strcmp("OPENHPI_DEL_SAVE", name)) {
                if (!strcmp("YES", value)) {
                        global_params.del_save = SAHPI_TRUE;
                } else {
                        global_params.del_save = SAHPI_FALSE;
                }
        } else if (!strcmp("OPENHPI_DAT_SIZE_LIMIT", name)) {
                global_params.dat_size_limit = atoi(value);
        } else if (!strcmp("OPENHPI_DAT_USER_LIMIT", name)) {
                global_params.dat_user_limit = atoi(value);
        } else if (!strcmp("OPENHPI_DAT_SAVE", name)) {
                if (!strcmp("YES", value)) {
                        global_params.dat_save = SAHPI_TRUE;
                } else {
                        global_params.dat_save = SAHPI_FALSE;
                }
        } else if (!strcmp("OPENHPI_PATH", name)) {
                memset(global_params.path, 0, OH_PATH_PARAM_MAX_LENGTH);
                strncpy(global_params.path, value, OH_PATH_PARAM_MAX_LENGTH-1);
        } else if (!strcmp("OPENHPI_VARPATH", name)) {
                memset(global_params.varpath, 0, OH_PATH_PARAM_MAX_LENGTH);
                strncpy(global_params.varpath, value, OH_PATH_PARAM_MAX_LENGTH-1);
        } else if (!strcmp("OPENHPI_CONF", name)) {
                memset(global_params.conf, 0, OH_PATH_PARAM_MAX_LENGTH);
                strncpy(global_params.conf, value, OH_PATH_PARAM_MAX_LENGTH-1);
        } else if (!strcmp("OPENHPI_UNCONFIGURED", name)) {
                if (!strcmp("YES", value)) {
                        global_params.unconfigured = SAHPI_TRUE;
                } else {
                        global_params.unconfigured = SAHPI_FALSE;
                }
        } else if (!strcmp("OPENHPI_AUTOINSERT_TIMEOUT", name)) {
                if (!strcmp(value, "BLOCK")) {
                    global_params.ai_timeout = SAHPI_TIMEOUT_BLOCK;
                } else if (!strcmp(value, "IMMEDIATE")) {
                    global_params.ai_timeout = SAHPI_TIMEOUT_IMMEDIATE;
                } else {
                    global_params.ai_timeout = strtoll(value, 0, 10);
                    if (global_params.ai_timeout < 0) {
                        global_params.ai_timeout = SAHPI_TIMEOUT_BLOCK;
                    }
                }
        } else if (!strcmp("OPENHPI_AUTOINSERT_TIMEOUT_READONLY", name)) {
                if (!strcmp("YES", value)) {
                        global_params.ai_timeout_readonly = SAHPI_TRUE;
                } else {
                        global_params.ai_timeout_readonly = SAHPI_FALSE;
                }
	} else {
                CRIT("Invalid global parameter %s in config file.", name);
        }

        wrap_g_static_rec_mutex_unlock(&global_params.lock);
}

static void read_globals_from_env(int force)
{
        char *tmp_env_str = NULL;
        int i;

        if (!force && global_params.read_env) return;

        wrap_g_static_rec_mutex_lock(&global_params.lock);

        for (i = 0; known_globals[i]; i++) {
                if ((tmp_env_str = getenv(known_globals[i])) != NULL) {
                        process_global_param(known_globals[i], tmp_env_str);
                        tmp_env_str = NULL;
                }
        }

        global_params.read_env = 1;
        wrap_g_static_rec_mutex_unlock(&global_params.lock);
}

/**
 * process_handler_token
 * @oh_scanner
 *
 * handles parsing of handler tokens into a hash table.
 *
 * Return value: 0 on sucess, < 0 on fail
 **/
static int process_handler_token (GScanner* oh_scanner)
{
        GHashTable *handler_stanza = NULL;
        char *tablekey, *tablevalue;
        int found_right_curly = 0;

        data_access_lock();

        if (g_scanner_get_next_token(oh_scanner) != (int)HPI_CONF_TOKEN_HANDLER) {
                CRIT("Processing handler: Unexpected token.");
                data_access_unlock();
                return -1;
        }

        /* Get the plugin type and store in Hash Table */
        if (g_scanner_get_next_token(oh_scanner) != G_TOKEN_STRING) {
                CRIT("Processing handler: Expected string token.");
                data_access_unlock();
                return -1;
        } else {
                handler_stanza = g_hash_table_new_full(g_str_hash, g_str_equal,
                                                       g_free, g_free);
                tablekey = g_strdup("plugin");
                tablevalue = g_strdup(oh_scanner->value.v_string);
                g_hash_table_insert(handler_stanza,
                                    (gpointer) tablekey,
                                    (gpointer) tablevalue);
                oh_load_plugin(tablevalue);
        }

        /* Check for Left Brace token type. If we have it, then continue parsing. */
        if (g_scanner_get_next_token(oh_scanner) != G_TOKEN_LEFT_CURLY) {
                CRIT("Processing handler: Expected left curly token.");
                goto free_table;
        }

        while (!found_right_curly) {
                /* get key token in key\value pair set (e.g. key = value) */
                if (g_scanner_get_next_token(oh_scanner) != G_TOKEN_STRING) {
                        CRIT("Processing handler: Expected string token.");
                        goto free_table;
                } else {
                        tablekey = g_strdup(oh_scanner->value.v_string);
                }

                /* Check for the equal sign next. If we have it, continue parsing */
                if (g_scanner_get_next_token(oh_scanner) != G_TOKEN_EQUAL_SIGN) {
                        CRIT("Processing handler: Expected equal sign token.");
                        goto free_table_and_key;
                }

                /*
                 * Now check for the value token in the key\value set.
                 * Store the key\value value pair in the hash table and continue on.
                 */
                if (g_scanner_peek_next_token(oh_scanner) != G_TOKEN_INT &&
                    g_scanner_peek_next_token(oh_scanner) != G_TOKEN_FLOAT &&
                    g_scanner_peek_next_token(oh_scanner) != G_TOKEN_STRING) {
                        CRIT("Processing handler: Expected string, integer, or float token.");
                        goto free_table_and_key;
                } else { /* The type of token tells us how to fetch the value from oh_scanner */
                        gpointer value = NULL;
                        int current_token = g_scanner_get_next_token(oh_scanner);

                        if (current_token == G_TOKEN_INT) {
                                // TODO seems everyone now relies that 
                                // value shall be string.
                                // So this code may break a plug-in expectations.
                                // Investigate
                                gulong *value_int = g_new0(gulong, 1);
                                *value_int = (gulong)oh_scanner->value.v_int64;
                                value = (gpointer)value_int;
                        } else if (current_token == G_TOKEN_FLOAT) {
                                // TODO seems everyone now relies that 
                                // value shall be string.
                                // So this code may break a plug-in expectations.
                                // Investigate
                                gdouble *value_float = g_new0(gdouble, 1);
                                *value_float = oh_scanner->value.v_float;
                                value = (gpointer)value_float;
                        } else {
                                gchar *value_string =
                                        g_strdup(oh_scanner->value.v_string);
                                value = (gpointer)value_string;
                        }

                        if (value == NULL) {
                                CRIT("Processing handler:"
                                    " Unable to allocate memory for value."
                                    " Token Type: %d.",
                                    current_token);
                                goto free_table_and_key;
                        }

                        g_hash_table_insert(handler_stanza,
                                            (gpointer) tablekey,
                                            value);
                }

                if (g_scanner_peek_next_token(oh_scanner) == G_TOKEN_RIGHT_CURLY) {
                        g_scanner_get_next_token(oh_scanner);
                        found_right_curly = 1;
                }
        } /* end of while(!found_right_curly) */

        /* Attach table describing handler stanza to the global linked list of handlers */
        if (handler_stanza != NULL) {
                handler_configs = g_slist_append(
                        handler_configs,
                        (gpointer) handler_stanza);
        }

        data_access_unlock();

        return 0;

free_table_and_key:
        g_free(tablekey);
free_table:
        /**
        There was an error reading a token so we need to error out,
        but not before cleaning up. Destroy the table.
        */
        g_hash_table_destroy(handler_stanza);

        data_access_unlock();

        return -1;
}

static int process_global_token(GScanner *scanner)
{
        char *name = NULL, *value = NULL;
        int current_token;

        data_access_lock();

        /* Get the global parameter name */
        current_token = g_scanner_get_next_token(scanner);
        if (current_token != G_TOKEN_STRING) {
                CRIT("Processing global: Expected string token. Got %d.",
                    current_token);
                goto quit;
        }

        name = g_strdup(scanner->value.v_string);
        if (!name) {
                CRIT("Unable to allocate for global param name.");
                goto quit;
        }

        current_token = g_scanner_get_next_token(scanner);
        if (current_token != G_TOKEN_EQUAL_SIGN) {
                CRIT("Did not get expected '=' token. Got %d.", current_token);
                goto free_and_quit;
        }

        current_token = g_scanner_get_next_token(scanner);
        if (current_token != G_TOKEN_STRING && current_token != G_TOKEN_INT) {
                CRIT("Did not get expected string value for global parameter."
                    " Got %d.", current_token);
                goto free_and_quit;
        }

        if (current_token == G_TOKEN_INT) {
                const guint max_digits = 32; // More than enough for uint64.
                value = (char *)g_malloc0(max_digits);
                snprintf(value, max_digits, "%" PRIu64, scanner->value.v_int64);
        } else {
                value = g_strdup(scanner->value.v_string);
        }

        if (!value) {
                CRIT("Unable to allocate for global param value.");
                goto free_and_quit;
        }

        process_global_param(name, value);
        g_free(name);
        g_free(value);
        data_access_unlock();
        return 0;

free_and_quit:
        g_free(name);
quit:
        data_access_unlock();
        return -1;
}

/**
 * scanner_msg_handler: a reference of this function is passed into the GScanner.
 * Used by the GScanner object to output messages that come up during parsing.
 *
 * @scanner: Object used to parse the config file.
 * @message: Message string.
 * @is_error: Bit to say the message is an error.
 *
 * Return value: None (void).
 **/
static void scanner_msg_handler (GScanner *scanner, gchar *message, gboolean is_error)
{
        g_return_if_fail (scanner != NULL);

        CRIT("%s:%d: %s%s.\n",
            scanner->input_name ? scanner->input_name : "<memory>",
            scanner->line, is_error ? "error: " : "", message );
}

/**
 * oh_load_config
 * @filename: OpenHPI configuration filename
 * @config: place where the parsed configuration will be placed.
 *
 * Parses an OpenHPI configuration file and gives the results
 * which can be processed by the caller.
 *
 * Return value: 0 on success, otherwise a failure.
 **/
int oh_load_config (char *filename, struct oh_parsed_config *config)
{
        FILE * fp = NULL;
#ifndef _WIN32
        struct stat fst, dst; 
        char *dir_name = NULL;
        char dup_filename[PATH_MAX];
#ifdef HAVE_ENCRYPT
        SaHpiUint32T type = OHPI_DECRYPT;
        gchar *confile_txt = NULL;
#endif /* HAVE_ENCRYPT */
#endif /* _WIN32 */
        gchar * gcrypt_str = NULL;
        gboolean g_decrypt = SAHPI_FALSE;
        GScanner *oh_scanner;
        int i, done = 0;
        int num_tokens = sizeof(oh_conf_tokens) / sizeof(oh_conf_tokens[0]);

        if (!filename || !config) {
                CRIT("Error. Invalid parameters.");
                return -1;
        }

        handler_configs = NULL;
        oh_scanner = g_scanner_new(&oh_scanner_config);
        if (!oh_scanner) {
                CRIT("Couldn't allocate g_scanner for file parsing.");
                return -2;
        }

        oh_scanner->msg_handler = scanner_msg_handler;
        oh_scanner->input_name = filename;

#ifndef _WIN32

        errno=0;
        if (stat (filename, &fst) == -1) {
                if (errno != ENOENT) {
                        CRIT("stat of %s failed.", filename);
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }
        } 
       	if (errno != ENOENT) { 
                if (fst.st_uid != geteuid()) {
                        if (fst.st_uid == 0) {
                             CRIT("%s owner is root (0)!", filename);
                             CRIT("Run as sudo or create a config with UID=%d.", (int)geteuid());
                             g_scanner_destroy(oh_scanner);
                             return -3;
                       } else  {
                             CRIT("%s owner(%d) is insecure!", filename, fst.st_uid);
                             CRIT("Owner UID shall be %d.", (int)geteuid());
                             g_scanner_destroy(oh_scanner);
                             return -3;
                       }
                }
        
                if (((fst.st_mode & (S_IRWXG | S_IRWXO)) != 0)) {
                        CRIT("%s permissions are insecure!", filename);
                        CRIT("Shall not contain bits for group/others.");
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }

                if ((fst.st_mode & S_IFMT) != S_IFREG) {
                        CRIT("%s permissions are insecure!", filename);
                        CRIT("It needs to be a regular file");
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }

                /* Now check the directory permissions */ 
                if (strlen(filename) >= PATH_MAX) {
                        CRIT("%s is too long",filename);
                        g_scanner_destroy(oh_scanner);
                        return -3;
                } 
                strcpy(dup_filename, filename);
                dir_name=dirname(dup_filename);
                if (stat(dir_name, &dst) == -1) {
                        CRIT("stat of %s failed.", dir_name);
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }

                if ((fst.st_uid != dst.st_uid)) {
                        CRIT("%s directory is insecure", dir_name);
                        CRIT("Owner UID shall be %d",fst.st_uid);
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }
                if ((dst.st_mode & (S_IWOTH | S_IWGRP)) != 0 ) {
                        CRIT("%s directory is insecure",dir_name);
                        CRIT("Shall not be writable by group or others");
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }
	}

#endif // _WIN32
         
        gcrypt_str = getenv("OPENHPI_GCRYPT");
        if(gcrypt_str)
               g_decrypt = atoi((const char *)gcrypt_str);
        if(g_decrypt) {
#ifndef _WIN32
#ifdef HAVE_ENCRYPT
                confile_txt = oh_crypt(filename, type);
                if(!confile_txt) {
                        CRIT("Unable to decrypt %s\n",filename);
                        g_scanner_destroy(oh_scanner);
                        return -3;
                }

                g_scanner_input_text(oh_scanner, confile_txt, fst.st_size);
#else
                CRIT("gcrypt asked for but not compiled with. Internal Error");
                return -4;
#endif /* HAVE_ENCRYPT */


#else
                WARN("Encrypting %s not supported",filename);
                WARN("Assuming plain file");
#endif
        } else {
               fp = fopen(filename, "r");
               if (!fp) {
                       CRIT("Configuration file '%s' could not be opened.", filename);
                       g_scanner_destroy(oh_scanner);
                       return -4;
               }
        }

#ifdef _WIN32
        g_scanner_input_file(oh_scanner, _fileno(fp));
#else
        if(!g_decrypt) {
                 g_scanner_input_file(oh_scanner, fileno(fp));
        }
#endif

        for (i = 0; i < num_tokens; i++) {
                g_scanner_scope_add_symbol(
                        oh_scanner, 0,
                        oh_conf_tokens[i].name,
                        GUINT_TO_POINTER(oh_conf_tokens[i].token));
        }

        while (!done) {
                int my_token;
                my_token = g_scanner_peek_next_token (oh_scanner);
                /*DBG("token: %d", my_token);*/
                switch (my_token)
                {
                case G_TOKEN_EOF:
                        done = 1;
                        break;
                case HPI_CONF_TOKEN_HANDLER:
                        process_handler_token(oh_scanner);
                        break;
                case G_TOKEN_STRING:
                        process_global_token(oh_scanner);
                        break;
                default:
                        /* need to advance it */
                        my_token = g_scanner_get_next_token(oh_scanner);
                        g_scanner_unexp_token(oh_scanner, G_TOKEN_SYMBOL,
                                              NULL, "\"handle\" or \"domain\"", NULL, NULL, 1);
                        break;
                }
        }

        read_globals_from_env(1);

        if (fp)
                fclose(fp);

        done = oh_scanner->parse_errors;

        g_scanner_destroy(oh_scanner);


        DBG("Done processing conf file. Parse errors: %d", done);

        config->handler_configs = handler_configs;

#ifdef HAVE_ENCRYPT
        free(confile_txt);
#endif

        handler_configs = NULL;

        return 0;
}

/**
 * oh_process_config
 * @config: pointer to parsed configuration for processing
 *
 * This will process a parsed configuration by loading
 * the specified plugins and corresponding handlers.
 *
 * Returns: SA_OK on success, otherwise the call failed.
 **/
SaErrorT oh_process_config(struct oh_parsed_config *config)
{
        GSList *node = NULL;

        if (!config) return SA_ERR_HPI_INVALID_PARAMS;

        /* Initialize handlers */
        for (node = config->handler_configs; node; node = node->next) {
                GHashTable *handler_config = (GHashTable *)node->data;
		unsigned int hid = 0;
		SaErrorT error = SA_OK;

		error = oh_create_handler(handler_config, &hid);
                if (error == SA_OK) {
                        DBG("Loaded handler for plugin %s",
                            (char *)g_hash_table_lookup(handler_config, "plugin"));
                        config->handlers_loaded++;
                } else {
                        CRIT("Couldn't load handler for plugin %s.",
                            (char *)g_hash_table_lookup(handler_config, "plugin"));
                        if (hid == 0) g_hash_table_destroy(handler_config);
                }
                config->handlers_defined++;
        }

        return SA_OK;
}

void oh_clean_config(struct oh_parsed_config *config)
{
        /* Free list of handler configuration blocks */
        g_slist_free(config->handler_configs);

}

/**
 * oh_get_global_param
 * @param
 *
 * Returns: 0 on Success.
 **/
int oh_get_global_param(struct oh_global_param *param)
{
        if (!param || !(param->type)) {

            if (!param) {
                CRIT("Invalid parameters param NULL.");
                return -1;
            }

            if (!param->type) {
                CRIT("Invalid parameters param->type NULL.");
                return -1;
            }

        }

        read_globals_from_env(0);

        wrap_g_static_rec_mutex_lock(&global_params.lock);
        switch (param->type) {
                case OPENHPI_LOG_ON_SEV:
                        param->u.log_on_sev = global_params.log_on_sev;
                        break;
                case OPENHPI_EVT_QUEUE_LIMIT:
                        param->u.evt_queue_limit = global_params.evt_queue_limit;
                        break;
                case OPENHPI_DEL_SIZE_LIMIT:
                        param->u.del_size_limit = global_params.del_size_limit;
                        break;
                case OPENHPI_DEL_SAVE:
                        param->u.del_save = global_params.del_save;
                        break;
                case OPENHPI_DAT_SIZE_LIMIT:
                        param->u.dat_size_limit = global_params.dat_size_limit;
                        break;
                case OPENHPI_DAT_USER_LIMIT:
                        param->u.dat_user_limit = global_params.dat_user_limit;
                        break;
                case OPENHPI_DAT_SAVE:
                        param->u.dat_save = global_params.dat_save;
                        break;
                case OPENHPI_PATH:
                        strncpy(param->u.path,
                                global_params.path,
                                OH_PATH_PARAM_MAX_LENGTH);
                        break;
                case OPENHPI_VARPATH:
                        strncpy(param->u.varpath,
                                global_params.varpath,
                                OH_PATH_PARAM_MAX_LENGTH);
                        break;
                case OPENHPI_CONF:
                        strncpy(param->u.conf,
                                global_params.conf,
                                OH_PATH_PARAM_MAX_LENGTH);
                        break;
                case OPENHPI_UNCONFIGURED:
                        param->u.unconfigured = global_params.unconfigured;
                        break;
                case OPENHPI_AUTOINSERT_TIMEOUT:
                        param->u.ai_timeout = global_params.ai_timeout;
                        break;
                case OPENHPI_AUTOINSERT_TIMEOUT_READONLY:
                        param->u.ai_timeout_readonly = global_params.ai_timeout_readonly;
                        break;
                default:
                        wrap_g_static_rec_mutex_unlock(&global_params.lock);
                        CRIT("Invalid global parameter %d!", param->type);
                        return -2;
        }
        wrap_g_static_rec_mutex_unlock(&global_params.lock);

        return 0;
}

/**
 * oh_get_global_param2
 * @param
 *
 * Returns: 0 on Success.
 **/
int oh_get_global_param2(oh_global_param_type type, struct oh_global_param *param)
{
        if (!param) {
                CRIT("Invalid parameters.");
                return -1;
        }

        param->type = type;

        return oh_get_global_param(param);
}

/**
 * oh_set_global_param
 * @param
 *
 * Returns: 0 on Success.
 **/
int oh_set_global_param(const struct oh_global_param *param)
{
        if (!param || !(param->type)) {
                CRIT("Invalid parameters.");
                return -1;
        }

        read_globals_from_env(0);

        wrap_g_static_rec_mutex_lock(&global_params.lock);
        switch (param->type) {
                case OPENHPI_LOG_ON_SEV:
                        global_params.log_on_sev = param->u.log_on_sev;
                        break;
                case OPENHPI_EVT_QUEUE_LIMIT:
                        global_params.evt_queue_limit = param->u.evt_queue_limit;
                        break;
                case OPENHPI_DEL_SIZE_LIMIT:
                        global_params.del_size_limit = param->u.del_size_limit;
                        break;
                case OPENHPI_DEL_SAVE:
                        global_params.del_save = param->u.del_save;
                        break;
                case OPENHPI_DAT_SIZE_LIMIT:
                        global_params.dat_size_limit = param->u.dat_size_limit;
                        break;
                case OPENHPI_DAT_USER_LIMIT:
                        global_params.dat_user_limit = param->u.dat_user_limit;
                        break;
                case OPENHPI_DAT_SAVE:
                        global_params.dat_save = param->u.dat_save;
                        break;
                case OPENHPI_PATH:
                        memset(global_params.path, 0, OH_PATH_PARAM_MAX_LENGTH);
                        strncpy(global_params.path,
                                param->u.path,
                                OH_PATH_PARAM_MAX_LENGTH-1);
                        break;
                case OPENHPI_VARPATH:
                        memset(global_params.varpath, 0, OH_PATH_PARAM_MAX_LENGTH);
                        strncpy(global_params.varpath,
                                param->u.varpath,
                                OH_PATH_PARAM_MAX_LENGTH-1);
                        break;
                case OPENHPI_CONF:
                        memset(global_params.conf, 0, OH_PATH_PARAM_MAX_LENGTH);
                        strncpy(global_params.conf,
                                param->u.conf,
                                OH_PATH_PARAM_MAX_LENGTH-1);
                        break;
                case OPENHPI_UNCONFIGURED:
                        global_params.unconfigured = param->u.unconfigured;
                        break;
                case OPENHPI_AUTOINSERT_TIMEOUT:
                        global_params.ai_timeout = param->u.ai_timeout;
                        break;
                case OPENHPI_AUTOINSERT_TIMEOUT_READONLY:
                        global_params.ai_timeout_readonly = param->u.ai_timeout_readonly;
                        break;
                default:
                        wrap_g_static_rec_mutex_unlock(&global_params.lock);
                        CRIT("Invalid global parameter %d!", param->type);
                        return -2;
        }
        wrap_g_static_rec_mutex_unlock(&global_params.lock);

        return 0;
}