libshare.c revision a335142544257a204dde1839676940d0e0432d06
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * CDDL HEADER START
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * The contents of this file are subject to the terms of the
6185db853e024a486ff8837e6784dd290d866112dougm * Common Development and Distribution License (the "License").
6185db853e024a486ff8837e6784dd290d866112dougm * You may not use this file except in compliance with the License.
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
6185db853e024a486ff8837e6784dd290d866112dougm * or http://www.opensolaris.org/os/licensing.
6185db853e024a486ff8837e6784dd290d866112dougm * See the License for the specific language governing permissions
6185db853e024a486ff8837e6784dd290d866112dougm * and limitations under the License.
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * When distributing Covered Code, include this CDDL HEADER in each
6185db853e024a486ff8837e6784dd290d866112dougm * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
6185db853e024a486ff8837e6784dd290d866112dougm * If applicable, add the following below this CDDL HEADER, with the
6185db853e024a486ff8837e6784dd290d866112dougm * fields enclosed by brackets "[]" replaced with your own identifying
6185db853e024a486ff8837e6784dd290d866112dougm * information: Portions Copyright [yyyy] [name of copyright owner]
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * CDDL HEADER END
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * Copyright 2007 Sun Microsystems, Inc. All rights reserved.
6185db853e024a486ff8837e6784dd290d866112dougm * Use is subject to license terms.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm#pragma ident "%Z%%M% %I% %E% SMI"
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * Share control API
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm#include <stdio.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <string.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <ctype.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <sys/types.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <sys/stat.h>
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm#include <fcntl.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <unistd.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <libxml/parser.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <libxml/tree.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include "libshare.h"
6185db853e024a486ff8837e6784dd290d866112dougm#include "libshare_impl.h"
6185db853e024a486ff8837e6784dd290d866112dougm#include <libscf.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include "scfutil.h"
6185db853e024a486ff8837e6784dd290d866112dougm#include <ctype.h>
6185db853e024a486ff8837e6784dd290d866112dougm#include <libintl.h>
549ec3fff108310966327d1dc9004551b63210b7dougm#include <thread.h>
549ec3fff108310966327d1dc9004551b63210b7dougm#include <synch.h>
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm#if _NOT_SMF
6185db853e024a486ff8837e6784dd290d866112dougm#define CONFIG_FILE "/var/tmp/share.cfg"
6185db853e024a486ff8837e6784dd290d866112dougm#define CONFIG_FILE_TMP "/var/tmp/share.cfg.tmp"
6185db853e024a486ff8837e6784dd290d866112dougm#endif
6185db853e024a486ff8837e6784dd290d866112dougm#define TSTAMP(tm) (uint64_t)(((uint64_t)tm.tv_sec << 32) | \
6185db853e024a486ff8837e6784dd290d866112dougm (tm.tv_nsec & 0xffffffff))
6185db853e024a486ff8837e6784dd290d866112dougm
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm#define DFS_LOCK_FILE "/etc/dfs/fstypes"
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define SA_STRSIZE 256 /* max string size for names */
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * internal data structures
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmextern struct sa_proto_plugin *sap_proto_list;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/* current SMF/SVC repository handle */
549ec3fff108310966327d1dc9004551b63210b7dougmextern void getlegacyconfig(sa_handle_t, char *, xmlNodePtr *);
549ec3fff108310966327d1dc9004551b63210b7dougmextern int gettransients(sa_handle_impl_t, xmlNodePtr *);
6185db853e024a486ff8837e6784dd290d866112dougmextern int sa_valid_property(void *, char *, sa_property_t);
6185db853e024a486ff8837e6784dd290d866112dougmextern char *sa_fstype(char *);
6185db853e024a486ff8837e6784dd290d866112dougmextern int sa_is_share(void *);
6185db853e024a486ff8837e6784dd290d866112dougmextern ssize_t scf_max_name_len; /* defined in scfutil during initialization */
6185db853e024a486ff8837e6784dd290d866112dougmextern int sa_group_is_zfs(sa_group_t);
6185db853e024a486ff8837e6784dd290d866112dougmextern int sa_path_is_zfs(char *);
6185db853e024a486ff8837e6784dd290d866112dougmextern int sa_zfs_set_sharenfs(sa_group_t, char *, int);
549ec3fff108310966327d1dc9004551b63210b7dougmextern void update_legacy_config(sa_handle_t);
6185db853e024a486ff8837e6784dd290d866112dougmextern int issubdir(char *, char *);
57b448de658d89a2c88a001d58073c46ed2180f3dougmextern int sa_zfs_init(sa_handle_impl_t);
549ec3fff108310966327d1dc9004551b63210b7dougmextern void sa_zfs_fini(sa_handle_impl_t);
a99982a76d4cc12b1e9021e88531cf425d1e7369dougmextern void sablocksigs(sigset_t *);
a99982a76d4cc12b1e9021e88531cf425d1e7369dougmextern void saunblocksigs(sigset_t *);
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * Data structures for finding/managing the document root to access
549ec3fff108310966327d1dc9004551b63210b7dougm * handle mapping. The list isn't expected to grow very large so a
549ec3fff108310966327d1dc9004551b63210b7dougm * simple list is acceptable. The purpose is to provide a way to start
549ec3fff108310966327d1dc9004551b63210b7dougm * with a group or share and find the library handle needed for
549ec3fff108310966327d1dc9004551b63210b7dougm * various operations.
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougmmutex_t sa_global_lock;
549ec3fff108310966327d1dc9004551b63210b7dougmstruct doc2handle {
549ec3fff108310966327d1dc9004551b63210b7dougm struct doc2handle *next;
549ec3fff108310966327d1dc9004551b63210b7dougm xmlNodePtr root;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t handle;
549ec3fff108310966327d1dc9004551b63210b7dougm};
549ec3fff108310966327d1dc9004551b63210b7dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm/* definitions used in a couple of property functions */
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define SA_PROP_OP_REMOVE 1
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define SA_PROP_OP_ADD 2
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define SA_PROP_OP_UPDATE 3
57b448de658d89a2c88a001d58073c46ed2180f3dougm
549ec3fff108310966327d1dc9004551b63210b7dougmstatic struct doc2handle *sa_global_handles = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/* helper functions */
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * sa_errorstr(err)
549ec3fff108310966327d1dc9004551b63210b7dougm *
549ec3fff108310966327d1dc9004551b63210b7dougm * convert an error value to an error string
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_errorstr(int err)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm static char errstr[32];
6185db853e024a486ff8837e6784dd290d866112dougm char *ret = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm switch (err) {
6185db853e024a486ff8837e6784dd290d866112dougm case SA_OK:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "ok");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_SUCH_PATH:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "path doesn't exist");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_MEMORY:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "no memory");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_DUPLICATE_NAME:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "name in use");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_BAD_PATH:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "bad path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_SUCH_GROUP:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "no such group");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_CONFIG_ERR:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "configuration error");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_SYSTEM_ERR:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "system error");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_SYNTAX_ERR:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "syntax error");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_PERMISSION:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "no permission");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_BUSY:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "busy");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_SUCH_PROP:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "no such property");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_INVALID_NAME:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "invalid name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_INVALID_PROTOCOL:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "invalid protocol");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NOT_ALLOWED:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "operation not allowed");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_BAD_VALUE:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "bad property value");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_INVALID_SECURITY:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "invalid security type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NO_SUCH_SECURITY:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "security type not found");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_VALUE_CONFLICT:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "property value conflict");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NOT_IMPLEMENTED:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "not implemented");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_INVALID_PATH:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "invalid path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NOT_SUPPORTED:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "operation not supported");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_PROP_SHARE_ONLY:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "property not valid for group");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm case SA_NOT_SHARED:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = dgettext(TEXT_DOMAIN, "not shared");
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm default:
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) snprintf(errstr, sizeof (errstr),
57b448de658d89a2c88a001d58073c46ed2180f3dougm dgettext(TEXT_DOMAIN, "unknown %d"), err);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = errstr;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * Document root to active handle mapping functions. These are only
549ec3fff108310966327d1dc9004551b63210b7dougm * used internally. A mutex is used to prevent access while the list
549ec3fff108310966327d1dc9004551b63210b7dougm * is changing. In general, the list will be relatively short - one
549ec3fff108310966327d1dc9004551b63210b7dougm * item per thread that has called sa_init().
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougmsa_handle_impl_t
549ec3fff108310966327d1dc9004551b63210b7dougmget_handle_for_root(xmlNodePtr root)
549ec3fff108310966327d1dc9004551b63210b7dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm struct doc2handle *item;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm (void) mutex_lock(&sa_global_lock);
549ec3fff108310966327d1dc9004551b63210b7dougm for (item = sa_global_handles; item != NULL; item = item->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (item->root == root)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
549ec3fff108310966327d1dc9004551b63210b7dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm (void) mutex_unlock(&sa_global_lock);
549ec3fff108310966327d1dc9004551b63210b7dougm if (item != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (item->handle);
549ec3fff108310966327d1dc9004551b63210b7dougm return (NULL);
549ec3fff108310966327d1dc9004551b63210b7dougm}
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougmstatic int
549ec3fff108310966327d1dc9004551b63210b7dougmadd_handle_for_root(xmlNodePtr root, sa_handle_impl_t handle)
549ec3fff108310966327d1dc9004551b63210b7dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm struct doc2handle *item;
549ec3fff108310966327d1dc9004551b63210b7dougm int ret = SA_NO_MEMORY;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm item = (struct doc2handle *)calloc(sizeof (struct doc2handle), 1);
549ec3fff108310966327d1dc9004551b63210b7dougm if (item != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm item->root = root;
57b448de658d89a2c88a001d58073c46ed2180f3dougm item->handle = handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) mutex_lock(&sa_global_lock);
57b448de658d89a2c88a001d58073c46ed2180f3dougm item->next = sa_global_handles;
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_global_handles = item;
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) mutex_unlock(&sa_global_lock);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_OK;
549ec3fff108310966327d1dc9004551b63210b7dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm return (ret);
549ec3fff108310966327d1dc9004551b63210b7dougm}
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * remove_handle_for_root(root)
549ec3fff108310966327d1dc9004551b63210b7dougm *
549ec3fff108310966327d1dc9004551b63210b7dougm * Walks the list of handles and removes the one for this "root" from
549ec3fff108310966327d1dc9004551b63210b7dougm * the list. It is up to the caller to free the data.
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougmstatic void
549ec3fff108310966327d1dc9004551b63210b7dougmremove_handle_for_root(xmlNodePtr root)
549ec3fff108310966327d1dc9004551b63210b7dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm struct doc2handle *item, *prev;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm (void) mutex_lock(&sa_global_lock);
549ec3fff108310966327d1dc9004551b63210b7dougm for (prev = NULL, item = sa_global_handles; item != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm item = item->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (item->root == root) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* first in the list */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (prev == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_global_handles = sa_global_handles->next;
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm prev->next = item->next;
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* Item is out of the list so free the list structure */
57b448de658d89a2c88a001d58073c46ed2180f3dougm free(item);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
549ec3fff108310966327d1dc9004551b63210b7dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm prev = item;
549ec3fff108310966327d1dc9004551b63210b7dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm (void) mutex_unlock(&sa_global_lock);
549ec3fff108310966327d1dc9004551b63210b7dougm}
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * sa_find_group_handle(sa_group_t group)
549ec3fff108310966327d1dc9004551b63210b7dougm *
549ec3fff108310966327d1dc9004551b63210b7dougm * Find the sa_handle_t for the configuration associated with this
549ec3fff108310966327d1dc9004551b63210b7dougm * group.
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougmsa_handle_t
549ec3fff108310966327d1dc9004551b63210b7dougmsa_find_group_handle(sa_group_t group)
549ec3fff108310966327d1dc9004551b63210b7dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm xmlNodePtr node = (xmlNodePtr)group;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_t handle;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm while (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (strcmp((char *)(node->name), "sharecfg") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* have the root so get the handle */
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle = (sa_handle_t)get_handle_for_root(node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->parent;
549ec3fff108310966327d1dc9004551b63210b7dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm return (NULL);
549ec3fff108310966327d1dc9004551b63210b7dougm}
549ec3fff108310966327d1dc9004551b63210b7dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * set_legacy_timestamp(root, path, timevalue)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * add the current timestamp value to the configuration for use in
6185db853e024a486ff8837e6784dd290d866112dougm * determining when to update the legacy files. For SMF, this
6185db853e024a486ff8837e6784dd290d866112dougm * property is kept in default/operation/legacy_timestamp
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic void
6185db853e024a486ff8837e6784dd290d866112dougmset_legacy_timestamp(xmlNodePtr root, char *path, uint64_t tval)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *lpath = NULL;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t handle;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm /* Have to have a handle or else we weren't initialized. */
549ec3fff108310966327d1dc9004551b63210b7dougm handle = get_handle_for_root(root);
549ec3fff108310966327d1dc9004551b63210b7dougm if (handle == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = root->xmlChildrenNode; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"legacy") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* a possible legacy node for this path */
57b448de658d89a2c88a001d58073c46ed2180f3dougm lpath = xmlGetProp(node, (xmlChar *)"path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (lpath != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(lpath, (xmlChar *)path) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(lpath);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (lpath != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(lpath);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (node == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* need to create the first legacy timestamp node */
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild(root, NULL, (xmlChar *)"legacy", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char tstring[32];
57b448de658d89a2c88a001d58073c46ed2180f3dougm int ret;
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) snprintf(tstring, sizeof (tstring), "%lld", tval);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"timestamp", (xmlChar *)tstring);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"path", (xmlChar *)path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* now commit to SMF */
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_get_instance(handle->scfhandle, "default");
6185db853e024a486ff8837e6784dd290d866112dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "operation");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "legacy-timestamp", tstring);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_end_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_abort_transaction(handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * is_shared(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * determine if the specified share is currently shared or not.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmis_shared(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *shared;
6185db853e024a486ff8837e6784dd290d866112dougm int result = 0; /* assume not */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm shared = sa_get_share_attr(share, "shared");
6185db853e024a486ff8837e6784dd290d866112dougm if (shared != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (strcmp(shared, "true") == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm result = 1;
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(shared);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (result);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * checksubdirgroup(group, newpath, strictness)
f345c0beb4c8f75cb54c2e070498e56febd468acdougm *
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * check all the specified newpath against all the paths in the
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * group. This is a helper function for checksubdir to make it easier
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * to also check ZFS subgroups.
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * The strictness values mean:
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * SA_CHECK_NORMAL == only check newpath against shares that are active
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * SA_CHECK_STRICT == check newpath against both active shares and those
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * stored in the repository
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
a99982a76d4cc12b1e9021e88531cf425d1e7369dougmchecksubdirgroup(sa_group_t group, char *newpath, int strictness)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t share;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm char *path;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm int issub = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm for (share = sa_get_share(group, NULL); share != NULL;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm share = sa_get_next_share(share)) {
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * The original behavior of share never checked
6185db853e024a486ff8837e6784dd290d866112dougm * against the permanent configuration
6185db853e024a486ff8837e6784dd290d866112dougm * (/etc/dfs/dfstab). PIT has a number of cases where
6185db853e024a486ff8837e6784dd290d866112dougm * it depends on this older behavior even though it
6185db853e024a486ff8837e6784dd290d866112dougm * could be considered incorrect. We may tighten this
6185db853e024a486ff8837e6784dd290d866112dougm * up in the future.
6185db853e024a486ff8837e6784dd290d866112dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (strictness == SA_CHECK_NORMAL && !is_shared(share))
57b448de658d89a2c88a001d58073c46ed2180f3dougm continue;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm path = sa_get_share_attr(share, "path");
f345c0beb4c8f75cb54c2e070498e56febd468acdougm /*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * If path is NULL, then a share is in the process of
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * construction or someone has modified the property
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * group inappropriately. It should be
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * ignored. issubdir() comes from the original share
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * implementation and does the difficult part of
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * checking subdirectories.
f345c0beb4c8f75cb54c2e070498e56febd468acdougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (path == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm continue;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (newpath != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm (strcmp(path, newpath) == 0 || issubdir(newpath, path) ||
57b448de658d89a2c88a001d58073c46ed2180f3dougm issubdir(path, newpath))) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm path = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm issub = SA_INVALID_PATH;
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm sa_free_attr_string(path);
6185db853e024a486ff8837e6784dd290d866112dougm path = NULL;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm }
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm return (issub);
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm}
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm/*
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * checksubdir(newpath, strictness)
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm *
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * checksubdir determines if the specified path (newpath) is a
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * subdirectory of another share. It calls checksubdirgroup() to do
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * the complicated work. The strictness parameter determines how
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * strict a check to make against the path. The strictness values
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * mean: SA_CHECK_NORMAL == only check newpath against shares that are
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * active SA_CHECK_STRICT == check newpath against both active shares
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm * and those * stored in the repository
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm */
a99982a76d4cc12b1e9021e88531cf425d1e7369dougmstatic int
549ec3fff108310966327d1dc9004551b63210b7dougmchecksubdir(sa_handle_t handle, char *newpath, int strictness)
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm{
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm sa_group_t group;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm int issub;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm char *path = NULL;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm
549ec3fff108310966327d1dc9004551b63210b7dougm for (issub = 0, group = sa_get_group(handle, NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm group != NULL && !issub; group = sa_get_next_group(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_group_is_zfs(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_group_t subgroup;
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (subgroup = sa_get_sub_group(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm subgroup != NULL && !issub;
57b448de658d89a2c88a001d58073c46ed2180f3dougm subgroup = sa_get_next_group(subgroup))
57b448de658d89a2c88a001d58073c46ed2180f3dougm issub = checksubdirgroup(subgroup, newpath,
57b448de658d89a2c88a001d58073c46ed2180f3dougm strictness);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm issub = checksubdirgroup(group, newpath, strictness);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (path != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(path);
6185db853e024a486ff8837e6784dd290d866112dougm return (issub);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * validpath(path, strictness)
6185db853e024a486ff8837e6784dd290d866112dougm * determine if the provided path is valid for a share. It shouldn't
6185db853e024a486ff8837e6784dd290d866112dougm * be a sub-dir of an already shared path or the parent directory of a
6185db853e024a486ff8837e6784dd290d866112dougm * share path.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
549ec3fff108310966327d1dc9004551b63210b7dougmvalidpath(sa_handle_t handle, char *path, int strictness)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int error = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm struct stat st;
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t share;
6185db853e024a486ff8837e6784dd290d866112dougm char *fstype;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (*path != '/')
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (SA_BAD_PATH);
57b448de658d89a2c88a001d58073c46ed2180f3dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (stat(path, &st) < 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm error = SA_NO_SUCH_PATH;
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm share = sa_find_share(handle, path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (share != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm error = SA_DUPLICATE_NAME;
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (error == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * check for special case with file system
57b448de658d89a2c88a001d58073c46ed2180f3dougm * that might have restrictions. For now, ZFS
57b448de658d89a2c88a001d58073c46ed2180f3dougm * is the only case since it has its own idea
57b448de658d89a2c88a001d58073c46ed2180f3dougm * of how to configure shares. We do this
57b448de658d89a2c88a001d58073c46ed2180f3dougm * before subdir checking since things like
57b448de658d89a2c88a001d58073c46ed2180f3dougm * ZFS will do that for us. This should also
57b448de658d89a2c88a001d58073c46ed2180f3dougm * be done via plugin interface.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm fstype = sa_fstype(path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (fstype != NULL && strcmp(fstype, "zfs") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_zfs_is_shared(handle, path))
57b448de658d89a2c88a001d58073c46ed2180f3dougm error = SA_INVALID_NAME;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (fstype != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_fstype(fstype);
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (error == SA_OK)
57b448de658d89a2c88a001d58073c46ed2180f3dougm error = checksubdir(handle, path, strictness);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (error);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * check to see if group/share is persistent.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmis_persistent(sa_group_t group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *type;
6185db853e024a486ff8837e6784dd290d866112dougm int persist = 1;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm type = sa_get_group_attr(group, "type");
6185db853e024a486ff8837e6784dd290d866112dougm if (type != NULL && strcmp(type, "transient") == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm persist = 0;
6185db853e024a486ff8837e6784dd290d866112dougm if (type != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(type);
6185db853e024a486ff8837e6784dd290d866112dougm return (persist);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_valid_group_name(name)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * check that the "name" contains only valid characters and otherwise
6185db853e024a486ff8837e6784dd290d866112dougm * fits the required naming conventions. Valid names must start with
6185db853e024a486ff8837e6784dd290d866112dougm * an alphabetic and the remainder may consist of only alphanumeric
6185db853e024a486ff8837e6784dd290d866112dougm * plus the '-' and '_' characters. This name limitation comes from
6185db853e024a486ff8837e6784dd290d866112dougm * inherent limitations in SMF.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_valid_group_name(char *name)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = 1;
6185db853e024a486ff8837e6784dd290d866112dougm ssize_t len;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (name != NULL && isalpha(*name)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char c;
57b448de658d89a2c88a001d58073c46ed2180f3dougm len = strlen(name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (len < (scf_max_name_len - sizeof ("group:"))) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (c = *name++; c != '\0' && ret != 0; c = *name++) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!isalnum(c) && c != '-' && c != '_')
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = 0;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
6185db853e024a486ff8837e6784dd290d866112dougm ret = 0;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = 0;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * is_zfs_group(group)
6185db853e024a486ff8837e6784dd290d866112dougm * Determine if the specified group is a ZFS sharenfs group
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmis_zfs_group(sa_group_t group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = 0;
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr parent;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *zfs;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (strcmp((char *)((xmlNodePtr)group)->name, "share") == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = (xmlNodePtr)sa_get_parent_group(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = (xmlNodePtr)group;
6185db853e024a486ff8837e6784dd290d866112dougm zfs = xmlGetProp(parent, (xmlChar *)"zfs");
6185db853e024a486ff8837e6784dd290d866112dougm if (zfs != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(zfs);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = 1;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_optionset_name(optionset, oname, len, id)
6185db853e024a486ff8837e6784dd290d866112dougm * return the SMF name for the optionset. If id is not NULL, it
6185db853e024a486ff8837e6784dd290d866112dougm * will have the GUID value for a share and should be used
6185db853e024a486ff8837e6784dd290d866112dougm * instead of the keyword "optionset" which is used for
6185db853e024a486ff8837e6784dd290d866112dougm * groups. If the optionset doesn't have a protocol type
6185db853e024a486ff8837e6784dd290d866112dougm * associated with it, "default" is used. This shouldn't happen
6185db853e024a486ff8837e6784dd290d866112dougm * at this point but may be desirable in the future if there are
6185db853e024a486ff8837e6784dd290d866112dougm * protocol independent properties added. The name is returned in
6185db853e024a486ff8837e6784dd290d866112dougm * oname.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmsa_optionset_name(sa_optionset_t optionset, char *oname, size_t len, char *id)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *proto;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (id == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = "optionset";
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm proto = sa_get_optionset_attr(optionset, "type");
6185db853e024a486ff8837e6784dd290d866112dougm len = snprintf(oname, len, "%s_%s", id, proto ? proto : "default");
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (proto != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(proto);
6185db853e024a486ff8837e6784dd290d866112dougm return (len);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_security_name(optionset, oname, len, id)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * return the SMF name for the security. If id is not NULL, it will
6185db853e024a486ff8837e6784dd290d866112dougm * have the GUID value for a share and should be used instead of the
6185db853e024a486ff8837e6784dd290d866112dougm * keyword "optionset" which is used for groups. If the optionset
6185db853e024a486ff8837e6784dd290d866112dougm * doesn't have a protocol type associated with it, "default" is
6185db853e024a486ff8837e6784dd290d866112dougm * used. This shouldn't happen at this point but may be desirable in
6185db853e024a486ff8837e6784dd290d866112dougm * the future if there are protocol independent properties added. The
6185db853e024a486ff8837e6784dd290d866112dougm * name is returned in oname. The security type is also encoded into
6185db853e024a486ff8837e6784dd290d866112dougm * the name. In the future, this wil *be handled a bit differently.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmsa_security_name(sa_security_t security, char *oname, size_t len, char *id)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *proto;
6185db853e024a486ff8837e6784dd290d866112dougm char *sectype;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (id == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = "optionset";
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm proto = sa_get_security_attr(security, "type");
6185db853e024a486ff8837e6784dd290d866112dougm sectype = sa_get_security_attr(security, "sectype");
57b448de658d89a2c88a001d58073c46ed2180f3dougm len = snprintf(oname, len, "%s_%s_%s", id, proto ? proto : "default",
57b448de658d89a2c88a001d58073c46ed2180f3dougm sectype ? sectype : "default");
6185db853e024a486ff8837e6784dd290d866112dougm if (proto != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(proto);
6185db853e024a486ff8837e6784dd290d866112dougm if (sectype != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(sectype);
6185db853e024a486ff8837e6784dd290d866112dougm return (len);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm/*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * verifydefgroupopts(handle)
57b448de658d89a2c88a001d58073c46ed2180f3dougm *
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Make sure a "default" group exists and has default protocols enabled.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougmstatic void
57b448de658d89a2c88a001d58073c46ed2180f3dougmverifydefgroupopts(sa_handle_t handle)
57b448de658d89a2c88a001d58073c46ed2180f3dougm{
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_group_t defgrp;
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_optionset_t opt;
57b448de658d89a2c88a001d58073c46ed2180f3dougm defgrp = sa_get_group(handle, "default");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (defgrp != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm opt = sa_get_optionset(defgrp, NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * NFS is the default for default group
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (opt == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm opt = sa_create_optionset(defgrp, "nfs");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm}
57b448de658d89a2c88a001d58073c46ed2180f3dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * sa_init(init_service)
6185db853e024a486ff8837e6784dd290d866112dougm * Initialize the API
6185db853e024a486ff8837e6784dd290d866112dougm * find all the shared objects
6185db853e024a486ff8837e6784dd290d866112dougm * init the tables with all objects
6185db853e024a486ff8837e6784dd290d866112dougm * read in the current configuration
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define GETPROP(prop) scf_simple_prop_next_astring(prop)
57b448de658d89a2c88a001d58073c46ed2180f3dougm#define CHECKTSTAMP(st, tval) stat(SA_LEGACY_DFSTAB, &st) >= 0 && \
57b448de658d89a2c88a001d58073c46ed2180f3dougm tval != TSTAMP(st.st_ctim)
57b448de658d89a2c88a001d58073c46ed2180f3dougm
549ec3fff108310966327d1dc9004551b63210b7dougmsa_handle_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_init(int init_service)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm struct stat st;
6185db853e024a486ff8837e6784dd290d866112dougm int legacy = 0;
6185db853e024a486ff8837e6784dd290d866112dougm uint64_t tval = 0;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm int lockfd;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm sigset_t old;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm int updatelegacy = B_FALSE;
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm scf_simple_prop_t *prop;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t handle;
549ec3fff108310966327d1dc9004551b63210b7dougm int err;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm handle = calloc(sizeof (struct sa_handle_impl), 1);
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm if (handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* get protocol specific structures */
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) proto_plugin_init();
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (init_service & SA_INIT_SHARE_API) {
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * initialize access into libzfs. We use this
57b448de658d89a2c88a001d58073c46ed2180f3dougm * when collecting info about ZFS datasets and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * shares.
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_zfs_init(handle) == B_FALSE) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm free(handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) proto_plugin_fini();
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * since we want to use SMF, initialize an svc handle
57b448de658d89a2c88a001d58073c46ed2180f3dougm * and find out what is there.
a99982a76d4cc12b1e9021e88531cf425d1e7369dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->scfhandle = sa_scf_init(handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (handle->scfhandle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Need to lock the extraction of the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * configuration if the dfstab file has
57b448de658d89a2c88a001d58073c46ed2180f3dougm * changed. Lock everything now and release if
57b448de658d89a2c88a001d58073c46ed2180f3dougm * not needed. Use a file that isn't being
57b448de658d89a2c88a001d58073c46ed2180f3dougm * manipulated by other parts of the system in
57b448de658d89a2c88a001d58073c46ed2180f3dougm * order to not interfere with locking. Using
57b448de658d89a2c88a001d58073c46ed2180f3dougm * dfstab doesn't work.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm sablocksigs(&old);
57b448de658d89a2c88a001d58073c46ed2180f3dougm lockfd = open(DFS_LOCK_FILE, O_RDWR);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (lockfd >= 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm extern int errno;
57b448de658d89a2c88a001d58073c46ed2180f3dougm errno = 0;
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) lockf(lockfd, F_LOCK, 0);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Check whether we are going to need
57b448de658d89a2c88a001d58073c46ed2180f3dougm * to merge any dfstab changes. This
57b448de658d89a2c88a001d58073c46ed2180f3dougm * is done by comparing the value of
57b448de658d89a2c88a001d58073c46ed2180f3dougm * legacy-timestamp with the current
57b448de658d89a2c88a001d58073c46ed2180f3dougm * st_ctim of the file. If they are
57b448de658d89a2c88a001d58073c46ed2180f3dougm * different, an update is needed and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the file must remain locked until
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the merge is done in order to
57b448de658d89a2c88a001d58073c46ed2180f3dougm * prevent multiple startups from
57b448de658d89a2c88a001d58073c46ed2180f3dougm * changing the SMF repository at the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * same time. The first to get the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * lock will make any changes before
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the others can read the repository.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm prop = scf_simple_prop_get
57b448de658d89a2c88a001d58073c46ed2180f3dougm (handle->scfhandle->handle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (const char *)SA_SVC_FMRI_BASE
57b448de658d89a2c88a001d58073c46ed2180f3dougm ":default", "operation",
57b448de658d89a2c88a001d58073c46ed2180f3dougm "legacy-timestamp");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (prop != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *i64;
57b448de658d89a2c88a001d58073c46ed2180f3dougm i64 = GETPROP(prop);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (i64 != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm tval = strtoull(i64,
57b448de658d89a2c88a001d58073c46ed2180f3dougm NULL, 0);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (CHECKTSTAMP(st, tval))
57b448de658d89a2c88a001d58073c46ed2180f3dougm updatelegacy = B_TRUE;
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_simple_prop_free(prop);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * We haven't set the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * timestamp before so do it.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm updatelegacy = B_TRUE;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (updatelegacy == B_FALSE) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* Don't need the lock anymore */
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) lockf(lockfd, F_ULOCK, 0);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) close(lockfd);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
1d1813a7a7c570174c2b6adc372045307b266117dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * It is essential that the document tree and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the internal list of roots to handles be
57b448de658d89a2c88a001d58073c46ed2180f3dougm * setup before anything that might try to
57b448de658d89a2c88a001d58073c46ed2180f3dougm * create a new object is called. The document
57b448de658d89a2c88a001d58073c46ed2180f3dougm * tree is the combination of handle->doc and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * handle->tree. This allows searches,
57b448de658d89a2c88a001d58073c46ed2180f3dougm * etc. when all you have is an object in the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * tree.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->doc = xmlNewDoc((xmlChar *)"1.0");
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->tree = xmlNewNode(NULL,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"sharecfg");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (handle->doc != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->tree != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlDocSetRootElement(handle->doc,
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->tree);
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = add_handle_for_root(handle->tree,
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (err == SA_OK)
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = sa_get_config(
57b448de658d89a2c88a001d58073c46ed2180f3dougm handle->scfhandle,
1d1813a7a7c570174c2b6adc372045307b266117dougm handle->tree, handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (handle->doc != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFreeDoc(handle->doc);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (handle->tree != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFreeNode(handle->tree);
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = SA_NO_MEMORY;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm saunblocksigs(&old);
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (err != SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * If we couldn't add the tree handle
57b448de658d89a2c88a001d58073c46ed2180f3dougm * to the list, then things are going
57b448de658d89a2c88a001d58073c46ed2180f3dougm * to fail badly. Might as well undo
57b448de658d89a2c88a001d58073c46ed2180f3dougm * everything now and fail the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * sa_init().
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_fini(handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
1d1813a7a7c570174c2b6adc372045307b266117dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (tval == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * first time so make sure
57b448de658d89a2c88a001d58073c46ed2180f3dougm * default is setup
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm verifydefgroupopts(handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (updatelegacy == B_TRUE) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sablocksigs(&old);
57b448de658d89a2c88a001d58073c46ed2180f3dougm getlegacyconfig((sa_handle_t)handle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm SA_LEGACY_DFSTAB, &handle->tree);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (stat(SA_LEGACY_DFSTAB, &st) >= 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm set_legacy_timestamp(handle->tree,
57b448de658d89a2c88a001d58073c46ed2180f3dougm SA_LEGACY_DFSTAB,
57b448de658d89a2c88a001d58073c46ed2180f3dougm TSTAMP(st.st_ctim));
57b448de658d89a2c88a001d58073c46ed2180f3dougm saunblocksigs(&old);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* Safe to unlock now to allow others to run */
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) lockf(lockfd, F_ULOCK, 0);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) close(lockfd);
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm legacy |= sa_get_zfs_shares(handle, "zfs");
57b448de658d89a2c88a001d58073c46ed2180f3dougm legacy |= gettransients(handle, &handle->tree);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm return ((sa_handle_t)handle);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * sa_fini(handle)
6185db853e024a486ff8837e6784dd290d866112dougm * Uninitialize the API structures including the configuration
1cea05af420c1992d793dc442f4e30c7269fc107dougm * data structures and ZFS related data.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
549ec3fff108310966327d1dc9004551b63210b7dougmsa_fini(sa_handle_t handle)
6185db853e024a486ff8837e6784dd290d866112dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle = (sa_handle_impl_t)handle;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle != NULL) {
549ec3fff108310966327d1dc9004551b63210b7dougm /*
549ec3fff108310966327d1dc9004551b63210b7dougm * Free the config trees and any other data structures
549ec3fff108310966327d1dc9004551b63210b7dougm * used in the handle.
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle->doc != NULL)
549ec3fff108310966327d1dc9004551b63210b7dougm xmlFreeDoc(impl_handle->doc);
549ec3fff108310966327d1dc9004551b63210b7dougm sa_scf_fini(impl_handle->scfhandle);
549ec3fff108310966327d1dc9004551b63210b7dougm sa_zfs_fini(impl_handle);
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm /* Remove and free the entry in the global list. */
549ec3fff108310966327d1dc9004551b63210b7dougm remove_handle_for_root(impl_handle->tree);
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm /* Make sure we free the handle */
549ec3fff108310966327d1dc9004551b63210b7dougm free(impl_handle);
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm /*
549ec3fff108310966327d1dc9004551b63210b7dougm * If this was the last handle to release, unload the
549ec3fff108310966327d1dc9004551b63210b7dougm * plugins that were loaded.
549ec3fff108310966327d1dc9004551b63210b7dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm if (sa_global_handles == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) proto_plugin_fini();
549ec3fff108310966327d1dc9004551b63210b7dougm
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_protocols(char **protocol)
6185db853e024a486ff8837e6784dd290d866112dougm * Get array of protocols that are supported
6185db853e024a486ff8837e6784dd290d866112dougm * Returns pointer to an allocated and NULL terminated
6185db853e024a486ff8837e6784dd290d866112dougm * array of strings. Caller must free.
6185db853e024a486ff8837e6784dd290d866112dougm * This really should be determined dynamically.
6185db853e024a486ff8837e6784dd290d866112dougm * If there aren't any defined, return -1.
6185db853e024a486ff8837e6784dd290d866112dougm * Use free() to return memory.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_protocols(char ***protocols)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int numproto = -1;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (protocols != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm struct sa_proto_plugin *plug;
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (numproto = 0, plug = sap_proto_list; plug != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm plug = plug->plugin_next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm numproto++;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm *protocols = calloc(numproto + 1, sizeof (char *));
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (*protocols != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm int ret = 0;
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (plug = sap_proto_list; plug != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm plug = plug->plugin_next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* faking for now */
57b448de658d89a2c88a001d58073c46ed2180f3dougm (*protocols)[ret++] =
57b448de658d89a2c88a001d58073c46ed2180f3dougm plug->plugin_ops->sa_protocol;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm numproto = -1;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (numproto);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * find_group_by_name(node, group)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * search the XML document subtree specified by node to find the group
6185db853e024a486ff8837e6784dd290d866112dougm * specified by group. Searching subtree allows subgroups to be
6185db853e024a486ff8837e6784dd290d866112dougm * searched for.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic xmlNodePtr
6185db853e024a486ff8837e6784dd290d866112dougmfind_group_by_name(xmlNodePtr node, xmlChar *group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *name = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = node->xmlChildrenNode; node != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"group") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* if no groupname, return the first found */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (group == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = xmlGetProp(node, (xmlChar *)"name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL && xmlStrcmp(name, group) == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (name != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(name);
6185db853e024a486ff8837e6784dd290d866112dougm return (node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_group(groupname)
6185db853e024a486ff8837e6784dd290d866112dougm * Return the "group" specified. If groupname is NULL,
6185db853e024a486ff8837e6784dd290d866112dougm * return the first group of the list of groups.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
549ec3fff108310966327d1dc9004551b63210b7dougmsa_get_group(sa_handle_t handle, char *groupname)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm char *subgroup = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm char *group = NULL;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle = (sa_handle_impl_t)handle;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle != NULL && impl_handle->tree != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (groupname != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = strdup(groupname);
a335142544257a204dde1839676940d0e0432d06dougm if (group != NULL) {
a335142544257a204dde1839676940d0e0432d06dougm subgroup = strchr(group, '/');
a335142544257a204dde1839676940d0e0432d06dougm if (subgroup != NULL)
a335142544257a204dde1839676940d0e0432d06dougm *subgroup++ = '\0';
a335142544257a204dde1839676940d0e0432d06dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
a335142544257a204dde1839676940d0e0432d06dougm /*
a335142544257a204dde1839676940d0e0432d06dougm * We want to find the, possibly, named group. If
a335142544257a204dde1839676940d0e0432d06dougm * group is not NULL, then lookup the name. If it is
a335142544257a204dde1839676940d0e0432d06dougm * NULL, we only do the find if groupname is also
a335142544257a204dde1839676940d0e0432d06dougm * NULL. This allows lookup of the "first" group in
a335142544257a204dde1839676940d0e0432d06dougm * the internal list.
a335142544257a204dde1839676940d0e0432d06dougm */
a335142544257a204dde1839676940d0e0432d06dougm if (group != NULL || groupname == NULL)
a335142544257a204dde1839676940d0e0432d06dougm node = find_group_by_name(impl_handle->tree,
a335142544257a204dde1839676940d0e0432d06dougm (xmlChar *)group);
a335142544257a204dde1839676940d0e0432d06dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* if a subgroup, find it before returning */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (subgroup != NULL && node != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = find_group_by_name(node, (xmlChar *)subgroup);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL && (char *)group != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_get_instance(impl_handle->scfhandle, (char *)group);
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm free(group);
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)(node));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_group(group)
6185db853e024a486ff8837e6784dd290d866112dougm * Return the "next" group after the specified group from
6185db853e024a486ff8837e6784dd290d866112dougm * the internal group list. NULL if there are no more.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_group(sa_group_t group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr ngroup = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (ngroup = ((xmlNodePtr)group)->next; ngroup != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm ngroup = ngroup->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(ngroup->name, (xmlChar *)"group") == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)ngroup);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_share(group, sharepath)
6185db853e024a486ff8837e6784dd290d866112dougm * Return the share object for the share specified. The share
6185db853e024a486ff8837e6784dd290d866112dougm * must be in the specified group. Return NULL if not found.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_share(sa_group_t group, char *sharepath)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *path;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * For future scalability, this should end up building a cache
6185db853e024a486ff8837e6784dd290d866112dougm * since it will get called regularly by the mountd and info
6185db853e024a486ff8837e6784dd290d866112dougm * services.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (node = ((xmlNodePtr)group)->children; node != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"share") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sharepath == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* is it the correct share? */
57b448de658d89a2c88a001d58073c46ed2180f3dougm path = xmlGetProp(node,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (path != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(path,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)sharepath) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_share_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_share(share)
6185db853e024a486ff8837e6784dd290d866112dougm * Return the next share following the specified share
6185db853e024a486ff8837e6784dd290d866112dougm * from the internal list of shares. Returns NULL if there
6185db853e024a486ff8837e6784dd290d866112dougm * are no more shares. The list is relative to the same
6185db853e024a486ff8837e6784dd290d866112dougm * group.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_share(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (share != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (node = ((xmlNodePtr)share)->next; node != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"share") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_share_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * _sa_get_child_node(node, type)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * find the child node of the specified node that has "type". This is
6185db853e024a486ff8837e6784dd290d866112dougm * used to implement several internal functions.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic xmlNodePtr
6185db853e024a486ff8837e6784dd290d866112dougm_sa_get_child_node(xmlNodePtr node, xmlChar *type)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr child;
6185db853e024a486ff8837e6784dd290d866112dougm for (child = node->xmlChildrenNode; child != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm child = child->next)
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(child->name, type) == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (child);
6185db853e024a486ff8837e6784dd290d866112dougm return ((xmlNodePtr)NULL);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * find_share(group, path)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Search all the shares in the specified group for one that has the
6185db853e024a486ff8837e6784dd290d866112dougm * specified path.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic sa_share_t
6185db853e024a486ff8837e6784dd290d866112dougmfind_share(sa_group_t group, char *sharepath)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t share;
6185db853e024a486ff8837e6784dd290d866112dougm char *path;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (share = sa_get_share(group, NULL); share != NULL;
6185db853e024a486ff8837e6784dd290d866112dougm share = sa_get_next_share(share)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm path = sa_get_share_attr(share, "path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (path != NULL && strcmp(path, sharepath) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(path);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (path != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(path);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (share);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_sub_group(group)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the first sub-group of group. The sa_get_next_group() function
6185db853e024a486ff8837e6784dd290d866112dougm * can be used to get the rest. This is currently only used for ZFS
6185db853e024a486ff8837e6784dd290d866112dougm * sub-groups but could be used to implement a more general mechanism.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_sub_group(sa_group_t group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)_sa_get_child_node((xmlNodePtr)group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"group"));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_find_share(sharepath)
6185db853e024a486ff8837e6784dd290d866112dougm * Finds a share regardless of group. In the future, this
6185db853e024a486ff8837e6784dd290d866112dougm * function should utilize a cache and hash table of some kind.
6185db853e024a486ff8837e6784dd290d866112dougm * The current assumption is that a path will only be shared
6185db853e024a486ff8837e6784dd290d866112dougm * once. In the future, this may change as implementation of
6185db853e024a486ff8837e6784dd290d866112dougm * resource names comes into being.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
549ec3fff108310966327d1dc9004551b63210b7dougmsa_find_share(sa_handle_t handle, char *sharepath)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t zgroup;
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t share = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm int done = 0;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm for (group = sa_get_group(handle, NULL); group != NULL && !done;
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = sa_get_next_group(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (is_zfs_group(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (zgroup =
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_group_t)_sa_get_child_node((xmlNodePtr)group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"group");
57b448de658d89a2c88a001d58073c46ed2180f3dougm zgroup != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm zgroup = sa_get_next_group(zgroup)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm share = find_share(zgroup, sharepath);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (share != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm share = find_share(group, sharepath);
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (share != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (share);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * sa_check_path(group, path, strictness)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * check that path is a valid path relative to the group. Currently,
6185db853e024a486ff8837e6784dd290d866112dougm * we are ignoring the group and checking only the NFS rules. Later,
6185db853e024a486ff8837e6784dd290d866112dougm * we may want to use the group to then check against the protocols
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * enabled on the group. The strictness values mean:
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * SA_CHECK_NORMAL == only check newpath against shares that are active
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * SA_CHECK_STRICT == check newpath against both active shares and those
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * stored in the repository
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
f345c0beb4c8f75cb54c2e070498e56febd468acdougmsa_check_path(sa_group_t group, char *path, int strictness)
6185db853e024a486ff8837e6784dd290d866112dougm{
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_t handle;
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm handle = sa_find_group_handle(group);
549ec3fff108310966327d1dc9004551b63210b7dougm return (validpath(handle, path, strictness));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * _sa_add_share(group, sharepath, persist, *error)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * common code for all types of add_share. sa_add_share() is the
6185db853e024a486ff8837e6784dd290d866112dougm * public API, we also need to be able to do this when parsing legacy
6185db853e024a486ff8837e6784dd290d866112dougm * files and construction of the internal configuration while
6185db853e024a486ff8837e6784dd290d866112dougm * extracting config info from SMF.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
6185db853e024a486ff8837e6784dd290d866112dougm_sa_add_share(sa_group_t group, char *sharepath, int persist, int *error)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm int err;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm err = SA_OK; /* assume success */
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild((xmlNodePtr)group, NULL, (xmlChar *)"share", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"path", (xmlChar *)sharepath);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"type",
57b448de658d89a2c88a001d58073c46ed2180f3dougm persist ? (xmlChar *)"persist" : (xmlChar *)"transient");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (persist != SA_SHARE_TRANSIENT) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * persistent shares come in two flavors: SMF and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * ZFS. Sort this one out based on target group and
57b448de658d89a2c88a001d58073c46ed2180f3dougm * path type. Currently, only NFS is supported in the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * ZFS group and it is always on.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_group_is_zfs(group) &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_path_is_zfs(sharepath)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = sa_zfs_set_sharenfs(group, sharepath, 1);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle =
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_handle_impl_t)sa_find_group_handle(
57b448de658d89a2c88a001d58073c46ed2180f3dougm group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = sa_commit_share(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_share_t)node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (err == SA_NO_PERMISSION && persist & SA_SHARE_PARSER) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* called by the dfstab parser so could be a show */
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = SA_OK;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (err != SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * we couldn't commit to the repository so undo
57b448de658d89a2c88a001d58073c46ed2180f3dougm * our internal state to reflect reality.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlUnlinkNode(node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFreeNode(node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = SA_NO_MEMORY;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (error != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm *error = err;
6185db853e024a486ff8837e6784dd290d866112dougm return (node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_add_share(group, sharepath, persist, *error)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Add a new share object to the specified group. The share will
6185db853e024a486ff8837e6784dd290d866112dougm * have the specified sharepath and will only be constructed if
6185db853e024a486ff8837e6784dd290d866112dougm * it is a valid path to be shared. NULL is returned on error
6185db853e024a486ff8837e6784dd290d866112dougm * and a detailed error value will be returned via the error
6185db853e024a486ff8837e6784dd290d866112dougm * pointer.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_add_share(sa_group_t group, char *sharepath, int persist, int *error)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t dup;
f345c0beb4c8f75cb54c2e070498e56febd468acdougm int strictness = SA_CHECK_NORMAL;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_t handle;
f345c0beb4c8f75cb54c2e070498e56febd468acdougm
f345c0beb4c8f75cb54c2e070498e56febd468acdougm /*
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * If the share is to be permanent, use strict checking so a
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * bad config doesn't get created. Transient shares only need
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * to check against the currently active
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * shares. SA_SHARE_PARSER is a modifier used internally to
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * indicate that we are being called by the dfstab parser and
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * that we need strict checking in all cases. Normally persist
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * is in integer value but SA_SHARE_PARSER may be or'd into
f345c0beb4c8f75cb54c2e070498e56febd468acdougm * it as an override.
f345c0beb4c8f75cb54c2e070498e56febd468acdougm */
f345c0beb4c8f75cb54c2e070498e56febd468acdougm if (persist & SA_SHARE_PARSER || persist == SA_SHARE_PERMANENT)
57b448de658d89a2c88a001d58073c46ed2180f3dougm strictness = SA_CHECK_STRICT;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm handle = sa_find_group_handle(group);
549ec3fff108310966327d1dc9004551b63210b7dougm
549ec3fff108310966327d1dc9004551b63210b7dougm if ((dup = sa_find_share(handle, sharepath)) == NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm (*error = sa_check_path(group, sharepath, strictness)) == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = _sa_add_share(group, sharepath, persist, error);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (dup != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm *error = SA_DUPLICATE_NAME;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_share_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_enable_share(share, protocol)
6185db853e024a486ff8837e6784dd290d866112dougm * Enable the specified share to the specified protocol.
6185db853e024a486ff8837e6784dd290d866112dougm * If protocol is NULL, then all protocols.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_enable_share(sa_share_t share, char *protocol)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *sharepath;
6185db853e024a486ff8837e6784dd290d866112dougm struct stat st;
6185db853e024a486ff8837e6784dd290d866112dougm int err = 0;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm sharepath = sa_get_share_attr(share, "path");
6185db853e024a486ff8837e6784dd290d866112dougm if (stat(sharepath, &st) < 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = SA_NO_SUCH_PATH;
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* tell the server about the share */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (protocol != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* lookup protocol specific handler */
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = sa_proto_share(protocol, share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (err == SA_OK)
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_set_share_attr(share, "shared",
57b448de658d89a2c88a001d58073c46ed2180f3dougm "true");
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Tell all protocols. Only NFS for now but
57b448de658d89a2c88a001d58073c46ed2180f3dougm * SMB is coming.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm err = sa_proto_share("nfs", share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_set_share_attr(share, "shared", "true");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (sharepath != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(sharepath);
6185db853e024a486ff8837e6784dd290d866112dougm return (err);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_disable_share(share, protocol)
6185db853e024a486ff8837e6784dd290d866112dougm * Disable the specified share to the specified protocol.
6185db853e024a486ff8837e6784dd290d866112dougm * If protocol is NULL, then all protocols.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_disable_share(sa_share_t share, char *protocol)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *path;
6185db853e024a486ff8837e6784dd290d866112dougm char *shared;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm path = sa_get_share_attr(share, "path");
6185db853e024a486ff8837e6784dd290d866112dougm shared = sa_get_share_attr(share, "shared");
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (protocol != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_proto_unshare(protocol, path);
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* need to do all protocols */
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_proto_unshare("nfs", path);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (ret == SA_OK)
6185db853e024a486ff8837e6784dd290d866112dougm (void) sa_set_share_attr(share, "shared", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm if (path != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(path);
6185db853e024a486ff8837e6784dd290d866112dougm if (shared != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(shared);
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_remove_share(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * remove the specified share from its containing group.
6185db853e024a486ff8837e6784dd290d866112dougm * Remove from the SMF or ZFS configuration space.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_remove_share(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm char *type;
6185db853e024a486ff8837e6784dd290d866112dougm int transient = 0;
6185db853e024a486ff8837e6784dd290d866112dougm char *groupname;
6185db853e024a486ff8837e6784dd290d866112dougm char *zfs;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm type = sa_get_share_attr(share, "type");
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_parent_group(share);
6185db853e024a486ff8837e6784dd290d866112dougm zfs = sa_get_group_attr(group, "zfs");
6185db853e024a486ff8837e6784dd290d866112dougm groupname = sa_get_group_attr(group, "name");
6185db853e024a486ff8837e6784dd290d866112dougm if (type != NULL && strcmp(type, "persist") != 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm transient = 1;
6185db853e024a486ff8837e6784dd290d866112dougm if (type != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(type);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /* remove the node from its group then free the memory */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * need to test if "busy"
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm /* only do SMF action if permanent */
6185db853e024a486ff8837e6784dd290d866112dougm if (!transient || zfs != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* remove from legacy dfstab as well as possible SMF */
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_legacy(share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!sa_group_is_zfs(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_share(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *sharepath = sa_get_share_attr(share,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "path");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sharepath != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_zfs_set_sharenfs(group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm sharepath, 0);
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(sharepath);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (groupname != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(groupname);
6185db853e024a486ff8837e6784dd290d866112dougm if (zfs != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(zfs);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode((xmlNodePtr)share);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode((xmlNodePtr)share);
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_move_share(group, share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * move the specified share to the specified group. Update SMF
6185db853e024a486ff8837e6784dd290d866112dougm * appropriately.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_move_share(sa_group_t group, sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t oldgroup;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /* remove the node from its group then free the memory */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm oldgroup = sa_get_parent_group(share);
6185db853e024a486ff8837e6784dd290d866112dougm if (oldgroup != group) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlUnlinkNode((xmlNodePtr)share);
6185db853e024a486ff8837e6784dd290d866112dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * now that the share isn't in its old group, add to
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the new one
6185db853e024a486ff8837e6784dd290d866112dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlAddChild((xmlNodePtr)group, (xmlNodePtr)share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* need to deal with SMF */
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * need to remove from old group first and then add to
57b448de658d89a2c88a001d58073c46ed2180f3dougm * new group. Ideally, we would do the other order but
57b448de658d89a2c88a001d58073c46ed2180f3dougm * need to avoid having the share in two groups at the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * same time.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_share(impl_handle->scfhandle, oldgroup,
57b448de658d89a2c88a001d58073c46ed2180f3dougm share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_commit_share(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm group, share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_parent_group(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the containg group for the share. If a group was actually
6185db853e024a486ff8837e6784dd290d866112dougm * passed in, we don't want a parent so return NULL.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_parent_group(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm if (share != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = ((xmlNodePtr)share)->parent;
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * make sure parent is a group and not sharecfg since
6185db853e024a486ff8837e6784dd290d866112dougm * we may be cheating and passing in a group.
6185db853e024a486ff8837e6784dd290d866112dougm * Eventually, groups of groups might come into being.
6185db853e024a486ff8837e6784dd290d866112dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (node == NULL ||
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(node->name, (xmlChar *)"sharecfg") == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
549ec3fff108310966327d1dc9004551b63210b7dougm * _sa_create_group(impl_handle, groupname)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a group in the document. The caller will need to deal with
6185db853e024a486ff8837e6784dd290d866112dougm * configuration store and activation.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
549ec3fff108310966327d1dc9004551b63210b7dougm_sa_create_group(sa_handle_impl_t impl_handle, char *groupname)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (sa_valid_group_name(groupname)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild(impl_handle->tree, NULL, (xmlChar *)"group",
57b448de658d89a2c88a001d58073c46ed2180f3dougm NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"name",
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"state",
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"enabled");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * _sa_create_zfs_group(group, groupname)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a ZFS subgroup under the specified group. This may
6185db853e024a486ff8837e6784dd290d866112dougm * eventually form the basis of general sub-groups, but is currently
6185db853e024a486ff8837e6784dd290d866112dougm * restricted to ZFS.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
6185db853e024a486ff8837e6784dd290d866112dougm_sa_create_zfs_group(sa_group_t group, char *groupname)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild((xmlNodePtr)group, NULL, (xmlChar *)"group", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm xmlSetProp(node, (xmlChar *)"name", (xmlChar *)groupname);
6185db853e024a486ff8837e6784dd290d866112dougm xmlSetProp(node, (xmlChar *)"state", (xmlChar *)"enabled");
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_create_group(groupname, *error)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a new group with groupname. Need to validate that it is a
6185db853e024a486ff8837e6784dd290d866112dougm * legal name for SMF and the construct the SMF service instance of
6185db853e024a486ff8837e6784dd290d866112dougm * svc:/network/shares/group to implement the group. All necessary
6185db853e024a486ff8837e6784dd290d866112dougm * operational properties must be added to the group at this point
6185db853e024a486ff8837e6784dd290d866112dougm * (via the SMF transaction model).
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_group_t
549ec3fff108310966327d1dc9004551b63210b7dougmsa_create_group(sa_handle_t handle, char *groupname, int *error)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm int ret;
57b448de658d89a2c88a001d58073c46ed2180f3dougm char rbacstr[SA_STRSIZE];
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle = (sa_handle_impl_t)handle;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle == NULL || impl_handle->scfhandle == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm goto err;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm group = sa_get_group(handle, groupname);
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_DUPLICATE_NAME;
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_valid_group_name(groupname)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild(impl_handle->tree, NULL,
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"group", NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"name",
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* default to the group being enabled */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"state",
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlChar *)"enabled");
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_create_instance(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "operation");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "state", "enabled");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_end_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_abort_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* initialize the RBAC strings */
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "general");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) snprintf(rbacstr,
57b448de658d89a2c88a001d58073c46ed2180f3dougm sizeof (rbacstr), "%s.%s",
57b448de658d89a2c88a001d58073c46ed2180f3dougm SA_RBAC_MANAGE, groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
6185db853e024a486ff8837e6784dd290d866112dougm "action_authorization",
6185db853e024a486ff8837e6784dd290d866112dougm rbacstr);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) snprintf(rbacstr,
57b448de658d89a2c88a001d58073c46ed2180f3dougm sizeof (rbacstr), "%s.%s",
57b448de658d89a2c88a001d58073c46ed2180f3dougm SA_RBAC_VALUE, groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
6185db853e024a486ff8837e6784dd290d866112dougm "value_authorization",
6185db853e024a486ff8837e6784dd290d866112dougm rbacstr);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_end_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_abort_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret != SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Couldn't commit the group
57b448de658d89a2c88a001d58073c46ed2180f3dougm * so we need to undo
57b448de658d89a2c88a001d58073c46ed2180f3dougm * internally.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlUnlinkNode(node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFreeNode(node);
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_NO_MEMORY;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_INVALID_NAME;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougmerr:
6185db853e024a486ff8837e6784dd290d866112dougm if (error != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm *error = ret;
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_remove_group(group)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Remove the specified group. This deletes from the SMF repository.
6185db853e024a486ff8837e6784dd290d866112dougm * All property groups and properties are removed.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_remove_group(sa_group_t group)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *name;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = sa_get_group_attr(group, "name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_instance(impl_handle->scfhandle, name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlUnlinkNode((xmlNodePtr)group); /* make sure unlinked */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFreeNode((xmlNodePtr)group); /* now it is gone */
549ec3fff108310966327d1dc9004551b63210b7dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_update_config()
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Used to update legacy files that need to be updated in bulk
6185db853e024a486ff8837e6784dd290d866112dougm * Currently, this is a placeholder and will go away in a future
6185db853e024a486ff8837e6784dd290d866112dougm * release.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
549ec3fff108310966327d1dc9004551b63210b7dougmsa_update_config(sa_handle_t handle)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * do legacy files first so we can tell when they change.
6185db853e024a486ff8837e6784dd290d866112dougm * This will go away when we start updating individual records
6185db853e024a486ff8837e6784dd290d866112dougm * rather than the whole file.
6185db853e024a486ff8837e6784dd290d866112dougm */
549ec3fff108310966327d1dc9004551b63210b7dougm update_legacy_config(handle);
6185db853e024a486ff8837e6784dd290d866112dougm return (SA_OK);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * get_node_attr(node, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the speficied tag(attribute) if it exists on the node. This is
6185db853e024a486ff8837e6784dd290d866112dougm * used internally by a number of attribute oriented functions.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic char *
6185db853e024a486ff8837e6784dd290d866112dougmget_node_attr(void *nodehdl, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = (xmlNodePtr)nodehdl;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *name = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (node != NULL)
6185db853e024a486ff8837e6784dd290d866112dougm name = xmlGetProp(node, (xmlChar *)tag);
6185db853e024a486ff8837e6784dd290d866112dougm return ((char *)name);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * get_node_attr(node, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the speficied tag(attribute) to the specified value This is
6185db853e024a486ff8837e6784dd290d866112dougm * used internally by a number of attribute oriented functions. It
6185db853e024a486ff8837e6784dd290d866112dougm * doesn't update the repository, only the internal document state.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
6185db853e024a486ff8837e6784dd290d866112dougmset_node_attr(void *nodehdl, char *tag, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = (xmlNodePtr)nodehdl;
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL && tag != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL)
6185db853e024a486ff8837e6784dd290d866112dougm xmlSetProp(node, (xmlChar *)tag, (xmlChar *)value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnsetProp(node, (xmlChar *)tag);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_group_attr(group, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the specied attribute, if defined, for the group.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_group_attr(sa_group_t group, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (get_node_attr((void *)group, tag));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_group_attr(group, tag, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * set the specified tag/attribute on the group using value as its
6185db853e024a486ff8837e6784dd290d866112dougm * value.
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * This will result in setting the property in the SMF repository as
6185db853e024a486ff8837e6784dd290d866112dougm * well as in the internal document.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_group_attr(sa_group_t group, char *tag, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret;
6185db853e024a486ff8837e6784dd290d866112dougm char *groupname;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle;
6185db853e024a486ff8837e6784dd290d866112dougm
549ec3fff108310966327d1dc9004551b63210b7dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
549ec3fff108310966327d1dc9004551b63210b7dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname = sa_get_group_attr(group, "name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_get_instance(impl_handle->scfhandle, groupname);
549ec3fff108310966327d1dc9004551b63210b7dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm set_node_attr((void *)group, tag, value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "operation");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm tag, value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK)
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_end_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_abort_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (groupname != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(groupname);
549ec3fff108310966327d1dc9004551b63210b7dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_share_attr(share, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the value of the tag/attribute set on the specified
6185db853e024a486ff8837e6784dd290d866112dougm * share. Returns NULL if the tag doesn't exist.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_share_attr(sa_share_t share, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (get_node_attr((void *)share, tag));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_resource(group, resource)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Search all the shares in the speified group for a share with a
6185db853e024a486ff8837e6784dd290d866112dougm * resource name matching the one specified.
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * In the future, it may be advantageous to allow group to be NULL and
6185db853e024a486ff8837e6784dd290d866112dougm * search all groups but that isn't needed at present.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_share_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_resource(sa_group_t group, char *resource)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t share = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm char *name = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (resource != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm for (share = sa_get_share(group, NULL); share != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm share = sa_get_next_share(share)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = sa_get_share_attr(share, "resource");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (strcmp(name, resource) == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(name);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_share_t)share);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * _sa_set_share_description(share, description)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Add a description tag with text contents to the specified share.
6185db853e024a486ff8837e6784dd290d866112dougm * A separate XML tag is used rather than a property.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmxmlNodePtr
6185db853e024a486ff8837e6784dd290d866112dougm_sa_set_share_description(sa_share_t share, char *content)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = xmlNewChild((xmlNodePtr)share, NULL, (xmlChar *)"description",
57b448de658d89a2c88a001d58073c46ed2180f3dougm NULL);
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodeSetContent(node, (xmlChar *)content);
6185db853e024a486ff8837e6784dd290d866112dougm return (node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_share_attr(share, tag, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the share attribute specified by tag to the specified value. In
6185db853e024a486ff8837e6784dd290d866112dougm * the case of "resource", enforce a no duplicates in a group rule. If
6185db853e024a486ff8837e6784dd290d866112dougm * the share is not transient, commit the changes to the repository
6185db853e024a486ff8837e6784dd290d866112dougm * else just update the share internally.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_share_attr(sa_share_t share, char *tag, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm sa_share_t resource;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_parent_group(share);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * There are some attributes that may have specific
6185db853e024a486ff8837e6784dd290d866112dougm * restrictions on them. Initially, only "resource" has
6185db853e024a486ff8837e6784dd290d866112dougm * special meaning that needs to be checked. Only one instance
6185db853e024a486ff8837e6784dd290d866112dougm * of a resource name may exist within a group.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (strcmp(tag, "resource") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm resource = sa_get_resource(group, value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (resource != share && resource != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_DUPLICATE_NAME;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm set_node_attr((void *)share, tag, value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *type;
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* we can probably optimize this some */
57b448de658d89a2c88a001d58073c46ed2180f3dougm type = sa_get_share_attr(share, "type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (type == NULL || strcmp(type, "transient") != 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle =
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_handle_impl_t)sa_find_group_handle(
57b448de658d89a2c88a001d58073c46ed2180f3dougm group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_commit_share(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (type != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(type);
549ec3fff108310966327d1dc9004551b63210b7dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_property_attr(prop, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the value of the specified property attribute. Standard
6185db853e024a486ff8837e6784dd290d866112dougm * attributes are "type" and "value".
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_property_attr(sa_property_t prop, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (get_node_attr((void *)prop, tag));
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_optionset_attr(prop, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the value of the specified property attribute. Standard
6185db853e024a486ff8837e6784dd290d866112dougm * attribute is "type".
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_optionset_attr(sa_property_t optionset, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (get_node_attr((void *)optionset, tag));
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_optionset_attr(optionset, tag, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the specified attribute(tag) to the specified value on the
6185db853e024a486ff8837e6784dd290d866112dougm * optionset.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_optionset_attr(sa_group_t optionset, char *tag, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm set_node_attr((void *)optionset, tag, value);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_free_attr_string(string)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Free the string that was returned in one of the sa_get_*_attr()
6185db853e024a486ff8837e6784dd290d866112dougm * functions.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
6185db853e024a486ff8837e6784dd290d866112dougmsa_free_attr_string(char *string)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlFree((xmlChar *)string);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_optionset(group, proto)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the optionset, if it exists, that is associated with the
6185db853e024a486ff8837e6784dd290d866112dougm * specified protocol.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_optionset_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_optionset(void *group, char *proto)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *value = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)group)->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"optionset") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = xmlGetProp(node, (xmlChar *)"type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (proto != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(value, (xmlChar *)proto) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (value != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_optionset_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_optionset(optionset)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the next optionset in the group. NULL if this was the last.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_optionset_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_optionset(sa_optionset_t optionset)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)optionset)->next; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"optionset") == 0) {
6185db853e024a486ff8837e6784dd290d866112dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_optionset_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_security(group, sectype, proto)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the security optionset. The internal name is a hold over
6185db853e024a486ff8837e6784dd290d866112dougm * from the implementation and will be changed before the API is
6185db853e024a486ff8837e6784dd290d866112dougm * finalized. This is really a named optionset that can be negotiated
6185db853e024a486ff8837e6784dd290d866112dougm * as a group of properties (like NFS security options).
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_security_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_security(sa_group_t group, char *sectype, char *proto)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *value = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)group)->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"security") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (proto != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = xmlGetProp(node, (xmlChar *)"type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value == NULL ||
57b448de658d89a2c88a001d58073c46ed2180f3dougm (value != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(value, (xmlChar *)proto) != 0)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* it doesn't match so continue */
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm continue;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* potential match */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sectype != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = xmlGetProp(node, (xmlChar *)"sectype");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(value, (xmlChar *)sectype) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (value != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_security_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_security(security)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the next security optionset if one exists.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_security_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_security(sa_security_t security)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)security)->next; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"security") == 0) {
6185db853e024a486ff8837e6784dd290d866112dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_security_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_property(optionset, prop)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the property object with the name specified in prop from the
6185db853e024a486ff8837e6784dd290d866112dougm * optionset.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_property_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_property(sa_optionset_t optionset, char *prop)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = (xmlNodePtr)optionset;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *value = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (optionset == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (NULL);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = node->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"option") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (prop == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = xmlGetProp(node, (xmlChar *)"type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcmp(value, (xmlChar *)prop) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (value != NULL)
6185db853e024a486ff8837e6784dd290d866112dougm xmlFree(value);
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL && xmlStrcmp(node->name, (xmlChar *)"option") != 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * avoid a non option node -- it is possible to be a
57b448de658d89a2c88a001d58073c46ed2180f3dougm * text node
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_property_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_property(property)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the next property following the specified property. NULL if
6185db853e024a486ff8837e6784dd290d866112dougm * this was the last.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_property_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_property(sa_property_t property)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)property)->next; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"option") == 0) {
6185db853e024a486ff8837e6784dd290d866112dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_property_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_share_description(share, content)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the description of share to content.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_share_description(sa_share_t share, char *content)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)share)->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"description") == 0) {
6185db853e024a486ff8837e6784dd290d866112dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_parent_group(share);
6185db853e024a486ff8837e6784dd290d866112dougm /* no existing description but want to add */
6185db853e024a486ff8837e6784dd290d866112dougm if (node == NULL && content != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm /* add a description */
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = _sa_set_share_description(share, content);
6185db853e024a486ff8837e6784dd290d866112dougm } else if (node != NULL && content != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm /* update a description */
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodeSetContent(node, (xmlChar *)content);
6185db853e024a486ff8837e6784dd290d866112dougm } else if (node != NULL && content == NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm /* remove an existing description */
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode(node);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode(node);
6185db853e024a486ff8837e6784dd290d866112dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm if (group != NULL && is_persistent((sa_group_t)share)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_commit_share(impl_handle->scfhandle, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * fixproblemchars(string)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * don't want any newline or tab characters in the text since these
6185db853e024a486ff8837e6784dd290d866112dougm * could break display of data and legacy file formats.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmstatic void
6185db853e024a486ff8837e6784dd290d866112dougmfixproblemchars(char *str)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int c;
6185db853e024a486ff8837e6784dd290d866112dougm for (c = *str; c != '\0'; c = *++str) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (c == '\t' || c == '\n')
57b448de658d89a2c88a001d58073c46ed2180f3dougm *str = ' ';
57b448de658d89a2c88a001d58073c46ed2180f3dougm else if (c == '"')
57b448de658d89a2c88a001d58073c46ed2180f3dougm *str = '\'';
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_share_description(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the description text for the specified share if it
6185db853e024a486ff8837e6784dd290d866112dougm * exists. NULL if no description exists.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_share_description(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *description = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)share)->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"description") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm description = xmlNodeGetContent((xmlNodePtr)share);
57b448de658d89a2c88a001d58073c46ed2180f3dougm fixproblemchars((char *)description);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((char *)description);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_free(share_description(description)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Free the description string.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
6185db853e024a486ff8837e6784dd290d866112dougmsa_free_share_description(char *description)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlFree((xmlChar *)description);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_create_optionset(group, proto)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create an optionset for the specified protocol in the specied
6185db853e024a486ff8837e6784dd290d866112dougm * group. This is manifested as a property group within SMF.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_optionset_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_create_optionset(sa_group_t group, char *proto)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_optionset_t optionset;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t parent = group;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm optionset = sa_get_optionset(group, proto);
6185db853e024a486ff8837e6784dd290d866112dougm if (optionset != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm /* can't have a duplicate protocol */
57b448de658d89a2c88a001d58073c46ed2180f3dougm optionset = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm optionset = (sa_optionset_t)xmlNewChild((xmlNodePtr)group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm NULL, (xmlChar *)"optionset", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * only put to repository if on a group and we were
6185db853e024a486ff8837e6784dd290d866112dougm * able to create an optionset.
6185db853e024a486ff8837e6784dd290d866112dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (optionset != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char oname[SA_STRSIZE];
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *groupname;
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *id = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_is_share(group))
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = sa_get_parent_group((sa_share_t)group);
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_set_optionset_attr(optionset, "type", proto);
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_is_share(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = sa_get_share_attr((sa_share_t)group, "id");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_optionset_name(optionset, oname,
57b448de658d89a2c88a001d58073c46ed2180f3dougm sizeof (oname), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname = sa_get_group_attr(parent, "name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (groupname != NULL && is_persistent(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm assert(impl_handle != NULL);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_get_instance(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_create_pgroup(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, oname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (groupname != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (id != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(id);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (optionset);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_property_parent(property)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Given a property, return the object it is a property of. This will
6185db853e024a486ff8837e6784dd290d866112dougm * be an optionset of some type.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic sa_optionset_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_property_parent(sa_property_t property)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (property != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = ((xmlNodePtr)property)->parent;
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_optionset_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_optionset_parent(optionset)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the parent of the specified optionset. This could be a group
6185db853e024a486ff8837e6784dd290d866112dougm * or a share.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic sa_group_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_optionset_parent(sa_optionset_t optionset)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (optionset != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = ((xmlNodePtr)optionset)->parent;
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_group_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * zfs_needs_update(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * In order to avoid making multiple updates to a ZFS share when
6185db853e024a486ff8837e6784dd290d866112dougm * setting properties, the share attribute "changed" will be set to
6185db853e024a486ff8837e6784dd290d866112dougm * true when a property is added or modifed. When done adding
6185db853e024a486ff8837e6784dd290d866112dougm * properties, we can then detect that an update is needed. We then
6185db853e024a486ff8837e6784dd290d866112dougm * clear the state here to detect additional changes.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmzfs_needs_update(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *attr;
6185db853e024a486ff8837e6784dd290d866112dougm int result = 0;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm attr = sa_get_share_attr(share, "changed");
6185db853e024a486ff8837e6784dd290d866112dougm if (attr != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(attr);
6185db853e024a486ff8837e6784dd290d866112dougm result = 1;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm set_node_attr((void *)share, "changed", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm return (result);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * zfs_set_update(share)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the changed attribute of the share to true.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic void
6185db853e024a486ff8837e6784dd290d866112dougmzfs_set_update(sa_share_t share)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm set_node_attr((void *)share, "changed", "true");
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_commit_properties(optionset, clear)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Check if SMF or ZFS config and either update or abort the pending
6185db853e024a486ff8837e6784dd290d866112dougm * changes.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_commit_properties(sa_optionset_t optionset, int clear)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t parent;
6185db853e024a486ff8837e6784dd290d866112dougm int zfs = 0;
6185db853e024a486ff8837e6784dd290d866112dougm int needsupdate = 0;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_optionset_parent(optionset);
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL && (sa_is_share(group) || is_zfs_group(group))) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /* only update ZFS if on a share */
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = sa_get_parent_group(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm zfs++;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (parent != NULL && is_zfs_group(parent))
57b448de658d89a2c88a001d58073c46ed2180f3dougm needsupdate = zfs_needs_update(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm zfs = 0;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (zfs) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!clear && needsupdate)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_zfs_update((sa_share_t)group);
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (clear) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_abort_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_end_transaction(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_destroy_optionset(optionset)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Remove the optionset from its group. Update the repostory to
6185db853e024a486ff8837e6784dd290d866112dougm * reflect this change.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_destroy_optionset(sa_optionset_t optionset)
6185db853e024a486ff8837e6784dd290d866112dougm{
57b448de658d89a2c88a001d58073c46ed2180f3dougm char name[SA_STRSIZE];
6185db853e024a486ff8837e6784dd290d866112dougm int len;
6185db853e024a486ff8837e6784dd290d866112dougm int ret;
6185db853e024a486ff8837e6784dd290d866112dougm char *id = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm int ispersist = 1;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /* now delete the prop group */
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_optionset_parent(optionset);
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL && sa_is_share(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ispersist = is_persistent(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = sa_get_share_attr((sa_share_t)group, "id");
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (ispersist) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm len = sa_optionset_name(optionset, name, sizeof (name), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (len > 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_pgroup(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
549ec3fff108310966327d1dc9004551b63210b7dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode((xmlNodePtr)optionset);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode((xmlNodePtr)optionset);
6185db853e024a486ff8837e6784dd290d866112dougm if (id != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(id);
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/* private to the implementation */
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougm_sa_remove_optionset(sa_optionset_t optionset)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode((xmlNodePtr)optionset);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode((xmlNodePtr)optionset);
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_create_security(group, sectype, proto)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a security optionset (one that has a type name and a
6185db853e024a486ff8837e6784dd290d866112dougm * proto). Security is left over from a pure NFS implementation. The
6185db853e024a486ff8837e6784dd290d866112dougm * naming will change in the future when the API is released.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougmsa_security_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_create_security(sa_group_t group, char *sectype, char *proto)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_security_t security;
6185db853e024a486ff8837e6784dd290d866112dougm char *id = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t parent;
6185db853e024a486ff8837e6784dd290d866112dougm char *groupname = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL && sa_is_share(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = sa_get_share_attr((sa_share_t)group, "id");
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = sa_get_parent_group(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (parent != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname = sa_get_group_attr(parent, "name");
6185db853e024a486ff8837e6784dd290d866112dougm } else if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname = sa_get_group_attr(group, "name");
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm security = sa_get_security(group, sectype, proto);
6185db853e024a486ff8837e6784dd290d866112dougm if (security != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm /* can't have a duplicate security option */
6185db853e024a486ff8837e6784dd290d866112dougm security = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm } else {
6185db853e024a486ff8837e6784dd290d866112dougm security = (sa_security_t)xmlNewChild((xmlNodePtr)group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm NULL, (xmlChar *)"security", NULL);
6185db853e024a486ff8837e6784dd290d866112dougm if (security != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char oname[SA_STRSIZE];
6185db853e024a486ff8837e6784dd290d866112dougm sa_set_security_attr(security, "type", proto);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm sa_set_security_attr(security, "sectype", sectype);
6185db853e024a486ff8837e6784dd290d866112dougm (void) sa_security_name(security, oname,
57b448de658d89a2c88a001d58073c46ed2180f3dougm sizeof (oname), id);
6185db853e024a486ff8837e6784dd290d866112dougm if (groupname != NULL && is_persistent(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle =
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_handle_impl_t)sa_find_group_handle(
57b448de658d89a2c88a001d58073c46ed2180f3dougm group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_get_instance(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_create_pgroup(
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle, oname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (groupname != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(groupname);
6185db853e024a486ff8837e6784dd290d866112dougm return (security);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_destroy_security(security)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Remove the specified optionset from the document and the
6185db853e024a486ff8837e6784dd290d866112dougm * configuration.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_destroy_security(sa_security_t security)
6185db853e024a486ff8837e6784dd290d866112dougm{
57b448de658d89a2c88a001d58073c46ed2180f3dougm char name[SA_STRSIZE];
6185db853e024a486ff8837e6784dd290d866112dougm int len;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm char *id = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm int iszfs = 0;
6185db853e024a486ff8837e6784dd290d866112dougm int ispersist = 1;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm group = sa_get_optionset_parent(security);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm iszfs = sa_group_is_zfs(group);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (group != NULL && !iszfs) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_is_share(group))
57b448de658d89a2c88a001d58073c46ed2180f3dougm ispersist = is_persistent(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = sa_get_share_attr((sa_share_t)group, "id");
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (ispersist) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm len = sa_security_name(security, name, sizeof (name), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!iszfs && len > 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle =
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_delete_pgroup(impl_handle->scfhandle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode((xmlNodePtr)security);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode((xmlNodePtr)security);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (iszfs)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_zfs_update(group);
6185db853e024a486ff8837e6784dd290d866112dougm if (id != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(id);
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_security_attr(optionset, tag)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Return the specified attribute value from the optionset.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmchar *
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_security_attr(sa_property_t optionset, char *tag)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (get_node_attr((void *)optionset, tag));
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_security_attr(optionset, tag, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the optioset attribute specied by tag to the specified value.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmvoid
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_security_attr(sa_group_t optionset, char *tag, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm set_node_attr((void *)optionset, tag, value);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * is_nodetype(node, type)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Check to see if node is of the type specified.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmis_nodetype(void *node, char *type)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm return (strcmp((char *)((xmlNodePtr)node)->name, type) == 0);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm/*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * add_or_update()
57b448de658d89a2c88a001d58073c46ed2180f3dougm *
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Add or update a property. Pulled out of sa_set_prop_by_prop for
57b448de658d89a2c88a001d58073c46ed2180f3dougm * readability.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougmstatic int
57b448de658d89a2c88a001d58073c46ed2180f3dougmadd_or_update(scfutilhandle_t *scf_handle, int type, scf_value_t *value,
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_transaction_entry_t *entry, char *name, char *valstr)
57b448de658d89a2c88a001d58073c46ed2180f3dougm{
57b448de658d89a2c88a001d58073c46ed2180f3dougm int ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (type == SA_PROP_OP_ADD)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = scf_transaction_property_new(scf_handle->trans,
57b448de658d89a2c88a001d58073c46ed2180f3dougm entry, name, SCF_TYPE_ASTRING);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = scf_transaction_property_change(scf_handle->trans,
57b448de658d89a2c88a001d58073c46ed2180f3dougm entry, name, SCF_TYPE_ASTRING);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = scf_value_set_astring(value, valstr);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = scf_entry_add_value(entry, value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == 0)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (ret);
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_value_destroy(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_entry_destroy(entry);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (SA_SYSTEM_ERR);
57b448de658d89a2c88a001d58073c46ed2180f3dougm}
57b448de658d89a2c88a001d58073c46ed2180f3dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_prop_by_prop(optionset, group, prop, type)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Add/remove/update the specified property prop into the optionset or
6185db853e024a486ff8837e6784dd290d866112dougm * share. If a share, sort out which property group based on GUID. In
6185db853e024a486ff8837e6784dd290d866112dougm * all cases, the appropriate transaction is set (or ZFS share is
6185db853e024a486ff8837e6784dd290d866112dougm * marked as needing an update)
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmstatic int
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_prop_by_prop(sa_optionset_t optionset, sa_group_t group,
6185db853e024a486ff8837e6784dd290d866112dougm sa_property_t prop, int type)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm char *name;
6185db853e024a486ff8837e6784dd290d866112dougm char *valstr;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm scf_transaction_entry_t *entry;
6185db853e024a486ff8837e6784dd290d866112dougm scf_value_t *value;
6185db853e024a486ff8837e6784dd290d866112dougm int opttype; /* 1 == optionset, 0 == security */
6185db853e024a486ff8837e6784dd290d866112dougm char *id = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm int iszfs = 0;
6185db853e024a486ff8837e6784dd290d866112dougm int isshare = 0;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t parent = NULL;
549ec3fff108310966327d1dc9004551b63210b7dougm sa_handle_impl_t impl_handle;
549ec3fff108310966327d1dc9004551b63210b7dougm scfutilhandle_t *scf_handle;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (!is_persistent(group)) {
6185db853e024a486ff8837e6784dd290d866112dougm /*
6185db853e024a486ff8837e6784dd290d866112dougm * if the group/share is not persistent we don't need
6185db853e024a486ff8837e6784dd290d866112dougm * to do anything here
6185db853e024a486ff8837e6784dd290d866112dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (SA_OK);
6185db853e024a486ff8837e6784dd290d866112dougm }
549ec3fff108310966327d1dc9004551b63210b7dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle == NULL || impl_handle->scfhandle == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (SA_SYSTEM_ERR);
549ec3fff108310966327d1dc9004551b63210b7dougm scf_handle = impl_handle->scfhandle;
6185db853e024a486ff8837e6784dd290d866112dougm name = sa_get_property_attr(prop, "type");
6185db853e024a486ff8837e6784dd290d866112dougm valstr = sa_get_property_attr(prop, "value");
6185db853e024a486ff8837e6784dd290d866112dougm entry = scf_entry_create(scf_handle->handle);
6185db853e024a486ff8837e6784dd290d866112dougm opttype = is_nodetype((void *)optionset, "optionset");
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (valstr != NULL && entry != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_is_share(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm isshare = 1;
57b448de658d89a2c88a001d58073c46ed2180f3dougm parent = sa_get_parent_group(group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (parent != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm iszfs = is_zfs_group(parent);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm iszfs = is_zfs_group(group);
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!iszfs) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (scf_handle->trans == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char oname[SA_STRSIZE];
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *groupname = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (isshare) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (parent != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname =
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_get_group_attr(parent,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "name");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm id =
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_get_share_attr((sa_share_t)group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "id");
549ec3fff108310966327d1dc9004551b63210b7dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname = sa_get_group_attr(group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "name");
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (groupname != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_get_instance(scf_handle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(groupname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (opttype)
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_optionset_name(optionset,
57b448de658d89a2c88a001d58073c46ed2180f3dougm oname, sizeof (oname), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_security_name(optionset,
57b448de658d89a2c88a001d58073c46ed2180f3dougm oname, sizeof (oname), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(scf_handle, oname);
6185db853e024a486ff8837e6784dd290d866112dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm switch (type) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm case SA_PROP_OP_REMOVE:
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = scf_transaction_property_delete(
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_handle->trans, entry, name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm case SA_PROP_OP_ADD:
57b448de658d89a2c88a001d58073c46ed2180f3dougm case SA_PROP_OP_UPDATE:
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = scf_value_create(
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_handle->handle);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = add_or_update(scf_handle, type,
57b448de658d89a2c88a001d58073c46ed2180f3dougm value, entry, name, valstr);
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * ZFS update. The calling function would have updated
57b448de658d89a2c88a001d58073c46ed2180f3dougm * the internal XML structure. Just need to flag it as
57b448de658d89a2c88a001d58073c46ed2180f3dougm * changed for ZFS.
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm zfs_set_update((sa_share_t)group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (name != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(name);
6185db853e024a486ff8837e6784dd290d866112dougm if (valstr != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(valstr);
6185db853e024a486ff8837e6784dd290d866112dougm else if (entry != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_entry_destroy(entry);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (ret == -1)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_create_property(name, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a new property with the specified name and value.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_property_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_create_property(char *name, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm node = xmlNewNode(NULL, (xmlChar *)"option");
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm xmlSetProp(node, (xmlChar *)"type", (xmlChar *)name);
6185db853e024a486ff8837e6784dd290d866112dougm xmlSetProp(node, (xmlChar *)"value", (xmlChar *)value);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_property_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_add_property(object, property)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Add the specified property to the object. Issue the appropriate
6185db853e024a486ff8837e6784dd290d866112dougm * transaction or mark a ZFS object as needing an update.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_add_property(void *object, sa_property_t property)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t parent;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm char *proto;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm proto = sa_get_optionset_attr(object, "type");
6185db853e024a486ff8837e6784dd290d866112dougm if (property != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if ((ret = sa_valid_property(object, proto, property)) ==
57b448de658d89a2c88a001d58073c46ed2180f3dougm SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm property = (sa_property_t)xmlAddChild(
57b448de658d89a2c88a001d58073c46ed2180f3dougm (xmlNodePtr)object, (xmlNodePtr)property);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (proto != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(proto);
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (ret);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (proto != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(proto);
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm parent = sa_get_parent_group(object);
6185db853e024a486ff8837e6784dd290d866112dougm if (!is_persistent(parent)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (sa_is_share(parent))
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = sa_get_parent_group(parent);
6185db853e024a486ff8837e6784dd290d866112dougm else
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = parent;
549ec3fff108310966327d1dc9004551b63210b7dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (property == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_NO_MEMORY;
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char oname[SA_STRSIZE];
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (!is_zfs_group(group)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *id = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_handle_impl_t impl_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm scfutilhandle_t *scf_handle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle = (sa_handle_impl_t)sa_find_group_handle(
57b448de658d89a2c88a001d58073c46ed2180f3dougm group);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (impl_handle == NULL ||
57b448de658d89a2c88a001d58073c46ed2180f3dougm impl_handle->scfhandle == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_SYSTEM_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_handle = impl_handle->scfhandle;
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (sa_is_share((sa_group_t)parent)) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm id = sa_get_share_attr(
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_share_t)parent, "id");
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (scf_handle->trans == NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (is_nodetype(object, "optionset")) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_optionset_name(
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_optionset_t)object,
57b448de658d89a2c88a001d58073c46ed2180f3dougm oname, sizeof (oname), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm (void) sa_security_name(
57b448de658d89a2c88a001d58073c46ed2180f3dougm (sa_optionset_t)object,
57b448de658d89a2c88a001d58073c46ed2180f3dougm oname, sizeof (oname), id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_start_transaction(scf_handle,
57b448de658d89a2c88a001d58073c46ed2180f3dougm oname);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (ret == SA_OK) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *name;
57b448de658d89a2c88a001d58073c46ed2180f3dougm char *value;
57b448de658d89a2c88a001d58073c46ed2180f3dougm name = sa_get_property_attr(property,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = sa_get_property_attr(property,
57b448de658d89a2c88a001d58073c46ed2180f3dougm "value");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL && value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (scf_handle->scf_state ==
57b448de658d89a2c88a001d58073c46ed2180f3dougm SCH_STATE_INIT) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_property(
57b448de658d89a2c88a001d58073c46ed2180f3dougm scf_handle, name,
57b448de658d89a2c88a001d58073c46ed2180f3dougm value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_CONFIG_ERR;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (name != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(
57b448de658d89a2c88a001d58073c46ed2180f3dougm name);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (id != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(id);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * ZFS is a special case. We do want
57b448de658d89a2c88a001d58073c46ed2180f3dougm * to allow editing property/security
57b448de658d89a2c88a001d58073c46ed2180f3dougm * lists since we can have a better
57b448de658d89a2c88a001d58073c46ed2180f3dougm * syntax and we also want to keep
57b448de658d89a2c88a001d58073c46ed2180f3dougm * things consistent when possible.
57b448de658d89a2c88a001d58073c46ed2180f3dougm *
57b448de658d89a2c88a001d58073c46ed2180f3dougm * Right now, we defer until the
57b448de658d89a2c88a001d58073c46ed2180f3dougm * sa_commit_properties so we can get
57b448de658d89a2c88a001d58073c46ed2180f3dougm * them all at once. We do need to
57b448de658d89a2c88a001d58073c46ed2180f3dougm * mark the share as "changed"
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm zfs_set_update((sa_share_t)parent);
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_remove_property(property)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Remove the specied property from its containing object. Update the
6185db853e024a486ff8837e6784dd290d866112dougm * repository as appropriate.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_remove_property(sa_property_t property)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm if (property != NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm sa_optionset_t optionset;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm optionset = sa_get_property_parent(property);
6185db853e024a486ff8837e6784dd290d866112dougm if (optionset != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = sa_get_optionset_parent(optionset);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_prop_by_prop(optionset, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm property, SA_PROP_OP_REMOVE);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm xmlUnlinkNode((xmlNodePtr)property);
6185db853e024a486ff8837e6784dd290d866112dougm xmlFreeNode((xmlNodePtr)property);
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_NO_SUCH_PROP;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_update_property(property, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Update the specified property to the new value. If value is NULL,
6185db853e024a486ff8837e6784dd290d866112dougm * we currently treat this as a remove.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_update_property(sa_property_t property, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_OK;
6185db853e024a486ff8837e6784dd290d866112dougm if (value == NULL) {
6185db853e024a486ff8837e6784dd290d866112dougm return (sa_remove_property(property));
6185db853e024a486ff8837e6784dd290d866112dougm } else {
6185db853e024a486ff8837e6784dd290d866112dougm sa_optionset_t optionset;
6185db853e024a486ff8837e6784dd290d866112dougm sa_group_t group;
6185db853e024a486ff8837e6784dd290d866112dougm set_node_attr((void *)property, "value", value);
6185db853e024a486ff8837e6784dd290d866112dougm optionset = sa_get_property_parent(property);
6185db853e024a486ff8837e6784dd290d866112dougm if (optionset != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm group = sa_get_optionset_parent(optionset);
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (group != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_set_prop_by_prop(optionset, group,
57b448de658d89a2c88a001d58073c46ed2180f3dougm property, SA_PROP_OP_UPDATE);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm } else {
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = SA_NO_SUCH_PROP;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_protocol_property(propset, prop)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the specified protocol specific property. These are global to
6185db853e024a486ff8837e6784dd290d866112dougm * the protocol and not specific to a group or share.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_property_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_protocol_property(sa_protocol_properties_t propset, char *prop)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node = (xmlNodePtr)propset;
6185db853e024a486ff8837e6784dd290d866112dougm xmlChar *value = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = node->children; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (xmlStrcmp(node->name, (xmlChar *)"option") == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (prop == NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = xmlGetProp(node, (xmlChar *)"type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL &&
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlStrcasecmp(value, (xmlChar *)prop) == 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm break;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (value != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlFree(value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm value = NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm if (value != NULL)
6185db853e024a486ff8837e6784dd290d866112dougm xmlFree(value);
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL && xmlStrcmp(node->name, (xmlChar *)"option") != 0) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm /*
57b448de658d89a2c88a001d58073c46ed2180f3dougm * avoid a non option node -- it is possible to be a
57b448de658d89a2c88a001d58073c46ed2180f3dougm * text node
57b448de658d89a2c88a001d58073c46ed2180f3dougm */
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = NULL;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_property_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_get_next_protocol_property(prop)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Get the next protocol specific property in the list.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_property_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_get_next_protocol_property(sa_property_t prop)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm for (node = ((xmlNodePtr)prop)->next; node != NULL;
57b448de658d89a2c88a001d58073c46ed2180f3dougm node = node->next) {
6185db853e024a486ff8837e6784dd290d866112dougm if (xmlStrcmp(node->name, (xmlChar *)"option") == 0) {
6185db853e024a486ff8837e6784dd290d866112dougm break;
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return ((sa_property_t)node);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_set_protocol_property(prop, value)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Set the specified property to have the new value. The protocol
6185db853e024a486ff8837e6784dd290d866112dougm * specific plugin will then be called to update the property.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_set_protocol_property(sa_property_t prop, char *value)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm sa_protocol_properties_t propset;
6185db853e024a486ff8837e6784dd290d866112dougm char *proto;
6185db853e024a486ff8837e6784dd290d866112dougm int ret = SA_INVALID_PROTOCOL;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm propset = ((xmlNodePtr)prop)->parent;
6185db853e024a486ff8837e6784dd290d866112dougm if (propset != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm proto = sa_get_optionset_attr(propset, "type");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (proto != NULL) {
57b448de658d89a2c88a001d58073c46ed2180f3dougm set_node_attr((xmlNodePtr)prop, "value", value);
57b448de658d89a2c88a001d58073c46ed2180f3dougm ret = sa_proto_set_property(proto, prop);
57b448de658d89a2c88a001d58073c46ed2180f3dougm sa_free_attr_string(proto);
57b448de658d89a2c88a001d58073c46ed2180f3dougm }
6185db853e024a486ff8837e6784dd290d866112dougm }
6185db853e024a486ff8837e6784dd290d866112dougm return (ret);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_add_protocol_property(propset, prop)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Add a new property to the protocol sepcific property set.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmint
6185db853e024a486ff8837e6784dd290d866112dougmsa_add_protocol_property(sa_protocol_properties_t propset, sa_property_t prop)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm /* should check for legitimacy */
6185db853e024a486ff8837e6784dd290d866112dougm node = xmlAddChild((xmlNodePtr)propset, (xmlNodePtr)prop);
6185db853e024a486ff8837e6784dd290d866112dougm if (node != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm return (SA_OK);
6185db853e024a486ff8837e6784dd290d866112dougm return (SA_NO_MEMORY);
6185db853e024a486ff8837e6784dd290d866112dougm}
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougm/*
6185db853e024a486ff8837e6784dd290d866112dougm * sa_create_protocol_properties(proto)
6185db853e024a486ff8837e6784dd290d866112dougm *
6185db853e024a486ff8837e6784dd290d866112dougm * Create a protocol specifity property set.
6185db853e024a486ff8837e6784dd290d866112dougm */
6185db853e024a486ff8837e6784dd290d866112dougm
6185db853e024a486ff8837e6784dd290d866112dougmsa_protocol_properties_t
6185db853e024a486ff8837e6784dd290d866112dougmsa_create_protocol_properties(char *proto)
6185db853e024a486ff8837e6784dd290d866112dougm{
6185db853e024a486ff8837e6784dd290d866112dougm xmlNodePtr node;
57b448de658d89a2c88a001d58073c46ed2180f3dougm
6185db853e024a486ff8837e6784dd290d866112dougm node = xmlNewNode(NULL, (xmlChar *)"propertyset");
57b448de658d89a2c88a001d58073c46ed2180f3dougm if (node != NULL)
57b448de658d89a2c88a001d58073c46ed2180f3dougm xmlSetProp(node, (xmlChar *)"type", (xmlChar *)proto);
6185db853e024a486ff8837e6784dd290d866112dougm return (node);
6185db853e024a486ff8837e6784dd290d866112dougm}