root/usr/src/cmd/isns/isnsd/admintf.c
/*
 * 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 <libxml/xmlreader.h>
#include <libxml/xmlwriter.h>
#include <libxml/tree.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <pthread.h>

#include "isns_server.h"
#include "isns_cfg.h"
#include "isns_htab.h"
#include "isns_cache.h"
#include "isns_obj.h"
#include "isns_dd.h"
#include "isns_utils.h"
#include "isns_mgmt.h"
#include "isns_protocol.h"
#include "admintf.h"

extern const int UID_ATTR_INDEX[MAX_OBJ_TYPE_FOR_SIZE];

static isns_type_t
get_lc_type(
        object_type obj
)
{
        isns_type_t type;

        switch (obj) {
        case Node:
                type = OBJ_ISCSI;
                break;
        case DiscoveryDomain:
        case DiscoveryDomainMember:
                type = OBJ_DD;
                break;
        case DiscoveryDomainSet:
        case DiscoveryDomainSetMember:
                type = OBJ_DDS;
                break;
        default:
                ASSERT(0);
                break;
        }

        return (type);
}

static uint32_t
get_lc_id(
        object_type obj
)
{
        uint32_t id;

        switch (obj) {
        case Node:
                id = ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID);
                break;
        case DiscoveryDomain:
        case DiscoveryDomainMember:
                id = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
                break;
        case DiscoveryDomainSet:
        case DiscoveryDomainSetMember:
                id = ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID);
                break;
        default:
                ASSERT(0);
                break;
        }

        return (id);
}

/*
 * ****************************************************************************
 *
 * cb_get_node_info: callback for get_node_op
 *      The routine process matching node and add a Node object elements
 *      to the response doc.
 *
 * p1   - matching node object
 * p2   - lookup control data that was used for node look up
 *          returns parent index(newtork entity) in look up control.
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_node_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, sub_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)NODEOBJECT);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NODE_TYPE_ATTR_ID)];
        switch (attr->value.ui) {
            case ISNS_CONTROL_NODE_TYPE | ISNS_INITIATOR_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODEINITIATORTYPE);
                break;
            case ISNS_CONTROL_NODE_TYPE | ISNS_TARGET_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODETARGETTYPE);
                break;
            case ISNS_TARGET_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)TARGETTYPE);
                break;
            case ISNS_INITIATOR_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)INITIATORTYPE);
                break;
            case ISNS_CONTROL_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODETYPE);
                break;
            default:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)UNKNOWNTYPE);
        }
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_ALIAS_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)ALIASATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /*
         * A node can have all or no SCN subsribtion.
         * May avoid redundant code with scsusrciption table.
         */
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_SCN_BITMAP_ATTR_ID)];
        if (IS_SCN_INIT_SELF_INFO_ONLY(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNINITSELFONLY);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_TARGET_SELF_INFO_ONLY(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNTARGETSELFONLY);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_MGMT_REG(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNTARGETSELFONLY);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_OBJ_REMOVED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNOBJECTREMOVED);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_OBJ_ADDED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNOBJECTADDED);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_OBJ_UPDATED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNOBJECTUPDATED);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_MEMBER_REMOVED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNMEMBERREMOVED);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }
        if (IS_SCN_MEMBER_ADDED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)SCNSUBSCRIPTION,
                (xmlChar *)SCNMEMBERADDED);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }

        /* set the parent object id, i.e. the network entity object id */
        lcp->id[2] = get_parent_uid(obj);
        /* pass back the node object element to add entity, portal info to it */
        lcp->data[2].ptr =  (uchar_t *)n_node;

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_get_entity_info: callback for get_node_op
 *      The routine process matching network entity and add children elements
 *      to a Node object for given entity.
 *
 * p1   - matching entity object
 * p2   - lookup control data that was used for node look up
 *          returns parent index(newtork entity) in look up control.
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_entity_info(
        void *p1,
        void *p2
)
{
        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlNodePtr n_node = (xmlNodePtr)lcp->data[2].ptr;
        xmlNodePtr sub_node, subchild_node, subgrandchild_node;
        char numbuf[32];
        char buff[INET6_ADDRSTRLEN + 1] = { 0 };
        isns_attr_t *attr;

        sub_node = xmlNewChild(n_node, NULL, (xmlChar *)NETWORKENTITY, NULL);

        if (sub_node) {
            attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_EID_ATTR_ID)];
            subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)ENTITYID, (xmlChar *)attr->value.ptr);
            if (subchild_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_ENTITY_PROTOCOL_ATTR_ID)];
            (void) sprintf(numbuf, "%u", attr->value.ui);
            subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)ENTITYPROTOCOL, (xmlChar *)numbuf);
            if (subchild_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_MGMT_IP_ADDR_ATTR_ID)];
            if (attr->value.ip) {
                /* convert the ipv6 to ipv4 */
                if (((int *)attr->value.ip)[0] == 0x00 &&
                    ((int *)attr->value.ip)[1] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[8] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[9] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[10] == 0xFF &&
                    ((uchar_t *)attr->value.ip)[11] == 0xFF) {
                    subchild_node = xmlNewChild(sub_node, NULL,
                        (xmlChar *)MANAGEMENTIPADDR,
                        (xmlChar *)inet_ntop(AF_INET,
                        (void *)&(((uint32_t *)attr->value.ip)[3]),
                        buff, sizeof (buff)));
                } else {
                    subchild_node = xmlNewChild(sub_node, NULL,
                        (xmlChar *)MANAGEMENTIPADDR,
                        (xmlChar *)inet_ntop(AF_INET6,
                        (void *)attr->value.ip, buff, sizeof (buff)));
                }
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_TIMESTAMP_ATTR_ID)];
            if (attr->value.ui) {
                (void) sprintf(numbuf, "%u", attr->value.ui);
                subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)ENTITYREGTIMESTAMP, (xmlChar *)numbuf);
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr = &obj->attrs[ATTR_INDEX_ENTITY(ISNS_VERSION_RANGE_ATTR_ID)];
            if (attr->value.ui) {
                subchild_node = xmlNewNode(NULL,
                    (xmlChar *)PROTOCOLVERSIONRANGE);
                subchild_node = xmlAddChild(sub_node, subchild_node);
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }

                (void) sprintf(numbuf, "%u",
                    (attr->value.ui >> ISNS_VER_SHIFT) & ISNS_VERSION);
                subgrandchild_node = xmlNewChild(subchild_node, NULL,
                    (xmlChar *)PROTOCOLMAXVERSION, (xmlChar *)numbuf);
                if (subgrandchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                (void) sprintf(numbuf, "%u", attr->value.ui & ISNS_VERSION);
                subgrandchild_node = xmlNewChild(subchild_node, NULL,
                    (xmlChar *)PROTOCOLMINVERSION, (xmlChar *)numbuf);
                if (subgrandchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr =
                &obj->attrs[ATTR_INDEX_ENTITY(ISNS_ENTITY_REG_PERIOD_ATTR_ID)];
            if (attr->value.ui) {
                (void) sprintf(numbuf, "%u", attr->value.ui);
                subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)REGISTRATIONPERIOD, (xmlChar *)numbuf);
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        } else {
            return (ERR_XML_NEWCHILD_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_get_pg_info: callback for get_node_op
 *      The routine process matching portal group and returns ip address
 *      and port number for further portal processing.
 *
 * p1   - matching portal group object
 * p2   - lookup control data that was used for portal group look up
 *          returns portal ip address, port and group tag in look up control.
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_pg_info(
        void *p1,
        void *p2
)
{
        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;

        isns_attr_t *attr;

        /* get pg portal ip address and port attributes */
        attr = &obj->attrs[ATTR_INDEX_PG(ISNS_PG_PORTAL_IP_ADDR_ATTR_ID)];
        (void) memcpy(lcp->data[1].ip, attr->value.ip, sizeof (in6_addr_t));
        attr = &obj->attrs[ATTR_INDEX_PG(ISNS_PG_PORTAL_PORT_ATTR_ID)];
        lcp->data[2].ui = attr->value.ui;
        attr = &obj->attrs[ATTR_INDEX_PG(ISNS_PG_TAG_ATTR_ID)];
        lcp->id[2] = attr->value.ui;

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_get_portal_info: callback for get_node_op
 *      The routine process matching portal and add portal object info to
 *      the node object.
 *
 * p1   - matching portal object
 * p2   - lookup control data that was used for portal look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_portal_info(
        void *p1,
        void *p2
)
{
        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlNodePtr n_node = (xmlNodePtr)lcp->data[2].ptr;
        uint32_t    tag = lcp->id[2];
        xmlNodePtr sub_node, subchild_node, subgrandchild_node;
        char numbuf[32];
        char buff[INET6_ADDRSTRLEN + 1] = { 0 };
        isns_attr_t *attr;

        sub_node = xmlNewChild(n_node, NULL, (xmlChar *)PORTAL, NULL);

        /* get portal object attributes. */
        if (sub_node) {
            attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_PORTAL_IP_ADDR_ATTR_ID)];
            if (attr->value.ip) {
                /* convert the ipv6 to ipv4 */
                if (((int *)attr->value.ip)[0] == 0x00 &&
                    ((int *)attr->value.ip)[1] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[8] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[9] == 0x00 &&
                    ((uchar_t *)attr->value.ip)[10] == 0xFF &&
                    ((uchar_t *)attr->value.ip)[11] == 0xFF) {
                    subchild_node = xmlNewChild(sub_node, NULL,
                        (xmlChar *)IPADDR,
                        (xmlChar *)inet_ntop(AF_INET,
                        (void *)&(((uint32_t *)attr->value.ip)[3]),
                        buff, sizeof (buff)));
                } else {
                    subchild_node = xmlNewChild(sub_node, NULL,
                        (xmlChar *)IPADDR,
                        (xmlChar *)inet_ntop(AF_INET6,
                        (void *)attr->value.ip, buff, sizeof (buff)));
                }
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            subchild_node = xmlNewChild(sub_node, NULL, (xmlChar *)UDPTCPPORT,
                NULL);
            if (subchild_node) {
                attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_PORTAL_PORT_ATTR_ID)];
                subgrandchild_node = xmlNewChild(subchild_node, NULL,
                    (xmlChar *)PORTTYPE, IS_PORT_UDP(attr->value.ui) ?
                    (xmlChar *)UDPPORT : (xmlChar *)TCPPORT);
                if (subgrandchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                (void) sprintf(numbuf, "%u", PORT_NUMBER(attr->value.ui));
                subgrandchild_node = xmlNewChild(subchild_node, NULL,
                    (xmlChar *)PORTNUMBER, (xmlChar *)numbuf);
                if (subgrandchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            } else {
                return (ERR_XML_NEWCHILD_FAILED);
            }
            (void) sprintf(numbuf, "%u", tag);
            subchild_node = xmlNewChild(sub_node, NULL, (xmlChar *)GROUPTAG,
                (xmlChar *)numbuf);
            if (subchild_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_PORTAL_NAME_ATTR_ID)];
            if (attr->value.ptr) {
                subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)SYMBOLICNAME, (xmlChar *)attr->value.ptr);
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_ESI_INTERVAL_ATTR_ID)];
            if (attr->value.ui) {
                (void) sprintf(numbuf, "%u", attr->value.ui);
                subchild_node = xmlNewChild(sub_node, NULL,
                (xmlChar *)ESIINTERVAL, (xmlChar *)numbuf);
                if (subchild_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_ESI_PORT_ATTR_ID)];
            if (attr->value.ui) {
                subchild_node = xmlNewChild(sub_node, NULL,
                    (xmlChar *)ESIPORT, NULL);
                if (subchild_node) {
                    subgrandchild_node = xmlNewChild(subchild_node, NULL,
                        (xmlChar *)PORTTYPE, IS_PORT_UDP(attr->value.ui) ?
                        (xmlChar *)UDPPORT : (xmlChar *)TCPPORT);
                    if (subgrandchild_node == NULL) {
                        return (ERR_XML_NEWCHILD_FAILED);
                    }
                    (void) sprintf(numbuf, "%u", PORT_NUMBER(attr->value.ui));
                    subgrandchild_node = xmlNewChild(subchild_node, NULL,
                        (xmlChar *)PORTNUMBER, (xmlChar *)numbuf);
                    if (subgrandchild_node == NULL) {
                        return (ERR_XML_NEWCHILD_FAILED);
                    }
                } else {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
            attr = &obj->attrs[ATTR_INDEX_PORTAL(ISNS_SCN_PORT_ATTR_ID)];
            if (attr->value.ui) {
                subchild_node = xmlNewChild(sub_node, NULL,
                    (xmlChar *)SCNPORT, NULL);
                if (subchild_node) {
                    subgrandchild_node = xmlNewChild(subchild_node, NULL,
                        (xmlChar *)PORTTYPE, IS_PORT_UDP(attr->value.ui) ?
                        (xmlChar *)UDPPORT : (xmlChar *)TCPPORT);
                    (void) sprintf(numbuf, "%u", PORT_NUMBER(attr->value.ui));
                    if (subgrandchild_node == NULL) {
                        return (ERR_XML_NEWCHILD_FAILED);
                    }
                    subgrandchild_node = xmlNewChild(subchild_node, NULL,
                        (xmlChar *)PORTNUMBER, (xmlChar *)numbuf);
                    if (subgrandchild_node == NULL) {
                        return (ERR_XML_NEWCHILD_FAILED);
                    }
                } else {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        } else if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_get_dd_info: callback for get_dd_op
 *      The routine process matching dd object
 *
 * p1   - matching dd object
 * p2   - lookup control data that was used for dd look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_dd_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, sub_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;
        char numbuf[32];

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDOBJECT);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_ID_ATTR_ID)];
        (void) sprintf(numbuf, "%u", attr->value.ui);
        n_attr = xmlSetProp(n_node, (xmlChar *)IDATTR,
                (xmlChar *)numbuf);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_FEATURES_ATTR_ID)];
        if (DD_BOOTLIST_ENABLED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)BOOTLISTENABLEDELEM,
                (xmlChar *)XMLTRUE);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        } else {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)BOOTLISTENABLEDELEM,
                (xmlChar *)XMLFALSE);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_get_ddset_info: callback for get_ddset_op
 *      The routine process matching dd object
 *
 * p1   - matching dds object
 * p2   - lookup control data that was used for dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_get_ddset_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, sub_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;
        char numbuf[32];

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_ID_ATTR_ID)];
        (void) sprintf(numbuf, "%u", attr->value.ui);
        n_attr = xmlSetProp(n_node, (xmlChar *)IDATTR,
                (xmlChar *)numbuf);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_STATUS_ATTR_ID)];
        if (DDS_ENABLED(attr->value.ui)) {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)ENABLEDELEM,
                (xmlChar *)XMLTRUE);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        } else {
            sub_node = xmlNewChild(n_node, NULL, (xmlChar *)ENABLEDELEM,
                (xmlChar *)XMLFALSE);
            if (sub_node == NULL) {
                return (ERR_XML_NEWCHILD_FAILED);
            }
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_enumerate_node_info: callback for enumerate_node_op
 *      The routine is invoked for each node object.
 *
 * p1   - node object
 * p2   - lookup control data that was used for node look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_enumerate_node_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)NODEOBJECT);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NODE_TYPE_ATTR_ID)];
        switch (attr->value.ui) {
            case ISNS_CONTROL_NODE_TYPE | ISNS_INITIATOR_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODEINITIATORTYPE);
                break;
            case ISNS_CONTROL_NODE_TYPE | ISNS_TARGET_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODETARGETTYPE);
                break;
            case ISNS_TARGET_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)TARGETTYPE);
                break;
            case ISNS_INITIATOR_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)INITIATORTYPE);
                break;
            case ISNS_CONTROL_NODE_TYPE:
                n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)CONTROLNODETYPE);
                break;
            default:
            n_attr = xmlSetProp(n_node, (xmlChar *)TYPEATTR,
                    (xmlChar *)UNKNOWNTYPE);
        }
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_ALIAS_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)ALIASATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * i_enumerate_dd_dds_info:
 *      The routine is implemnetation for enumerate dd and enumerate dds.
 *
 * p1   - dd or dd set object
 * p2   - lookup control data that was used for dd and dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
i_enumerate_dd_dds_info(
        void *p1,
        void *p2,
        isns_type_t obj_type
)
{
        xmlNodePtr      n_obj, n_node, sub_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;
        char numbuf[32];

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        if (obj_type == OBJ_DD) {
            n_node = xmlNewNode(NULL, (xmlChar *)DDOBJECT);
        } else {
            n_node = xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
        }

        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        if (obj_type == OBJ_DD) {
            /* get name, id, feaure and generate xml info */
            attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID)];
            n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
            if (n_attr == NULL) {
                return (ERR_XML_SETPROP_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_ID_ATTR_ID)];
            (void) sprintf(numbuf, "%u", attr->value.ui);
            n_attr = xmlSetProp(n_node, (xmlChar *)IDATTR,
                (xmlChar *)numbuf);
            if (n_attr == NULL) {
                return (ERR_XML_SETPROP_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_FEATURES_ATTR_ID)];
            if (DD_BOOTLIST_ENABLED(attr->value.ui)) {
                sub_node = xmlNewChild(n_node, NULL,
                    (xmlChar *)BOOTLISTENABLEDELEM, (xmlChar *)XMLTRUE);
                if (sub_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            } else {
                sub_node = xmlNewChild(n_node, NULL,
                    (xmlChar *)BOOTLISTENABLEDELEM, (xmlChar *)XMLFALSE);
                if (sub_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        } else {
            /* get name, id, status and generate xml info */
            attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID)];
            n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                (xmlChar *)attr->value.ptr);
            if (n_attr == NULL) {
                return (ERR_XML_SETPROP_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_ID_ATTR_ID)];
            (void) sprintf(numbuf, "%u", attr->value.ui);
            n_attr = xmlSetProp(n_node, (xmlChar *)IDATTR,
                (xmlChar *)numbuf);
            if (n_attr == NULL) {
                return (ERR_XML_SETPROP_FAILED);
            }
            attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_STATUS_ATTR_ID)];
            if (DDS_ENABLED(attr->value.ui)) {
                sub_node = xmlNewChild(n_node, NULL,
                    (xmlChar *)ENABLEDELEM, (xmlChar *)XMLTRUE);
                if (sub_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            } else {
                sub_node = xmlNewChild(n_node, NULL,
                    (xmlChar *)ENABLEDELEM, (xmlChar *)XMLFALSE);
                if (sub_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_enumerate_dd_info: callback for enumerate_dd_op
 *      The routine is invoked for each dd object.
 *
 * p1   - dd object
 * p2   - lookup control data that was used for dd look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_enumerate_dd_info(
        void *p1,
        void *p2
)
{
        return (i_enumerate_dd_dds_info(p1, p2, OBJ_DD));
}

/*
 * ****************************************************************************
 *
 * cb_enumerate_ddset_info: callback for enumerate_dd_op
 *      The routine is invoked for each dd object.
 *
 * p1   - dd object
 * p2   - lookup control data that was used for dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_enumerate_ddset_info(
        void *p1,
        void *p2
)
{
        return (i_enumerate_dd_dds_info(p1, p2, OBJ_DDS));
}

/*
 * ****************************************************************************
 *
 * cb_getAssociated_node_info:
 *      The routine is implemnetation for enumerate dd and enumerate dds.
 *
 * p1   - dd or dd set object
 * p2   - lookup control data that was used for dd and dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_getAssociated_node_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;
        uchar_t *ddname = lcp->data[2].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDOBJECTMEMBER);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        attr = &obj->attrs[ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)NODENAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                (xmlChar *)ddname);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_getAssociated_node_to_dd_info:
 *      The routine is implemnetation for enumerate dd and enumerate dds.
 *
 * p1   - dd or dd set object
 * p2   - lookup control data that was used for dd and dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_getAssociated_node_to_dd_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;
        uchar_t *nodename = lcp->data[2].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDOBJECTMEMBER);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        n_attr = xmlSetProp(n_node, (xmlChar *)NODENAMEATTR,
                (xmlChar *)nodename);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_getAssociated_dd_info:
 *      The routine is implemnetation for getting dds membership.
 *
 * p1   - dd or dd set object
 * p2   - lookup control data that was used for dd and dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_getAssociated_dd_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;
        uchar_t *ddsetname = lcp->data[2].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDSETOBJECTMEMBER);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        attr = &obj->attrs[ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        n_attr = xmlSetProp(n_node, (xmlChar *)DDSETNAMEATTR,
                (xmlChar *)ddsetname);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * cb_getAssociated_dd_to_ddset_info:
 *      The routine is implemnetation for enumerate dd and enumerate dds.
 *
 * p1   - dd or dd set object
 * p2   - lookup control data that was used for dd and dd set look up
 * return - error code
 *
 * ****************************************************************************
 */
static int
cb_getAssociated_dd_to_ddset_info(
        void *p1,
        void *p2
)
{
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        isns_attr_t *attr;

        isns_obj_t *obj = (isns_obj_t *)p1;
        lookup_ctrl_t *lcp = (lookup_ctrl_t *)p2;
        xmlDocPtr doc = (xmlDocPtr)lcp->data[1].ptr;
        uchar_t *ddname = lcp->data[2].ptr;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        n_node = xmlNewNode(NULL, (xmlChar *)DDSETOBJECTMEMBER);
        if (n_node) {
            n_node = xmlAddChild(n_obj, n_node);
            if (n_node == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        /* get node name, alias, type and generate xml info */
        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                (xmlChar *)ddname);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }
        attr = &obj->attrs[ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID)];
        n_attr = xmlSetProp(n_node, (xmlChar *)DDSETNAMEATTR,
                (xmlChar *)attr->value.ptr);
        if (n_attr == NULL) {
            return (ERR_XML_SETPROP_FAILED);
        }

        /* successful */
        return (0);
}

/*
 * ****************************************************************************
 *
 * handle_partial_success:
 *
 * doc  - response doc to fill up
 * ret  - return code from the caller.
 *
 * ****************************************************************************
 */
static int
handle_partial_success(
        xmlDocPtr doc,
        int ret
)
{
        xmlNodePtr      n_obj, n_node, root;
        char numbuf[32];

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)RESULTELEMENT);
        if (n_obj) {
            if (root->children) {
                n_obj = xmlAddPrevSibling(root->children, n_obj);
                (void) sprintf(numbuf, "%d", (ret != 0) ? PARTIAL_SUCCESS : 0);
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)STATUSELEMENT,
                    (xmlChar *)numbuf);
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)MESSAGEELEMENT,
                    (xmlChar *)result_code_to_str((ret != 0) ?
                        PARTIAL_SUCCESS : 0));
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            } else {
                n_obj = xmlAddChild(root, n_obj);
                if (n_obj == NULL) {
                    return (ERR_XML_ADDCHILD_FAILED);
                }
                (void) sprintf(numbuf, "%d", ret);
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)STATUSELEMENT,
                    (xmlChar *)numbuf);
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)MESSAGEELEMENT,
                    (xmlChar *)result_code_to_str(ret));
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        return (0);
}

/*
 * ****************************************************************************
 *
 * handle_partial_failure:
 *
 * doc  - response doc to fill up
 * ret  - return code from the caller.
 *
 * ****************************************************************************
 */
static int
handle_partial_failure(
        xmlDocPtr doc,
        int ret,
        boolean_t all_failed
)
{
        xmlNodePtr      n_obj, n_node, root;
        char numbuf[32];

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)RESULTELEMENT);
        if (n_obj) {
            if (root->children) {
                /* some or all associations failed to create */
                n_obj = xmlAddPrevSibling(root->children, n_obj);
                /* capture last error. should come up with all failed?? */
                (void) sprintf(numbuf, "%d",
                    all_failed ? ret : PARTIAL_FAILURE);
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)STATUSELEMENT,
                    (xmlChar *)numbuf);
                if (n_node == NULL) {
                        return (ERR_XML_NEWCHILD_FAILED);
                }
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)MESSAGEELEMENT,
                    (xmlChar *)result_code_to_str(all_failed ? ret :
                        PARTIAL_FAILURE));
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            } else {
                n_obj = xmlAddChild(root, n_obj);
                if (n_obj == NULL) {
                    return (ERR_XML_ADDCHILD_FAILED);
                }
                (void) sprintf(numbuf, "%d", (ret != 0) ? ret : 0);
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)STATUSELEMENT,
                    (xmlChar *)numbuf);
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                n_node = xmlNewChild(n_obj, NULL, (xmlChar *)MESSAGEELEMENT,
                    (xmlChar *)result_code_to_str((ret != 0) ? ret : 0));
                if (n_node == NULL) {
                    return (ERR_XML_NEWCHILD_FAILED);
                }
            }
        } else {
            return (ERR_XML_NEWNODE_FAILED);
        }

        return (0);
}

/*
 * ****************************************************************************
 *
 * get_serverconfig_op:
 *      The routine process server administrative setting.
 *
 * doc  - response doc to fill up.
 *
 * ****************************************************************************
 */
int
get_serverconfig_op(
        xmlDocPtr doc
)
{
        extern uint64_t esi_threshold;
        extern uint8_t mgmt_scn;
        extern ctrl_node_t *control_nodes;
        extern pthread_mutex_t ctrl_node_mtx;
        extern char data_store[MAXPATHLEN];

        xmlNodePtr      n_obj, root;
        char numbuf[32];
        ctrl_node_t *ctrl_node_p;
        int ret = 0;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        n_obj = xmlNewNode(NULL, (xmlChar *)ISNSSERVER);
        if (n_obj) {
            n_obj = xmlAddChild(root, n_obj);
            if (n_obj == NULL) {
                return (ERR_XML_ADDCHILD_FAILED);
            }
        } else {
            return (ERR_XML_ADDCHILD_FAILED);
        }

        if (xmlNewChild(n_obj, NULL, (xmlChar *)DATASTORELOCATION,
            (xmlChar *)data_store) == NULL) {
            return (ERR_XML_NEWCHILD_FAILED);
        }

        (void) sprintf(numbuf, "%llu", esi_threshold);
        if (xmlNewChild(n_obj, NULL, (xmlChar *)ESIRETRYTHRESHOLD,
            (xmlChar *)numbuf) == NULL) {
            return (ERR_XML_NEWCHILD_FAILED);
        }
        if (xmlNewChild(n_obj, NULL, (xmlChar *)MANAGEMENTSCNENABLED,
            (mgmt_scn) ? (uchar_t *)XMLTRUE : (uchar_t *)XMLFALSE) == NULL) {
            return (ERR_XML_NEWCHILD_FAILED);
        }

        (void) pthread_mutex_lock(&ctrl_node_mtx);
        if (control_nodes == NULL) {
            if (xmlNewChild(n_obj, NULL, (xmlChar *)CONTROLNODENAME,
                    (xmlChar *)NULL) == NULL) {
                    (void) pthread_mutex_unlock(&ctrl_node_mtx);
                    return (ERR_XML_NEWCHILD_FAILED);
            }
        } else {
            ctrl_node_p = control_nodes;
            while (ctrl_node_p != NULL) {
                if (xmlNewChild(n_obj, NULL, (xmlChar *)CONTROLNODENAME,
                    (xmlChar *)ctrl_node_p->name) == NULL) {
                    (void) pthread_mutex_unlock(&ctrl_node_mtx);
                    return (ERR_XML_NEWCHILD_FAILED);
                }
                ctrl_node_p = ctrl_node_p->next;
            }
        }
        (void) pthread_mutex_unlock(&ctrl_node_mtx);

        return (handle_partial_success(doc, ret));
}

/*
 * ****************************************************************************
 *
 * get_node_op:
 *      service get operation on a given node.
 *
 * req  - contains all info for a request.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
get_node_op(
        request_t *req,
        xmlDocPtr doc
        /* any additional arguments go here */
)
{
        int ret = 0, ret_save = 0;
        int i = 0;
        lookup_ctrl_t lc, lc2, lc3;
        uint32_t uid;
        char buff2[INET6_ADDRSTRLEN];

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = get_lc_type(req->op_info.obj);
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;
        lc.data[1].ptr = (uchar_t *)doc; /* xml writer descriptor */
        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                ret = cache_lookup(&lc, &uid, cb_get_node_info);
                if (uid == 0) {
                    ret = ERR_MATCHING_ISCSI_NODE_NOT_FOUND;
                }

                /* generate network entity object information */
                if (ret == 0 && lc.id[2] != 0) {
                        /*
                         * !!! there might be no entity and portal info for
                         * !!! the node if it is not a registered node
                         */
                        /* prepare lookup ctrl data for looking for entity */
                        SET_UID_LCP(&lc2, OBJ_ENTITY, lc.id[2]);

                        lc2.data[1].ptr = (uchar_t *)doc;
                        /* cb_get_node_info callback returned Node object. */
                        lc2.data[2].ptr = lc.data[2].ptr;
                        ret = cache_lookup(&lc2, &uid, cb_get_entity_info);
                        if (uid == 0) {
                            ret = ERR_MATCHING_NETWORK_ENTITY_NOT_FOUND;
                        }
                }

                /* generate portal information */
                if (ret == 0 && lc.id[2] != 0) {
                        /* prepare lookup ctrl data for looking for pg */
                        lc2.curr_uid = 0;
                        lc2.type = OBJ_PG;
                        lc2.id[0] = ATTR_INDEX_PG(ISNS_PG_ISCSI_NAME_ATTR_ID);
                        lc2.op[0] = OP_STRING;
                        /* lc.data[0].ptr contains node name */
                        lc2.data[0].ptr = lc.data[0].ptr;
                        lc2.op[1] = 0;
                        lc2.data[1].ip = (in6_addr_t *)buff2;

                        /* prepare lookup ctrl data for looking for portal */
                        lc3.curr_uid = 0;
                        lc3.type = OBJ_PORTAL;
                        lc3.id[0] = ATTR_INDEX_PORTAL(
                                ISNS_PORTAL_IP_ADDR_ATTR_ID);
                        lc3.op[0] = OP_MEMORY_IP6;
                        lc3.id[1] = ATTR_INDEX_PORTAL(
                                ISNS_PORTAL_PORT_ATTR_ID);
                        lc3.op[1] = OP_INTEGER;
                        lc3.op[2] = 0;
                        /* cb_get_node_info callback returned Node object. */
                        lc3.data[2].ptr = lc.data[2].ptr;
                        for (;;) {
                                ret = cache_lookup(&lc2, &uid, cb_get_pg_info);
                                if (uid != 0) {
                                        /* we found a portal group */
                                        lc2.curr_uid = uid;
                                        /* it is a null pg if pgt is zero. */
                                        if (lc2.id[2] != 0) {
                                                /* pass ip addr */
                                                lc3.data[0].ip = lc2.data[1].ip;
                                                /* pass port num */
                                                lc3.data[1].ui = lc2.data[2].ui;
                                                /* pass pgt */
                                                lc3.id[2] = lc2.id[2];
                                                ret = cache_lookup(&lc3, &uid,
                                                    cb_get_portal_info);
                                        }
                                } else {
                                        /*
                                         * no more portal group which is
                                         * tied to this stroage node object.
                                         */
                                        break;
                                }
                        }
                }
                /* save error for this iteration */
                if (ret != 0) {
                    ret_save = ret;
                }
                ret = 0;
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * i_get_dd_dds_op:
 *      serves get operatrion on dd or dds.
 *
 * req  - contains all info for a request.
 * doc  - response doc to fill up
 * obj_type     - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
static int
i_get_dd_dds_op(
        request_t *req,
        xmlDocPtr doc,
        isns_type_t obj_type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        int i = 0;
        lookup_ctrl_t lc;
        uint32_t uid;

        if ((obj_type != OBJ_DD) && (obj_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = obj_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;
        lc.data[1].ptr = (uchar_t *)doc; /* xml writer descriptor */
        while (i < req->count) {
                if (obj_type == OBJ_DD) {
                    lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                    ret = cache_lookup(&lc, &uid, cb_get_dd_info);
                    if (uid == 0) {
                        /* set an error and continue. */
                        ret = ERR_MATCHING_DD_NOT_FOUND;
                    }
                } else {
                    lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                    ret = cache_lookup(&lc, &uid, cb_get_ddset_info);
                    if (uid == 0) {
                        /* set an error and continue. */
                        ret = ERR_MATCHING_DDSET_NOT_FOUND;
                    }
                }
                /* save error for this iteration */
                if (ret != 0) {
                    ret_save = ret;
                }
                ret = 0;
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * i_delete_ddmember_op:
 *      serves delete member operatrion on dd.
 *
 * container    - dd name
 * member       - node name
 *
 * ****************************************************************************
 */
static int
i_delete_ddmember_op(
        uchar_t *container,
        uchar_t *member
)
{
        int ret = 0;

        isns_assoc_iscsi_t aiscsi;
        isns_obj_t *assoc;
        isns_attr_t *attr;
        int len;

        lookup_ctrl_t lc;
        uint32_t dd_id;

        /* prepare lookup ctrl data for looking for the dd object */
        lc.curr_uid = 0;
        lc.type = OBJ_DD;
        lc.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        lc.op[0] = OP_STRING;
        lc.data[0].ptr = container;
        lc.op[1] = 0;

        if ((dd_id = is_obj_there(&lc)) != 0) {
            aiscsi.type = OBJ_ASSOC_ISCSI;
            aiscsi.puid = dd_id;

            len = strlen((char *)member) + 1;
            len += 4 - (len % 4);

            attr = &aiscsi.attrs[ATTR_INDEX_ASSOC_ISCSI(
                ISNS_DD_ISCSI_NAME_ATTR_ID)];
            attr->tag = ISNS_DD_ISCSI_NAME_ATTR_ID;
            attr->len = len;
            attr->value.ptr = (uchar_t *)member;
            attr = &aiscsi.attrs[ATTR_INDEX_ASSOC_ISCSI(
                ISNS_DD_ISCSI_INDEX_ATTR_ID)];
            attr->tag = 0; /* clear it */
            assoc = (isns_obj_t *)&aiscsi;
            ret = remove_dd_member(assoc);
        } else {
            ret = ERR_MATCHING_DD_NOT_FOUND;
        }

        return (ret);
}

/*
 * ****************************************************************************
 *
 * i_delete_ddsetmember_op:
 *      serves delete member operatrion on dd set.
 *
 * container    - dd set name
 * member       - dd name
 *
 * ****************************************************************************
 */
static int
i_delete_ddsetmember_op(
        uchar_t *container,
        uchar_t *member
)
{
        int ret = 0;

        lookup_ctrl_t lc, lc2;
        uint32_t container_id, member_id;

        /* prepare lookup ctrl data for looking for the dd-set object */
        lc.curr_uid = 0;
        lc.type = OBJ_DDS;
        lc.id[0] = ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID);
        lc.op[0] = OP_STRING;
        lc.data[0].ptr = container;
        lc.op[1] = 0;

        /* prepare lookup ctrl data for looking for the dd object */
        lc2.curr_uid = 0;
        lc2.type = OBJ_DD;
        lc2.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        lc2.op[0] = OP_STRING;
        lc2.data[0].ptr = member;
        lc2.op[1] = 0;

        if ((container_id = is_obj_there(&lc)) != 0) {
            if ((member_id = is_obj_there(&lc2)) != 0) {
                ret = remove_dds_member(container_id, member_id);
            } else {
                ret = ERR_MATCHING_DD_NOT_FOUND;
            }
        } else {
            ret = ERR_MATCHING_DDSET_NOT_FOUND;
        }

        return (ret);
}

/*
 * ****************************************************************************
 *
 * get_dd_op:
 *      service get operation on given dd(s).
 *
 * req  - contains all info for a request.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
get_dd_op(
        request_t *req,
        xmlDocPtr doc
        /* any additional arguments go here */
)
{
        return (i_get_dd_dds_op(req, doc, OBJ_DD));
}

/*
 * ****************************************************************************
 *
 * get_ddset_op:
 *      service get operation on given dd set(s).
 *
 * req  - contains all info for a request.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
get_ddset_op(
        request_t *req,
        xmlDocPtr doc
        /* any additional arguments go here */
)
{
        return (i_get_dd_dds_op(req, doc, OBJ_DDS));
}

/*
 * ****************************************************************************
 *
 * enumerate_node_op:
 *      services enumerate node op.
 *
 * req  - contains enumerate request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
enumerate_node_op(
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        htab_t *htab = cache_get_htab(OBJ_ISCSI);
        uint32_t uid = 0;
        lookup_ctrl_t lc;
        int         ret = 0, ret_save = 0;

        SET_UID_LCP(&lc, OBJ_ISCSI, 0);

        lc.data[1].ptr = (uchar_t *)doc;
        lc.data[2].ui = 0;

        FOR_EACH_ITEM(htab, uid, {
                lc.data[0].ui = uid;
                ret = cache_lookup(&lc, NULL, cb_enumerate_node_info);
                if (ret != 0) {
                    ret_save = ret;
                }
        });

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * enumerate_dd_op:
 *      services enumerate discovery domain op.
 *
 * req  - contains enumerate request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
enumerate_dd_op(
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{

        htab_t *htab = cache_get_htab(OBJ_DD);
        uint32_t uid = 0;
        lookup_ctrl_t lc;
        int         ret = 0, ret_save = 0;

        SET_UID_LCP(&lc, OBJ_DD, 0);

        lc.data[1].ptr = (uchar_t *)doc;
        lc.data[2].ui = 0;

        FOR_EACH_ITEM(htab, uid, {
                lc.data[0].ui = uid;
                ret = cache_lookup(&lc, NULL, cb_enumerate_dd_info);
                if (ret != 0) {
                    ret_save = ret;
                }
        });

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * enumerate_ddset_op:
 *      services enumerate discovery domain set op.
 *
 * req  - contains enumerate request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
enumerate_ddset_op(
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        htab_t *htab = cache_get_htab(OBJ_DDS);
        uint32_t uid = 0;
        lookup_ctrl_t lc;
        int         ret = 0, ret_save = 0;

        SET_UID_LCP(&lc, OBJ_DDS, 0);

        lc.data[1].ptr = (uchar_t *)doc;
        lc.data[2].ui = 0;

        FOR_EACH_ITEM(htab, uid, {
                lc.data[0].ui = uid;
                ret = cache_lookup(&lc, NULL, cb_enumerate_ddset_info);
                if (ret != 0) {
                    ret_save = ret;
                }
        });

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * getassociated_dd_to_node_op:
 *      construct a list of node that is associated with a given Discovery
 *      Domain.
 *
 * req  - contains getAssociated request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
getAssociated_dd_to_node_op(
        request_t *req,
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        uint32_t uid = 0, n;
        lookup_ctrl_t lc, lc2;
        int     i = 0, ret = 0, ret_save = 0;
        bmp_t   *p;

        lc.curr_uid = 0;
        lc.type = OBJ_DD;
        lc.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        SET_UID_LCP(&lc2, OBJ_ISCSI, 0);

        lc2.data[1].ptr = (uchar_t *)doc;

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                if ((uid = is_obj_there(&lc)) != 0) {
                    ret = get_dd_matrix(uid, &p, &n);
                    FOR_EACH_MEMBER(p, n, uid, {
                        lc2.data[0].ui = uid;
                        lc2.data[2].ptr = (uchar_t *)req->req_data.data[i];
                        ret = cache_lookup(&lc2, NULL,
                            cb_getAssociated_node_info);
                    });
                    free(p);
                } else {
                    ret = ERR_MATCHING_DD_NOT_FOUND;
                }
                /* save error for this iteration */
                if (ret != 0) {
                    ret_save = ret;
                }
                ret = 0;
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * getassociated_node_to_dd_op:
 *      construct a list of Discovery Doamins that is associated with a given
 *      node.
 *
 * req  - contains getAssociated request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
getAssociated_node_to_dd_op(
        request_t *req,
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        uint32_t uid = 0, dd_id;
        lookup_ctrl_t lc, lc2;
        int     i = 0, ret = 0, ret_save = 0;

        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.op[1] = 0;

        SET_UID_LCP(&lc2, OBJ_DD, 0);

        lc2.data[1].ptr = (uchar_t *)doc;

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                if ((uid = is_obj_there(&lc)) != 0) {
                    if ((dd_id = get_dd_id(uid, 0)) == 0) {
                        ret = ERR_NO_ASSOCIATED_DD_FOUND;
                        i++;
                        continue;
                    } else {
                        do {
                            lc2.data[0].ui = dd_id;
                            lc2.data[2].ptr = (uchar_t *)req->req_data.data[i];
                            ret = cache_lookup(&lc2, NULL,
                                cb_getAssociated_node_to_dd_info);
                            dd_id = get_dd_id(uid, dd_id);
                        } while (dd_id != 0);
                    };
                } else {
                    ret = ERR_MATCHING_NODE_NOT_FOUND;
                }
                /* save error for this iteration */
                if (ret != 0) {
                    ret_save = ret;
                }
                ret = 0;
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * getassociated_ddset_to_dd_op:
 *      construct a list of Discovery Doamins that is associated with a given
 *      Discover Domain set.
 *
 * req  - contains getAssociated request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
getAssociated_ddset_to_dd_op(
        request_t *req,
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        uint32_t uid = 0, n;
        lookup_ctrl_t lc, lc2;
        int     i = 0, ret = 0, ret_save = 0;
        bmp_t   *p;

        lc.curr_uid = 0;
        lc.type = OBJ_DDS;
        lc.id[0] = ATTR_INDEX_DDS(ISNS_DD_SET_NAME_ATTR_ID);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        SET_UID_LCP(&lc2, OBJ_DD, 0);

        lc2.data[1].ptr = (uchar_t *)doc;

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                if ((uid = is_obj_there(&lc)) != 0) {
                    ret = get_dds_matrix(uid, &p, &n);
                    FOR_EACH_MEMBER(p, n, uid, {
                        lc2.data[0].ui = uid;
                        lc2.data[2].ptr = (uchar_t *)req->req_data.data[i];
                        ret = cache_lookup(&lc2, NULL,
                            cb_getAssociated_dd_info);
                    });
                    free(p);
                } else {
                    ret = ERR_MATCHING_DDSET_NOT_FOUND;
                }
                /* save error for this iteration */
                if (ret != 0) {
                    ret_save = ret;
                }
                ret = 0;
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * getassociated_dd_to_ddset_op:
 *      construct a list of Discovery Doamin sets that is associated with a
 *      given Discovery Domain.
 *
 * req  - contains getAssociated request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
getAssociated_dd_to_ddset_op(
        request_t *req,
        xmlDocPtr   doc
        /* any additional arguments go here */
)
{
        uint32_t uid = 0, ddset_id;
        lookup_ctrl_t lc, lc2;
        int     i = 0, ret = 0, ret_save = 0;

        lc.curr_uid = 0;
        lc.type = OBJ_DD;
        lc.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        SET_UID_LCP(&lc2, OBJ_DDS, 0);

        lc2.data[1].ptr = (uchar_t *)doc;

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];
                if ((uid = is_obj_there(&lc)) != 0) {
                    lc2.data[2].ui = 0;
                    if ((ddset_id = get_dds_id(uid, 0)) == 0) {
                        ret = ERR_NO_ASSOCIATED_DDSET_FOUND;
                        i++;
                        continue;
                    } else {
                        do {
                            lc2.data[0].ui = ddset_id;
                            lc2.data[2].ptr = (uchar_t *)req->req_data.data[i];
                            ret = cache_lookup(&lc2, NULL,
                                cb_getAssociated_dd_to_ddset_info);
                            ddset_id = get_dds_id(uid, ddset_id);
                        } while (ddset_id != 0);
                    };
                } else {
                    ret = ERR_MATCHING_DD_NOT_FOUND;
                }
                if (ret != 0) {
                    ret_save = ret;
                }
                i++;
        }

        return (handle_partial_success(doc, ret_save));
}

/*
 * ****************************************************************************
 *
 * delete_dd_ddset_op:
 *      removes a list of dd or dd set.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 * obj_type     - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
int
delete_dd_ddset_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        int i = 0, err_count = 0;
        lookup_ctrl_t lc;
        uint32_t uid;
        xmlNodePtr      n_obj, n_node, root;
        xmlAttrPtr      n_attr;
        int different_err = 0;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = lc_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;
        lc.data[1].ptr = (uchar_t *)doc; /* xml writer descriptor */
        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.data[i];

                /* lock the cache for writing */
                (void) cache_lock_write();

                if ((uid = is_obj_there(&lc)) != 0) {
                    /* remove the dd/ddset */
                    ret = (lc_type == OBJ_DD) ?
                        remove_dd_object(uid) :
                        remove_dds_object(uid);
                    /* unlock the cache and sync the data */
                    ret = cache_unlock_sync(ret);
                } else {
                    /* unlock the cache and no need to sync data */
                    (void) cache_unlock_nosync();
                    /* set an error and continue. */
                    ret = (lc_type == OBJ_DD) ?  ERR_MATCHING_DD_NOT_FOUND :
                        ERR_MATCHING_DDSET_NOT_FOUND;
                }

                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
                    if (n_obj) {
                        if ((n_obj = xmlAddChild(root, n_obj)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }

                    n_node = (lc_type == OBJ_DD) ?
                        xmlNewNode(NULL, (xmlChar *)DDOBJECT) :
                        xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
                    if (n_node) {
                        if ((n_node = xmlAddChild(n_obj, n_node)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                        n_attr = xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                                (xmlChar *)req->req_data.data[i]);
                        if (n_attr == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }
                    ret_save = ret;
                }
                i ++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * delete_ddmember_ddsetmember_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 * type - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
int
delete_ddmember_ddsetmember_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        int i = 0, err_count = 0;
        lookup_ctrl_t lc, lc2;
        uint32_t container_id, member_id;
        xmlNodePtr      n_node, n_obj, root;
        xmlAttrPtr      n_attr;
        int different_err = 0;
        int is_a_member;

        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = lc_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        lc2.curr_uid = 0;
        if (lc_type == OBJ_DD) {
            lc2.type = OBJ_ISCSI;
            lc2.id[0] = ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID);
        } else {
            lc2.type = OBJ_DD;
            lc2.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        }
        lc2.op[0] = OP_STRING;
        lc2.op[1] = 0;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.pair[i]->container;

                /* get the dd_id/dds_id */
                (void) cache_lock_write();
                container_id = is_obj_there(&lc);

                if (container_id != 0) {
                    lc2.data[0].ptr = (uchar_t *)req->req_data.pair[i]->member;

                    member_id = is_obj_there(&lc2);
                    if (member_id != 0) {
                        is_a_member =
                            (container_id ==
                            ((lc_type == OBJ_DD) ?
                            get_dd_id(member_id, container_id - 1) :
                            get_dds_id(member_id, container_id - 1)));
                    }
                    if (member_id != 0 && is_a_member != 0) {
                        /* delete the dd member */
                        ret = (lc_type == OBJ_DD) ?
                            i_delete_ddmember_op(
                                (uchar_t *)req->req_data.pair[i]->container,
                                (uchar_t *)req->req_data.pair[i]->member) :
                            i_delete_ddsetmember_op(
                                (uchar_t *)req->req_data.pair[i]->container,
                                (uchar_t *)req->req_data.pair[i]->member);
                        /* unlock the cache and sync the data */
                        ret = cache_unlock_sync(ret);
                    } else {
                        /* unlock the cache and no need to sync */
                        (void) cache_unlock_nosync();
                        ret = ERR_NO_SUCH_ASSOCIATION;
                    }
                } else {
                    /* unlock the cache and no need to sync */
                    (void) cache_unlock_nosync();
                    ret = (lc_type == OBJ_DD) ?  ERR_MATCHING_DD_NOT_FOUND :
                        ERR_MATCHING_DDSET_NOT_FOUND;
                }

                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    ret_save = ret;
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
                    if (n_obj) {
                        n_obj = xmlAddChild(root, n_obj);
                        if (n_obj == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }
                    if (lc_type == OBJ_DD) {
                        n_node =
                            xmlNewNode(NULL, (xmlChar *)DDOBJECTMEMBER);
                        n_attr = xmlSetProp(n_node, (xmlChar *)NODENAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->member);
                        if (n_attr == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->container);
                        if (n_attr == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                    } else {
                        n_node =
                            xmlNewNode(NULL, (xmlChar *)DDSETOBJECTMEMBER);
                        n_attr = xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->member);
                        if (n_attr == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                        n_attr = xmlSetProp(n_node, (xmlChar *)DDSETNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->container);
                        if (n_attr == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                    }
                    if (xmlAddChild(n_obj, n_node) == NULL) {
                        return (ERR_XML_ADDCHILD_FAILED);
                    }
                }
                i++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * create_ddmember_ddsetmember_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 * type - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
int
create_ddmember_ddsetmember_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        int i = 0, err_count = 0;
        lookup_ctrl_t lc, lc2;
        uint32_t container_id, member_id;
        xmlNodePtr      n_node, n_obj, root;
        isns_assoc_iscsi_t aiscsi = { 0 };
        isns_assoc_dd_t add = { 0 };
        isns_obj_t *assoc;
        isns_attr_t *attr;
        uint32_t len;
        int different_err = 0;

        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = lc_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        lc2.curr_uid = 0;
        if (lc_type == OBJ_DD) {
            lc2.type = OBJ_ISCSI;
            lc2.id[0] = ATTR_INDEX_ISCSI(ISNS_ISCSI_NAME_ATTR_ID);
        } else {
            lc2.type = OBJ_DD;
            lc2.id[0] = ATTR_INDEX_DD(ISNS_DD_NAME_ATTR_ID);
        }
        lc2.op[0] = OP_STRING;
        lc2.op[1] = 0;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        while (i < req->count) {
                lc.data[0].ptr = (uchar_t *)req->req_data.pair[i]->container;

                /* get the dd_id/dds_id */
                (void) cache_lock_write();
                container_id = is_obj_there(&lc);

                if (container_id != 0) {
                    (void) memset(&aiscsi, 0, sizeof (aiscsi));
                    if (lc_type == OBJ_DD) {
                        aiscsi.puid = container_id;
                        aiscsi.type = OBJ_ASSOC_ISCSI;
                        attr = &aiscsi.attrs[ATTR_INDEX_ASSOC_ISCSI(
                            ISNS_DD_ISCSI_NAME_ATTR_ID)];
                        attr->tag = ISNS_DD_ISCSI_NAME_ATTR_ID;
                        len = xmlStrlen(
                            (xmlChar *)req->req_data.pair[i]->member) + 1;
                        len += 4 - (len % 4); /* on 4 bytes aligned */
                        attr->len = len;
                        attr->value.ptr =
                            (uchar_t *)req->req_data.pair[i]->member;
                        assoc = (isns_obj_t *)&aiscsi;

                        /* add the dd member */
                        ret = add_dd_member(assoc);

                        /* unlock the cache and sync the data */
                        ret = cache_unlock_sync(ret);
                    } else {
                        lc2.data[0].ptr =
                            (uchar_t *)req->req_data.pair[i]->member;

                        if ((member_id = is_obj_there(&lc2)) != 0) {
                            add.puid = container_id;
                            add.type = OBJ_ASSOC_DD;
                            attr = &add.attrs[ATTR_INDEX_ASSOC_DD(
                                ISNS_DD_ID_ATTR_ID)];
                            attr->tag = ISNS_DD_ID_ATTR_ID;
                            attr->len = 4;
                            attr->value.ui = member_id;
                            assoc = (isns_obj_t *)&add;

                            /* add the dd-set member */
                            ret = add_dds_member(assoc);

                            /* unlock the cache and sync the data */
                            ret = cache_unlock_sync(ret);
                        } else {
                            /* unlock the cache and no need to sync */
                            (void) cache_unlock_nosync();
                            ret = ERR_MATCHING_DD_NOT_FOUND;
                        }
                    }
                } else {
                    /* unlock the cache and no need to sync */
                    (void) cache_unlock_nosync();
                    ret = (lc_type == OBJ_DD) ?  ERR_MATCHING_DD_NOT_FOUND :
                        ERR_MATCHING_DDSET_NOT_FOUND;
                }
                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ASSOCIATION);
                    if (n_obj) {
                        n_obj = xmlAddChild(root, n_obj);
                        if (n_obj == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }
                    if (lc_type == OBJ_DD) {
                        n_node =
                            xmlNewNode(NULL, (xmlChar *)DDOBJECTMEMBER);
                        if (xmlSetProp(n_node, (xmlChar *)NODENAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->member) == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                        if (xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->container) ==
                            NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                    } else {
                        n_node =
                            xmlNewNode(NULL, (xmlChar *)DDSETOBJECTMEMBER);
                        if (xmlSetProp(n_node, (xmlChar *)DDNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->member) == NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                        if (xmlSetProp(n_node, (xmlChar *)DDSETNAMEATTR,
                            (xmlChar *)req->req_data.pair[i]->container) ==
                            NULL) {
                            return (ERR_XML_SETPROP_FAILED);
                        }
                    }
                    if (xmlAddChild(n_obj, n_node) == NULL) {
                        return (ERR_XML_ADDCHILD_FAILED);
                    }
                    ret_save = ret;
                }
                i++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * rename_dd_ddset_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 * type - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
static int
rename_dd_ddset_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        int i = 0, err_count = 0;
        lookup_ctrl_t lc;
        uint32_t container_id;
        xmlNodePtr      n_node, n_obj, root;
        uchar_t *name;
        uint32_t len;
        int different_err = 0;

        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        SET_UID_LCP(&lc, lc_type, 0);

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        while (i < req->count) {
                /* id is checked to be not NULL before calling this routine. */
                lc.data[0].ui = *(req->req_data.attrlist[i]->id);

                /* get the dd_id/dds_id */
                (void) cache_lock_write();

                if ((container_id = is_obj_there(&lc)) != 0) {
                    name = (uchar_t *)req->req_data.attrlist[i]->name;
                    /* the length of the name need to include the */
                    /* null terminator and be on 4 bytes aligned */
                    len = xmlStrlen(name) + 1;
                    len += 4 - (len % 4);

                    /* rename the dd/dds */
                    ret = (lc_type == OBJ_DD) ?
                        update_dd_name(container_id, len, name) :
                        update_dds_name(container_id, len, name);

                    /* release the lock and sync the data */
                    ret = cache_unlock_sync(ret);
                } else {
                    /* release the lock and no need to sync */
                    (void) cache_unlock_nosync();
                    ret = (lc_type == OBJ_DD) ?  ERR_MATCHING_DD_NOT_FOUND :
                        ERR_MATCHING_DDSET_NOT_FOUND;
                }
                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    ret_save = ret;
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
                    if (n_obj) {
                        if ((n_obj = xmlAddChild(root, n_obj)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }

                    n_node = (lc_type == OBJ_DD) ?
                        xmlNewNode(NULL, (xmlChar *)DDOBJECT) :
                        xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
                    if (n_node) {
                        if ((n_node = xmlAddChild(n_obj, n_node)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        } else {
                            if (xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                                (xmlChar *)req->req_data.attrlist[i]->name) ==
                                NULL) {
                                return (ERR_XML_SETPROP_FAILED);
                            }
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }

                }
                i++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * update_dd_ddset_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 * type - object type(either dd or dd set)
 *
 * ****************************************************************************
 */
static int
update_dd_ddset_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        int i = 0, err_count = 0;
        lookup_ctrl_t lc;
        uint32_t container_id;
        xmlNodePtr      n_node, n_obj, root;
        int different_err = 0;

        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = lc_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }

        while (i < req->count) {
                lc.data[0].ptr = req->req_data.attrlist[i]->name;

                /* lock the cache for writing */
                (void) cache_lock_write();

                if ((container_id = is_obj_there(&lc)) != 0) {
                    ret = (lc_type == OBJ_DD) ?
                        /* enabled is checked to be not NULL before calling. */
                        update_dd_features(container_id,
                        *(req->req_data.attrlist[i]->enabled) ? 1 : 0):
                        update_dds_status(container_id,
                        *(req->req_data.attrlist[i]->enabled) ? 1 : 0);
                    /* unlock the cache and sync the data */
                    ret = cache_unlock_sync(ret);
                } else {
                    (void) cache_unlock_nosync();
                    ret = (lc_type == OBJ_DD) ?  ERR_MATCHING_DD_NOT_FOUND :
                        ERR_MATCHING_DDSET_NOT_FOUND;
                }
                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    ret_save = ret;
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
                    if (n_obj) {
                        if ((n_obj = xmlAddChild(root, n_obj)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }

                    n_node = (lc_type == OBJ_DD) ?
                        xmlNewNode(NULL, (xmlChar *)DDOBJECT) :
                        xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
                    if (n_node) {
                        if ((n_node = xmlAddChild(n_obj, n_node)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        } else {
                            if (xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                                (xmlChar *)req->req_data.attrlist[i]->name) ==
                                NULL) {
                                return (ERR_XML_SETPROP_FAILED);
                            }
                        }
                    } else {
                            return (ERR_XML_NEWNODE_FAILED);
                    }
                }
                i++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * createModify_dd_ddset_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
static int
create_dd_ddset_op(
        request_t *req,
        xmlDocPtr doc,
        object_type type
        /* any additional arguments go here */
)
{
        isns_obj_t  *obj;
        result_code_t ret = 0, ret_save = 0;
        isns_type_t lc_type;
        lookup_ctrl_t lc;
        uint32_t uid;
        int i = 0, err_count = 0;
        xmlNodePtr      n_obj, n_node, root;
        int different_err = 0;

        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        root = xmlDocGetRootElement(doc);
        if (root == NULL) {
            return (ERR_SYNTAX_MISSING_ROOT);
        }
        lc_type = get_lc_type(type);
        if ((lc_type != OBJ_DD) && (lc_type != OBJ_DDS)) {
            return (ERR_INVALID_MGMT_REQUEST);
        }

        /* prepare lookup ctrl data for looking for the node object */
        lc.curr_uid = 0;
        lc.type = lc_type;
        lc.id[0] = get_lc_id(req->op_info.obj);
        lc.op[0] = OP_STRING;
        lc.op[1] = 0;
        lc.data[1].ptr = (uchar_t *)doc; /* xml writer descriptor */
        while (i < req->count) {
                lc.data[0].ptr = req->req_data.attrlist[i]->name,
                /* grab the write lock */
                (void) cache_lock_write();

                uid = is_obj_there(&lc);
                if (uid == 0) {
                    ret = (lc_type == OBJ_DD) ?
                        adm_create_dd(&obj, req->req_data.attrlist[i]->name,
                        0, 0) :
                        adm_create_dds(&obj, req->req_data.attrlist[i]->name,
                        0, 0);
                    if (ret == 0) {
                        ret = register_object(obj, NULL, NULL);
                        if (ret != 0) {
                            free_object(obj);
                        }
                        /* release the lock and sync the cache and data store */
                        ret = cache_unlock_sync(ret);
                    }
                } else {
                        /* release the lock and no need to sync the data */
                        (void) cache_unlock_nosync();
                        ret = ERR_NAME_IN_USE;
                }

                if (ret != 0) {
                /* keep track if there are different errors encountered. */
                    if (ret_save != 0 && ret != ret_save) {
                        different_err++;
                    }
                    ret_save = ret;
                    err_count++;
                    n_obj = xmlNewNode(NULL, (xmlChar *)ISNSOBJECT);
                    if (n_obj) {
                        if ((n_obj = xmlAddChild(root, n_obj)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        }
                    } else {
                        return (ERR_XML_ADDCHILD_FAILED);
                    }

                    n_node = (lc_type == OBJ_DD) ?
                        xmlNewNode(NULL, (xmlChar *)DDOBJECT) :
                        xmlNewNode(NULL, (xmlChar *)DDSETOBJECT);
                    if (n_node) {
                        if ((n_node = xmlAddChild(n_obj, n_node)) == NULL) {
                            return (ERR_XML_ADDCHILD_FAILED);
                        } else {
                            if (xmlSetProp(n_node, (xmlChar *)NAMEATTR,
                                (xmlChar *)req->req_data.attrlist[i]->name) ==
                                NULL) {
                                return (ERR_XML_SETPROP_FAILED);
                            }
                        }
                    } else {
                        return (ERR_XML_NEWNODE_FAILED);
                    }
                }
                i++;
        }

        return (handle_partial_failure(doc, ret_save,
            (req->count == err_count && !different_err) ? B_TRUE : B_FALSE));
}

/*
 * ****************************************************************************
 *
 * createModify_dd_ddset_op:
 *      removes a list of dd memeber or dd seti member.
 *
 * req  - contains delete request info.
 * doc  - response doc to fill up
 *
 * ****************************************************************************
 */
int
createModify_dd_ddset_op(
        request_t *req,
        xmlDocPtr doc
        /* any additional arguments go here */
)
{
        result_code_t ret = 0;

        if (req->req_data.attrlist[0]->id != NULL) {
            ret = rename_dd_ddset_op(req, doc, req->op_info.obj);
        } else if (req->req_data.attrlist[0]->enabled != NULL) {
            ret = update_dd_ddset_op(req, doc, req->op_info.obj);
        } else {
            ret = create_dd_ddset_op(req, doc, req->op_info.obj);
        }

        return (ret);
}