/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2009 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <pthread.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include "isns_server.h" #include "isns_msgq.h" #include "isns_htab.h" #include "isns_cache.h" #include "isns_pdu.h" #include "isns_obj.h" #include "isns_dd.h" #include "isns_func.h" #include "isns_dseng.h" #include "isns_log.h" #include "isns_scn.h" #include "isns_utils.h" #include "isns_esi.h" /* * external variables */ #ifdef DEBUG extern int verbose_mc; extern void print_object(char *, isns_obj_t *); #endif extern msg_queue_t *sys_q; extern msg_queue_t *scn_q; extern pthread_mutex_t el_mtx; extern int cache_flag; /* * global data */ /* * local variables */ /* type of parent object */ const int TYPE_OF_PARENT[MAX_OBJ_TYPE_FOR_SIZE] = { 0, 0, ISCSI_PARENT_TYPE, PORTAL_PARENT_TYPE, PG_PARENT_TYPE, 0, /* OBJ_DD */ 0, /* OBJ_DDS */ 0, /* MAX_OBJ_TYPE */ 0, /* OBJ_DUMMY1 */ 0, /* OBJ_DUMMY2 */ 0, /* OBJ_DUMMY3 */ 0, /* OBJ_DUMMY4 */ ASSOC_ISCSI_PARENT_TYPE, ASSOC_DD_PARENT_TYPE }; /* number of children object type */ const int NUM_OF_CHILD[MAX_OBJ_TYPE] = { 0, MAX_ENTITY_CHILD, MAX_ISCSI_CHILD, MAX_PORTAL_CHILD, MAX_PG_CHILD, 0, 0 }; /* type of a child object */ const int TYPE_OF_CHILD[MAX_OBJ_TYPE][MAX_CHILD_TYPE] = { { 0, 0 }, { OBJ_ISCSI, OBJ_PORTAL }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 }, { 0, 0 } }; /* number of attributes of certain type of object */ const int NUM_OF_ATTRS[MAX_OBJ_TYPE_FOR_SIZE] = { 0, NUM_OF_ENTITY_ATTRS, NUM_OF_ISCSI_ATTRS, NUM_OF_PORTAL_ATTRS, NUM_OF_PG_ATTRS, NUM_OF_DD_ATTRS, NUM_OF_DDS_ATTRS, 0, /* MAX_OBJ_TYPE */ 0, /* OBJ_DUMMY1 */ 0, /* OBJ_DUMMY2 */ 0, /* OBJ_DUMMY3 */ 0, /* OBJ_DUMMY4 */ NUM_OF_ASSOC_ISCSI_ATTRS, NUM_OF_ASSOC_DD_ATTRS }; /* the tag of UID of each type of object */ static const int UID_TAG[MAX_OBJ_TYPE_FOR_SIZE] = { 0, ISNS_ENTITY_INDEX_ATTR_ID, ISNS_ISCSI_NODE_INDEX_ATTR_ID, ISNS_PORTAL_INDEX_ATTR_ID, ISNS_PG_INDEX_ATTR_ID, ISNS_DD_ID_ATTR_ID, ISNS_DD_SET_ID_ATTR_ID, 0, /* MAX_OBJ_TYPE */ 0, /* OBJ_DUMMY1 */ 0, /* OBJ_DUMMY2 */ 0, /* OBJ_DUMMY3 */ 0, /* OBJ_DUMMY4 */ ISNS_DD_ISCSI_INDEX_ATTR_ID, ISNS_DD_ID_ATTR_ID }; /* the index of UID of each type of object */ const int UID_ATTR_INDEX[MAX_OBJ_TYPE_FOR_SIZE] = { 0, ATTR_INDEX_ENTITY(ISNS_ENTITY_INDEX_ATTR_ID), ATTR_INDEX_ISCSI(ISNS_ISCSI_NODE_INDEX_ATTR_ID), ATTR_INDEX_PORTAL(ISNS_PORTAL_INDEX_ATTR_ID), ATTR_INDEX_PG(ISNS_PG_INDEX_ATTR_ID), ATTR_INDEX_DD(ISNS_DD_ID_ATTR_ID), ATTR_INDEX_DDS(ISNS_DD_SET_ID_ATTR_ID), 0, /* MAX_OBJ_TYPE */ 0, /* OBJ_DUMMY1 */ 0, /* OBJ_DUMMY2 */ 0, /* OBJ_DUMMY3 */ 0, /* OBJ_DUMMY4 */ ATTR_INDEX_ASSOC_ISCSI(ISNS_DD_ISCSI_INDEX_ATTR_ID), ATTR_INDEX_ASSOC_DD(ISNS_DD_ID_ATTR_ID) }; /* the index of the key attributes of each type of object */ static const int KEY_ATTR_INDEX[MAX_OBJ_TYPE][MAX_KEY_ATTRS] = { { 0 }, { ATTR_INDEX_ENTITY(ISNS_EID_ATTR_ID), 0 }, { ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID), 0 }, { ATTR_INDEX_PORTAL(ISNS_PORTAL_IP_ADDR_ATTR_ID), ATTR_INDEX_PORTAL(ISNS_PORTAL_PORT_ATTR_ID), 0 }, { ATTR_INDEX_PG(ISNS_PG_ISCSI_NAME_ATTR_ID), ATTR_INDEX_PG(ISNS_PG_PORTAL_IP_ADDR_ATTR_ID), ATTR_INDEX_PG(ISNS_PG_PORTAL_PORT_ATTR_ID) }, { ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID), 0 }, { ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID), 0 } }; /* the operating methods for key attributes of each type of object */ static const int KEY_ATTR_OP[MAX_OBJ_TYPE][MAX_KEY_ATTRS] = { { 0 }, { OP_STRING, 0 }, { OP_STRING, 0 }, { OP_MEMORY_IP6, OP_INTEGER, 0 }, { OP_STRING, OP_MEMORY_IP6, OP_INTEGER }, { OP_STRING, 0 }, { OP_STRING, 0 } }; /* the size of each type of object */ static const int SIZEOF_OBJ[MAX_OBJ_TYPE_FOR_SIZE] = { 0, sizeof (isns_entity_t), sizeof (isns_iscsi_t), sizeof (isns_portal_t), sizeof (isns_pg_t), sizeof (isns_dd_t), sizeof (isns_dds_t), 0, 0, 0, 0, 0, sizeof (isns_assoc_iscsi_t), sizeof (isns_assoc_dd_t) }; #ifdef DEBUG const int NUM_OF_REF[MAX_OBJ_TYPE_FOR_SIZE] = { #else static const int NUM_OF_REF[MAX_OBJ_TYPE_FOR_SIZE] = { #endif 0, 0, 0, 0, PG_REF_COUNT, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; /* the type of the reference object */ static const int TYPE_OF_REF[MAX_OBJ_TYPE][MAX_REF_COUNT + 1] = { { 0 }, { 0 }, { OBJ_PG, OBJ_PORTAL, 0 }, { OBJ_PG, OBJ_ISCSI, 0 }, { 0, OBJ_ISCSI, OBJ_PORTAL }, { 0 }, { 0 } }; /* the operating method for match operation of the reference object */ #define MAX_REF_MATCH (2) static const int REF_MATCH_OPS[MAX_OBJ_TYPE][MAX_REF_MATCH] = { { 0, 0 }, { 0, 0 }, { OP_STRING, 0 }, { OP_MEMORY_IP6, OP_INTEGER }, { 0, 0 }, { 0, 0 }, { 0, 0 } }; /* the index of the attribute of being matched object */ static const int REF_MATCH_ID1[MAX_OBJ_TYPE][MAX_REF_MATCH] = { { 0, 0 }, { 0, 0 }, { ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID), 0 }, { ATTR_INDEX_PORTAL(ISNS_PORTAL_IP_ADDR_ATTR_ID), ATTR_INDEX_PORTAL(ISNS_PORTAL_PORT_ATTR_ID) }, { 0, 0 }, { 0, 0 }, { 0, 0 } }; /* the index of the attribute of matching object */ static const int REF_MATCH_ID2[MAX_OBJ_TYPE][MAX_REF_MATCH] = { { 0, 0 }, { 0, 0 }, { ATTR_INDEX_PG(ISNS_PG_ISCSI_NAME_ATTR_ID), 0 }, { ATTR_INDEX_PG(ISNS_PG_PORTAL_IP_ADDR_ATTR_ID), ATTR_INDEX_PG(ISNS_PG_PORTAL_PORT_ATTR_ID) }, { 0, 0 }, { 0, 0 }, { 0, 0 } }; /* * local functions. */ static uint32_t get_reg_period(); static char *make_unique_name(int *, uint32_t); static lookup_ctrl_t *set_lookup_ctrl(lookup_ctrl_t *, isns_obj_t *); static int setup_ref_lcp(lookup_ctrl_t *, const isns_obj_t *, const isns_obj_t *); static int setup_deref_lcp(lookup_ctrl_t *, const isns_obj_t *, isns_type_t); static int cb_get_parent(void *, void *); static int cb_node_child(void *, void *); static int cb_set_ref(void *, void *); static int cb_clear_ref(void *, void *); static int cb_add_child(void *, void *); static int cb_remove_child(void *, void *); static int cb_verify_ref(void *, void *); static int cb_ref_new2old(void *, void *); static int cb_new_ref(void *, void *); static int ref_new2old( lookup_ctrl_t *, isns_type_t, uint32_t, const isns_obj_t *); static int ref_new2new( lookup_ctrl_t *, const isns_obj_t *, const isns_obj_t *); static int new_ref(const isns_obj_t *, const isns_obj_t *); static uint32_t setup_parent_lcp(lookup_ctrl_t *, isns_obj_t *); static int set_obj_offline(isns_obj_t *); static int copy_attrs(isns_obj_t *, const isns_obj_t *); static isns_obj_t *make_default_pg(const isns_obj_t *, const isns_obj_t *); static isns_obj_t *(*const make_ref[MAX_OBJ_TYPE]) (const isns_obj_t *, const isns_obj_t *) = { NULL, NULL, &make_default_pg, &make_default_pg, NULL, NULL, NULL }; static uint32_t entity_hval(void *, uint16_t, uint32_t *); static uint32_t iscsi_hval(void *, uint16_t, uint32_t *); static uint32_t portal_hval(void *, uint16_t, uint32_t *); static uint32_t pg_hval(void *, uint16_t, uint32_t *); static uint32_t dd_hval(void *, uint16_t, uint32_t *); static uint32_t dds_hval(void *, uint16_t, uint32_t *); static uint32_t (*const hval_func[MAX_OBJ_TYPE]) (void *, uint16_t, uint32_t *) = { NULL, &entity_hval, &iscsi_hval, &portal_hval, &pg_hval, &dd_hval, &dds_hval }; /* * **************************************************************************** * * entity_hval: * caculate the hash value of a network entity object. * * p - the pointer pointers to network entity object or * the lookup control data, both have the key attribute * of a network entity object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t entity_hval( void *p, /* LINTED E_FUNC_ARG_UNUSED */ uint16_t chunk, uint32_t *flags ) { uchar_t *key; isns_obj_t *obj; lookup_ctrl_t *lcp; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p pointers to a network entity object */ obj = (isns_obj_t *)p; key = obj->attrs[ATTR_INDEX_ENTITY(ISNS_EID_ATTR_ID)]. value.ptr; } else { /* p is lookup control data */ lcp = (lookup_ctrl_t *)p; key = lcp->data[0].ptr; } return (htab_compute_hval(key)); } /* * **************************************************************************** * * iscsi_hval: * caculate the hash value of an iscsi storage node object. * * p - the pointer pointers to iscsi storage node object or * the lookup control data, both have the key attribute * of an iscsi storage node object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t iscsi_hval( void *p, /* LINTED E_FUNC_ARG_UNUSED */ uint16_t chunk, uint32_t *flags ) { uchar_t *key; isns_obj_t *obj; lookup_ctrl_t *lcp; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p pointers to an iscsi storage node object */ obj = (isns_obj_t *)p; key = obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID)]. value.ptr; } else { /* p is lookup control data */ lcp = (lookup_ctrl_t *)p; key = lcp->data[0].ptr; } return (htab_compute_hval(key)); } /* * **************************************************************************** * * portal_hval: * caculate the hash value of a portal object. * * p - the pointer pointers to a portal object or the lookup control * data, both have the key attributes of a portal object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t portal_hval( void *p, /* LINTED E_FUNC_ARG_UNUSED */ uint16_t chunk, uint32_t *flags ) { char buff[INET6_ADDRSTRLEN + 8] = { 0 }; char buff2[8] = { 0 }; uchar_t *key; isns_obj_t *obj; lookup_ctrl_t *lcp; in6_addr_t *ip; uint32_t port; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p pointers to a portal object */ obj = (isns_obj_t *)p; ip = obj->attrs[ATTR_INDEX_PORTAL (ISNS_PORTAL_IP_ADDR_ATTR_ID)].value.ip; port = obj->attrs[ATTR_INDEX_PORTAL (ISNS_PORTAL_PORT_ATTR_ID)].value.ui; } else { /* p is lookup control data */ lcp = (lookup_ctrl_t *)p; ip = lcp->data[0].ip; port = lcp->data[1].ui; } key = (uchar_t *)inet_ntop(AF_INET6, (void *)ip, buff, sizeof (buff)); (void) snprintf(buff2, sizeof (buff2), "%d", port); (void) strcat((char *)key, buff2); return (htab_compute_hval(key)); } /* * **************************************************************************** * * pg_hval: * caculate the hash value of a portal group object. * * p - the pointer pointers to a portal group object or the lookup * control data, both have the key attributes of a portal object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t pg_hval( void *p, uint16_t chunk, uint32_t *flags ) { char buff[INET6_ADDRSTRLEN + 8] = { 0 }; char buff2[8] = { 0 }; uchar_t *key = NULL; isns_obj_t *obj; lookup_ctrl_t *lcp; in6_addr_t *ip = NULL; uint32_t port; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p is a portal group object */ obj = (isns_obj_t *)p; if (chunk == 0) { /* the first chunk */ key = obj->attrs[ATTR_INDEX_PG (ISNS_PG_ISCSI_NAME_ATTR_ID)].value.ptr; } else { /* another chunk */ ip = obj->attrs[ATTR_INDEX_PG (ISNS_PG_PORTAL_IP_ADDR_ATTR_ID)].value.ip; port = obj->attrs[ATTR_INDEX_PG (ISNS_PG_PORTAL_PORT_ATTR_ID)].value.ui; } } else { /* p is a lookup control data */ lcp = (lookup_ctrl_t *)p; /* clear the chunk flags */ *flags &= ~FLAGS_CHUNK_MASK; if (lcp->op[0] == OP_STRING) { /* the first chunk */ key = lcp->data[0].ptr; } else { /* another chunk */ ip = lcp->data[0].ip; port = lcp->data[1].ui; *flags |= 1; } } if (key == NULL) { key = (uchar_t *)inet_ntop(AF_INET6, (void *)ip, buff, sizeof (buff)); (void) snprintf(buff2, sizeof (buff2), "%d", port); (void) strcat((char *)key, buff2); } return (htab_compute_hval(key)); } /* * **************************************************************************** * * dd_hval: * caculate the hash value of a DD object. * * p - the pointer pointers to a DD object or the lookup control data, * both have the key attributes of a DD object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t dd_hval( void *p, /* LINTED E_FUNC_ARG_UNUSED */ uint16_t chunk, uint32_t *flags ) { uchar_t *key; isns_obj_t *obj; lookup_ctrl_t *lcp; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p is a DD object */ obj = (isns_obj_t *)p; key = obj->attrs[ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID)]. value.ptr; } else { /* p is a lookup control data */ lcp = (lookup_ctrl_t *)p; key = lcp->data[0].ptr; } return (htab_compute_hval(key)); } /* * **************************************************************************** * * dds_hval: * caculate the hash value of a DD-set object. * * p - the pointer pointers to a DD-set object or the lookup control data, * both have the key attributes of a DD-set object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ static uint32_t dds_hval( void *p, /* LINTED E_FUNC_ARG_UNUSED */ uint16_t chunk, uint32_t *flags ) { uchar_t *key; isns_obj_t *obj; lookup_ctrl_t *lcp; if ((*flags & FLAGS_CTRL_MASK) == 0) { /* p is a DD-set object */ obj = (isns_obj_t *)p; key = obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID)]. value.ptr; } else { /* p is lookup control data */ lcp = (lookup_ctrl_t *)p; key = lcp->data[0].ptr; } return (htab_compute_hval(key)); } /* * **************************************************************************** * * obj_hval: * caculate the hash value of an object. * * p - the pointer pointers to an object or lookup control data, * both has the object type and the key attributes of an object. * chunk- which chunk of the hash table. * flags- pointer to flags. * return - the hash value. * * **************************************************************************** */ uint32_t obj_hval( void *p, uint16_t chunk, uint32_t *flags ) { isns_type_t type = ((isns_obj_t *)p)->type; return (hval_func[type](p, chunk, flags)); } /* * **************************************************************************** * * get_obj_uid: * get the UID of an object. * * p - the pointer pointers to an object. * return - the UID. * * **************************************************************************** */ uint32_t get_obj_uid( const void *p ) { isns_obj_t *obj = (isns_obj_t *)p; isns_attr_t *attr = &obj->attrs[UID_ATTR_INDEX[obj->type]]; uint32_t uid = attr->value.ui; return (uid); } /* * **************************************************************************** * * set_obj_uid: * set the UID of an object. * * p - the pointer pointers to an object. * uid - the UID. * return - the UID. * * **************************************************************************** */ uint32_t set_obj_uid( void *p, uint32_t uid ) { isns_obj_t *obj = (isns_obj_t *)p; isns_attr_t *attr = &obj->attrs[UID_ATTR_INDEX[obj->type]]; /* set the tag, len and value */ attr->tag = UID_TAG[obj->type]; attr->len = 4; attr->value.ui = uid; return (uid); } /* * **************************************************************************** * * obj_cmp: * compare between two objects or an object with a lookup control data. * * p1 - the pointer points to an object. * p2 - the pointer points to an object or a lookup control data. * flags- 0: p2 is an object; otherwise p2 is a lookup control data. * return - the comparsion result. * * **************************************************************************** */ int obj_cmp( void *p1, void *p2, int flags ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t buff = { 0 }; lookup_ctrl_t *lcp; uint32_t uid; if (flags == 0) { lcp = set_lookup_ctrl(&buff, (isns_obj_t *)p2); } else { lcp = (lookup_ctrl_t *)p2; uid = get_obj_uid(obj); /* the object are linked with decending order by */ /* the object UID, if the object UID is greater than */ /* or equal to the current UID, it needs to compare */ /* for the next one. */ if (lcp->curr_uid != 0 && uid >= lcp->curr_uid) { return (-1); } } return (key_cmp(lcp, obj)); } /* * **************************************************************************** * * replace_object: * replace an existing object with the new one. * * p1 - the pointer points to an object being replaced. * p2 - the pointer points to a new object. * uid_p- points to uid for returning. * flag - 0: do not free the source object, otherwise free it. * return - error code. * * **************************************************************************** */ int replace_object( void *p1, void *p2, uint32_t *uid_p, int flag ) { int ec = 0; #ifndef SKIP_SRC_AUTH uint32_t *pp_dst, *pp_src, swap; #endif int online; isns_obj_t *dst = (isns_obj_t *)p1; isns_obj_t *src = (isns_obj_t *)p2; if (src->type == OBJ_DD || src->type == OBJ_DDS) { /* replace not allowed */ return (ERR_NAME_IN_USE); } online = is_obj_online(dst); /* set cache update flag */ SET_CACHE_UPDATED(); /* update parent uid */ #ifndef SKIP_SRC_AUTH pp_dst = get_parent_p(dst); if (pp_dst != NULL) { pp_src = get_parent_p(src); swap = *pp_dst; *pp_dst = *pp_src; if (swap != 0) { *pp_src = swap; } } #endif /* update all of attributes */ if (copy_attrs(dst, src) != 0) { return (ISNS_RSP_INTERNAL_ERROR); } /* free up the src object */ if (flag != 0) { (void) free_object(src); } else if (online == 0) { (void) set_obj_uid(src, get_obj_uid(dst)); (void) set_obj_offline(src); } /* update data store */ if (sys_q != NULL) { ec = write_data(DATA_UPDATE, dst); } else { /* we should never have duplicated entry in data store */ ec = ISNS_RSP_INTERNAL_ERROR; } /* trigger a scn */ if (ec == 0) { if (scn_q != NULL) { (void) make_scn((online == 0) ? ISNS_OBJECT_ADDED : ISNS_OBJECT_UPDATED, dst); } if (uid_p != NULL) { *uid_p = get_obj_uid(dst); } } return (ec); } /* * **************************************************************************** * * add_object: * post function after adding a new object. * * p - object which has been added. * return - error code. * * **************************************************************************** */ int add_object( void *p ) { int ec = 0; isns_obj_t *obj = (isns_obj_t *)p; /* add the new object to data store */ if (sys_q != NULL) { ec = write_data(DATA_ADD, obj); } /* trigger a scn */ if (ec == 0 && scn_q != NULL) { (void) make_scn(ISNS_OBJECT_ADDED, obj); } return (ec); } /* * **************************************************************************** * * obj_tab_init: * initialize the object hash tables. * * c - points to the cache. * return - error code. * * **************************************************************************** */ int obj_tab_init( struct cache *c ) { htab_t *t; htab_init(); /* * allocate an array of pointer for the object hash tables. */ c->t = (struct htab **)calloc(sizeof (struct htab *), MAX_OBJ_TYPE); if (c->t == NULL) { return (1); } /* * hash table for network entity objects. */ t = htab_create(UID_FLAGS_SEQ, 8, 1); if (t != NULL) { t->c = c; c->t[OBJ_ENTITY] = t; } else { return (1); } /* * hash table for iscsi storage node objects. */ t = htab_create(UID_FLAGS_SEQ, 8, 1); if (t != NULL) { t->c = c; c->t[OBJ_ISCSI] = t; } else { return (1); } /* * hash table for portal objects. */ t = htab_create(UID_FLAGS_SEQ, 8, 1); if (t != NULL) { t->c = c; c->t[OBJ_PORTAL] = t; } else { return (1); } /* * hash table for portal group objects. */ t = htab_create(UID_FLAGS_SEQ, 8, 2); if (t != NULL) { t->c = c; c->t[OBJ_PG] = t; } else { return (1); } /* * hash table for discovery domain objects. */ t = htab_create(0, 6, 1); if (t != NULL) { t->c = c; c->t[OBJ_DD] = t; } else { return (1); } /* * hash table for discovery domain set objects. */ t = htab_create(0, 4, 1); if (t != NULL) { t->c = c; c->t[OBJ_DDS] = t; } else { return (1); } return (0); } /* * **************************************************************************** * * get_ref_np: * get the ref pointer of the portal group object. * * obj - portal group object. * return - ref pointer. * * **************************************************************************** */ static uint32_t * get_ref_np( isns_obj_t *obj, int n ) { uint32_t *refp = obj->type == OBJ_PG ? &((isns_pg_t *)obj)->ref[n] : NULL; return (refp); } #ifdef DEBUG uint32_t #else static uint32_t #endif get_ref_n( isns_obj_t *obj, int n ) { return (*get_ref_np(obj, n)); } static uint32_t * get_ref_p( isns_obj_t *obj, isns_type_t rt ) { isns_type_t t = obj->type; int i = 0; while (i < NUM_OF_REF[t]) { if (rt == TYPE_OF_REF[t][i + 1]) { return (get_ref_np(obj, i)); } i ++; } return (NULL); } uint32_t get_ref_t( isns_obj_t *obj, isns_type_t type ) { uint32_t *refp = get_ref_p(obj, type); if (refp != NULL) { return (*refp); /* LINTED E_NOP_ELSE_STMT */ } else { ASSERT(0); } return (0); } /* * **************************************************************************** * * get_parent_p: * get the pointer of the parent object. * * obj - an object. * return - parent object pointer. * * **************************************************************************** */ uint32_t *const get_parent_p( const isns_obj_t *obj ) { uint32_t *pp; switch (obj->type) { case OBJ_ISCSI: pp = &((isns_iscsi_t *)obj)->puid; break; case OBJ_PORTAL: pp = &((isns_portal_t *)obj)->puid; break; case OBJ_PG: pp = &((isns_pg_t *)obj)->puid; break; case OBJ_ASSOC_ISCSI: pp = &((isns_assoc_iscsi_t *)obj)->puid; break; case OBJ_ASSOC_DD: pp = &((isns_assoc_dd_t *)obj)->puid; break; default: pp = NULL; break; } return (pp); } uint32_t get_parent_uid( const isns_obj_t *obj ) { uint32_t *pp = get_parent_p(obj); if (pp != NULL) { return (*pp); } return (0); } /* * **************************************************************************** * * get_child_np: * get the pointer of the UID array of the n'th child of an object. * * obj - an object. * n - the child index. * return - the pointer of the UID array. * * **************************************************************************** */ static uint32_t ** get_child_np( isns_obj_t *obj, int n ) { uint32_t **pp = obj->type == OBJ_ENTITY ? &((isns_entity_t *)obj)->cuid[n] : NULL; return (pp); } /* * **************************************************************************** * * get_child_n: * get the UID array of the n'th child of an object. * * obj - an object. * n - the child index. * return - the UID array. * * **************************************************************************** */ #ifdef DEBUG uint32_t * #else static uint32_t * #endif get_child_n( isns_obj_t *obj, int n ) { uint32_t **pp = get_child_np(obj, n); if (pp != NULL) { return (*pp); } ASSERT(0); return (NULL); } /* * **************************************************************************** * * get_child_p: * get the pointer of the UID array of the child matching the type. * * base - an object. * child_type - the child object type. * return - the pointer of the UID array. * * **************************************************************************** */ static uint32_t ** get_child_p( isns_obj_t *base, int child_type ) { uint32_t **pp = NULL; int i = 0; while (i < NUM_OF_CHILD[base->type]) { if (child_type == TYPE_OF_CHILD[base->type][i]) { pp = get_child_np(base, i); break; } i ++; } return (pp); } /* * **************************************************************************** * * get_child_t: * get the UID array of the child object matching the type. * * base - an object. * child_type - the child object type. * return - the UID array. * * **************************************************************************** */ uint32_t * get_child_t( isns_obj_t *base, int child_type ) { uint32_t **pp = get_child_p(base, child_type); if (pp != NULL) { return (*pp); } else { return (NULL); } } /* * **************************************************************************** * * key_cmp: * compare the object against the lookup control data. * * lcp - the lookup control data. * obj - an object. * return - comparison result. * * **************************************************************************** */ int key_cmp( lookup_ctrl_t *lcp, isns_obj_t *obj ) { int i = 0; int match = 1; while (i < MAX_LOOKUP_CTRL && lcp->op[i] > 0 && match) { isns_attr_t *attr = &obj->attrs[lcp->id[i]]; switch (lcp->op[i]) { case OP_STRING: match = (strcmp((const char *)lcp->data[i].ptr, (const char *)attr->value.ptr) == 0); break; case OP_INTEGER: match = (lcp->data[i].ui == attr->value.ui); break; case OP_MEMORY_IP6: match = !memcmp((void *)lcp->data[i].ip, (void *)attr->value.ip, sizeof (in6_addr_t)); break; default: ASSERT(0); match = 0; break; } i ++; } if (i && match) { return (0); } else { return (1); } } /* * **************************************************************************** * * set_lookup_ctrl: * fill in the lookup control data for an object. * * lcp - the lookup control data. * obj - an object. * return - the lookup control data. * * **************************************************************************** */ static lookup_ctrl_t * set_lookup_ctrl( lookup_ctrl_t *lcp, isns_obj_t *obj ) { isns_type_t type = obj->type; uint32_t id, op; int i = 0; lcp->type = type; while (i < MAX_KEY_ATTRS) { op = KEY_ATTR_OP[type][i]; if (op != 0) { id = KEY_ATTR_INDEX[type][i]; lcp->id[i] = id; lcp->op[i] = op; lcp->data[i].ui = obj->attrs[id].value.ui; } else { break; } i ++; } return (lcp); } /* * **************************************************************************** * * assign_attr: * assign an attribute. * * attr - the attribute being assigned. * tmp - the attribute. * return - error code. * * **************************************************************************** */ int assign_attr( isns_attr_t *attr, const isns_attr_t *tmp ) { uint32_t t; switch (tmp->tag) { case ISNS_EID_ATTR_ID: case ISNS_DD_SET_NAME_ATTR_ID: case ISNS_DD_NAME_ATTR_ID: if (tmp->len == 0 && attr->len == 0) { int len; char *name = make_unique_name(&len, tmp->tag); if (name != NULL) { attr->value.ptr = (uchar_t *)name; attr->tag = tmp->tag; attr->len = len; } else { /* memory exhausted */ return (1); } } case ISNS_PORTAL_NAME_ATTR_ID: case ISNS_ISCSI_NAME_ATTR_ID: case ISNS_ISCSI_ALIAS_ATTR_ID: case ISNS_ISCSI_AUTH_METHOD_ATTR_ID: case ISNS_PG_ISCSI_NAME_ATTR_ID: case ISNS_DD_ISCSI_NAME_ATTR_ID: if (tmp->len == 0) { return (0); } else if (tmp->len >= attr->len) { attr->value.ptr = realloc( attr->value.ptr, tmp->len + 1); } if (attr->value.ptr != NULL) { (void) strncpy((char *)attr->value.ptr, (char *)tmp->value.ptr, tmp->len); attr->value.ptr[tmp->len] = 0; attr->tag = tmp->tag; attr->len = tmp->len; } else { /* memory exhausted */ return (1); } break; case ISNS_MGMT_IP_ADDR_ATTR_ID: case ISNS_PORTAL_IP_ADDR_ATTR_ID: case ISNS_PG_PORTAL_IP_ADDR_ATTR_ID: if (attr->value.ip == NULL) { attr->value.ip = (in6_addr_t *)calloc(1, tmp->len); } if (attr->value.ip != NULL) { (void) memcpy((void *)attr->value.ip, (void *)tmp->value.ip, tmp->len); attr->tag = tmp->tag; attr->len = tmp->len; } else { /* memory exhausted */ return (1); } break; case ISNS_ENTITY_INDEX_ATTR_ID: case ISNS_PORTAL_INDEX_ATTR_ID: case ISNS_ISCSI_NODE_INDEX_ATTR_ID: case ISNS_PG_INDEX_ATTR_ID: case ISNS_DD_SET_ID_ATTR_ID: case ISNS_DD_ID_ATTR_ID: if (attr->value.ui != 0) { break; } case ISNS_ENTITY_PROTOCOL_ATTR_ID: case ISNS_VERSION_RANGE_ATTR_ID: case ISNS_PORTAL_PORT_ATTR_ID: case ISNS_ESI_PORT_ATTR_ID: case ISNS_SCN_PORT_ATTR_ID: case ISNS_ISCSI_NODE_TYPE_ATTR_ID: case ISNS_ISCSI_SCN_BITMAP_ATTR_ID: case ISNS_PG_PORTAL_PORT_ATTR_ID: case ISNS_PG_TAG_ATTR_ID: case ISNS_DD_SET_STATUS_ATTR_ID: case ISNS_DD_ISCSI_INDEX_ATTR_ID: attr->tag = tmp->tag; attr->len = tmp->len; attr->value.ui = tmp->value.ui; break; case ISNS_ENTITY_REG_PERIOD_ATTR_ID: attr->tag = tmp->tag; attr->len = tmp->len; attr->value.ui = tmp->value.ui; t = get_reg_period(); if (attr->value.ui > t) { attr->value.ui = t; } else if (attr->value.ui < ONE_DAY) { attr->value.ui = ONE_DAY; } break; case ISNS_ESI_INTERVAL_ATTR_ID: attr->tag = tmp->tag; attr->len = tmp->len; attr->value.ui = tmp->value.ui; if (attr->value.ui > ONE_DAY) { attr->value.ui = ONE_DAY; } else if (attr->value.ui < MIN_ESI_INTVAL) { attr->value.ui = MIN_ESI_INTVAL; /* 20 seconds */ } break; default: ASSERT(0); /* don't assign the attribute */ break; } return (0); } /* * **************************************************************************** * * copy_attrs: * copy all of attributes from one object to another. * * dst - the destination object. * tmp - the source object. * return - error code. * * **************************************************************************** */ static int copy_attrs( isns_obj_t *dst, const isns_obj_t *src ) { int i = 0; int n = NUM_OF_ATTRS[dst->type]; isns_attr_t *dst_attr; const isns_attr_t *src_attr; while (i < n) { src_attr = &(src->attrs[i]); if (src_attr->tag != 0) { dst_attr = &(dst->attrs[i]); if (assign_attr(dst_attr, src_attr) != 0) { return (1); } } i ++; } return (0); } /* * **************************************************************************** * * extract_attr: * extract an attribute from a TLV format data. * * attr - the attribute. * tlv - the TLV format data. * return - error code. * * **************************************************************************** */ int extract_attr( isns_attr_t *attr, const isns_tlv_t *tlv, int flag ) { int ec = 0; uint32_t min_len = 4, max_len = 224; switch (tlv->attr_id) { case ISNS_EID_ATTR_ID: min_len = 0; case ISNS_PORTAL_NAME_ATTR_ID: case ISNS_ISCSI_ALIAS_ATTR_ID: case ISNS_DD_SET_NAME_ATTR_ID: case ISNS_DD_NAME_ATTR_ID: max_len = 256; case ISNS_ISCSI_NAME_ATTR_ID: case ISNS_PG_ISCSI_NAME_ATTR_ID: if (tlv->attr_len < min_len || tlv->attr_len > max_len) { ec = ISNS_RSP_MSG_FORMAT_ERROR; } else { attr->tag = tlv->attr_id; attr->len = tlv->attr_len; attr->value.ptr = (uchar_t *)&(tlv->attr_value[0]); } break; case ISNS_ISCSI_AUTH_METHOD_ATTR_ID: attr->tag = tlv->attr_id; attr->len = tlv->attr_len; attr->value.ptr = (uchar_t *)&(tlv->attr_value[0]); break; case ISNS_MGMT_IP_ADDR_ATTR_ID: case ISNS_PORTAL_IP_ADDR_ATTR_ID: case ISNS_PG_PORTAL_IP_ADDR_ATTR_ID: if (tlv->attr_len != 16) { ec = ISNS_RSP_MSG_FORMAT_ERROR; } else { attr->tag = tlv->attr_id; attr->len = tlv->attr_len; attr->value.ip = (void *)&(tlv->attr_value[0]); } break; case ISNS_ENTITY_PROTOCOL_ATTR_ID: case ISNS_VERSION_RANGE_ATTR_ID: case ISNS_ENTITY_REG_PERIOD_ATTR_ID: /* fall throught */ case ISNS_PORTAL_PORT_ATTR_ID: case ISNS_ESI_INTERVAL_ATTR_ID: case ISNS_ESI_PORT_ATTR_ID: case ISNS_SCN_PORT_ATTR_ID: /* fall throught */ case ISNS_ISCSI_NODE_TYPE_ATTR_ID: /* fall throught */ case ISNS_PG_PORTAL_PORT_ATTR_ID: /* fall throught */ case ISNS_DD_SET_ID_ATTR_ID: case ISNS_DD_SET_STATUS_ATTR_ID: /* fall throught */ case ISNS_DD_ID_ATTR_ID: if (tlv->attr_len != 4) { ec = ISNS_RSP_MSG_FORMAT_ERROR; break; } case ISNS_PG_TAG_ATTR_ID: attr->tag = tlv->attr_id; attr->len = tlv->attr_len; if (tlv->attr_len == 4) { attr->value.ui = ntohl(*(uint32_t *) &(tlv->attr_value[0])); } else { attr->value.ui = 0; } break; case ISNS_ISCSI_SCN_BITMAP_ATTR_ID: /* ignore scn bitmap attribute during object registration, */ /* it is registered by scn_reg message. */ case ISNS_ENTITY_ISAKMP_P1_ATTR_ID: case ISNS_ENTITY_CERT_ATTR_ID: case ISNS_PORTAL_SEC_BMP_ATTR_ID: case ISNS_PORTAL_ISAKMP_P1_ATTR_ID: case ISNS_PORTAL_ISAKMP_P2_ATTR_ID: case ISNS_PORTAL_CERT_ATTR_ID: break; case ISNS_PORTAL_INDEX_ATTR_ID: case ISNS_ISCSI_NODE_INDEX_ATTR_ID: case ISNS_PG_INDEX_ATTR_ID: if (flag == 0) { if (tlv->attr_len != 4) { ec = ISNS_RSP_MSG_FORMAT_ERROR; } else { attr->tag = tlv->attr_id; attr->len = tlv->attr_len; attr->value.ui = ntohl(*(uint32_t *) &(tlv->attr_value[0])); } break; } case ISNS_ENTITY_INDEX_ATTR_ID: case ISNS_TIMESTAMP_ATTR_ID: default: if (flag == 0) { ec = ISNS_RSP_INVALID_QRY; } else { ec = ISNS_RSP_INVALID_REGIS; } break; } return (ec); } /* * **************************************************************************** * * copy_attr: * copy an attribute from a TLV format data. * * attr - the attribute. * tlv - the TLV format data. * return - error code. * * **************************************************************************** */ static int copy_attr( isns_attr_t *attr, const isns_tlv_t *tlv ) { int ec = 0; isns_attr_t tmp = { 0 }; /* extract the attribute first */ ec = extract_attr(&tmp, tlv, 1); /* assign the attribute */ if (ec == 0 && tmp.tag != 0) { if (assign_attr(attr, &tmp) != 0) { ec = ISNS_RSP_INTERNAL_ERROR; } } return (ec); } /* * **************************************************************************** * * get_timestamp: * get current timestamp. * * return - current timestamp. * * **************************************************************************** */ uint32_t get_timestamp( ) { uint32_t t; int flag; /* block the scheduler */ (void) pthread_mutex_lock(&el_mtx); /* get most current time */ if (sys_q != NULL) { /* need to wakeup idle */ flag = 1; } else { flag = 0; } t = get_stopwatch(flag); /* unblock it */ (void) pthread_mutex_unlock(&el_mtx); return (t); } /* * **************************************************************************** * * get_reg_period: * get the longest registration period. * * return - the longest registration period. * * **************************************************************************** */ static uint32_t get_reg_period( ) { uint32_t t; uint32_t period; /* get most current time */ t = get_timestamp(); /* just one second before the end of the world */ period = INFINITY - t - 1; return (period); } /* * **************************************************************************** * * obj_calloc: * allocate memory space for an object. * * type - the object type. * return - pointer of the object being allocated. * * **************************************************************************** */ isns_obj_t * obj_calloc( int type ) { isns_obj_t *obj = NULL; obj = (isns_obj_t *)calloc(1, SIZEOF_OBJ[type]); if (obj != NULL) { obj->type = type; #ifdef DEBUG if (verbose_mc) { printf("object(%d) allocated\n", type); } #endif } return (obj); } /* * **************************************************************************** * * make_default_entity: * generate a default network entity object. * * return - pointer of the default network entity object. * * **************************************************************************** */ isns_obj_t * make_default_entity( ) { uint32_t t; isns_obj_t *obj = obj_calloc(OBJ_ENTITY); isns_attr_t *attr; if (obj != NULL) { int len; char *eid = make_unique_name(&len, ISNS_EID_ATTR_ID); if (!eid) { free(obj); return (NULL); } attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_EID_ATTR_ID)]; /* set default entity name */ attr->tag = ISNS_EID_ATTR_ID; attr->len = len; attr->value.ptr = (uchar_t *)eid; /* set default registration period */ attr = &obj->attrs[ ATTR_INDEX_ENTITY(ISNS_ENTITY_REG_PERIOD_ATTR_ID)]; if (attr->tag == 0) { attr->tag = ISNS_ENTITY_REG_PERIOD_ATTR_ID; attr->len = 4; t = get_reg_period(); attr->value.ui = t; } } return (obj); } /* * **************************************************************************** * * make_default_pg: * generate a default portal group object. * * iscsi - the iscsi storage node object. * portal - the portal object. * return - pointer of the default portal group object. * * **************************************************************************** */ static isns_obj_t * make_default_pg( const isns_obj_t *p1, const isns_obj_t *p2 ) { const isns_obj_t *iscsi, *portal; const isns_attr_t *name, *addr, *port; isns_obj_t *pg; uchar_t *pg_name; in6_addr_t *pg_addr; isns_attr_t *attr; uint32_t *refp; if (p1->type == OBJ_ISCSI) { iscsi = p1; portal = p2; } else { iscsi = p2; portal = p1; } name = &iscsi->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID)]; addr = &portal->attrs[ATTR_INDEX_PORTAL(ISNS_PORTAL_IP_ADDR_ATTR_ID)]; port = &portal->attrs[ATTR_INDEX_PORTAL(ISNS_PORTAL_PORT_ATTR_ID)]; pg = obj_calloc(OBJ_PG); pg_name = (uchar_t *)malloc(name->len); pg_addr = (in6_addr_t *)malloc(addr->len); if (pg != NULL && pg_name != NULL && pg_addr != NULL) { (void) strcpy((char *)pg_name, (char *)name->value.ptr); attr = &pg->attrs[ATTR_INDEX_PG(ISNS_PG_ISCSI_NAME_ATTR_ID)]; attr->tag = ISNS_PG_ISCSI_NAME_ATTR_ID; attr->len = name->len; attr->value.ptr = pg_name; (void) memcpy((void *)pg_addr, (void *)addr->value.ip, addr->len); attr = &pg->attrs[ATTR_INDEX_PG( ISNS_PG_PORTAL_IP_ADDR_ATTR_ID)]; attr->tag = ISNS_PG_PORTAL_IP_ADDR_ATTR_ID; attr->len = addr->len; attr->value.ip = pg_addr; attr = &pg->attrs[ATTR_INDEX_PG( ISNS_PG_PORTAL_PORT_ATTR_ID)]; attr->tag = ISNS_PG_PORTAL_PORT_ATTR_ID; attr->len = port->len; attr->value.ui = port->value.ui; attr = &pg->attrs[ATTR_INDEX_PG( ISNS_PG_TAG_ATTR_ID)]; attr->tag = ISNS_PG_TAG_ATTR_ID; attr->len = 4; attr->value.ui = ISNS_DEFAULT_PGT; refp = get_ref_p(pg, OBJ_ISCSI); *refp = get_obj_uid(iscsi); refp = get_ref_p(pg, OBJ_PORTAL); *refp = get_obj_uid(portal); (void) set_parent_obj(pg, get_parent_uid(iscsi)); } else { free(pg); free(pg_name); free(pg_addr); pg = NULL; } return (pg); } /* * **************************************************************************** * * reg_get_entity: * parse the Operating Attributes of the DevAttrReg message and * create the Network Entity object if it has one. * * p - the pointer of the object for returning. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ int reg_get_entity( isns_obj_t **p, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *entity = NULL; tmp = *op; tmp_len = *op_len; /* parse the entity object */ if (tmp_len >= 8 && IS_ENTITY_KEY(tmp->attr_id)) { entity = obj_calloc(OBJ_ENTITY); if (entity != NULL) { do { attr = &entity->attrs[ ATTR_INDEX_ENTITY(tmp->attr_id)]; ec = copy_attr(attr, tmp); NEXT_TLV(tmp, tmp_len); } while (ec == 0 && tmp_len >= 8 && IS_ENTITY_ATTR(tmp->attr_id)); } else { ec = ISNS_RSP_INTERNAL_ERROR; } if (ec == 0) { /* set default registration period */ attr = &entity->attrs[ ATTR_INDEX_ENTITY(ISNS_ENTITY_REG_PERIOD_ATTR_ID)]; if (attr->tag == 0) { attr->tag = ISNS_ENTITY_REG_PERIOD_ATTR_ID; attr->len = 4; attr->value.ui = get_reg_period(); } } else if (entity != NULL) { free(entity); entity = NULL; } } *p = entity; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_iscsi: * parse the Operating Attributes of the DevAttrReg message and * create an iSCSI Storage Node object. * * p - the pointer of the object for returning. * pg_key1 - the pointer of iscsi storage node name for returning. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ static int reg_get_iscsi( isns_obj_t **p, isns_attr_t *pg_key1, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *obj = NULL; tmp = *op; tmp_len = *op_len; /* keep the iscsi storage node name for */ /* parsing a pg object which is immediately */ /* followed with a PGT by the iscsi storage node */ pg_key1->tag = PG_KEY1; pg_key1->len = tmp->attr_len; pg_key1->value.ptr = (uchar_t *)&tmp->attr_value[0]; /* parse one iscsi storage node object */ obj = obj_calloc(OBJ_ISCSI); if (obj != NULL) { /* parse key & non-key attributes */ do { attr = &obj->attrs[ ATTR_INDEX_ISCSI(tmp->attr_id)]; ec = copy_attr(attr, tmp); NEXT_TLV(tmp, tmp_len); } while (ec == 0 && tmp_len >= 8 && IS_ISCSI_ATTR(tmp->attr_id)); } else { /* no memory */ ec = ISNS_RSP_INTERNAL_ERROR; } *p = obj; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_portal: * parse the Operating Attributes of the DevAttrReg message and * create a Portal object. * * p - the pointer of the object for returning. * pg_key1 - the pointer of portal ip addr for returning. * pg_key2 - the pointer of portal port for returning. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ static int reg_get_portal( isns_obj_t **p, isns_attr_t *pg_key1, isns_attr_t *pg_key2, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *obj = NULL; isns_tlv_t *ip; tmp = *op; tmp_len = *op_len; /* keep the portal ip addr */ pg_key1->tag = PG_KEY2; pg_key1->len = tmp->attr_len; pg_key1->value.ip = (void *)&tmp->attr_value[0]; ip = tmp; NEXT_TLV(tmp, tmp_len); if (tmp_len > 8 && tmp->attr_id == PORTAL_KEY2 && tmp->attr_len == 4) { /* keep the portal port */ pg_key2->tag = PG_KEY3; pg_key2->len = tmp->attr_len; pg_key2->value.ui = ntohl(*(uint32_t *)&tmp->attr_value[0]); /* parse one portal object */ obj = obj_calloc(OBJ_PORTAL); if (obj != NULL) { /* copy ip addr attribute */ attr = &obj->attrs[ ATTR_INDEX_PORTAL(ip->attr_id)]; ec = copy_attr(attr, ip); /* copy port attribute */ if (ec == 0) { attr = &obj->attrs[ ATTR_INDEX_PORTAL(tmp->attr_id)]; ec = copy_attr(attr, tmp); } /* parse non-key attributes */ NEXT_TLV(tmp, tmp_len); while (ec == 0 && tmp_len >= 8 && IS_PORTAL_ATTR(tmp->attr_id)) { attr = &obj->attrs[ ATTR_INDEX_PORTAL( tmp->attr_id)]; ec = copy_attr(attr, tmp); NEXT_TLV(tmp, tmp_len); } } else { /* no memory */ ec = ISNS_RSP_INTERNAL_ERROR; } } else { /* ip address is not followed by port */ ec = ISNS_RSP_MSG_FORMAT_ERROR; } *p = obj; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_pg: * parse the Operating Attributes of the DevAttrReg message and * create a Portal Group object. * * p - the pointer of the object for returning. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ static int reg_get_pg( isns_obj_t **p, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *obj = NULL; tmp = *op; tmp_len = *op_len; /* parse a complete pg object */ obj = obj_calloc(OBJ_PG); if (obj != NULL) { /* parse attributes */ do { attr = &obj->attrs[ ATTR_INDEX_PG(tmp->attr_id)]; ec = copy_attr(attr, tmp); NEXT_TLV(tmp, tmp_len); } while (ec == 0 && tmp_len >= 8 && IS_PG_ATTR(tmp->attr_id)); } else { ec = ISNS_RSP_INTERNAL_ERROR; } *p = obj; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_pg1: * parse the Operating Attributes of the DevAttrReg message and * create a Portal Group object which is followed to a Portal object. * * p - the pointer of the object for returning. * pgt - the size-3 array of pointers which have the pg portal ip addr, port * and the pg tag attributes. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ static int reg_get_pg1( isns_obj_t **p, isns_attr_t const *pgt, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *obj = NULL; int i = 0; tmp = *op; tmp_len = *op_len; if (pgt[0].tag == PG_KEY2 && pgt[1].tag == PG_KEY3) { /* the pg iscsi storage node name is */ /* followed to a portal group tag */ obj = obj_calloc(OBJ_PG); if (obj != NULL) { /* copy pg iscsi storage node name */ attr = &obj->attrs[ ATTR_INDEX_PG(tmp->attr_id)]; ec = copy_attr(attr, tmp); /* copy pg ip addr, pg port & pgt */ while (ec == 0 && i < 3) { attr = &obj->attrs[ ATTR_INDEX_PG(pgt[i].tag)]; ec = assign_attr(attr, &pgt[i]); i ++; } NEXT_TLV(tmp, tmp_len); } else { /* no memory */ ec = ISNS_RSP_INTERNAL_ERROR; } } else { ec = ISNS_RSP_MSG_FORMAT_ERROR; } *p = obj; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_pg2: * parse the Operating Attributes of the DevAttrReg message and * create a Portal Group object which is followed to a iSCSI * Storage Node object. * * p - the pointer of the object for returning. * pgt - the size-3 array of pointers which have the pg iscsi storage * node name and the pg tag attributes. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ static int reg_get_pg2( isns_obj_t **p, isns_attr_t const *pgt, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; isns_tlv_t *tmp; uint16_t tmp_len; isns_attr_t *attr; isns_obj_t *obj = NULL; int i = 0; isns_tlv_t *ip; tmp = *op; tmp_len = *op_len; /* keep ip address */ ip = tmp; NEXT_TLV(tmp, tmp_len); if (tmp_len > 8 && /* expect pg portal port */ tmp->attr_id == PG_KEY3 && tmp->attr_len == 4 && /* expect pg tag */ pgt[2].tag == PG_PGT && /* expect pg iscsi storage node name only */ pgt[1].tag == 0 && pgt[0].tag == PG_KEY1) { /* the pg portal ip addr & port is followed */ /* to a pg tag and we have the iscsi storage */ /* node parsed previously */ obj = obj_calloc(OBJ_PG); if (obj != NULL) { /* copy the pg ip addr */ attr = &obj->attrs[ ATTR_INDEX_PG(ip->attr_id)]; ec = copy_attr(attr, ip); /* copy the pg port */ if (ec == 0) { attr = &obj->attrs[ ATTR_INDEX_PG(tmp->attr_id)]; ec = copy_attr(attr, tmp); } /* copy pg iscsi storage node name & pgt */ while (ec == 0 && i < 3) { attr = &obj->attrs[ ATTR_INDEX_PG(pgt[i].tag)]; ec = assign_attr(attr, &pgt[i]); i += 2; } NEXT_TLV(tmp, tmp_len); } else { ec = ISNS_RSP_INTERNAL_ERROR; } } else { ec = ISNS_RSP_MSG_FORMAT_ERROR; } *p = obj; *op = tmp; *op_len = tmp_len; return (ec); } /* * **************************************************************************** * * reg_get_obj: * parse and create one object from the rest of Operating Attributes * of the DevAttrReg message, the object can be iSCSI Storage Node, * Portal or Portal Group. * * p - the pointer of the object for returning. * pgt - an attribute array with size 3, the elements are: * 0: the first pg key attribute, it is either the name of an * iscsi storage node object or the ip addr of a portal object. * 1: the second pg key attribute, i.e. the portal port. * 2: the portal group tag attribute. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ int reg_get_obj( isns_obj_t **p, isns_attr_t *pgt, isns_tlv_t **op, uint16_t *op_len ) { int ec = 0; int derefd = 0; uint32_t pg_tag; if (*op_len == 0) { *p = NULL; return (0); } switch ((*op)->attr_id) { case ISCSI_KEY: ec = reg_get_iscsi(p, &pgt[0], op, op_len); pgt[1].tag = 0; pgt[2].tag = 0; break; case PORTAL_KEY1: ec = reg_get_portal(p, &pgt[0], &pgt[1], op, op_len); pgt[2].tag = 0; break; case PG_KEY1: if (pgt[2].tag == PG_PGT) { /* pg iscsi storage node name is */ /* followed to a pgt */ ec = reg_get_pg1(p, pgt, op, op_len); } else { /* a complete pg object */ ec = reg_get_pg(p, op, op_len); pgt[0].tag = 0; pgt[1].tag = 0; pgt[2].tag = 0; } break; case PG_KEY2: /* pg portal ip addr is followed to a pgt */ ec = reg_get_pg2(p, pgt, op, op_len); break; case PG_PGT: switch (pgt[0].tag) { case 0: /* portal group tag does not follow */ /* iscsi storage node or portal object */ *p = NULL; ec = ISNS_RSP_MSG_FORMAT_ERROR; break; case PG_KEY1: case PG_KEY2: pgt[2].tag = PG_PGT; pgt[2].len = (*op)->attr_len; pg_tag = 0; switch ((*op)->attr_len) { case 4: pg_tag = ntohl(*(uint32_t *) &(*op)->attr_value[0]); case 0: pgt[2].value.ui = pg_tag; break; default: *p = NULL; ec = ISNS_RSP_MSG_FORMAT_ERROR; break; } if (ec == 0) { derefd = 1; NEXT_TLV(*op, *op_len); ec = reg_get_obj(p, pgt, op, op_len); } break; default: /* should never happen */ ASSERT(0); *p = NULL; ec = ISNS_RSP_INTERNAL_ERROR; break; } break; default: *p = NULL; ec = ISNS_RSP_MSG_FORMAT_ERROR; break; } if (ec == 0 && derefd == 0) { ec = update_deref_obj(*p); } if (ec != 0 && *p != NULL) { free_one_object(*p); *p = NULL; } return (ec); } /* * **************************************************************************** * * reg_auth_src: * Authorize the source attribute the DevAttrReg message. * The update can only performed by the node who has the owenership. * * p - the pointer of the object for returning. * pgt - an attribute array with size 3, the elements are: * 0: the first pg key attribute, it is either the name of an * iscsi storage node object or the ip addr of a portal object. * 1: the second pg key attribute, i.e. the portal port. * 2: the portal group tag attribute. * op - the operating attributes. * op_len - the length of the operating attributes. * return - error code. * * **************************************************************************** */ int reg_auth_src( isns_type_t type, uint32_t uid, uchar_t *src ) { lookup_ctrl_t lc; uint32_t puid; puid = is_parent_there(src); if (TYPE_OF_PARENT[type] != 0) { SET_UID_LCP(&lc, type, uid); uid = cache_lookup(&lc, NULL, cb_get_parent); type = TYPE_OF_PARENT[type]; } if (uid != 0 && puid == 0) { SET_UID_LCP(&lc, type, uid); uid = cache_lookup(&lc, NULL, cb_node_child); } if (puid != uid) { return (0); } return (1); } /* * **************************************************************************** * * is_obj_online: * determine if the object is currently registered with the server. * * obj - the object being checked. * return - 0: not registered, otherwise registered. * * **************************************************************************** */ int is_obj_online( const isns_obj_t *obj ) { int online = 1; switch (obj->type) { case OBJ_ISCSI: online = obj->attrs[ATTR_INDEX_ISCSI( ISNS_ISCSI_NODE_TYPE_ATTR_ID)].value.ui == 0 ? 0 : 1; break; default: break; } return (online); } static int set_obj_offline( isns_obj_t *obj ) { switch (obj->type) { case OBJ_ISCSI: obj->attrs[ATTR_INDEX_ISCSI( ISNS_ISCSI_NODE_TYPE_ATTR_ID)].value.ui = 0; break; default: break; } return (0); } /* * **************************************************************************** * * assoc_clone: * clone the association object. * * p - the object being cloned. * clone_flag - 0: the object is being removed; * 1: only the association is being removed. * return - the clone object. * * **************************************************************************** */ void * assoc_clone( void *p, int clone_flag ) { isns_type_t type; isns_obj_t *clone; const isns_attr_t *src_attr; isns_attr_t *dst_attr; uint32_t id, op; int i = 0; const isns_obj_t *obj; uint32_t dd_flag; int online; int state; obj = (isns_obj_t *)p; if (obj->type != OBJ_ISCSI) { return (NULL); } dd_flag = (get_dd_id(get_obj_uid(obj), ISNS_DEFAULT_DD_ID) == 0) ? 0 : 1; online = is_obj_online(obj); state = (clone_flag << 2) | (dd_flag << 1) | online; /* clone_flag dd_flag online action */ /* 0 0 0 ASSERT(0) */ /* 0 0 1 NULL */ /* 0 1 0 itself */ /* 0 1 1 clone it */ /* 1 0 0 NULL */ /* 1 0 1 itself */ /* 1 1 0 itself */ /* 1 1 1 itself */ switch (state) { case 0: ASSERT(0); case 1: case 4: return (NULL); case 2: case 5: case 6: case 7: return (p); case 3: default: break; } type = obj->type; clone = obj_calloc(type); if (clone != NULL) { id = UID_ATTR_INDEX[type]; src_attr = &(obj->attrs[id]); dst_attr = &(clone->attrs[id]); if (assign_attr(dst_attr, src_attr) != 0) { free_one_object(clone); return (NULL); } while (i < MAX_KEY_ATTRS) { op = KEY_ATTR_OP[type][i]; if (op != 0) { id = KEY_ATTR_INDEX[type][i]; src_attr = &(obj->attrs[id]); dst_attr = &(clone->attrs[id]); if (assign_attr(dst_attr, src_attr) != 0) { free_one_object(clone); return (NULL); } } else { break; } i ++; } } return ((void *)clone); } /* * **************************************************************************** * * free_one_object: * free up one object. * * obj - the object being freed. * * **************************************************************************** */ void free_one_object( isns_obj_t *obj ) { int i; uint32_t *cuid; if (obj == NULL) { return; } for (i = 0; i < NUM_OF_ATTRS[obj->type]; i++) { isns_attr_t *attr = &obj->attrs[i]; switch (attr->tag) { case ISNS_EID_ATTR_ID: case ISNS_ISCSI_NAME_ATTR_ID: case ISNS_ISCSI_ALIAS_ATTR_ID: case ISNS_ISCSI_AUTH_METHOD_ATTR_ID: case ISNS_PG_ISCSI_NAME_ATTR_ID: case ISNS_PORTAL_IP_ADDR_ATTR_ID: case ISNS_PORTAL_NAME_ATTR_ID: case ISNS_PG_PORTAL_IP_ADDR_ATTR_ID: case ISNS_DD_SET_NAME_ATTR_ID: case ISNS_DD_NAME_ATTR_ID: case ISNS_DD_ISCSI_NAME_ATTR_ID: case ISNS_DD_FC_PORT_NAME_ATTR_ID: case ISNS_DD_PORTAL_IP_ADDR_ATTR_ID: #ifdef DEBUG if (verbose_mc) { printf("memory(%d) deallocated\n", attr->len); } #endif free(attr->value.ptr); attr->value.ptr = NULL; break; default: break; } } /* free child uids */ i = 0; while (i < NUM_OF_CHILD[obj->type]) { cuid = get_child_n(obj, i); free(cuid); i ++; } /* at last, free the object itself */ #ifdef DEBUG if (verbose_mc) { printf("object(%d) deallocated\n", obj->type); } #endif free(obj); } /* * **************************************************************************** * * free_object: * free up one object. * * obj - the object being freed. * * **************************************************************************** */ void free_object( isns_obj_t *obj ) { free_one_object(obj); } /* * **************************************************************************** * * set_parent_obj: * set the parent object UID. * * obj - the child object. * puid- the parent object UID. * return - error code. * * **************************************************************************** */ int set_parent_obj( isns_obj_t *obj, uint32_t puid ) { uint32_t *const p = get_parent_p(obj); if (p != NULL) { *p = puid; } return (0); } /* * **************************************************************************** * * buff_child_obj: * add a child object UID to the child object array. * * obj - the parent object. * child_type - the type of the child object. * number - the number of the child object. * return - the length of the child object UID array. * * **************************************************************************** */ int buff_child_obj( const isns_type_t ptype, const isns_type_t ctype, const void *c, void const ***child ) { int ec = 0; int i = 0; void const ***pp, **p; uint32_t num, new_num; pp = NULL; /* get the pointer of the array which the child belongs to */ while (i < NUM_OF_CHILD[ptype]) { if (TYPE_OF_CHILD[ptype][i] == ctype) { pp = &child[i]; break; } i ++; } /* the child type is not applicable */ if (pp == NULL) { return (ec); } p = *pp; /* get an empty slot from the uid array for this child */ if (p != NULL) { num = (uint32_t)*p; i = 0; while (i < num) { if (p[++i] == NULL) { /* found it */ p[i] = c; return (ec); } } p = *pp; new_num = num + 1; } else { num = 0; new_num = 1; } /* the array is full, enlarge the child uid array */ p = (void const **)realloc(p, (new_num + 1) * sizeof (void *)); if (p != NULL) { *pp = p; *p = (void *)new_num; p[new_num] = c; } else { ec = ISNS_RSP_INTERNAL_ERROR; } return (ec); } /* * **************************************************************************** * * update_child_object: * update the child object of a network entity object. * * puid - the UID of the parent object, i.e. the network entity object. * child_type - the type of the child object. * child_uid - the uid of the child object. * return - error code. * * **************************************************************************** */ int update_child_obj( const isns_type_t ptype, const uint32_t puid, void const ***child, int child_flag ) { int ec = 0; lookup_ctrl_t lc; SET_UID_LCP(&lc, ptype, puid); lc.data[1].ptr = (uchar_t *)child; lc.data[2].ui = child_flag; ec = cache_lookup(&lc, NULL, cb_add_child); return (ec); } int update_ref_obj( const isns_obj_t *obj ) { uint32_t uid; lookup_ctrl_t lc; isns_type_t t; t = obj->type; if (TYPE_OF_REF[t][0] != 0) { (void) setup_ref_lcp(&lc, obj, NULL); lc.id[2] = t; lc.data[2].ui = get_obj_uid(obj); uid = 0; do { lc.curr_uid = uid; (void) cache_lookup(&lc, &uid, cb_set_ref); } while (uid != 0); } return (0); } /* * **************************************************************************** * * verify_ref_obj: * update the reference bit of a portal group object. * * obj - the object being ref'ed. * return - error code. * * **************************************************************************** */ int verify_ref_obj( const isns_type_t ptype, const uint32_t puid, void const ***child ) { int ec = 0; lookup_ctrl_t lc; SET_UID_LCP(&lc, ptype, puid); lc.data[1].ptr = (uchar_t *)child; ec = cache_lookup(&lc, NULL, cb_verify_ref); return (ec); } int update_deref_obj( isns_obj_t *obj ) { int ec = 0; isns_type_t t, rt; lookup_ctrl_t lc; int i, ref_count; uint32_t uid, *refp; t = obj->type; i = ref_count = 0; while (i < NUM_OF_REF[t]) { rt = TYPE_OF_REF[t][i + 1]; (void) setup_deref_lcp(&lc, obj, rt); uid = is_obj_there(&lc); if (uid != 0) { refp = get_ref_p(obj, lc.type); *refp = uid; ref_count ++; } i ++; } if (i > 0 && ref_count == 0) { ec = ISNS_RSP_INVALID_REGIS; } return (ec); } /* * **************************************************************************** * * register_object: * add one object to the object container. * * obj - the object being added. * uid_p- the pointer for returning object UID. * update_p- the pointer for returning flag which indicates if the object * is newly registered or updated with an existing one. * return - error code. * * **************************************************************************** */ int register_object( isns_obj_t *obj, uint32_t *uid_p, int *update_p ) { return (cache_add(obj, 0, uid_p, update_p)); } /* * **************************************************************************** * * register_assoc: * add one association object to the object container, the association * object has only the information for discovery domain membership, i.e. * a name and UID only. * * obj - the association object being added. * uid_p- the pointer for returning object UID. * return - error code. * * **************************************************************************** */ int register_assoc( isns_obj_t *obj, uint32_t *uid_p ) { return (cache_add(obj, 1, uid_p, NULL)); } /* * **************************************************************************** * * is_obj_there: * check if the object is registered or not. * * lcp - the lookup control data. * return - the object UID. * * **************************************************************************** */ uint32_t is_obj_there( lookup_ctrl_t *lcp ) { uint32_t uid; (void) cache_lookup(lcp, &uid, NULL); return (uid); } uint32_t is_parent_there( uchar_t *src ) { lookup_ctrl_t lc; lc.curr_uid = 0; lc.type = OBJ_ISCSI; lc.id[0] = ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID); lc.op[0] = OP_STRING; lc.data[0].ptr = src; lc.op[1] = 0; return (cache_lookup(&lc, NULL, cb_get_parent)); } /* * **************************************************************************** * * setup_ref_lcp: * prepare the lookup control data for looking up a portal group * object which references to a iscsi stroage node and/or a portal * object. * * lcp - the lookup control data. * iscsi- the ref'ed iscsi storage node object. * portal- the ref'ed portal object. * return - error code. * * **************************************************************************** */ static int setup_ref_lcp( lookup_ctrl_t *lcp, const isns_obj_t *iscsi, const isns_obj_t *portal ) { int i = 0, j = 0; lcp->curr_uid = 0; lcp->type = TYPE_OF_REF[iscsi->type][0]; /* extrace the matching attributes from iscsi storage node object */ while (iscsi != NULL && i < MAX_REF_MATCH && REF_MATCH_OPS[iscsi->type][i] > 0) { lcp->id[i] = REF_MATCH_ID2[iscsi->type][i]; lcp->op[i] = REF_MATCH_OPS[iscsi->type][i]; lcp->data[i].ptr = iscsi->attrs[ REF_MATCH_ID1[iscsi->type][i]].value.ptr; i ++; } /* extrace the matching attributes from portal object */ while (portal != NULL && i < MAX_LOOKUP_CTRL && j < MAX_REF_MATCH && REF_MATCH_OPS[portal->type][j] > 0) { lcp->id[i] = REF_MATCH_ID2[portal->type][j]; lcp->op[i] = REF_MATCH_OPS[portal->type][j]; lcp->data[i].ptr = portal->attrs[ REF_MATCH_ID1[portal->type][j]].value.ptr; j ++; i ++; } if (i < MAX_LOOKUP_CTRL) { lcp->op[i] = 0; } return (0); } static int setup_deref_lcp( lookup_ctrl_t *lcp, const isns_obj_t *pg, isns_type_t t ) { int i = 0; lcp->curr_uid = 0; lcp->type = t; /* extrace the matching attributes from iscsi storage node object */ while (i < MAX_REF_MATCH && REF_MATCH_OPS[t][i] > 0) { lcp->id[i] = REF_MATCH_ID1[t][i]; lcp->op[i] = REF_MATCH_OPS[t][i]; lcp->data[i].ptr = pg->attrs[ REF_MATCH_ID2[t][i]].value.ptr; i ++; } if (i < MAX_LOOKUP_CTRL) { lcp->op[i] = 0; } return (0); } /* * **************************************************************************** * * setup_parent_lcp: * prepare the lookup control data for looking up parent object * with a child object. * * lcp - the lookup control data. * obj - the child object. * return - parent object UID. * * **************************************************************************** */ static uint32_t setup_parent_lcp( lookup_ctrl_t *lcp, isns_obj_t *obj ) { isns_type_t ptype; uint32_t puid; puid = get_parent_uid(obj); if (puid != 0) { ptype = TYPE_OF_PARENT[obj->type]; SET_UID_LCP(lcp, ptype, puid); lcp->data[1].ui = obj->type; lcp->data[2].ui = get_obj_uid(obj); } return (puid); } static int cb_get_parent( void *p1, /* LINTED E_FUNC_ARG_UNUSED */ void *p2 ) { return (get_parent_uid(p1)); } static int cb_node_child( void *p1, /* LINTED E_FUNC_ARG_UNUSED */ void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; uint32_t num, uid; uint32_t *cuid = get_child_t(obj, OBJ_ISCSI); if (cuid != NULL) { num = *cuid; } else { num = 0; } while (num > 0) { uid = *++cuid; if (uid != 0) { return (uid); } num --; } return (0); } /* * **************************************************************************** * * cb_set_ref: * callback function which sets the reference bit to 1 according to * the type of object. * * p1 - the object. * p2 - the lcp. * return - error code. * * **************************************************************************** */ static int cb_set_ref( void *p1, void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; isns_type_t t; uint32_t u; uint32_t *refp; t = lcp->id[2]; u = lcp->data[2].ui; refp = get_ref_p(obj, t); *refp = u; /* successful */ return (0); } /* * **************************************************************************** * * cb_clear_ref: * callback function which clears the reference bit according to * the type of object. * * p1 - the object. * p2 - the lcp. * return - 1: the object is no longer ref'ed, 0: otherwise. * * **************************************************************************** */ static int cb_clear_ref( void *p1, void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; isns_type_t t; uint32_t *refp; int i = 0; uint32_t ref; t = lcp->data[2].ui; refp = get_ref_p(obj, t); *refp = 0; while (i < NUM_OF_REF[obj->type]) { ref = get_ref_n(obj, i); if (ref != 0) { return (0); } i ++; } return (1); } static int cb_add_child( void *p1, void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; const void ***child; const void **vpp; uint32_t vnum; int child_flag; uint32_t **upp, *up; uint32_t num; isns_obj_t *o; int i = 0; child = (const void ***)lcp->data[1].ptr; child_flag = lcp->data[2].ui; while (i < NUM_OF_CHILD[obj->type]) { vpp = child[i]; if (vpp != NULL && (vnum = (uint32_t)*vpp) > 0 && *(vpp + 1) != NULL) { upp = get_child_np(obj, i); if (*upp == NULL) { if (child_flag == 0 && sizeof (typeof (**upp)) == sizeof (typeof (**child))) { *upp = (uint32_t *)vpp; vpp = NULL; child[i] = NULL; } num = vnum; } else { num = **upp + vnum; } if (vpp != NULL) { /* copy required */ up = (uint32_t *)realloc(*upp, (num + 1) * sizeof (uint32_t)); if (up == NULL) { return (ISNS_RSP_INTERNAL_ERROR); } *upp = up; *up = num; up += num; vpp += vnum; while (vnum > 0) { if (*vpp == NULL) { *up = 0; } else if (child_flag == 0) { *up = (uint32_t)*vpp; *vpp = NULL; } else { o = (isns_obj_t *)*vpp; *up = get_obj_uid(o); if (is_obj_online(o) == 0) { free_object(o); } *vpp = NULL; } up --; vpp --; vnum --; } } } i ++; } return (0); } /* * **************************************************************************** * * cb_remove_child: * callback function which removes a child object UID from the * children objet UID array of the parent object. * * p1 - the object. * p2 - the lcp. * return - 1: no more such type of child object, 0: otherwise. * * **************************************************************************** */ static int cb_remove_child( void *p1, void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; uint32_t child_type = lcp->data[1].ui; uint32_t child_uid = lcp->data[2].ui; uint32_t *cuidp, cuid, num_of_child = 0; int i; /* get the children object UID array */ cuidp = get_child_t(obj, child_type); if (cuidp != NULL) { num_of_child = *cuidp; } /* remove it */ while (num_of_child > 0) { cuid = *++cuidp; if (cuid == child_uid) { *cuidp = 0; break; } num_of_child --; } /* check if all of child object UIDs are removed */ i = 0; while (i < NUM_OF_CHILD[obj->type]) { cuidp = get_child_n(obj, i); if (cuidp != NULL) { num_of_child = *cuidp; while (num_of_child > 0) { cuid = *++cuidp; if (cuid != 0) { return (0); } num_of_child --; } } i ++; } return (1); } static int cb_verify_ref( void *p1, void *p2 ) { int ec = 0; isns_obj_t *parent = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; const void ***child; const void **vpp; const void *vp; uint32_t vnum; const void **evpp; const void *evp; uint32_t evnum; isns_type_t pt; /* parent object type */ isns_type_t ct; /* child object type */ isns_type_t rt; /* ref object type */ isns_type_t et; /* peer object type */ uint32_t *up; uint32_t u; uint32_t unum; lookup_ctrl_t lc; uint8_t flag[MAX_OBJ_TYPE + 1] = { 0 }; int i, j, k; pt = parent->type; child = (const void ***)lcp->data[1].ptr; for (i = 0; i < NUM_OF_CHILD[pt]; i++) { ct = TYPE_OF_CHILD[pt][i]; rt = TYPE_OF_REF[ct][0]; if (rt == 0) { continue; } et = TYPE_OF_REF[ct][1]; vpp = child[i]; if (vpp != NULL) { vnum = (uint32_t)*vpp; up = get_child_t(parent, et); if (up != NULL) { unum = *up; } else { unum = 0; } } else { vnum = 0; } j = vnum; while (j > 0) { vp = vpp[j]; if (vp != NULL) { (void) setup_ref_lcp(&lc, vp, NULL); k = unum; while (k > 0) { u = up[k]; if (u != 0) { ec = ref_new2old( &lc, et, u, vp); if (ec != 0) { return (ec); } } k --; } /* End of while each unum */ } j --; } /* End of while each vnum */ if (flag[ct] != 0) { continue; } evnum = 0; j = 0; while (j < NUM_OF_CHILD[pt]) { if (TYPE_OF_CHILD[pt][j] == et) { evpp = child[j]; if (evpp != NULL) { evnum = (uint32_t)*evpp; } break; } j ++; } j = vnum; while (j > 0) { vp = vpp[j]; k = evnum; while (k > 0) { evp = evpp[k]; if (vp != NULL && evp != NULL) { (void) setup_ref_lcp(&lc, vp, evp); ec = ref_new2new(&lc, vp, evp); if (ec != 0) { return (ec); } } k --; } j --; } /* End of while each vnum */ flag[et] = 1; } /* End of for each type of child */ return (ec); } static int cb_ref_new2old( void *p1, void *p2 ) { isns_obj_t *obj = (isns_obj_t *)p1; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; isns_type_t et; uint32_t uu; uint32_t ref; int match; et = lcp->id[2]; uu = lcp->data[2].ui; ref = get_ref_t(obj, et); if (ref == uu) { match = 1; } else { match = 0; } return (match); } static int cb_new_ref( void *p1, void *p2 ) { int ec = 0; lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2; isns_obj_t *a = (isns_obj_t *)p1; isns_obj_t *b = (isns_obj_t *)lcp->data[2].ptr; ec = new_ref(a, b); return (ec); } static int ref_new2old( lookup_ctrl_t *lcp, isns_type_t et, uint32_t uu, const isns_obj_t *vp ) { int ec = 0; int match; uint32_t uid; lookup_ctrl_t lc; lcp->id[2] = et; lcp->data[2].ui = uu; uid = 0; do { lcp->curr_uid = uid; match = cache_lookup(lcp, &uid, cb_ref_new2old); } while (match == 0 && uid != 0); if (match == 0) { /* no such ref, create a default one */ SET_UID_LCP(&lc, et, uu); lc.data[2].ptr = (uchar_t *)vp; ec = cache_lookup(&lc, NULL, cb_new_ref); } return (ec); } static int ref_new2new( lookup_ctrl_t *lcp, const isns_obj_t *p1, const isns_obj_t *p2 ) { int ec = 0; if (is_obj_there(lcp) != 0) { return (0); } ec = new_ref(p1, p2); return (ec); } static int new_ref( const isns_obj_t *p1, const isns_obj_t *p2 ) { int ec = 0; isns_obj_t *obj; obj = make_ref[p1->type](p1, p2); if (obj != NULL) { ec = register_object(obj, NULL, NULL); } else { ec = ISNS_RSP_INTERNAL_ERROR; } return (ec); } /* * **************************************************************************** * * do_dereg: * Physically remove an object along with the children objects, * the reference object and the parent object recursively. * Apporiate SCN is triggered. * * lcp - the lookup control for the object being removed. * parent_flag - 1: the object being removed is the parent object; * 0: otherwise. * child_flag - 1: the object being removed is a child object; * 0: otherwise. * pending - 1: do not remove the ESI entry immediately; * 0: remove the ESI entry without any delay. * return - error code. * * **************************************************************************** */ static int do_dereg( lookup_ctrl_t *lcp, int parent_flag, int child_flag, int pending ) { int ec = 0; isns_obj_t *obj; uint32_t *cuidp, num; isns_type_t type; uint32_t uid; int i; /* remove the object from object container */ obj = cache_remove(lcp, 0); if (obj == NULL) { return (0); } /* trigger a scn */ if (scn_q != NULL) { (void) make_scn(ISNS_OBJECT_REMOVED, obj); } /* dereg children */ i = 0; while (ec == 0 && !parent_flag && i < NUM_OF_CHILD[obj->type]) { type = TYPE_OF_CHILD[obj->type][i]; cuidp = get_child_n(obj, i); if (cuidp != NULL) { num = *cuidp; } else { num = 0; } while (ec == 0 && num > 0) { uid = cuidp[num]; if (uid != 0) { SET_UID_LCP(lcp, type, uid); ec = do_dereg(lcp, parent_flag, 1, pending); } num --; } i ++; } /* clear the ref bit on the ref'd object */ if (ec == 0 && TYPE_OF_REF[obj->type][0] > 0) { uid = 0; do { (void) setup_ref_lcp(lcp, obj, NULL); lcp->curr_uid = uid; lcp->data[2].ui = obj->type; if (cache_lookup(lcp, &uid, cb_clear_ref) != 0) { UPDATE_LCP_UID(lcp, uid); ec = do_dereg(lcp, parent_flag, child_flag, pending); } } while (uid != 0); } /* remove it from the parent */ if (ec == 0 && !child_flag && TYPE_OF_PARENT[obj->type] > 0 && (uid = setup_parent_lcp(lcp, obj)) != 0) { if (cache_lookup(lcp, NULL, cb_remove_child) != 0) { UPDATE_LCP_UID(lcp, uid); ec = do_dereg(lcp, 1, child_flag, 0); } } if (ec == 0 && !child_flag) { /* remove it from persistent data store */ if (sys_q) { ec = write_data(DATA_DELETE, obj); } /* remove esi event entry */ if (ec == 0) { (void) esi_remove_obj(obj, pending); } /* save the parent uid for caller */ if (TYPE_OF_PARENT[obj->type] != 0) { lcp->curr_uid = get_parent_uid(obj); } else { /* it's the parent itself */ lcp->curr_uid = get_obj_uid(obj); } } /* remove this portal from scn registry */ if (ec == 0 && obj->type == OBJ_PORTAL) { (void) remove_scn_portal(get_obj_uid(obj)); } /* free the object */ (void) free_object(obj); return (ec); } /* * **************************************************************************** * * dereg_assoc: * Remove one association object from object container. * * lcp - the lookup control for the object being removed. * return - error code. * * **************************************************************************** */ int dereg_assoc( lookup_ctrl_t *lcp ) { isns_obj_t *obj; obj = cache_remove(lcp, 1); /* free the object */ if (obj != NULL) { free_object(obj); } return (0); } /* * **************************************************************************** * * dereg_object: * Remove one object from object container. * * lcp - the lookup control for the object being removed. * return - error code. * * **************************************************************************** */ int dereg_object( lookup_ctrl_t *lcp, int pending ) { return (do_dereg(lcp, 0, 0, pending)); } /* * **************************************************************************** * * data_sync: * Synchronize the cache with persistent data store. * Flush the cache data to data store if the input ec is zero, * retreat the changes in cache and ignore data store update * if there is an error. * * ec - error code. * return - error code. * * **************************************************************************** */ int data_sync( int ec ) { /* cache is updated successfully, commit the data to data store */ if (IS_CACHE_UPDATED()) { if (ec == 0) { ec = write_data(DATA_COMMIT, NULL); } if (ec == 0) { /* successful, trigger the SCN */ (void) queue_msg_set(scn_q, SCN_TRIGGER, (void *)NULL); } else { shutdown_server(); } } else { /* ignore all SCNs which have been generated */ (void) queue_msg_set(scn_q, SCN_IGNORE, (void *)NULL); (void) write_data(DATA_RETREAT, NULL); } return (ec); } static pthread_mutex_t name_mtx[3] = { PTHREAD_MUTEX_INITIALIZER, PTHREAD_MUTEX_INITIALIZER, PTHREAD_MUTEX_INITIALIZER }; static const char *name_pattern[3] = { "ENTITY_ID_%d", "DD_%d", "DD-Set_%d" }; static uint32_t name_count[3] = { 0, 0, 0 }; /* * **************************************************************************** * * make_unique_name: * make a default unique name for a newly registered network entity, * discovery domain or discovery domain set object. * * len - pointer of the length of the new name for returning. * tag - which attribute of the new name is for. * return - the name being made. * * **************************************************************************** */ static char * make_unique_name( int *len, uint32_t tag ) { int i; int count; char name[32] = { 0 }; char *p; lookup_ctrl_t lc; lc.curr_uid = 0; switch (tag) { case ISNS_EID_ATTR_ID: i = 0; lc.type = OBJ_ENTITY; lc.id[0] = ATTR_INDEX_ENTITY(ISNS_EID_ATTR_ID); break; case ISNS_DD_NAME_ATTR_ID: i = 1; lc.type = OBJ_DD; lc.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID); break; case ISNS_DD_SET_NAME_ATTR_ID: i = 2; lc.type = OBJ_DDS; lc.id[0] = ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID); break; default: ASSERT(0); break; } lc.op[0] = OP_STRING; lc.op[1] = 0; do { (void) pthread_mutex_lock(&name_mtx[i]); count = ++ name_count[i]; (void) pthread_mutex_unlock(&name_mtx[i]); /* no more space, failure */ if (count == 0) { return (NULL); } (void) sprintf(name, name_pattern[i], count); lc.data[0].ptr = (uchar_t *)name; } while (is_obj_there(&lc) != 0); /* 4-bytes aligned length */ *len = strlen(name); *len = *len + (4 - *len % 4); p = (char *)malloc(*len); if (p != NULL) { (void) strcpy(p, name); } return (p); } #ifdef DEBUG void obj_dump( void *p ) { print_object(NULL, (isns_obj_t *)p); } #endif