|
Packit Service |
0af388 |
/*
|
|
Packit Service |
0af388 |
* Copyright (C) 2015 - 2016 Red Hat, Inc.
|
|
Packit Service |
0af388 |
*
|
|
Packit Service |
0af388 |
* This program is free software: you can redistribute it and/or modify
|
|
Packit Service |
0af388 |
* it under the terms of the GNU General Public License as published by
|
|
Packit Service |
0af388 |
* the Free Software Foundation, either version 3 of the License, or
|
|
Packit Service |
0af388 |
* (at your option) any later version.
|
|
Packit Service |
0af388 |
*
|
|
Packit Service |
0af388 |
* This program is distributed in the hope that it will be useful,
|
|
Packit Service |
0af388 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit Service |
0af388 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit Service |
0af388 |
* GNU General Public License for more details.
|
|
Packit Service |
0af388 |
*
|
|
Packit Service |
0af388 |
* You should have received a copy of the GNU General Public License
|
|
Packit Service |
0af388 |
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
Packit Service |
0af388 |
*
|
|
Packit Service |
0af388 |
* Author: Gris Ge <fge@redhat.com>
|
|
Packit Service |
0af388 |
* Todd Gill <tgill@redhat.com>
|
|
Packit Service |
0af388 |
*/
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
#include <stdlib.h>
|
|
Packit Service |
0af388 |
#include <stdint.h>
|
|
Packit Service |
0af388 |
#include <inttypes.h>
|
|
Packit Service |
0af388 |
#include <string.h>
|
|
Packit Service |
0af388 |
#include <assert.h>
|
|
Packit Service |
0af388 |
#include <json.h>
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
#include "libdmmp/libdmmp.h"
|
|
Packit Service |
0af388 |
#include "libdmmp_private.h"
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PGS_CMD "show groups raw format %w|%g|%p|%t|%s"
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PG_INDEX_WWID 0
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PG_INDEX_PG_ID 1
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PG_INDEX_PRI 2
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PG_INDEX_STATUS 3
|
|
Packit Service |
0af388 |
#define _DMMP_SHOW_PG_INDEX_SELECTOR 4
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
struct dmmp_path_group {
|
|
Packit Service |
0af388 |
uint32_t id;
|
|
Packit Service |
0af388 |
/* ^ pgindex of struct path, will be used for path group switch */
|
|
Packit Service |
0af388 |
uint32_t status;
|
|
Packit Service |
0af388 |
uint32_t priority;
|
|
Packit Service |
0af388 |
char *selector;
|
|
Packit Service |
0af388 |
uint32_t dmmp_p_count;
|
|
Packit Service |
0af388 |
struct dmmp_path **dmmp_ps;
|
|
Packit Service |
0af388 |
};
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
static const struct _num_str_conv _DMMP_PATH_GROUP_STATUS_CONV[] = {
|
|
Packit Service |
0af388 |
{DMMP_PATH_GROUP_STATUS_UNKNOWN, "undef"},
|
|
Packit Service |
0af388 |
{DMMP_PATH_GROUP_STATUS_ACTIVE, "active"},
|
|
Packit Service |
0af388 |
{DMMP_PATH_GROUP_STATUS_DISABLED, "disabled"},
|
|
Packit Service |
0af388 |
{DMMP_PATH_GROUP_STATUS_ENABLED, "enabled"},
|
|
Packit Service |
0af388 |
};
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_dmmp_str_func_gen(dmmp_path_group_status_str, uint32_t, pg_status,
|
|
Packit Service |
0af388 |
_DMMP_PATH_GROUP_STATUS_CONV);
|
|
Packit Service |
0af388 |
_dmmp_str_conv_func_gen(_dmmp_path_group_status_str_conv, ctx, pg_status_str,
|
|
Packit Service |
0af388 |
uint32_t, DMMP_PATH_GROUP_STATUS_UNKNOWN,
|
|
Packit Service |
0af388 |
_DMMP_PATH_GROUP_STATUS_CONV);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_dmmp_getter_func_gen(dmmp_path_group_id_get, struct dmmp_path_group, dmmp_pg,
|
|
Packit Service |
0af388 |
id, uint32_t);
|
|
Packit Service |
0af388 |
_dmmp_getter_func_gen(dmmp_path_group_status_get, struct dmmp_path_group,
|
|
Packit Service |
0af388 |
dmmp_pg, status, uint32_t);
|
|
Packit Service |
0af388 |
_dmmp_getter_func_gen(dmmp_path_group_priority_get, struct dmmp_path_group,
|
|
Packit Service |
0af388 |
dmmp_pg, priority, uint32_t);
|
|
Packit Service |
0af388 |
_dmmp_getter_func_gen(dmmp_path_group_selector_get, struct dmmp_path_group,
|
|
Packit Service |
0af388 |
dmmp_pg, selector, const char *);
|
|
Packit Service |
0af388 |
_dmmp_array_free_func_gen(_dmmp_path_group_array_free, struct dmmp_path_group,
|
|
Packit Service |
0af388 |
_dmmp_path_group_free);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
struct dmmp_path_group *_dmmp_path_group_new(void)
|
|
Packit Service |
0af388 |
{
|
|
Packit Service |
0af388 |
struct dmmp_path_group *dmmp_pg = NULL;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg = (struct dmmp_path_group *)
|
|
Packit Service |
0af388 |
malloc(sizeof(struct dmmp_path_group));
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
if (dmmp_pg != NULL) {
|
|
Packit Service |
0af388 |
dmmp_pg->id = _DMMP_PATH_GROUP_ID_UNKNOWN;
|
|
Packit Service |
0af388 |
dmmp_pg->status = DMMP_PATH_GROUP_STATUS_UNKNOWN;
|
|
Packit Service |
0af388 |
dmmp_pg->priority = 0;
|
|
Packit Service |
0af388 |
dmmp_pg->selector = NULL;
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_p_count = 0;
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_ps = NULL;
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
return dmmp_pg;
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
int _dmmp_path_group_update(struct dmmp_context *ctx,
|
|
Packit Service |
0af388 |
struct dmmp_path_group *dmmp_pg,
|
|
Packit Service |
0af388 |
json_object *j_obj_pg)
|
|
Packit Service |
0af388 |
{
|
|
Packit Service |
0af388 |
int rc = DMMP_OK;
|
|
Packit Service |
0af388 |
uint32_t id = 0;
|
|
Packit Service |
0af388 |
int priority_int = -1 ;
|
|
Packit Service |
0af388 |
const char *status_str = NULL;
|
|
Packit Service |
0af388 |
const char *selector = NULL;
|
|
Packit Service |
0af388 |
struct array_list *ar_ps = NULL;
|
|
Packit Service |
0af388 |
int ar_ps_len = -1;
|
|
Packit Service |
0af388 |
uint32_t i = 0;
|
|
Packit Service |
0af388 |
struct dmmp_path *dmmp_p = NULL;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
assert(ctx != NULL);
|
|
Packit Service |
0af388 |
assert(dmmp_pg != NULL);
|
|
Packit Service |
0af388 |
assert(j_obj_pg != NULL);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_json_obj_get_value(ctx, j_obj_pg, status_str, "dm_st",
|
|
Packit Service |
0af388 |
json_type_string, json_object_get_string, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_json_obj_get_value(ctx, j_obj_pg, selector, "selector",
|
|
Packit Service |
0af388 |
json_type_string, json_object_get_string, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_json_obj_get_value(ctx, j_obj_pg, priority_int, "pri",
|
|
Packit Service |
0af388 |
json_type_int, json_object_get_int, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_json_obj_get_value(ctx, j_obj_pg, id, "group",
|
|
Packit Service |
0af388 |
json_type_int, json_object_get_int, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg->priority = (priority_int <= 0) ? 0 : priority_int & UINT32_MAX;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_dmmp_null_or_empty_str_check(ctx, status_str, rc, out);
|
|
Packit Service |
0af388 |
_dmmp_null_or_empty_str_check(ctx, selector, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg->selector = strdup(selector);
|
|
Packit Service |
0af388 |
_dmmp_alloc_null_check(ctx, dmmp_pg->selector, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg->id = id;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
if (dmmp_pg->id == _DMMP_PATH_GROUP_ID_UNKNOWN) {
|
|
Packit Service |
0af388 |
rc = DMMP_ERR_BUG;
|
|
Packit Service |
0af388 |
_error(ctx, "BUG: Got unknown(%d) path group ID",
|
|
Packit Service |
0af388 |
_DMMP_PATH_GROUP_ID_UNKNOWN);
|
|
Packit Service |
0af388 |
goto out;
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg->status = _dmmp_path_group_status_str_conv(ctx, status_str);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_json_obj_get_value(ctx, j_obj_pg, ar_ps, "paths",
|
|
Packit Service |
0af388 |
json_type_array, json_object_get_array, rc, out);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
ar_ps_len = array_list_length(ar_ps);
|
|
Packit Service |
0af388 |
if (ar_ps_len < 0) {
|
|
Packit Service |
0af388 |
rc = DMMP_ERR_BUG;
|
|
Packit Service |
0af388 |
_error(ctx, "BUG: Got negative length for ar_ps");
|
|
Packit Service |
0af388 |
goto out;
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
else if (ar_ps_len == 0)
|
|
Packit Service |
0af388 |
goto out;
|
|
Packit Service |
0af388 |
else
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_p_count = ar_ps_len & UINT32_MAX;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_ps = (struct dmmp_path **)
|
|
Packit Service |
0af388 |
malloc(sizeof(struct dmmp_path *) * dmmp_pg->dmmp_p_count);
|
|
Packit Service |
0af388 |
_dmmp_alloc_null_check(ctx, dmmp_pg->dmmp_ps, rc, out);
|
|
Packit Service |
0af388 |
for (; i < dmmp_pg->dmmp_p_count; ++i)
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_ps[i] = NULL;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
for (i = 0; i < dmmp_pg->dmmp_p_count; ++i) {
|
|
Packit Service |
0af388 |
dmmp_p = _dmmp_path_new();
|
|
Packit Service |
0af388 |
_dmmp_alloc_null_check(ctx, dmmp_p, rc, out);
|
|
Packit Service |
0af388 |
dmmp_pg->dmmp_ps[i] = dmmp_p;
|
|
Packit Service |
0af388 |
_good(_dmmp_path_update(ctx, dmmp_p,
|
|
Packit Service |
0af388 |
array_list_get_idx(ar_ps, i)),
|
|
Packit Service |
0af388 |
rc, out);
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
_debug(ctx, "Got path group id: %" PRIu32 "", dmmp_pg->id);
|
|
Packit Service |
0af388 |
_debug(ctx, "Got path group priority: %" PRIu32 "", dmmp_pg->priority);
|
|
Packit Service |
0af388 |
_debug(ctx, "Got path group status: %s(%" PRIu32 ")",
|
|
Packit Service |
0af388 |
dmmp_path_group_status_str(dmmp_pg->status), dmmp_pg->status);
|
|
Packit Service |
0af388 |
_debug(ctx, "Got path group selector: '%s'", dmmp_pg->selector);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
out:
|
|
Packit Service |
0af388 |
if (rc != DMMP_OK)
|
|
Packit Service |
0af388 |
_dmmp_path_group_free(dmmp_pg);
|
|
Packit Service |
0af388 |
return rc;
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
void _dmmp_path_group_free(struct dmmp_path_group *dmmp_pg)
|
|
Packit Service |
0af388 |
{
|
|
Packit Service |
0af388 |
uint32_t i = 0;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
if (dmmp_pg == NULL)
|
|
Packit Service |
0af388 |
return;
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
free((char *) dmmp_pg->selector);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
if (dmmp_pg->dmmp_ps != NULL) {
|
|
Packit Service |
0af388 |
for (i = 0; i < dmmp_pg->dmmp_p_count; ++i) {
|
|
Packit Service |
0af388 |
_dmmp_path_free(dmmp_pg->dmmp_ps[i]);
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
free(dmmp_pg->dmmp_ps);
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
free(dmmp_pg);
|
|
Packit Service |
0af388 |
}
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
void dmmp_path_array_get(struct dmmp_path_group *mp_pg,
|
|
Packit Service |
0af388 |
struct dmmp_path ***mp_paths,
|
|
Packit Service |
0af388 |
uint32_t *dmmp_p_count)
|
|
Packit Service |
0af388 |
{
|
|
Packit Service |
0af388 |
assert(mp_pg != NULL);
|
|
Packit Service |
0af388 |
assert(mp_paths != NULL);
|
|
Packit Service |
0af388 |
assert(dmmp_p_count != NULL);
|
|
Packit Service |
0af388 |
|
|
Packit Service |
0af388 |
*mp_paths = mp_pg->dmmp_ps;
|
|
Packit Service |
0af388 |
*dmmp_p_count = mp_pg->dmmp_p_count;
|
|
Packit Service |
0af388 |
}
|