usr/src/cmd/fs.d/nfs/mountd/exportlist.c
author th199096
Mon, 02 Apr 2007 21:38:04 -0700
changeset 3957 86c9dda5df37
parent 0 68f95e015346
permissions -rw-r--r--
PSARC/2007/052 In-kernel Sharetab 6371468 /etc/dfs/sharetab should be a mntfs style file

/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */

/*
 * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

#pragma ident	"%Z%%M%	%I%	%E% SMI"

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <string.h>
#include <syslog.h>
#include <sys/param.h>
#include <sys/stat.h>
#include <sys/file.h>
#include <sys/time.h>
#include <errno.h>
#include <rpcsvc/mount.h>
#include <sys/pathconf.h>
#include <sys/systeminfo.h>
#include <sys/utsname.h>
#include <signal.h>
#include <locale.h>
#include <unistd.h>
#include <thread.h>
#include <sharefs/share.h>
#include <sharefs/sharetab.h>
#include "../lib/sharetab.h"
#include "mountd.h"

static void freeexports(struct exportnode *);
static struct groupnode **newgroup(char *, struct groupnode **);
static struct exportnode **newexport(char *, struct groupnode *,
						struct exportnode **);

static char *optlist[] = {
#define	OPT_RO		0
	SHOPT_RO,
#define	OPT_RW		1
	SHOPT_RW,
	NULL
};

/*
 * Send current export list to a client
 */
void
export(struct svc_req *rqstp)
{
	SVCXPRT *transp;
	struct exportnode *exportlist;
	struct exportnode **tail;
	struct groupnode *groups;
	struct groupnode **grtail;
	struct share *sh;
	struct sh_list *shp;
	char *gr, *p, *opts, *val, *lasts;

	int export_to_everyone;

	transp = rqstp->rq_xprt;
	if (!svc_getargs(transp, xdr_void, NULL)) {
		svcerr_decode(transp);
		return;
	}

	check_sharetab();

	exportlist = NULL;
	tail = &exportlist;

	(void) rw_rdlock(&sharetab_lock);

	for (shp = share_list; shp; shp = shp->shl_next) {

		groups = NULL;
		grtail = &groups;

		sh = shp->shl_sh;

		/*
		 * Check for "ro" or "rw" list without argument values.  This
		 * indicates export to everyone.  Unfortunately, SunOS 4.x
		 * automounter uses this, and it is indicated indirectly with
		 * 'showmount -e'.
		 *
		 * If export_to_everyone is 1, then groups should be NULL to
		 * indicate export to everyone.
		 */

		opts = strdup(sh->sh_opts);
		p = opts;


		export_to_everyone = 0;
		while (*p) {
			switch (getsubopt(&p, optlist, &val)) {
			case OPT_RO:
			case OPT_RW:
				if (val == NULL)
					export_to_everyone = 1;
				break;
			}
		}

		free(opts);

		if (export_to_everyone == 0) {

			opts = strdup(sh->sh_opts);
			p = opts;

			/*
			 * Just concatenate all the hostnames/groups
			 * from the "ro" and "rw" lists for each flavor.
			 * This list is rather meaningless now, but
			 * that's what the protocol demands.
			 */
			while (*p) {
				switch (getsubopt(&p, optlist, &val)) {
				case OPT_RO:
				case OPT_RW:

					while ((gr = strtok_r(val, ":", &lasts))
					    != NULL) {
						val = NULL;
						grtail = newgroup(gr, grtail);
					}
					break;
				}
			}

			free(opts);
		}
		tail = newexport(sh->sh_path, groups, tail);
	}

	(void) rw_unlock(&sharetab_lock);

	errno = 0;
	if (!svc_sendreply(transp, xdr_exports, (char *)&exportlist))
		log_cant_reply(transp);

	freeexports(exportlist);
}


static void
freeexports(struct exportnode *ex)
{
	struct groupnode *groups, *tmpgroups;
	struct exportnode *tmpex;

	while (ex) {
		groups = ex->ex_groups;
		while (groups) {
			tmpgroups = groups->gr_next;
			free(groups->gr_name);
			free(groups);
			groups = tmpgroups;
		}
		tmpex = ex->ex_next;
		free(ex->ex_dir);
		free(ex);
		ex = tmpex;
	}
}


static struct groupnode **
newgroup(char *grname, struct groupnode **tail)
{
	struct groupnode *new;
	char *newname;

	new = exmalloc(sizeof (*new));
	newname = exmalloc(strlen(grname) + 1);
	(void) strcpy(newname, grname);

	new->gr_name = newname;
	new->gr_next = NULL;
	*tail = new;
	return (&new->gr_next);
}


static struct exportnode **
newexport(char *grname, struct groupnode *grplist, struct exportnode **tail)
{
	struct exportnode *new;
	char *newname;

	new = exmalloc(sizeof (*new));
	newname = exmalloc(strlen(grname) + 1);
	(void) strcpy(newname, grname);

	new->ex_dir = newname;
	new->ex_groups = grplist;
	new->ex_next = NULL;
	*tail = new;
	return (&new->ex_next);
}