usr/src/uts/common/fs/nfs/nfs4_srv.c
author jpk
Fri, 24 Mar 2006 12:29:20 -0800
changeset 1676 37f4a3e2bd99
parent 1146 c3555cdf52c2
child 2035 a29bc457bcb9
permissions -rw-r--r--
PSARC/2002/762 Layered Trusted Solaris PSARC/2005/060 TSNET: Trusted Networking with Security Labels PSARC/2005/259 Layered Trusted Solaris Label Interfaces PSARC/2005/573 Solaris Trusted Extensions for Printing PSARC/2005/691 Trusted Extensions for Device Allocation PSARC/2005/723 Solaris Trusted Extensions Filesystem Labeling PSARC/2006/009 Labeled Auditing PSARC/2006/155 Trusted Extensions RBAC Changes PSARC/2006/191 is_system_labeled 6293271 Zone processes should use zone_kcred instead of kcred 6394554 integrate Solaris Trusted Extensions
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
     5
 * Common Development and Distribution License (the "License").
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
     6
 * You may not use this file except in compliance with the License.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
/*
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
    22
 * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
 *	Copyright (c) 1983,1984,1985,1986,1987,1988,1989  AT&T.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
 *	All Rights Reserved
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#include <sys/param.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
#include <sys/systm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include <sys/cred.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <sys/buf.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <sys/vfs.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <sys/vnode.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include <sys/uio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
#include <sys/errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#include <sys/sysmacros.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
#include <sys/statvfs.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
#include <sys/kmem.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#include <sys/dirent.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
#include <sys/cmn_err.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
#include <sys/debug.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
#include <sys/systeminfo.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
#include <sys/flock.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
#include <sys/pathname.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
#include <sys/nbmlock.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
#include <sys/share.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
#include <sys/atomic.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
#include <sys/policy.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
#include <sys/fem.h>
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
    56
#include <sys/sdt.h>
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
#include <rpc/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
#include <rpc/auth.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
#include <rpc/rpcsec_gss.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
#include <rpc/svc.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
#include <nfs/nfs.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
#include <nfs/export.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#include <nfs/lm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
#include <nfs/nfs4.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
#include <sys/strsubr.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
#include <sys/strsun.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
#include <inet/common.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
#include <inet/ip.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
#include <inet/ip6.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
    75
#include <sys/tsol/label.h>
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
    76
#include <sys/tsol/tndb.h>
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
    77
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
#define	RFS4_MAXLOCK_TRIES 4	/* Try to get the lock this many times */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
static int rfs4_maxlock_tries = RFS4_MAXLOCK_TRIES;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
#define	RFS4_LOCK_DELAY 10	/* Milliseconds */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
static clock_t rfs4_lock_delay = RFS4_LOCK_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
/* End of Tunables */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
 * Used to bump the stateid4.seqid value and show changes in the stateid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
#define	next_stateid(sp) (++(sp)->bits.chgseq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
 * RFS4_MINLEN_ENTRY4: XDR-encoded size of smallest possible dirent.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
 *	This is used to return NFS4ERR_TOOSMALL when clients specify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
 *	maxcount that isn't large enough to hold the smallest possible
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
 *	XDR encoded dirent.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
 *	    sizeof cookie (8 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
 *	    sizeof name_len (4 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
 *	    sizeof smallest (padded) name (4 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
 *	    sizeof bitmap4_len (12 bytes) +   NOTE: we always encode len=2 bm4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
 *	    sizeof attrlist4_len (4 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
 *	    sizeof next boolean (4 bytes)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
 * RFS4_MINLEN_RDDIR4: XDR-encoded size of READDIR op reply containing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
 * the smallest possible entry4 (assumes no attrs requested).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
 *	sizeof nfsstat4 (4 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
 *	sizeof verifier4 (8 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
 *	sizeof entry4list bool (4 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
 *	sizeof entry4 	(36 bytes) +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
 *	sizeof eof bool  (4 bytes)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
 * RFS4_MINLEN_RDDIR_BUF: minimum length of buffer server will provide to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
 *	VOP_READDIR.  Its value is the size of the maximum possible dirent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
 *	for solaris.  The DIRENT64_RECLEN macro returns	the size of dirent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
 *	required for a given name length.  MAXNAMELEN is the maximum
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
 *	filename length allowed in Solaris.  The first two DIRENT64_RECLEN()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
 *	macros are to allow for . and .. entries -- just a minor tweak to try
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
 *	and guarantee that buffer we give to VOP_READDIR will be large enough
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
 *	to hold ., .., and the largest possible solaris dirent64.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
#define	RFS4_MINLEN_ENTRY4 36
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
#define	RFS4_MINLEN_RDDIR4 (4 + NFS4_VERIFIER_SIZE + 4 + RFS4_MINLEN_ENTRY4 + 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
#define	RFS4_MINLEN_RDDIR_BUF \
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	(DIRENT64_RECLEN(1) + DIRENT64_RECLEN(2) + DIRENT64_RECLEN(MAXNAMELEN))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
 * It would be better to pad to 4 bytes since that's what XDR would do,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
 * but the dirents UFS gives us are already padded to 8, so just take
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
 * what we're given.  Dircount is only a hint anyway.  Currently the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
 * solaris kernel is ASCII only, so there's no point in calling the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
 * UTF8 functions.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
 * dirent64: named padded to provide 8 byte struct alignment
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
 *	d_ino(8) + d_off(8) + d_reclen(2) + d_name(namelen + null(1) + pad)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
 * cookie: uint64_t   +  utf8namelen: uint_t  +   utf8name padded to 8 bytes
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
#define	DIRENT64_TO_DIRCOUNT(dp) \
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
	(3 * BYTES_PER_XDR_UNIT + DIRENT64_NAMELEN((dp)->d_reclen))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   141
/*
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   142
 * types of label comparison
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   143
 */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   144
#define	EQUALITY_CHECK	0
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   145
#define	DOMINANCE_CHECK	1
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
   146
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   147
time_t rfs4_start_time;			/* Initialized in rfs4_srvrinit */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   148
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   149
static sysid_t lockt_sysid;		/* dummy sysid for all LOCKT calls */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   150
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   151
u_longlong_t nfs4_srv_caller_id;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   152
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   153
verifier4	Write4verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   154
verifier4	Readdir4verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   155
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   156
void		rfs4_init_compound_state(struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   157
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   158
static void	nullfree(caddr_t);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
static void	rfs4_op_inval(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   160
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   161
static void	rfs4_op_access(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   162
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   163
static void	rfs4_op_close(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   164
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   165
static void	rfs4_op_commit(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   166
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   167
static void	rfs4_op_create(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   168
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   169
static void	rfs4_op_create_free(nfs_resop4 *resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   170
static void	rfs4_op_delegreturn(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   171
				    struct svc_req *, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   172
static void	rfs4_op_getattr(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   173
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   174
static void	rfs4_op_getattr_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   175
static void	rfs4_op_getfh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   176
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   177
static void	rfs4_op_getfh_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   178
static void	rfs4_op_illegal(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   179
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   180
static void	rfs4_op_link(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   181
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   182
static void	rfs4_op_lock(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   183
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   184
static void	lock_denied_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   185
static void	rfs4_op_locku(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   186
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   187
static void	rfs4_op_lockt(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   188
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   189
static void	rfs4_op_lookup(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   190
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   191
static void	rfs4_op_lookupp(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   192
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   193
static void	rfs4_op_openattr(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   194
				struct svc_req *req, struct compound_state *cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   195
static void	rfs4_op_nverify(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   196
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   197
static void	rfs4_op_open(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   198
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   199
static void	rfs4_op_open_confirm(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   200
			struct svc_req *, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   201
static void	rfs4_op_open_downgrade(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   202
			struct svc_req *, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   203
static void	rfs4_op_putfh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   204
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   205
static void	rfs4_op_putpubfh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   206
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   207
static void	rfs4_op_putrootfh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   208
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   209
static void	rfs4_op_read(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   210
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   211
static void	rfs4_op_read_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   212
static void	rfs4_op_readdir_free(nfs_resop4 *resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   213
static void	rfs4_op_readlink(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   214
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   215
static void	rfs4_op_readlink_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   216
static void	rfs4_op_release_lockowner(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   217
			struct svc_req *, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   218
static void	rfs4_op_remove(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   219
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   220
static void	rfs4_op_rename(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   221
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   222
static void	rfs4_op_renew(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   223
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   224
static void	rfs4_op_restorefh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   225
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   226
static void	rfs4_op_savefh(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   227
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   228
static void	rfs4_op_setattr(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   229
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   230
static void	rfs4_op_verify(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   231
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   232
static void	rfs4_op_write(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   233
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   234
static void	rfs4_op_setclientid(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   235
			struct svc_req *, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   236
static void	rfs4_op_setclientid_confirm(nfs_argop4 *, nfs_resop4 *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   237
			struct svc_req *req, struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   238
static void	rfs4_op_secinfo(nfs_argop4 *, nfs_resop4 *, struct svc_req *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   239
			struct compound_state *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   240
static void	rfs4_op_secinfo_free(nfs_resop4 *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   241
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   242
static nfsstat4 check_open_access(uint32_t,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   243
				struct compound_state *, struct svc_req *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   244
nfsstat4 rfs4_client_sysid(rfs4_client_t *, sysid_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   245
static int	vop_shrlock(vnode_t *, int, struct shrlock *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   246
static int 	rfs4_shrlock(rfs4_state_t *, int);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   247
static int	rfs4_share(rfs4_state_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   248
void rfs4_ss_clid(rfs4_client_t *, struct svc_req *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   249
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   250
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   251
 * translation table for attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   252
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   253
struct nfs4_ntov_table {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   254
	union nfs4_attr_u *na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   255
	uint8_t amap[NFS4_MAXNUM_ATTRS];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   256
	int attrcnt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   257
	bool_t vfsstat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   258
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   259
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   260
static void	nfs4_ntov_table_init(struct nfs4_ntov_table *ntovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   261
static void	nfs4_ntov_table_free(struct nfs4_ntov_table *ntovp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   262
				    struct nfs4_svgetit_arg *sargp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   263
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   264
static nfsstat4	do_rfs4_set_attrs(bitmap4 *resp, fattr4 *fattrp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   265
		    struct compound_state *cs, struct nfs4_svgetit_arg *sargp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   266
		    struct nfs4_ntov_table *ntovp, nfs4_attr_cmd_t cmd);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   267
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   268
fem_t	*deleg_rdops;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   269
fem_t	*deleg_wrops;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   270
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   271
rfs4_servinst_t	*rfs4_cur_servinst = NULL;	/* current server instance */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   272
kmutex_t	rfs4_servinst_lock;		/* protects linked list */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   273
int		rfs4_seen_first_compound;	/* set first time we see one */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   274
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   275
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   276
int	rfs4_servinst_debug = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   277
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   278
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   279
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   280
 * NFS4 op dispatch table
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   281
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   282
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   283
struct rfsv4disp {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   284
	void	(*dis_proc)();		/* proc to call */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   285
	void	(*dis_resfree)();	/* frees space allocated by proc */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   286
	int	dis_flags;		/* RPC_IDEMPOTENT, etc... */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   287
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   288
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   289
static struct rfsv4disp rfsv4disptab[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   290
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   291
	 * NFS VERSION 4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   292
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   293
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   294
	/* RFS_NULL = 0 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   295
	{rfs4_op_illegal, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   296
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   297
	/* UNUSED = 1 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   298
	{rfs4_op_illegal, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   299
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   300
	/* UNUSED = 2 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   301
	{rfs4_op_illegal, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   302
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   303
	/* OP_ACCESS = 3 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   304
	{rfs4_op_access, nullfree, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   306
	/* OP_CLOSE = 4 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   307
	{rfs4_op_close, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   308
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   309
	/* OP_COMMIT = 5 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   310
	{rfs4_op_commit, nullfree, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   311
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   312
	/* OP_CREATE = 6 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   313
	{rfs4_op_create, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   314
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   315
	/* OP_DELEGPURGE = 7 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   316
	{rfs4_op_inval, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   317
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   318
	/* OP_DELEGRETURN = 8 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   319
	{rfs4_op_delegreturn, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   320
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   321
	/* OP_GETATTR = 9 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   322
	{rfs4_op_getattr, rfs4_op_getattr_free, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   323
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   324
	/* OP_GETFH = 10 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   325
	{rfs4_op_getfh, rfs4_op_getfh_free, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   326
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   327
	/* OP_LINK = 11 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   328
	{rfs4_op_link, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   329
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   330
	/* OP_LOCK = 12 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   331
	{rfs4_op_lock, lock_denied_free, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   332
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   333
	/* OP_LOCKT = 13 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   334
	{rfs4_op_lockt, lock_denied_free, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   335
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   336
	/* OP_LOCKU = 14 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   337
	{rfs4_op_locku, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   338
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   339
	/* OP_LOOKUP = 15 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   340
	{rfs4_op_lookup, nullfree, (RPC_IDEMPOTENT|RPC_PUBLICFH_OK)},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   341
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   342
	/* OP_LOOKUPP = 16 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   343
	{rfs4_op_lookupp, nullfree, (RPC_IDEMPOTENT|RPC_PUBLICFH_OK)},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   344
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   345
	/* OP_NVERIFY = 17 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   346
	{rfs4_op_nverify, nullfree, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   347
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   348
	/* OP_OPEN = 18 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   349
	{rfs4_op_open, rfs4_free_reply, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   350
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   351
	/* OP_OPENATTR = 19 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   352
	{rfs4_op_openattr, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   353
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   354
	/* OP_OPEN_CONFIRM = 20 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   355
	{rfs4_op_open_confirm, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   356
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   357
	/* OP_OPEN_DOWNGRADE = 21 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   358
	{rfs4_op_open_downgrade, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   360
	/* OP_OPEN_PUTFH = 22 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   361
	{rfs4_op_putfh, nullfree, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   362
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   363
	/* OP_PUTPUBFH = 23 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   364
	{rfs4_op_putpubfh, nullfree, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   365
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   366
	/* OP_PUTROOTFH = 24 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   367
	{rfs4_op_putrootfh, nullfree, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   368
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   369
	/* OP_READ = 25 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   370
	{rfs4_op_read, rfs4_op_read_free, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   371
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   372
	/* OP_READDIR = 26 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   373
	{rfs4_op_readdir, rfs4_op_readdir_free, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   374
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   375
	/* OP_READLINK = 27 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   376
	{rfs4_op_readlink, rfs4_op_readlink_free, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   377
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   378
	/* OP_REMOVE = 28 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   379
	{rfs4_op_remove, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   380
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   381
	/* OP_RENAME = 29 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   382
	{rfs4_op_rename, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   383
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   384
	/* OP_RENEW = 30 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   385
	{rfs4_op_renew, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   386
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   387
	/* OP_RESTOREFH = 31 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   388
	{rfs4_op_restorefh, nullfree, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   389
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   390
	/* OP_SAVEFH = 32 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   391
	{rfs4_op_savefh, nullfree, RPC_ALL},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   392
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   393
	/* OP_SECINFO = 33 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   394
	{rfs4_op_secinfo, rfs4_op_secinfo_free, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   395
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   396
	/* OP_SETATTR = 34 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   397
	{rfs4_op_setattr, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   398
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   399
	/* OP_SETCLIENTID = 35 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   400
	{rfs4_op_setclientid, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   401
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   402
	/* OP_SETCLIENTID_CONFIRM = 36 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   403
	{rfs4_op_setclientid_confirm, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   404
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   405
	/* OP_VERIFY = 37 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   406
	{rfs4_op_verify, nullfree, RPC_IDEMPOTENT},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   407
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   408
	/* OP_WRITE = 38 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   409
	{rfs4_op_write, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   410
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   411
	/* OP_RELEASE_LOCKOWNER = 39 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   412
	{rfs4_op_release_lockowner, nullfree, 0},
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   413
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   414
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   415
static uint_t rfsv4disp_cnt = sizeof (rfsv4disptab) / sizeof (rfsv4disptab[0]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   417
#define	OP_ILLEGAL_IDX (rfsv4disp_cnt)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   418
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   419
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   420
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   421
int rfs4_fillone_debug = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   422
int rfs4_shrlock_debug = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   423
int rfs4_no_stub_access = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   424
int rfs4_rddir_debug = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   425
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   426
static char *rfs4_op_string[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   427
	"rfs4_op_null",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   428
	"rfs4_op_1 unused",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   429
	"rfs4_op_2 unused",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   430
	"rfs4_op_access",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   431
	"rfs4_op_close",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   432
	"rfs4_op_commit",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   433
	"rfs4_op_create",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   434
	"rfs4_op_delegpurge",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   435
	"rfs4_op_delegreturn",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   436
	"rfs4_op_getattr",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   437
	"rfs4_op_getfh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   438
	"rfs4_op_link",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   439
	"rfs4_op_lock",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   440
	"rfs4_op_lockt",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   441
	"rfs4_op_locku",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   442
	"rfs4_op_lookup",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   443
	"rfs4_op_lookupp",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   444
	"rfs4_op_nverify",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   445
	"rfs4_op_open",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   446
	"rfs4_op_openattr",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   447
	"rfs4_op_open_confirm",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   448
	"rfs4_op_open_downgrade",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   449
	"rfs4_op_putfh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   450
	"rfs4_op_putpubfh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   451
	"rfs4_op_putrootfh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   452
	"rfs4_op_read",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   453
	"rfs4_op_readdir",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   454
	"rfs4_op_readlink",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   455
	"rfs4_op_remove",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   456
	"rfs4_op_rename",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   457
	"rfs4_op_renew",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   458
	"rfs4_op_restorefh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   459
	"rfs4_op_savefh",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   460
	"rfs4_op_secinfo",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   461
	"rfs4_op_setattr",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   462
	"rfs4_op_setclientid",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   463
	"rfs4_op_setclient_confirm",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   464
	"rfs4_op_verify",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   465
	"rfs4_op_write",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   466
	"rfs4_op_release_lockowner",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   467
	"rfs4_op_illegal"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   468
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   469
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   470
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   471
void rfs4_ss_chkclid(rfs4_client_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   472
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   473
#ifdef	nextdp
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   474
#undef nextdp
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   475
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   476
#define	nextdp(dp)	((struct dirent64 *)((char *)(dp) + (dp)->d_reclen))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   477
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   478
static const fs_operation_def_t nfs4_rd_deleg_tmpl[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   479
	VOPNAME_OPEN, deleg_rdopen,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   480
	VOPNAME_WRITE, deleg_write,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   481
	VOPNAME_SETATTR, deleg_setattr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   482
	VOPNAME_RWLOCK, deleg_rd_rwlock,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   483
	VOPNAME_SPACE, deleg_space,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   484
	VOPNAME_SETSECATTR, deleg_setsecattr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   485
	VOPNAME_VNEVENT, deleg_vnevent,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   486
	NULL, NULL
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   487
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   488
static const fs_operation_def_t nfs4_wr_deleg_tmpl[] = {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   489
	VOPNAME_OPEN, deleg_wropen,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   490
	VOPNAME_READ, deleg_read,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   491
	VOPNAME_WRITE, deleg_write,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   492
	VOPNAME_SETATTR, deleg_setattr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   493
	VOPNAME_RWLOCK, deleg_wr_rwlock,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   494
	VOPNAME_SPACE, deleg_space,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   495
	VOPNAME_SETSECATTR, deleg_setsecattr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   496
	VOPNAME_VNEVENT, deleg_vnevent,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   497
	NULL, NULL
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   498
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   499
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   500
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   501
rfs4_srvrinit(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   502
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   503
	timespec32_t verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   504
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   505
	extern void rfs4_attr_init();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   506
	extern krwlock_t rfs4_deleg_policy_lock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   507
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   508
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   509
	 * The following algorithm attempts to find a unique verifier
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   510
	 * to be used as the write verifier returned from the server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   511
	 * to the client.  It is important that this verifier change
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   512
	 * whenever the server reboots.  Of secondary importance, it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   513
	 * is important for the verifier to be unique between two
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   514
	 * different servers.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   515
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   516
	 * Thus, an attempt is made to use the system hostid and the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   517
	 * current time in seconds when the nfssrv kernel module is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   518
	 * loaded.  It is assumed that an NFS server will not be able
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   519
	 * to boot and then to reboot in less than a second.  If the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   520
	 * hostid has not been set, then the current high resolution
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   521
	 * time is used.  This will ensure different verifiers each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   522
	 * time the server reboots and minimize the chances that two
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   523
	 * different servers will have the same verifier.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   524
	 * XXX - this is broken on LP64 kernels.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   525
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   526
	verf.tv_sec = (time_t)nfs_atoi(hw_serial);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   527
	if (verf.tv_sec != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   528
		verf.tv_nsec = gethrestime_sec();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   529
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   530
		timespec_t tverf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   531
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   532
		gethrestime(&tverf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   533
		verf.tv_sec = (time_t)tverf.tv_sec;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   534
		verf.tv_nsec = tverf.tv_nsec;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   535
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   536
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   537
	Write4verf = *(uint64_t *)&verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   538
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   539
	rfs4_attr_init();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   540
	mutex_init(&rfs4_deleg_lock, NULL, MUTEX_DEFAULT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   541
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   542
	/* Used to manage create/destroy of server state */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   543
	mutex_init(&rfs4_state_lock, NULL, MUTEX_DEFAULT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   544
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   545
	/* Used to manage access to server instance linked list */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   546
	mutex_init(&rfs4_servinst_lock, NULL, MUTEX_DEFAULT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   547
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   548
	/* Used to manage access to rfs4_deleg_policy */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   549
	rw_init(&rfs4_deleg_policy_lock, NULL, RW_DEFAULT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   550
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   551
	error = fem_create("deleg_rdops", nfs4_rd_deleg_tmpl, &deleg_rdops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   552
	if (error != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   553
		rfs4_disable_delegation();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   554
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   555
		error = fem_create("deleg_wrops", nfs4_wr_deleg_tmpl,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   556
				&deleg_wrops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   557
		if (error != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   558
			rfs4_disable_delegation();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   559
			fem_free(deleg_rdops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   560
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   561
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   562
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   563
	nfs4_srv_caller_id = fs_new_caller_id();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   564
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   565
	lockt_sysid = lm_alloc_sysidt();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   566
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   567
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   568
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   569
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   570
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   571
rfs4_srvrfini(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   572
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   573
	extern krwlock_t rfs4_deleg_policy_lock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   574
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   575
	if (lockt_sysid != LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   576
		lm_free_sysidt(lockt_sysid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   577
		lockt_sysid = LM_NOSYSID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   578
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   579
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   580
	mutex_destroy(&rfs4_deleg_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   581
	mutex_destroy(&rfs4_state_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   582
	rw_destroy(&rfs4_deleg_policy_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   583
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   584
	fem_free(deleg_rdops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   585
	fem_free(deleg_wrops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   586
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   587
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   588
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   589
rfs4_init_compound_state(struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   590
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   591
	bzero(cs, sizeof (*cs));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   592
	cs->cont = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   593
	cs->access = CS_ACCESS_DENIED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   594
	cs->deleg = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   595
	cs->mandlock = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   596
	cs->fh.nfs_fh4_val = cs->fhbuf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   597
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   598
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   599
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   600
rfs4_grace_start(rfs4_servinst_t *sip)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   601
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   602
	time_t now = gethrestime_sec();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   603
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   604
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   605
	    "rfs4_grace_start: inst %p: 0x%lx", (void *)sip, now));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   606
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   607
	rw_enter(&sip->rwlock, RW_WRITER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   608
	sip->start_time = now;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   609
	sip->grace_period = rfs4_grace_period;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   610
	rw_exit(&sip->rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   611
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   612
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   613
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   614
 * returns true if the instance's grace period has never been started
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   615
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   616
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   617
rfs4_servinst_grace_new(rfs4_servinst_t *sip)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   618
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   619
	time_t start_time;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   621
	rw_enter(&sip->rwlock, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   622
	start_time = sip->start_time;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   623
	rw_exit(&sip->rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   624
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   625
	return (start_time == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   626
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   627
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   628
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   629
 * Indicates if server instance is within the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   630
 * grace period.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   631
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   632
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   633
rfs4_servinst_in_grace(rfs4_servinst_t *sip)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   634
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   635
	time_t grace_expiry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   636
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   637
	rw_enter(&sip->rwlock, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   638
	grace_expiry = sip->start_time + sip->grace_period;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   639
	rw_exit(&sip->rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   640
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   641
	return (gethrestime_sec() < grace_expiry);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   642
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   643
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   644
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   645
rfs4_clnt_in_grace(rfs4_client_t *cp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   646
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   647
	ASSERT(rfs4_dbe_refcnt(cp->dbe) > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   648
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   649
	return (rfs4_servinst_in_grace(cp->server_instance));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   650
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   651
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   652
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   653
 * reset all currently active grace periods
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   654
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   655
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   656
rfs4_grace_reset_all(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   657
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   658
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   659
	int n = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   660
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   661
	rfs4_servinst_t *sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   662
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   663
	mutex_enter(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   664
	for (sip = rfs4_cur_servinst; sip != NULL; sip = sip->prev) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   665
		if (rfs4_servinst_in_grace(sip)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   666
			rfs4_grace_start(sip);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   667
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   668
			n++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   669
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   670
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   671
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   672
	mutex_exit(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   673
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   674
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   675
	    "rfs4_grace_reset_all: reset %d instances", n));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   676
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   677
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   678
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   679
 * start any new instances' grace periods
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   680
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   681
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   682
rfs4_grace_start_new(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   683
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   684
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   685
	int n = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   686
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   687
	rfs4_servinst_t *sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   688
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   689
	mutex_enter(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   690
	for (sip = rfs4_cur_servinst; sip != NULL; sip = sip->prev) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   691
		if (rfs4_servinst_grace_new(sip))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   692
			rfs4_grace_start(sip);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   693
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   694
		n++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   695
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   696
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   697
	mutex_exit(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   698
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   699
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   700
	    "rfs4_grace_start_new: started %d new instances", n));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   701
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   702
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   703
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   704
 * Create a new server instance, and make it the currently active instance.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   705
 * Note that starting the grace period too early will reduce the clients'
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   706
 * recovery window.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   707
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   708
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   709
rfs4_servinst_create(int start_grace)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   710
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   711
	rfs4_servinst_t *sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   712
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   713
	sip = kmem_alloc(sizeof (rfs4_servinst_t), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   714
	rw_init(&sip->rwlock, NULL, RW_DEFAULT, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   715
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   716
	sip->start_time = (time_t)0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   717
	sip->grace_period = (time_t)0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   718
	sip->next = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   719
	sip->prev = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   720
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   721
	mutex_enter(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   722
	if (rfs4_cur_servinst == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   723
		NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   724
		    "rfs4_servinst_create: creating first instance"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   725
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   726
		/* add to linked list */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   727
		sip->prev = rfs4_cur_servinst;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   728
		rfs4_cur_servinst->next = sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   729
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   730
	if (start_grace)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   731
		rfs4_grace_start(sip);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   732
	/* make the new instance "current" */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   733
	rfs4_cur_servinst = sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   734
	mutex_exit(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   735
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   736
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   737
	    "rfs4_servinst_create: new current instance: %p; start_grace: %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   738
	    (void *)sip, start_grace));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   739
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   740
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   741
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   742
 * In future, we might add a rfs4_servinst_destroy(sip) but, for now, destroy
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   743
 * all instances directly.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   744
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   745
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   746
rfs4_servinst_destroy_all(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   747
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   748
	rfs4_servinst_t *sip, *prev, *current;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   749
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   750
	int n = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   751
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   752
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   753
	mutex_enter(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   754
	ASSERT(rfs4_cur_servinst != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   755
	current = rfs4_cur_servinst;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   756
	rfs4_cur_servinst = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   757
	for (sip = current; sip != NULL; sip = prev) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   758
		prev = sip->prev;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   759
		rw_destroy(&sip->rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   760
		kmem_free(sip, sizeof (rfs4_servinst_t));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   761
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   762
		n++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   763
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   764
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   765
	mutex_exit(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   766
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   767
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   768
	    "rfs4_servinst_destroy_all: destroyed %d instances", n));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   769
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   770
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   771
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   772
 * Assign the current server instance to a client_t.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   773
 * Should be called with cp->dbe held.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   774
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   775
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   776
rfs4_servinst_assign(rfs4_client_t *cp, rfs4_servinst_t *sip)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   777
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   778
	ASSERT(rfs4_dbe_refcnt(cp->dbe) > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   779
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   780
	NFS4_DEBUG(rfs4_servinst_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   781
	    "rfs4_servinst_assign: client: %p, old: %p, new: %p", (void *)cp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   782
	    (void *)cp->server_instance, (void *)sip));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   783
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   784
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   785
	 * The lock ensures that if the current instance is in the process
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   786
	 * of changing, we will see the new one.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   787
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   788
	mutex_enter(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   789
	cp->server_instance = sip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   790
	mutex_exit(&rfs4_servinst_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   791
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   792
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   793
rfs4_servinst_t *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   794
rfs4_servinst(rfs4_client_t *cp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   795
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   796
	ASSERT(rfs4_dbe_refcnt(cp->dbe) > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   797
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   798
	return (cp->server_instance);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   799
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   800
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   801
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   802
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   803
nullfree(caddr_t resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   804
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   805
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   806
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   807
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   808
 * This is a fall-through for invalid or not implemented (yet) ops
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   809
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   810
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   811
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   812
rfs4_op_inval(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   813
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   814
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   815
	*cs->statusp = *((nfsstat4 *)&(resop)->nfs_resop4_u) = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   816
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   817
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   818
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   819
 * Check if the security flavor, nfsnum, is in the flavor_list.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   820
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   821
bool_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   822
in_flavor_list(int nfsnum, int *flavor_list, int count)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   823
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   824
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   825
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   826
	for (i = 0; i < count; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   827
		if (nfsnum == flavor_list[i])
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   828
			return (TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   829
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   830
	return (FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   831
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   832
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   833
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   834
 * Used by rfs4_op_secinfo to get the security information from the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   835
 * export structure associated with the component.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   836
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   837
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   838
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   839
do_rfs4_op_secinfo(struct compound_state *cs, char *nm, SECINFO4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   840
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   841
	int error, different_export = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   842
	vnode_t *dvp, *vp, *tvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   843
	struct exportinfo *exi = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   844
	fid_t fid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   845
	uint_t count, i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   846
	secinfo4 *resok_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   847
	struct secinfo *secp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   848
	bool_t did_traverse;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   849
	int dotdot, walk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   850
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   851
	dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   852
	dotdot = (nm[0] == '.' && nm[1] == '.' && nm[2] == '\0');
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   853
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   854
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   855
	 * If dotdotting, then need to check whether it's above the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   856
	 * root of a filesystem, or above an export point.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   857
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   858
	if (dotdot) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   859
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   860
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   861
		 * If dotdotting at the root of a filesystem, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   862
		 * need to traverse back to the mounted-on filesystem
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   863
		 * and do the dotdot lookup there.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   864
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   865
		if (cs->vp->v_flag & VROOT) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   866
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   867
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   868
			 * If at the system root, then can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   869
			 * go up no further.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   870
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   871
			if (VN_CMP(dvp, rootdir))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   872
				return (puterrno4(ENOENT));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   873
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   874
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   875
			 * Traverse back to the mounted-on filesystem
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   876
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   877
			dvp = untraverse(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   878
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   879
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   880
			 * Set the different_export flag so we remember
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   881
			 * to pick up a new exportinfo entry for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   882
			 * this new filesystem.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   883
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   884
			different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   885
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   886
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   887
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   888
			 * If dotdotting above an export point then set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   889
			 * the different_export to get new export info.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   890
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   891
			different_export = nfs_exported(cs->exi, cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   892
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   893
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   894
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   895
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   896
	 * Get the vnode for the component "nm".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   897
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   898
	error = VOP_LOOKUP(dvp, nm, &vp, NULL, 0, NULL, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   899
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   900
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   901
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   902
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   903
	 * If the vnode is in a pseudo filesystem, or if the security flavor
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   904
	 * used in the request is valid but not an explicitly shared flavor,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   905
	 * or the access bit indicates that this is a limited access,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   906
	 * check whether this vnode is visible.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   907
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   908
	if (!different_export &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   909
	    (PSEUDO(cs->exi) || ! is_exported_sec(cs->nfsflavor, cs->exi) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   910
	    cs->access & CS_ACCESS_LIMITED)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   911
		if (! nfs_visible(cs->exi, vp, &different_export)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   912
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   913
			return (puterrno4(ENOENT));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   914
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   915
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   917
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   918
	 * If it's a mountpoint, then traverse it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   919
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   920
	if (vn_ismntpt(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   921
		tvp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   922
		if ((error = traverse(&tvp)) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   923
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   924
			return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   925
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   926
		/* remember that we had to traverse mountpoint */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   927
		did_traverse = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   928
		vp = tvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   929
		different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   930
	} else if (vp->v_vfsp != dvp->v_vfsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   931
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   932
		 * If vp isn't a mountpoint and the vfs ptrs aren't the same,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   933
		 * then vp is probably an LOFS object.  We don't need the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   934
		 * realvp, we just need to know that we might have crossed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   935
		 * a server fs boundary and need to call checkexport4.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   936
		 * (LOFS lookup hides server fs mountpoints, and actually calls
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   937
		 * traverse)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   938
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   939
		different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   940
		did_traverse = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   941
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   942
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   943
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   944
	 * Get the export information for it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   945
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   946
	if (different_export) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   947
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   948
		bzero(&fid, sizeof (fid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   949
		fid.fid_len = MAXFIDSZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   950
		error = vop_fid_pseudo(vp, &fid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   951
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   952
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   953
			return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   954
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   955
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   956
		if (dotdot)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   957
			exi = nfs_vptoexi(NULL, vp, cs->cr, &walk, NULL, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   958
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   959
			exi = checkexport4(&vp->v_vfsp->vfs_fsid, &fid, vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   960
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   961
		if (exi == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   962
			if (did_traverse == TRUE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   963
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   964
				 * If this vnode is a mounted-on vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   965
				 * but the mounted-on file system is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   966
				 * exported, send back the secinfo for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   967
				 * the exported node that the mounted-on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   968
				 * vnode lives in.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   969
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   970
				exi = cs->exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   971
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   972
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   973
				return (puterrno4(EACCES));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   974
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   975
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   976
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   977
		exi = cs->exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   978
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   979
	ASSERT(exi != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   980
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   981
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   982
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   983
	 * Create the secinfo result based on the security information
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   984
	 * from the exportinfo structure (exi).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   985
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   986
	 * Return all flavors for a pseudo node.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   987
	 * For a real export node, return the flavor that the client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   988
	 * has access with.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   989
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   990
	ASSERT(RW_LOCK_HELD(&exported_lock));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   991
	if (PSEUDO(exi)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   992
		count = exi->exi_export.ex_seccnt; /* total sec count */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   993
		resok_val = kmem_alloc(count * sizeof (secinfo4), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   994
		secp = exi->exi_export.ex_secinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   995
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   996
		for (i = 0; i < count; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   997
		    resok_val[i].flavor = secp[i].s_secinfo.sc_rpcnum;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   998
		    if (resok_val[i].flavor == RPCSEC_GSS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   999
			rpcsec_gss_info *info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1000
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1001
			info = &resok_val[i].flavor_info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1002
			info->qop = secp[i].s_secinfo.sc_qop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1003
			info->service =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1004
				(rpc_gss_svc_t)secp[i].s_secinfo.sc_service;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1005
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1006
			/* get oid opaque data */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1007
			info->oid.sec_oid4_len =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1008
				secp[i].s_secinfo.sc_gss_mech_type->length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1009
			info->oid.sec_oid4_val =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1010
				kmem_alloc(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1011
				    secp[i].s_secinfo.sc_gss_mech_type->length,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1012
				    KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1013
			bcopy(secp[i].s_secinfo.sc_gss_mech_type->elements,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1014
				info->oid.sec_oid4_val, info->oid.sec_oid4_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1015
		    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1016
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1017
		resp->SECINFO4resok_len = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1018
		resp->SECINFO4resok_val = resok_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1019
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1020
		int ret_cnt = 0, k = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1021
		int *flavor_list;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1022
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1023
		count = exi->exi_export.ex_seccnt; /* total sec count */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1024
		secp = exi->exi_export.ex_secinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1025
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1026
		flavor_list = kmem_alloc(count * sizeof (int), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1027
		/* find out which flavors to return */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1028
		for (i = 0; i < count; i ++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1029
			int access, flavor, perm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1030
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1031
			flavor = secp[i].s_secinfo.sc_nfsnum;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1032
			perm = secp[i].s_flags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1033
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1034
			access = nfsauth4_secinfo_access(exi, cs->req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1035
						flavor, perm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1036
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1037
			if (! (access & NFSAUTH_DENIED) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1038
			    ! (access & NFSAUTH_WRONGSEC)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1039
				flavor_list[ret_cnt] = flavor;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1040
				ret_cnt++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1041
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1042
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1043
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1044
		/* Create the returning SECINFO value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1045
		resok_val = kmem_alloc(ret_cnt * sizeof (secinfo4), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1046
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1047
		for (i = 0; i < count; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1048
		/* If the flavor is in the flavor list, fill in resok_val. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1049
		    if (in_flavor_list(secp[i].s_secinfo.sc_nfsnum,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1050
						flavor_list, ret_cnt)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1051
			resok_val[k].flavor = secp[i].s_secinfo.sc_rpcnum;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1052
			if (resok_val[k].flavor == RPCSEC_GSS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1053
			    rpcsec_gss_info *info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1054
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1055
			    info = &resok_val[k].flavor_info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1056
			    info->qop = secp[i].s_secinfo.sc_qop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1057
			    info->service =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1058
				(rpc_gss_svc_t)secp[i].s_secinfo.sc_service;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1059
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1060
			    /* get oid opaque data */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1061
			    info->oid.sec_oid4_len =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1062
				secp[i].s_secinfo.sc_gss_mech_type->length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1063
			    info->oid.sec_oid4_val =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1064
				kmem_alloc(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1065
				    secp[i].s_secinfo.sc_gss_mech_type->length,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1066
				    KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1067
			    bcopy(secp[i].s_secinfo.sc_gss_mech_type->elements,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1068
				info->oid.sec_oid4_val, info->oid.sec_oid4_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1069
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1070
			k++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1071
		    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1072
		    if (k >= ret_cnt)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1073
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1074
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1075
		resp->SECINFO4resok_len = ret_cnt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1076
		resp->SECINFO4resok_val = resok_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1077
		kmem_free(flavor_list, count * sizeof (int));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1078
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1079
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1080
	VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1081
	return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1082
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1083
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1084
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1085
 * SECINFO (Operation 33): Obtain required security information on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1086
 * the component name in the format of (security-mechanism-oid, qop, service)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1087
 * triplets.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1088
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1089
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1090
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1091
rfs4_op_secinfo(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1092
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1093
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1094
	SECINFO4res *resp = &resop->nfs_resop4_u.opsecinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1095
	utf8string *utfnm = &argop->nfs_argop4_u.opsecinfo.name;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1096
	uint_t len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1097
	char *nm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1098
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1099
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1100
	 * Current file handle (cfh) should have been set before getting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1101
	 * into this function. If not, return error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1102
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1103
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1104
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1105
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1106
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1107
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1108
	if (cs->vp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1109
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1110
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1111
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1112
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1113
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1114
	 * Verify the component name. If failed, error out, but
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1115
	 * do not error out if the component name is a "..".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1116
	 * SECINFO will return its parents secinfo data for SECINFO "..".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1117
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1118
	if (!utf8_dir_verify(utfnm)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1119
		if (utfnm->utf8string_len != 2 ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1120
				utfnm->utf8string_val[0] != '.' ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1121
				utfnm->utf8string_val[1] != '.') {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1122
			*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1123
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1124
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1125
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1126
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1127
	nm = utf8_to_str(utfnm, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1128
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1129
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1130
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1131
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1132
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1133
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1134
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1135
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1136
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1137
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1138
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1139
	*cs->statusp = resp->status = do_rfs4_op_secinfo(cs, nm, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1140
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1141
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1142
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1143
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1144
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1145
 * Free SECINFO result.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1146
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1147
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1148
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1149
rfs4_op_secinfo_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1150
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1151
	SECINFO4res *resp = &resop->nfs_resop4_u.opsecinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1152
	int count, i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1153
	secinfo4 *resok_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1154
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1155
	/* If this is not an Ok result, nothing to free. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1156
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1157
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1158
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1159
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1160
	count = resp->SECINFO4resok_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1161
	resok_val = resp->SECINFO4resok_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1162
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1163
	for (i = 0; i < count; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1164
	    if (resok_val[i].flavor == RPCSEC_GSS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1165
		rpcsec_gss_info *info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1166
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1167
		info = &resok_val[i].flavor_info;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1168
		kmem_free(info->oid.sec_oid4_val, info->oid.sec_oid4_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1169
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1170
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1171
	kmem_free(resok_val, count * sizeof (secinfo4));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1172
	resp->SECINFO4resok_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1173
	resp->SECINFO4resok_val = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1174
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1175
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1176
/*
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1177
 * do label check on client label and server's file lable.
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1178
 */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1179
static boolean_t
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1180
do_rfs4_label_check(bslabel_t *clabel, vnode_t *vp, int flag)
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1181
{
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1182
	bslabel_t *slabel;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1183
	ts_label_t *tslabel;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1184
	boolean_t result;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1185
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1186
	if ((tslabel = nfs4_getflabel(vp)) == NULL) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1187
		return (B_FALSE);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1188
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1189
	slabel = label2bslabel(tslabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1190
	DTRACE_PROBE4(tx__rfs4__log__info__labelcheck, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1191
	    "comparing server's file label(1) with client label(2) (vp(3))",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1192
	    bslabel_t *, slabel, bslabel_t *, clabel, vnode_t *, vp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1193
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1194
	if (flag == EQUALITY_CHECK)
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1195
		result = blequal(clabel, slabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1196
	else
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1197
		result = bldominates(clabel, slabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1198
	label_rele(tslabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1199
	return (result);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1200
}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1201
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1202
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1203
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1204
rfs4_op_access(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1205
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1206
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1207
	ACCESS4args *args = &argop->nfs_argop4_u.opaccess;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1208
	ACCESS4res *resp = &resop->nfs_resop4_u.opaccess;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1209
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1210
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1211
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1212
	int checkwriteperm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1213
	cred_t *cr = cs->cr;
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1214
	bslabel_t *clabel, *slabel;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1215
	ts_label_t *tslabel;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1216
	boolean_t admin_low_client;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1217
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1218
#if 0	/* XXX allow access even if !cs->access. Eventually only pseudo fs */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1219
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1220
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1221
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1222
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1223
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1224
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1225
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1226
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1227
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1228
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1229
	ASSERT(cr != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1230
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1231
	vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1232
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1233
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1234
	 * If the file system is exported read only, it is not appropriate
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1235
	 * to check write permissions for regular files and directories.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1236
	 * Special files are interpreted by the client, so the underlying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1237
	 * permissions are sent back to the client for interpretation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1238
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1239
	if (rdonly4(cs->exi, cs->vp, req) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1240
		(vp->v_type == VREG || vp->v_type == VDIR))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1241
		checkwriteperm = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1242
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1243
		checkwriteperm = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1244
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1245
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1246
	 * XXX
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1247
	 * We need the mode so that we can correctly determine access
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1248
	 * permissions relative to a mandatory lock file.  Access to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1249
	 * mandatory lock files is denied on the server, so it might
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1250
	 * as well be reflected to the server during the open.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1251
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1252
	va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1253
	error = VOP_GETATTR(vp, &va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1254
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1255
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1256
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1257
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1258
	resp->access = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1259
	resp->supported = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1260
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1261
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1262
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1263
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1264
		DTRACE_PROBE2(tx__rfs4__log__info__opaccess__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1265
		    "got client label from request(1)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1266
		    struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1267
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1268
			if ((tslabel = nfs4_getflabel(vp)) == NULL) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1269
				*cs->statusp = resp->status = puterrno4(EACCES);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1270
				return;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1271
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1272
			slabel = label2bslabel(tslabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1273
			DTRACE_PROBE3(tx__rfs4__log__info__opaccess__slabel,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1274
			    char *, "got server label(1) for vp(2)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1275
			    bslabel_t *, slabel, vnode_t *, vp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1276
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1277
			admin_low_client = B_FALSE;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1278
		} else
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1279
			admin_low_client = B_TRUE;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1280
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1281
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1282
	if (args->access & ACCESS4_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1283
		error = VOP_ACCESS(vp, VREAD, 0, cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1284
		if (!error && !MANDLOCK(vp, va.va_mode) &&
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1285
		    (!is_system_labeled() || admin_low_client ||
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1286
		    bldominates(clabel, slabel)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1287
			resp->access |= ACCESS4_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1288
		resp->supported |= ACCESS4_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1289
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1290
	if ((args->access & ACCESS4_LOOKUP) && vp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1291
		error = VOP_ACCESS(vp, VEXEC, 0, cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1292
		if (!error && (!is_system_labeled() || admin_low_client ||
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1293
		    bldominates(clabel, slabel)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1294
			resp->access |= ACCESS4_LOOKUP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1295
		resp->supported |= ACCESS4_LOOKUP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1296
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1297
	if (checkwriteperm &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1298
	    (args->access & (ACCESS4_MODIFY|ACCESS4_EXTEND))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1299
		error = VOP_ACCESS(vp, VWRITE, 0, cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1300
		if (!error && !MANDLOCK(vp, va.va_mode) &&
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1301
		    (!is_system_labeled() || admin_low_client ||
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1302
		    blequal(clabel, slabel)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1303
			resp->access |=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1304
			    (args->access & (ACCESS4_MODIFY|ACCESS4_EXTEND));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1305
		resp->supported |= (ACCESS4_MODIFY|ACCESS4_EXTEND);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1306
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1307
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1308
	if (checkwriteperm &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1309
	    (args->access & ACCESS4_DELETE) && vp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1310
		error = VOP_ACCESS(vp, VWRITE, 0, cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1311
		if (!error && (!is_system_labeled() || admin_low_client ||
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1312
		    blequal(clabel, slabel)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1313
			resp->access |= ACCESS4_DELETE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1314
		resp->supported |= ACCESS4_DELETE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1315
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1316
	if (args->access & ACCESS4_EXECUTE && vp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1317
		error = VOP_ACCESS(vp, VEXEC, 0, cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1318
		if (!error && !MANDLOCK(vp, va.va_mode) &&
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1319
		    (!is_system_labeled() || admin_low_client ||
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1320
		    bldominates(clabel, slabel)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1321
			resp->access |= ACCESS4_EXECUTE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1322
		resp->supported |= ACCESS4_EXECUTE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1323
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1324
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1325
	if (is_system_labeled() && !admin_low_client)
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1326
		label_rele(tslabel);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  1327
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1328
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1329
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1330
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1331
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1332
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1333
rfs4_op_commit(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1334
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1335
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1336
	COMMIT4args *args = &argop->nfs_argop4_u.opcommit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1337
	COMMIT4res *resp = &resop->nfs_resop4_u.opcommit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1338
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1339
	vnode_t *vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1340
	cred_t *cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1341
	vattr_t va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1342
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1343
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1344
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1345
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1346
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1347
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1348
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1349
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1350
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1351
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1352
	if (args->offset + args->count < args->offset) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1353
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1354
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1355
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1356
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1357
	va.va_mask = AT_UID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1358
	error = VOP_GETATTR(vp, &va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1360
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1361
	 * If we can't get the attributes, then we can't do the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1362
	 * right access checking.  So, we'll fail the request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1363
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1364
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1365
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1366
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1367
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1368
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1369
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1370
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1371
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1372
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1373
	if (vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1374
		if (vp->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1375
			resp->status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1376
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1377
			resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1378
		*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1379
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1380
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1381
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1382
	if (crgetuid(cr) != va.va_uid &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1383
	    (error = VOP_ACCESS(vp, VWRITE, 0, cs->cr))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1384
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1385
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1386
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1387
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1388
	error = VOP_PUTPAGE(vp, args->offset, args->count, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1389
	if (!error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1390
		error = VOP_FSYNC(vp, FNODSYNC, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1391
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1392
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1393
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1394
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1395
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1396
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1397
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1398
	resp->writeverf = Write4verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1399
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1400
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1401
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1402
 * rfs4_op_mknod is called from rfs4_op_create after all initial verification
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1403
 * was completed. It does the nfsv4 create for special files.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1404
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1405
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1406
static vnode_t *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1407
do_rfs4_op_mknod(CREATE4args *args, CREATE4res *resp, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1408
	struct compound_state *cs, vattr_t *vap, char *nm)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1409
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1410
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1411
	cred_t *cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1412
	vnode_t *dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1413
	vnode_t *vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1414
	int mode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1415
	enum vcexcl excl;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1417
	switch (args->type) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1418
	case NF4CHR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1419
	case NF4BLK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1420
		if (secpolicy_sys_devices(cr) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1421
			*cs->statusp = resp->status = NFS4ERR_PERM;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1422
			return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1423
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1424
		if (args->type == NF4CHR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1425
			vap->va_type = VCHR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1426
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1427
			vap->va_type = VBLK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1428
		vap->va_rdev = makedevice(args->ftype4_u.devdata.specdata1,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1429
					args->ftype4_u.devdata.specdata2);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1430
		vap->va_mask |= AT_RDEV;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1431
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1432
	case NF4SOCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1433
		vap->va_type = VSOCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1434
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1435
	case NF4FIFO:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1436
		vap->va_type = VFIFO;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1437
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1438
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1439
		*cs->statusp = resp->status = NFS4ERR_BADTYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1440
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1441
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1442
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1443
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1444
	 * Must specify the mode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1445
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1446
	if (!(vap->va_mask & AT_MODE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1447
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1448
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1449
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1450
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1451
	excl = EXCL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1452
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1453
	mode = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1454
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1455
	error = VOP_CREATE(dvp, nm, vap, excl, mode, &vp, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1456
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1457
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1458
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1459
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1460
	return (vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1461
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1462
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1463
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1464
 * nfsv4 create is used to create non-regular files. For regular files,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1465
 * use nfsv4 open.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1466
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1467
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1468
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1469
rfs4_op_create(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1470
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1471
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1472
	CREATE4args *args = &argop->nfs_argop4_u.opcreate;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1473
	CREATE4res *resp = &resop->nfs_resop4_u.opcreate;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1474
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1475
	struct vattr bva, iva, iva2, ava, *vap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1476
	cred_t *cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1477
	vnode_t *dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1478
	vnode_t *vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1479
	char *nm, *lnm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1480
	uint_t len, llen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1481
	int syncval = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1482
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1483
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1484
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1485
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1486
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1487
	resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1488
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1489
	if (dvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1490
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1491
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1492
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1493
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1494
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1495
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1496
	 * do not allow to create an object in this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1497
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1498
	if (vn_ismntpt(dvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1499
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1500
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1501
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1502
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1503
	/* Verify that type is correct */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1504
	switch (args->type) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1505
	case NF4LNK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1506
	case NF4BLK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1507
	case NF4CHR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1508
	case NF4SOCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1509
	case NF4FIFO:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1510
	case NF4DIR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1511
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1512
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1513
		*cs->statusp = resp->status = NFS4ERR_BADTYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1514
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1515
	};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1516
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1517
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1518
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1519
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1520
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1521
	if (dvp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1522
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1523
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1524
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1525
	if (!utf8_dir_verify(&args->objname)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1526
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1527
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1528
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1529
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1530
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1531
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1532
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1533
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1534
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1535
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1536
	 * Name of newly created object
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1537
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1538
	nm = utf8_to_fn(&args->objname, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1539
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1540
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1541
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1542
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1543
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1544
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1545
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1546
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1547
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1548
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1549
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1550
	resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1551
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1552
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1553
	nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1554
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1555
	status = do_rfs4_set_attrs(&resp->attrset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1556
					&args->createattrs, cs, &sarg,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1557
					&ntov, NFS4ATTR_SETIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1558
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1559
	if (sarg.vap->va_mask == 0 && status == NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1560
		status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1561
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1562
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1563
		*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1564
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1565
		nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1566
		resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1567
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1568
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1569
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1570
	/* Get "before" change value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1571
	bva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1572
	error = VOP_GETATTR(dvp, &bva, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1573
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1574
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1575
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1576
		nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1577
		resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1578
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1579
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1580
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.before, bva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1581
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1582
	vap = sarg.vap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1583
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1584
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1585
	 * Set default initial values for attributes when not specified
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1586
	 * in createattrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1587
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1588
	if ((vap->va_mask & AT_UID) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1589
		vap->va_uid = crgetuid(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1590
		vap->va_mask |= AT_UID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1591
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1592
	if ((vap->va_mask & AT_GID) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1593
		vap->va_gid = crgetgid(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1594
		vap->va_mask |= AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1595
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1596
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1597
	vap->va_mask |= AT_TYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1598
	switch (args->type) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1599
	case NF4DIR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1600
		vap->va_type = VDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1601
		if ((vap->va_mask & AT_MODE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1602
			vap->va_mode = 0700;	/* default: owner rwx only */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1603
			vap->va_mask |= AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1604
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1605
		error = VOP_MKDIR(dvp, nm, vap, &vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1606
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1607
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1608
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1609
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1610
		 * Get the initial "after" sequence number, if it fails,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1611
		 * set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1612
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1613
		iva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1614
		if (VOP_GETATTR(dvp, &iva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1615
			iva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1616
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1617
	case NF4LNK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1618
		vap->va_type = VLNK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1619
		if ((vap->va_mask & AT_MODE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1620
			vap->va_mode = 0700;	/* default: owner rwx only */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1621
			vap->va_mask |= AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1622
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1623
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1624
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1625
		 * symlink names must be treated as data
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1626
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1627
		lnm = utf8_to_str(&args->ftype4_u.linkdata, &llen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1628
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1629
		if (lnm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1630
			*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1631
			kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1632
			nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1633
			resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1634
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1635
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1636
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1637
		if (llen > MAXPATHLEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1638
			*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1639
			kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1640
			kmem_free(lnm, llen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1641
			nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1642
			resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1643
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1644
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1645
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1646
		error = VOP_SYMLINK(dvp, nm, vap, lnm, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1647
		if (lnm != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1648
			kmem_free(lnm, llen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1649
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1650
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1651
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1652
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1653
		 * Get the initial "after" sequence number, if it fails,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1654
		 * set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1655
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1656
		iva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1657
		if (VOP_GETATTR(dvp, &iva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1658
			iva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1659
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1660
		error = VOP_LOOKUP(dvp, nm, &vp, NULL, 0, NULL, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1661
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1662
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1663
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1664
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1665
		 * va_seq is not safe over VOP calls, check it again
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1666
		 * if it has changed zero out iva to force atomic = FALSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1667
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1668
		iva2.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1669
		if (VOP_GETATTR(dvp, &iva2, 0, cs->cr) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1670
						iva2.va_seq != iva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1671
			iva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1672
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1673
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1674
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1675
		 * probably a special file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1676
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1677
		if ((vap->va_mask & AT_MODE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1678
			vap->va_mode = 0600;	/* default: owner rw only */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1679
			vap->va_mask |= AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1680
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1681
		syncval = FNODSYNC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1682
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1683
		 * We know this will only generate one VOP call
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1684
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1685
		vp = do_rfs4_op_mknod(args, resp, req, cs, vap, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1686
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1687
		if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1688
			kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1689
			nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1690
			resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1691
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1692
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1693
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1694
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1695
		 * Get the initial "after" sequence number, if it fails,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1696
		 * set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1697
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1698
		iva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1699
		if (VOP_GETATTR(dvp, &iva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1700
			iva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1701
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1702
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1703
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1704
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1705
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1706
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1707
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1708
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1709
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1710
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1711
	 * Force modified data and metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1712
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1713
	(void) VOP_FSYNC(dvp, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1714
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1715
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1716
		if (vp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1717
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1718
		nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1719
		resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1720
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1721
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1722
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1723
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1724
	 * Finish setup of cinfo response, "before" value already set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1725
	 * Get "after" change value, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1726
	 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1727
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1728
	ava.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1729
	if (VOP_GETATTR(dvp, &ava, 0, cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1730
		ava.va_ctime = bva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1731
		ava.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1732
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1733
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.after, ava.va_ctime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1734
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1735
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1736
	 * True verification that object was created with correct
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1737
	 * attrs is impossible.  The attrs could have been changed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1738
	 * immediately after object creation.  If attributes did
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1739
	 * not verify, the only recourse for the server is to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1740
	 * destroy the object.  Maybe if some attrs (like gid)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1741
	 * are set incorrectly, the object should be destroyed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1742
	 * however, seems bad as a default policy.  Do we really
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1743
	 * want to destroy an object over one of the times not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1744
	 * verifying correctly?  For these reasons, the server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1745
	 * currently sets bits in attrset for createattrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1746
	 * that were set; however, no verification is done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1747
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1748
	 * vmask_to_nmask accounts for vattr bits set on create
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1749
	 *	[do_rfs4_set_attrs() only sets resp bits for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1750
	 *	 non-vattr/vfs bits.]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1751
	 * Mask off any bits set by default so as not to return
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1752
	 * more attrset bits than were requested in createattrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1753
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1754
	nfs4_vmask_to_nmask(sarg.vap->va_mask, &resp->attrset);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1755
	resp->attrset &= args->createattrs.attrmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1756
	nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1757
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1758
	error = makefh4(&cs->fh, vp, cs->exi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1759
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1760
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1761
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1762
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1763
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1764
	 * The cinfo.atomic = TRUE only if we got no errors, we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1765
	 * non-zero va_seq's, and it has incremented by exactly one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1766
	 * during the creation and it didn't change during the VOP_LOOKUP
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1767
	 * or VOP_FSYNC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1768
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1769
	if (!error && bva.va_seq && iva.va_seq && ava.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1770
			iva.va_seq == (bva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1771
			iva.va_seq == ava.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1772
		resp->cinfo.atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1773
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1774
		resp->cinfo.atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1775
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1776
	(void) VOP_FSYNC(vp, syncval, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1777
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1778
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1779
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1780
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1781
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1782
	if (cs->vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1783
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1784
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1785
	cs->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1786
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1787
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1788
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1789
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1790
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1791
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1792
rfs4_op_delegreturn(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1793
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1794
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1795
	DELEGRETURN4args *args = &argop->nfs_argop4_u.opdelegreturn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1796
	DELEGRETURN4res *resp = &resop->nfs_resop4_u.opdelegreturn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1797
	rfs4_deleg_state_t *dsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1798
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1799
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1800
	status = rfs4_get_deleg_state(&args->deleg_stateid, &dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1801
	resp->status = *cs->statusp = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1802
	if (status != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1803
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1804
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1805
	/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1806
	if (cs->vp != dsp->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1807
		resp->status = *cs->statusp = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1808
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1809
		rfs4_return_deleg(dsp, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1810
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1811
	rfs4_update_lease(dsp->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1812
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1813
	rfs4_deleg_state_rele(dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1814
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1815
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1816
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1817
 * Check to see if a given "flavor" is an explicitly shared flavor.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1818
 * The assumption of this routine is the "flavor" is already a valid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1819
 * flavor in the secinfo list of "exi".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1820
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1821
 *	e.g.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1822
 *		# share -o sec=flavor1 /export
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1823
 *		# share -o sec=flavor2 /export/home
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1824
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1825
 *		flavor2 is not an explicitly shared flavor for /export,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1826
 *		however it is in the secinfo list for /export thru the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1827
 *		server namespace setup.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1828
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1829
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1830
is_exported_sec(int flavor, struct exportinfo *exi)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1831
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1832
	int	i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1833
	struct secinfo *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1834
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1835
	sp = exi->exi_export.ex_secinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1836
	for (i = 0; i < exi->exi_export.ex_seccnt; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1837
		if (flavor == sp[i].s_secinfo.sc_nfsnum ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1838
		    sp[i].s_secinfo.sc_nfsnum == AUTH_NONE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1839
			return (SEC_REF_EXPORTED(&sp[i]));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1840
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1841
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1842
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1843
	/* Should not reach this point based on the assumption */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1844
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1845
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1846
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1847
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1848
 * Check if the security flavor used in the request matches what is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1849
 * required at the export point or at the root pseudo node (exi_root).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1850
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1851
 * returns 1 if there's a match or if exported with AUTH_NONE; 0 otherwise.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1852
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1853
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1854
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1855
secinfo_match_or_authnone(struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1856
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1857
	int	i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1858
	struct secinfo *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1859
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1860
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1861
	 * Check cs->nfsflavor (from the request) against
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1862
	 * the current export data in cs->exi.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1863
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1864
	sp = cs->exi->exi_export.ex_secinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1865
	for (i = 0; i < cs->exi->exi_export.ex_seccnt; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1866
		if (cs->nfsflavor == sp[i].s_secinfo.sc_nfsnum ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1867
		    sp[i].s_secinfo.sc_nfsnum == AUTH_NONE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1868
			return (1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1869
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1870
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1871
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1872
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1873
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1874
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1875
 * Check the access authority for the client and return the correct error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1876
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1877
nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1878
call_checkauth4(struct compound_state *cs, struct svc_req *req)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1879
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1880
	int	authres;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1881
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1882
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1883
	 * First, check if the security flavor used in the request
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1884
	 * are among the flavors set in the server namespace.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1885
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1886
	if (!secinfo_match_or_authnone(cs)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1887
		*cs->statusp = NFS4ERR_WRONGSEC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1888
		return (*cs->statusp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1889
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1890
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1891
	authres = checkauth4(cs, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1892
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1893
	if (authres > 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1894
		*cs->statusp = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1895
		if (! (cs->access & CS_ACCESS_LIMITED))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1896
			cs->access = CS_ACCESS_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1897
	} else if (authres == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1898
		*cs->statusp = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1899
	} else if (authres == -2) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1900
		*cs->statusp = NFS4ERR_WRONGSEC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1901
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1902
		*cs->statusp = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1903
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1904
	return (*cs->statusp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1905
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1906
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1907
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1908
 * bitmap4_to_attrmask is called by getattr and readdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1909
 * It sets up the vattr mask and determines whether vfsstat call is needed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1910
 * based on the input bitmap.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1911
 * Returns nfsv4 status.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1912
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1913
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1914
bitmap4_to_attrmask(bitmap4 breq, struct nfs4_svgetit_arg *sargp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1915
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1916
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1917
	uint_t	va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1918
	struct statvfs64 *sbp = sargp->sbp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1919
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1920
	sargp->sbp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1921
	sargp->flag = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1922
	sargp->rdattr_error = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1923
	sargp->mntdfid_set = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1924
	if (sargp->cs->vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1925
		sargp->xattr = get_fh4_flag(&sargp->cs->fh,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1926
					    FH4_ATTRDIR | FH4_NAMEDATTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1927
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1928
		sargp->xattr = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1929
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1930
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1931
	 * Set rdattr_error_req to true if return error per
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1932
	 * failed entry rather than fail the readdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1933
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1934
	if (breq & FATTR4_RDATTR_ERROR_MASK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1935
		sargp->rdattr_error_req = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1936
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1937
		sargp->rdattr_error_req = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1938
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1939
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1940
	 * generate the va_mask
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1941
	 * Handle the easy cases first
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1942
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1943
	switch (breq) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1944
	case NFS4_NTOV_ATTR_MASK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1945
		sargp->vap->va_mask = NFS4_NTOV_ATTR_AT_MASK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1946
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1947
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1948
	case NFS4_FS_ATTR_MASK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1949
		sargp->vap->va_mask = NFS4_FS_ATTR_AT_MASK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1950
		sargp->sbp = sbp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1951
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1952
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1953
	case NFS4_NTOV_ATTR_CACHE_MASK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1954
		sargp->vap->va_mask = NFS4_NTOV_ATTR_CACHE_AT_MASK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1955
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1956
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1957
	case FATTR4_LEASE_TIME_MASK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1958
		sargp->vap->va_mask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1959
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1960
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1961
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1962
		va_mask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1963
		for (i = 0; i < nfs4_ntov_map_size; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1964
			if ((breq & nfs4_ntov_map[i].fbit) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1965
							nfs4_ntov_map[i].vbit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1966
				va_mask |= nfs4_ntov_map[i].vbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1967
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1968
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1969
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1970
		 * Check is vfsstat is needed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1971
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1972
		if (breq & NFS4_FS_ATTR_MASK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1973
			sargp->sbp = sbp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1974
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1975
		sargp->vap->va_mask = va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1976
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1977
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1978
	/* NOTREACHED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1979
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1980
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1981
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1982
 * bitmap4_get_sysattrs is called by getattr and readdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1983
 * It calls both VOP_GETATTR and VFS_STATVFS calls to get the attrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1984
 * Returns nfsv4 status.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1985
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1986
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1987
bitmap4_get_sysattrs(struct nfs4_svgetit_arg *sargp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1988
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1989
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1990
	struct compound_state *cs = sargp->cs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1991
	vnode_t *vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1992
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1993
	if (sargp->sbp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1994
		if (error = VFS_STATVFS(vp->v_vfsp, sargp->sbp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1995
			sargp->sbp = NULL;	/* to identify error */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1996
			return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1997
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1998
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1999
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2000
	return (rfs4_vop_getattr(vp, sargp->vap, 0, cs->cr));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2001
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2002
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2003
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2004
nfs4_ntov_table_init(struct nfs4_ntov_table *ntovp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2005
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2006
	ntovp->na = kmem_zalloc(sizeof (union nfs4_attr_u) * nfs4_ntov_map_size,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2007
			KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2008
	ntovp->attrcnt = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2009
	ntovp->vfsstat = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2010
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2011
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2012
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2013
nfs4_ntov_table_free(struct nfs4_ntov_table *ntovp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2014
	struct nfs4_svgetit_arg *sargp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2015
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2016
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2017
	union nfs4_attr_u *na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2018
	uint8_t *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2019
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2020
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2021
	 * XXX Should do the same checks for whether the bit is set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2022
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2023
	for (i = 0, na = ntovp->na, amap = ntovp->amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2024
		i < ntovp->attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2025
		(void) (*nfs4_ntov_map[*amap].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2026
			NFS4ATTR_FREEIT, sargp, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2027
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2028
	if ((sargp->op == NFS4ATTR_SETIT) || (sargp->op == NFS4ATTR_VERIT)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2029
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2030
		 * xdr_free for getattr will be done later
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2031
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2032
		for (i = 0, na = ntovp->na, amap = ntovp->amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2033
			i < ntovp->attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2034
			xdr_free(nfs4_ntov_map[*amap].xfunc, (caddr_t)na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2035
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2036
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2037
	kmem_free(ntovp->na, sizeof (union nfs4_attr_u) * nfs4_ntov_map_size);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2038
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2039
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2040
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2041
 * do_rfs4_op_getattr gets the system attrs and converts into fattr4.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2042
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2043
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2044
do_rfs4_op_getattr(bitmap4 breq, fattr4 *fattrp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2045
	struct nfs4_svgetit_arg *sargp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2046
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2047
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2048
	int i, k;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2049
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2050
	XDR xdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2051
	ulong_t xdr_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2052
	char *xdr_attrs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2053
	nfsstat4 status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2054
	nfsstat4 prev_rdattr_error = sargp->rdattr_error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2055
	union nfs4_attr_u *na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2056
	uint8_t *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2057
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2058
	sargp->op = NFS4ATTR_GETIT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2059
	sargp->flag = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2060
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2061
	fattrp->attrmask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2062
	/* if no bits requested, then return empty fattr4 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2063
	if (breq == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2064
		fattrp->attrlist4_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2065
		fattrp->attrlist4 = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2066
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2067
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2068
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2069
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2070
	 * return NFS4ERR_INVAL when client requests write-only attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2071
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2072
	if (breq & (FATTR4_TIME_ACCESS_SET_MASK | FATTR4_TIME_MODIFY_SET_MASK))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2073
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2074
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2075
	nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2076
	na = ntov.na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2077
	amap = ntov.amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2078
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2079
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2080
	 * Now loop to get or verify the attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2081
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2082
	for (i = 0; i < nfs4_ntov_map_size; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2083
		if (breq & nfs4_ntov_map[i].fbit) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2084
			if ((*nfs4_ntov_map[i].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2085
				    NFS4ATTR_SUPPORTED, sargp, NULL) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2086
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2087
				error = (*nfs4_ntov_map[i].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2088
						NFS4ATTR_GETIT, sargp, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2089
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2090
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2091
				 * Possible error values:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2092
				 * >0 if sv_getit failed to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2093
				 * get the attr; 0 if succeeded;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2094
				 * <0 if rdattr_error and the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2095
				 * attribute cannot be returned.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2096
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2097
				if (error && !(sargp->rdattr_error_req))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2098
					goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2099
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2100
				 * If error then just for entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2101
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2102
				if (error == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2103
					fattrp->attrmask |=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2104
						nfs4_ntov_map[i].fbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2105
					*amap++ =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2106
						(uint8_t)nfs4_ntov_map[i].nval;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2107
					na++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2108
					(ntov.attrcnt)++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2109
				} else if ((error > 0) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2110
					(sargp->rdattr_error == NFS4_OK)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2111
					sargp->rdattr_error = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2112
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2113
				error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2114
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2115
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2116
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2117
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2118
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2119
	 * If rdattr_error was set after the return value for it was assigned,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2120
	 * update it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2121
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2122
	if (prev_rdattr_error != sargp->rdattr_error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2123
		na = ntov.na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2124
		amap = ntov.amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2125
		for (i = 0; i < ntov.attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2126
			k = *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2127
			if (k < FATTR4_RDATTR_ERROR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2128
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2129
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2130
			if ((k == FATTR4_RDATTR_ERROR) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2131
			    ((*nfs4_ntov_map[k].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2132
				NFS4ATTR_SUPPORTED, sargp, NULL) == 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2133
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2134
				(void) (*nfs4_ntov_map[k].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2135
						NFS4ATTR_GETIT, sargp, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2136
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2137
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2138
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2139
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2140
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2141
	xdr_size = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2142
	na = ntov.na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2143
	amap = ntov.amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2144
	for (i = 0; i < ntov.attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2145
		xdr_size += xdr_sizeof(nfs4_ntov_map[*amap].xfunc, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2146
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2147
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2148
	fattrp->attrlist4_len = xdr_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2149
	if (xdr_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2150
		/* freed by rfs4_op_getattr_free() */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2151
		fattrp->attrlist4 = xdr_attrs = kmem_zalloc(xdr_size, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2152
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2153
		xdrmem_create(&xdr, xdr_attrs, xdr_size, XDR_ENCODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2154
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2155
		na = ntov.na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2156
		amap = ntov.amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2157
		for (i = 0; i < ntov.attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2158
			if (!(*nfs4_ntov_map[*amap].xfunc)(&xdr, na)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2159
				cmn_err(CE_WARN, "do_rfs4_op_getattr: xdr "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2160
					"encode of attribute %d failed\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2161
					*amap);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2162
				status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2163
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2164
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2165
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2166
		/* xdrmem_destroy(&xdrs); */	/* NO-OP */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2167
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2168
		fattrp->attrlist4 = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2169
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2170
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2171
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2172
	nfs4_ntov_table_free(&ntov, sargp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2173
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2174
	if (error != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2175
		status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2176
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2177
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2178
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2179
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2180
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2181
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2182
rfs4_op_getattr(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2183
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2184
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2185
	GETATTR4args *args = &argop->nfs_argop4_u.opgetattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2186
	GETATTR4res *resp = &resop->nfs_resop4_u.opgetattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2187
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2188
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2189
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2190
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2191
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2192
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2193
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2194
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2195
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2196
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2197
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2198
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2199
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2200
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2201
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2202
	sarg.cs = cs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2203
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2204
	status = bitmap4_to_attrmask(args->attr_request, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2205
	if (status == NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2206
		status = bitmap4_get_sysattrs(&sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2207
		if (status == NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2208
			status = do_rfs4_op_getattr(args->attr_request,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2209
				&resp->obj_attributes, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2210
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2211
	*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2212
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2213
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2214
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2215
rfs4_op_getattr_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2216
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2217
	GETATTR4res *resp = &resop->nfs_resop4_u.opgetattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2218
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2219
	nfs4_fattr4_free(&resp->obj_attributes);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2220
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2221
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2222
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2223
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2224
rfs4_op_getfh(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2225
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2226
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2227
	GETFH4res *resp = &resop->nfs_resop4_u.opgetfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2228
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2229
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2230
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2231
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2232
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2233
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2234
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2235
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2236
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2237
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2238
	resp->object.nfs_fh4_val =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2239
		kmem_alloc(cs->fh.nfs_fh4_len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2240
	nfs_fh4_copy(&cs->fh, &resp->object);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2241
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2242
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2243
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2244
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2245
rfs4_op_getfh_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2246
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2247
	GETFH4res *resp = &resop->nfs_resop4_u.opgetfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2248
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2249
	if (resp->status == NFS4_OK &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2250
	    resp->object.nfs_fh4_val != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2251
		kmem_free(resp->object.nfs_fh4_val, resp->object.nfs_fh4_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2252
		resp->object.nfs_fh4_val = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2253
		resp->object.nfs_fh4_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2254
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2255
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2256
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2257
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2258
 * illegal: args: void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2259
 *	    res : status (NFS4ERR_OP_ILLEGAL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2260
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2261
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2262
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2263
rfs4_op_illegal(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2264
	struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2265
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2266
	ILLEGAL4res *resp = &resop->nfs_resop4_u.opillegal;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2267
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2268
	resop->resop = OP_ILLEGAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2269
	*cs->statusp = resp->status = NFS4ERR_OP_ILLEGAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2270
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2271
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2272
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2273
 * link: args: SAVED_FH: file, CURRENT_FH: target directory
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2274
 *	 res: status. If success - CURRENT_FH unchanged, return change_info
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2275
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2276
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2277
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2278
rfs4_op_link(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2279
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2280
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2281
	LINK4args *args = &argop->nfs_argop4_u.oplink;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2282
	LINK4res *resp = &resop->nfs_resop4_u.oplink;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2283
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2284
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2285
	vnode_t *dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2286
	struct vattr bdva, idva, adva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2287
	char *nm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2288
	uint_t  len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2289
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2290
	/* SAVED_FH: source object */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2291
	vp = cs->saved_vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2292
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2293
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2294
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2295
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2296
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2297
	/* CURRENT_FH: target directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2298
	dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2299
	if (dvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2300
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2301
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2302
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2303
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2304
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2305
	 * If there is a non-shared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2306
	 * do not allow to link any file in this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2307
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2308
	if (vn_ismntpt(dvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2309
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2310
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2311
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2312
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2313
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2314
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2315
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2316
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2317
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2318
	/* Check source object's type validity */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2319
	if (vp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2320
		*cs->statusp = resp->status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2321
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2322
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2323
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2324
	/* Check target directory's type */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2325
	if (dvp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2326
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2327
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2328
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2329
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2330
	if (cs->saved_exi != cs->exi) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2331
		*cs->statusp = resp->status = NFS4ERR_XDEV;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2332
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2333
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2334
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2335
	if (!utf8_dir_verify(&args->newname)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2336
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2337
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2338
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2339
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2340
	nm = utf8_to_fn(&args->newname, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2341
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2342
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2343
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2344
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2345
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2346
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2347
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2348
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2349
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2350
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2351
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2352
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2353
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2354
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2355
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2356
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2357
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2358
	/* Get "before" change value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2359
	bdva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2360
	error = VOP_GETATTR(dvp, &bdva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2361
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2362
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2363
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2364
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2365
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2366
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2367
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.before, bdva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2368
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2369
	error = VOP_LINK(dvp, vp, nm, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2370
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2371
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2372
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2373
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2374
	 * Get the initial "after" sequence number, if it fails, set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2375
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2376
	idva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2377
	if (VOP_GETATTR(dvp, &idva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2378
		idva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2379
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2380
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2381
	 * Force modified data and metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2382
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2383
	(void) VOP_FSYNC(vp, FNODSYNC, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2384
	(void) VOP_FSYNC(dvp, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2385
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2386
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2387
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2388
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2389
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2390
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2391
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2392
	 * Get "after" change value, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2393
	 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2394
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2395
	adva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2396
	if (VOP_GETATTR(dvp, &adva, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2397
		adva.va_ctime = bdva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2398
		adva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2399
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2400
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2401
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.after, adva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2402
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2403
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2404
	 * The cinfo.atomic = TRUE only if we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2405
	 * non-zero va_seq's, and it has incremented by exactly one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2406
	 * during the VOP_LINK and it didn't change during the VOP_FSYNC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2407
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2408
	if (bdva.va_seq && idva.va_seq && adva.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2409
			idva.va_seq == (bdva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2410
			idva.va_seq == adva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2411
		resp->cinfo.atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2412
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2413
		resp->cinfo.atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2414
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2415
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2416
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2417
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2418
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2419
 * Used by rfs4_op_lookup and rfs4_op_lookupp to do the actual work.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2420
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2421
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2422
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2423
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2424
do_rfs4_op_lookup(char *nm, uint_t buflen, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2425
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2426
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2427
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2428
	int different_export = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2429
	vnode_t *vp, *tvp, *pre_tvp = NULL, *oldvp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2430
	struct exportinfo *exi = NULL, *pre_exi = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2431
	nfsstat4 stat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2432
	fid_t fid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2433
	int attrdir, dotdot, walk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2434
	bool_t is_newvp = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2435
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2436
	if (cs->vp->v_flag & V_XATTRDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2437
		attrdir = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2438
		ASSERT(get_fh4_flag(&cs->fh, FH4_ATTRDIR));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2439
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2440
		attrdir = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2441
		ASSERT(! get_fh4_flag(&cs->fh, FH4_ATTRDIR));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2442
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2443
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2444
	dotdot = (nm[0] == '.' && nm[1] == '.' && nm[2] == '\0');
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2445
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2446
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2447
	 * If dotdotting, then need to check whether it's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2448
	 * above the root of a filesystem, or above an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2449
	 * export point.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2450
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2451
	if (dotdot) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2452
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2453
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2454
		 * If dotdotting at the root of a filesystem, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2455
		 * need to traverse back to the mounted-on filesystem
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2456
		 * and do the dotdot lookup there.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2457
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2458
		if (cs->vp->v_flag & VROOT) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2459
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2460
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2461
			 * If at the system root, then can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2462
			 * go up no further.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2463
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2464
			if (VN_CMP(cs->vp, rootdir))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2465
				return (puterrno4(ENOENT));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2466
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2467
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2468
			 * Traverse back to the mounted-on filesystem
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2469
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2470
			cs->vp = untraverse(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2471
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2472
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2473
			 * Set the different_export flag so we remember
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2474
			 * to pick up a new exportinfo entry for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2475
			 * this new filesystem.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2476
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2477
			different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2478
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2479
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2480
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2481
			 * If dotdotting above an export point then set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2482
			 * the different_export to get new export info.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2483
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2484
			different_export = nfs_exported(cs->exi, cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2485
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2486
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2487
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2488
	error = VOP_LOOKUP(cs->vp, nm, &vp, NULL, 0, NULL, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2489
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2490
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2491
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2492
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2493
	 * If the vnode is in a pseudo filesystem, check whether it is visible.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2494
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2495
	 * XXX if the vnode is a symlink and it is not visible in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2496
	 * a pseudo filesystem, return ENOENT (not following symlink).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2497
	 * V4 client can not mount such symlink. This is a regression
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2498
	 * from V2/V3.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2499
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2500
	 * In the same exported filesystem, if the security flavor used
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2501
	 * is not an explicitly shared flavor, limit the view to the visible
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2502
	 * list entries only. This is not a WRONGSEC case because it's already
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2503
	 * checked via PUTROOTFH/PUTPUBFH or PUTFH.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2504
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2505
	if (!different_export &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2506
	    (PSEUDO(cs->exi) || ! is_exported_sec(cs->nfsflavor, cs->exi) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2507
	    cs->access & CS_ACCESS_LIMITED)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2508
		if (! nfs_visible(cs->exi, vp, &different_export)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2509
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2510
			return (puterrno4(ENOENT));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2511
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2512
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2513
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2514
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2515
	 * If it's a mountpoint, then traverse it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2516
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2517
	if (vn_ismntpt(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2518
		pre_exi = cs->exi;	/* save pre-traversed exportinfo */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2519
		pre_tvp = vp;		/* save pre-traversed vnode	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2520
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2521
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2522
		 * hold pre_tvp to counteract rele by traverse.  We will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2523
		 * need pre_tvp below if checkexport4 fails
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2524
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2525
		VN_HOLD(pre_tvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2526
		tvp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2527
		if ((error = traverse(&tvp)) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2528
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2529
			VN_RELE(pre_tvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2530
			return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2531
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2532
		vp = tvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2533
		different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2534
	} else if (vp->v_vfsp != cs->vp->v_vfsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2535
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2536
		 * The vfsp comparison is to handle the case where
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2537
		 * a LOFS mount is shared.  lo_lookup traverses mount points,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2538
		 * and NFS is unaware of local fs transistions because
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2539
		 * v_vfsmountedhere isn't set.  For this special LOFS case,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2540
		 * the dir and the obj returned by lookup will have different
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2541
		 * vfs ptrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2542
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2543
		different_export = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2544
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2545
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2546
	if (different_export) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2547
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2548
		bzero(&fid, sizeof (fid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2549
		fid.fid_len = MAXFIDSZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2550
		error = vop_fid_pseudo(vp, &fid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2551
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2552
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2553
			if (pre_tvp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2554
				VN_RELE(pre_tvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2555
			return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2556
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2557
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2558
		if (dotdot)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2559
			exi = nfs_vptoexi(NULL, vp, cs->cr, &walk, NULL, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2560
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2561
			exi = checkexport4(&vp->v_vfsp->vfs_fsid, &fid, vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2562
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2563
		if (exi == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2564
			if (pre_tvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2565
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2566
				 * If this vnode is a mounted-on vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2567
				 * but the mounted-on file system is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2568
				 * exported, send back the filehandle for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2569
				 * the mounted-on vnode, not the root of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2570
				 * the mounted-on file system.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2571
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2572
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2573
				vp = pre_tvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2574
				exi = pre_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2575
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2576
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2577
				return (puterrno4(EACCES));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2578
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2579
		} else if (pre_tvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2580
			/* we're done with pre_tvp now. release extra hold */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2581
			VN_RELE(pre_tvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2582
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2583
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2584
		cs->exi = exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2585
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2586
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2587
		 * Now we do a checkauth4. The reason is that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2588
		 * this client/user may not have access to the new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2589
		 * exported file system, and if he does,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2590
		 * the client/user may be mapped to a different uid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2591
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2592
		 * We start with a new cr, because the checkauth4 done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2593
		 * in the PUT*FH operation over wrote the cred's uid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2594
		 * gid, etc, and we want the real thing before calling
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2595
		 * checkauth4()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2596
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2597
		crfree(cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2598
		cs->cr = crdup(cs->basecr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2599
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2600
		if (cs->vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2601
			oldvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2602
		cs->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2603
		is_newvp = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2604
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2605
		stat = call_checkauth4(cs, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2606
		if (stat != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2607
			VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2608
			cs->vp = oldvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2609
			return (stat);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2610
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2611
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2612
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2613
	/*
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2614
	 * After various NFS checks, do a label check on the path
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2615
	 * component. The label on this path should either be the
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2616
	 * global zone's label or a zone's label. We are only
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2617
	 * interested in the zone's label because exported files
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2618
	 * in global zone is accessible (though read-only) to
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2619
	 * clients. The exportability/visibility check is already
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2620
	 * done before reaching this code.
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2621
	 */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2622
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2623
		bslabel_t *clabel;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2624
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2625
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2626
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2627
		DTRACE_PROBE2(tx__rfs4__log__info__oplookup__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2628
		    "got client label from request(1)", struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2629
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2630
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2631
			if (!do_rfs4_label_check(clabel, vp, DOMINANCE_CHECK)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2632
				error = EACCES;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2633
				goto err_out;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2634
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2635
		} else {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2636
			/*
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2637
			 * We grant access to admin_low label clients
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2638
			 * only if the client is trusted, i.e. also
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2639
			 * running Solaris Trusted Extension.
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2640
			 */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2641
			struct sockaddr	*ca;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2642
			int		addr_type;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2643
			void		*ipaddr;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2644
			tsol_tpc_t	*tp;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2645
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2646
			ca = (struct sockaddr *)svc_getrpccaller(
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2647
			    req->rq_xprt)->buf;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2648
			if (ca->sa_family == AF_INET) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2649
				addr_type = IPV4_VERSION;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2650
				ipaddr = &((struct sockaddr_in *)ca)->sin_addr;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2651
			} else if (ca->sa_family == AF_INET6) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2652
				addr_type = IPV6_VERSION;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2653
				ipaddr = &((struct sockaddr_in6 *)
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2654
				    ca)->sin6_addr;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2655
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2656
			tp = find_tpc(ipaddr, addr_type, B_FALSE);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2657
			if (tp == NULL || tp->tpc_tp.tp_doi !=
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2658
			    l_admin_low->tsl_doi || tp->tpc_tp.host_type !=
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2659
			    SUN_CIPSO) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2660
				error = EACCES;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2661
				goto err_out;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2662
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2663
		}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2664
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2665
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2666
	error = makefh4(&cs->fh, vp, cs->exi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2667
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  2668
err_out:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2669
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2670
		if (is_newvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2671
			VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2672
			cs->vp = oldvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2673
		} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2674
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2675
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2676
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2677
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2678
	if (!is_newvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2679
		if (cs->vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2680
			VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2681
		cs->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2682
	} else if (oldvp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2683
		VN_RELE(oldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2684
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2685
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2686
	 * if did lookup on attrdir and didn't lookup .., set named
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2687
	 * attr fh flag
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2688
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2689
	if (attrdir && ! dotdot)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2690
		set_fh4_flag(&cs->fh, FH4_NAMEDATTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2691
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2692
	/* Assume false for now, open proc will set this */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2693
	cs->mandlock = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2694
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2695
	return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2696
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2697
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2698
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2699
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2700
rfs4_op_lookup(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2701
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2702
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2703
	LOOKUP4args *args = &argop->nfs_argop4_u.oplookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2704
	LOOKUP4res *resp = &resop->nfs_resop4_u.oplookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2705
	char *nm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2706
	uint_t len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2707
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2708
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2709
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2710
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2711
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2712
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2713
	if (cs->vp->v_type == VLNK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2714
		*cs->statusp = resp->status = NFS4ERR_SYMLINK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2715
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2716
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2717
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2718
	if (cs->vp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2719
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2720
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2721
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2722
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2723
	if (!utf8_dir_verify(&args->objname)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2724
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2725
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2726
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2727
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2728
	nm = utf8_to_str(&args->objname, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2729
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2730
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2731
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2732
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2733
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2734
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2735
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2736
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2737
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2738
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2739
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2740
	*cs->statusp = resp->status = do_rfs4_op_lookup(nm, len, req, cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2741
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2742
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2743
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2744
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2745
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2746
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2747
rfs4_op_lookupp(nfs_argop4 *args, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2748
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2749
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2750
	LOOKUPP4res *resp = &resop->nfs_resop4_u.oplookupp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2751
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2752
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2753
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2754
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2755
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2756
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2757
	if (cs->vp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2758
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2759
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2760
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2761
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2762
	*cs->statusp = resp->status = do_rfs4_op_lookup("..", 3, req, cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2763
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2764
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2765
	 * From NFSV4 Specification, LOOKUPP should not check for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2766
	 * NFS4ERR_WRONGSEC. Retrun NFS4_OK instead.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2767
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2768
	if (resp->status == NFS4ERR_WRONGSEC) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2769
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2770
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2771
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2772
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2773
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2774
/*ARGSUSED2*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2775
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2776
rfs4_op_openattr(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2777
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2778
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2779
	OPENATTR4args	*args = &argop->nfs_argop4_u.opopenattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2780
	OPENATTR4res	*resp = &resop->nfs_resop4_u.opopenattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2781
	vnode_t		*avp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2782
	int		lookup_flags = LOOKUP_XATTR, error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2783
	int		exp_ro = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2784
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2785
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2786
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2787
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2788
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2789
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2790
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2791
	 * Make a couple of checks made by copen()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2792
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2793
	 * Check to make sure underlying fs supports xattrs.  This
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2794
	 * is required because solaris filesystem implementations
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2795
	 * (UFS/TMPFS) don't enforce the noxattr mount option
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2796
	 * in VOP_LOOKUP(LOOKUP_XATTR).  If fs doesn't support this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2797
	 * pathconf cmd or if fs supports cmd but doesn't claim
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2798
	 * support for xattr, return NOTSUPP.  It would be better
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2799
	 * to use VOP_PATHCONF( _PC_XATTR_ENABLED) for this; however,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2800
	 * that cmd is not available to VOP_PATHCONF interface
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2801
	 * (it's only implemented inside pathconf syscall)...
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2802
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2803
	 * Verify permission to put attributes on files (access
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2804
	 * checks from copen).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2805
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2806
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2807
	if ((cs->vp->v_vfsp->vfs_flag & VFS_XATTR) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2808
		error = ENOTSUP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2809
		goto error_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2810
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2811
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2812
	if ((VOP_ACCESS(cs->vp, VREAD, 0, cs->cr) != 0) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2813
	    (VOP_ACCESS(cs->vp, VWRITE, 0, cs->cr) != 0) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2814
	    (VOP_ACCESS(cs->vp, VEXEC, 0, cs->cr) != 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2815
		error = EACCES;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2816
		goto error_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2817
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2818
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2819
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2820
	 * The CREATE_XATTR_DIR VOP flag cannot be specified if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2821
	 * the file system is exported read-only -- regardless of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2822
	 * createdir flag.  Otherwise the attrdir would be created
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2823
	 * (assuming server fs isn't mounted readonly locally).  If
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2824
	 * VOP_LOOKUP returns ENOENT in this case, the error will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2825
	 * be translated into EROFS.  ENOSYS is mapped to ENOTSUP
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2826
	 * because specfs has no VOP_LOOKUP op, so the macro would
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2827
	 * return ENOSYS.  EINVAL is returned by all (current)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2828
	 * Solaris file system implementations when any of their
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2829
	 * restrictions are violated (xattr(dir) can't have xattrdir).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2830
	 * Returning NOTSUPP is more appropriate in this case
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2831
	 * because the object will never be able to have an attrdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2832
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2833
	if (args->createdir && ! (exp_ro = rdonly4(cs->exi, cs->vp, req)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2834
		lookup_flags |= CREATE_XATTR_DIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2835
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2836
	error = VOP_LOOKUP(cs->vp, "", &avp, NULL, lookup_flags, NULL, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2837
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2838
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2839
		if (error == ENOENT && args->createdir && exp_ro)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2840
			error = EROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2841
		else if (error == EINVAL || error == ENOSYS)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2842
			error = ENOTSUP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2843
		goto error_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2844
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2845
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2846
	ASSERT(avp->v_flag & V_XATTRDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2847
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2848
	error = makefh4(&cs->fh, avp, cs->exi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2849
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2850
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2851
		VN_RELE(avp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2852
		goto error_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2853
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2854
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2855
	VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2856
	cs->vp = avp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2857
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2858
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2859
	 * There is no requirement for an attrdir fh flag
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2860
	 * because the attrdir has a vnode flag to distinguish
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2861
	 * it from regular (non-xattr) directories.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2862
	 * FH4_ATTRDIR flag is set for future sanity checks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2863
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2864
	set_fh4_flag(&cs->fh, FH4_ATTRDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2865
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2866
	return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2867
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2868
error_out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2869
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2870
	*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2871
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2872
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2873
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2874
do_io(int direction, vnode_t *vp, struct uio *uio, int ioflag, cred_t *cred)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2875
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2876
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2877
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2878
	clock_t delaytime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2879
	caller_context_t ct;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2880
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2881
	delaytime = MSEC_TO_TICK_ROUNDUP(rfs4_lock_delay);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2882
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2883
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2884
	 * Don't block on mandatory locks. If this routine returns
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2885
	 * EAGAIN, the caller should return NFS4ERR_LOCKED.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2886
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2887
	uio->uio_fmode = FNONBLOCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2888
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2889
	ct.cc_sysid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2890
	ct.cc_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2891
	ct.cc_caller_id = nfs4_srv_caller_id;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2892
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2893
	for (i = 0; i < rfs4_maxlock_tries; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2894
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2895
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2896
		if (direction == FREAD) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2897
			(void) VOP_RWLOCK(vp, V_WRITELOCK_FALSE, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2898
			error = VOP_READ(vp, uio, ioflag, cred, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2899
			VOP_RWUNLOCK(vp, V_WRITELOCK_FALSE, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2900
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2901
			(void) VOP_RWLOCK(vp, V_WRITELOCK_TRUE, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2902
			error = VOP_WRITE(vp, uio, ioflag, cred, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2903
			VOP_RWUNLOCK(vp, V_WRITELOCK_TRUE, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2904
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2905
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2906
		if (error != EAGAIN)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2907
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2908
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2909
		if (i < rfs4_maxlock_tries - 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2910
			delay(delaytime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2911
			delaytime *= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2912
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2913
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2914
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2915
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2916
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2917
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2918
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2919
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2920
rfs4_op_read(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2921
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2922
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2923
	READ4args *args = &argop->nfs_argop4_u.opread;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2924
	READ4res *resp = &resop->nfs_resop4_u.opread;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2925
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2926
	int verror;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2927
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2928
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2929
	struct iovec iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2930
	struct uio uio;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2931
	u_offset_t offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2932
	bool_t *deleg = &cs->deleg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2933
	nfsstat4 stat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2934
	int in_crit = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2935
	mblk_t *mp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2936
	int alloc_err = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2937
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2938
	vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2939
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2940
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2941
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2942
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2943
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2944
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2945
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2946
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2947
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2948
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2949
	 * Enter the critical region before calling VOP_RWLOCK
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2950
	 * to avoid a deadlock with write requests.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2951
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2952
	if (nbl_need_check(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2953
		nbl_start_crit(vp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2954
		in_crit = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2955
		if (nbl_conflict(vp, NBL_READ, args->offset, args->count, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2956
			*cs->statusp = resp->status = NFS4ERR_LOCKED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2957
			goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2958
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2959
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2960
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2961
	if ((stat = rfs4_check_stateid(FREAD, vp, &args->stateid, FALSE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2962
					deleg, TRUE)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2963
		*cs->statusp = resp->status = stat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2964
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2965
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2966
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2967
	va.va_mask = AT_MODE|AT_SIZE|AT_UID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2968
	verror = VOP_GETATTR(vp, &va, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2969
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2970
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2971
	 * If we can't get the attributes, then we can't do the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2972
	 * right access checking.  So, we'll fail the request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2973
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2974
	if (verror) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2975
		*cs->statusp = resp->status = puterrno4(verror);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2976
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2977
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2978
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2979
	if (vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2980
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2981
			((vp->v_type == VDIR) ? NFS4ERR_ISDIR : NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2982
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2983
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2984
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2985
	if (crgetuid(cs->cr) != va.va_uid &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2986
	    (error = VOP_ACCESS(vp, VREAD, 0, cs->cr)) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2987
	    (error = VOP_ACCESS(vp, VEXEC, 0, cs->cr))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2988
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2989
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2990
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2991
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2992
	if (MANDLOCK(vp, va.va_mode)) { /* XXX - V4 supports mand locking */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2993
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2994
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2995
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2996
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2997
	offset = args->offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2998
	if (offset >= va.va_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2999
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3000
		resp->eof = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3001
		resp->data_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3002
		resp->data_val = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3003
		resp->mblk = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3004
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3005
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3006
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3007
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3008
	if (args->count == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3009
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3010
		resp->eof = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3011
		resp->data_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3012
		resp->data_val = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3013
		resp->mblk = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3014
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3015
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3016
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3017
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3018
	 * Do not allocate memory more than maximum allowed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3019
	 * transfer size
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3020
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3021
	if (args->count > rfs4_tsize(req))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3022
		args->count = rfs4_tsize(req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3023
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3024
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3025
	 * mp will contain the data to be sent out in the read reply.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3026
	 * It will be freed after the reply has been sent.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3027
	 * Let's roundup the data to a BYTES_PER_XDR_UNIT multiple,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3028
	 * so that the call to xdrmblk_putmblk() never fails.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3029
	 * If the first alloc of the requested size fails, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3030
	 * decrease the size to something more reasonable and wait
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3031
	 * for the allocation to occur.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3032
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3033
	mp = allocb(RNDUP(args->count), BPRI_MED);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3034
	if (mp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3035
		if (args->count > MAXBSIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3036
			args->count = MAXBSIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3037
		mp = allocb_wait(RNDUP(args->count), BPRI_MED,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3038
				STR_NOSIG, &alloc_err);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3039
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3040
	ASSERT(mp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3041
	ASSERT(alloc_err == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3042
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3043
	iov.iov_base = (caddr_t)mp->b_datap->db_base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3044
	iov.iov_len = args->count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3045
	uio.uio_iov = &iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3046
	uio.uio_iovcnt = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3047
	uio.uio_segflg = UIO_SYSSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3048
	uio.uio_extflg = UIO_COPY_CACHED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3049
	uio.uio_loffset = args->offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3050
	uio.uio_resid = args->count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3051
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3052
	error = do_io(FREAD, vp, &uio, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3053
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3054
	va.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3055
	verror = VOP_GETATTR(vp, &va, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3056
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3057
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3058
		freeb(mp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3059
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3060
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3061
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3062
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3063
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3064
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3065
	ASSERT(uio.uio_resid >= 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3066
	resp->data_len = args->count - uio.uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3067
	resp->data_val = (char *)mp->b_datap->db_base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3068
	resp->mblk = mp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3069
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3070
	if (!verror && offset + resp->data_len == va.va_size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3071
		resp->eof = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3072
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3073
		resp->eof = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3074
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3075
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3076
	if (in_crit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3077
		nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3078
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3079
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3080
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3081
rfs4_op_read_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3082
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3083
	READ4res *resp = &resop->nfs_resop4_u.opread;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3084
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3085
	if (resp->status == NFS4_OK && resp->mblk != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3086
		freeb(resp->mblk);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3087
		resp->mblk = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3088
		resp->data_val = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3089
		resp->data_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3090
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3091
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3092
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3093
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3094
rfs4_op_readdir_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3095
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3096
	READDIR4res *resp = &resop->nfs_resop4_u.opreaddir;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3097
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3098
	if (resp->status == NFS4_OK && resp->mblk != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3099
		freeb(resp->mblk);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3100
		resp->mblk = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3101
		resp->data_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3102
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3103
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3104
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3105
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3106
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3107
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3108
rfs4_op_putpubfh(nfs_argop4 *args, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3109
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3110
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3111
	PUTPUBFH4res *resp = &resop->nfs_resop4_u.opputpubfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3112
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3113
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3114
	struct exportinfo *exi, *sav_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3115
	nfs_fh4_fmt_t *fh_fmtp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3116
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3117
	if (cs->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3118
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3119
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3120
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3121
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3122
	if (cs->cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3123
		crfree(cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3124
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3125
	cs->cr = crdup(cs->basecr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3126
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3127
	vp = exi_public->exi_vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3128
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3129
		*cs->statusp = resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3130
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3131
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3132
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3133
	error = makefh4(&cs->fh, vp, exi_public);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3134
	if (error != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3135
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3136
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3137
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3138
	sav_exi = cs->exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3139
	if (exi_public == exi_root) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3140
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3141
		 * No filesystem is actually shared public, so we default
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3142
		 * to exi_root. In this case, we must check whether root
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3143
		 * is exported.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3144
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3145
		fh_fmtp = (nfs_fh4_fmt_t *)cs->fh.nfs_fh4_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3146
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3147
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3148
		 * if root filesystem is exported, the exportinfo struct that we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3149
		 * should use is what checkexport4 returns, because root_exi is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3150
		 * actually a mostly empty struct.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3151
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3152
		exi = checkexport4(&fh_fmtp->fh4_fsid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3153
			(fid_t *)&fh_fmtp->fh4_xlen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3154
		cs->exi = ((exi != NULL) ? exi : exi_public);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3155
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3156
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3157
		 * it's a properly shared filesystem
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3158
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3159
		cs->exi = exi_public;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3160
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3161
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3162
	VN_HOLD(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3163
	cs->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3164
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3165
	if ((resp->status = call_checkauth4(cs, req)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3166
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3167
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3168
		cs->exi = sav_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3169
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3170
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3171
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3172
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3173
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3174
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3175
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3176
 * XXX - issue with put*fh operations. Suppose /export/home is exported.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3177
 * Suppose an NFS client goes to mount /export/home/joe. If /export, home,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3178
 * or joe have restrictive search permissions, then we shouldn't let
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3179
 * the client get a file handle. This is easy to enforce. However, we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3180
 * don't know what security flavor should be used until we resolve the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3181
 * path name. Another complication is uid mapping. If root is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3182
 * the user, then it will be mapped to the anonymous user by default,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3183
 * but we won't know that till we've resolved the path name. And we won't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3184
 * know what the anonymous user is.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3185
 * Luckily, SECINFO is specified to take a full filename.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3186
 * So what we will have to in rfs4_op_lookup is check that flavor of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3187
 * the target object matches that of the request, and if root was the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3188
 * caller, check for the root= and anon= options, and if necessary,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3189
 * repeat the lookup using the right cred_t. But that's not done yet.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3190
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3191
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3192
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3193
rfs4_op_putfh(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3194
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3195
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3196
	PUTFH4args *args = &argop->nfs_argop4_u.opputfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3197
	PUTFH4res *resp = &resop->nfs_resop4_u.opputfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3198
	nfs_fh4_fmt_t *fh_fmtp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3199
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3200
	if (cs->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3201
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3202
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3203
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3204
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3205
	if (cs->cr) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3206
		crfree(cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3207
		cs->cr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3208
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3209
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3210
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3211
	if (args->object.nfs_fh4_len < NFS_FH4_LEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3212
		*cs->statusp = resp->status = NFS4ERR_BADHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3213
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3214
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3215
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3216
	fh_fmtp = (nfs_fh4_fmt_t *)args->object.nfs_fh4_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3217
	cs->exi = checkexport4(&fh_fmtp->fh4_fsid, (fid_t *)&fh_fmtp->fh4_xlen,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3218
				NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3219
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3220
	if (cs->exi == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3221
		*cs->statusp = resp->status = NFS4ERR_STALE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3222
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3223
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3224
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3225
	cs->cr = crdup(cs->basecr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3226
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3227
	ASSERT(cs->cr != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3228
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3229
	if (! (cs->vp = nfs4_fhtovp(&args->object, cs->exi, &resp->status))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3230
		*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3231
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3232
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3233
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3234
	if ((resp->status = call_checkauth4(cs, req)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3235
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3236
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3237
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3238
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3239
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3240
	nfs_fh4_copy(&args->object, &cs->fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3241
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3242
	cs->deleg = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3243
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3244
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3245
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3246
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3247
rfs4_op_putrootfh(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3248
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3249
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3250
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3251
	PUTROOTFH4res *resp = &resop->nfs_resop4_u.opputrootfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3252
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3253
	fid_t fid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3254
	struct exportinfo *exi, *sav_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3255
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3256
	if (cs->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3257
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3258
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3259
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3260
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3261
	if (cs->cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3262
		crfree(cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3263
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3264
	cs->cr = crdup(cs->basecr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3265
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3266
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3267
	 * Using rootdir, the system root vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3268
	 * get its fid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3269
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3270
	bzero(&fid, sizeof (fid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3271
	fid.fid_len = MAXFIDSZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3272
	error = vop_fid_pseudo(rootdir, &fid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3273
	if (error != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3274
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3275
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3276
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3277
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3278
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3279
	 * Then use the root fsid & fid it to find out if it's exported
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3280
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3281
	 * If the server root isn't exported directly, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3282
	 * it should at least be a pseudo export based on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3283
	 * one or more exports further down in the server's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3284
	 * file tree.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3285
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3286
	exi = checkexport4(&rootdir->v_vfsp->vfs_fsid, &fid, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3287
	if (exi == NULL || exi->exi_export.ex_flags & EX_PUBLIC) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3288
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3289
			(CE_WARN, "rfs4_op_putrootfh: export check failure"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3290
		*cs->statusp = resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3291
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3292
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3293
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3294
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3295
	 * Now make a filehandle based on the root
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3296
	 * export and root vnode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3297
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3298
	error = makefh4(&cs->fh, rootdir, exi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3299
	if (error != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3300
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3301
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3302
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3303
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3304
	sav_exi = cs->exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3305
	cs->exi = exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3306
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3307
	VN_HOLD(rootdir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3308
	cs->vp = rootdir;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3309
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3310
	if ((resp->status = call_checkauth4(cs, req)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3311
		VN_RELE(rootdir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3312
		cs->vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3313
		cs->exi = sav_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3314
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3315
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3316
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3317
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3318
	cs->deleg = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3319
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3320
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3321
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3322
 * A directory entry is a valid nfsv4 entry if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3323
 * - it has a non-zero ino
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3324
 * - it is not a dot or dotdot name
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3325
 * - it is visible in a pseudo export or in a real export that can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3326
 *   only have a limited view.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3327
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3328
static bool_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3329
valid_nfs4_entry(struct exportinfo *exi, struct dirent64 *dp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3330
		int *expseudo, int check_visible)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3331
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3332
	if (dp->d_ino == 0 || NFS_IS_DOTNAME(dp->d_name)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3333
		*expseudo = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3334
		return (FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3335
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3336
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3337
	if (! check_visible) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3338
		*expseudo = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3339
		return (TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3340
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3341
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3342
	return (nfs_visible_inode(exi, dp->d_ino, expseudo));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3343
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3344
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3345
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3346
 * set_rdattr_params sets up the variables used to manage what information
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3347
 * to get for each directory entry.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3348
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3349
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3350
set_rdattr_params(struct nfs4_svgetit_arg *sargp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3351
		bitmap4 attrs, bool_t *need_to_lookup)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3352
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3353
	uint_t	va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3354
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3355
	bitmap4 objbits;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3356
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3357
	status = bitmap4_to_attrmask(attrs, sargp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3358
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3359
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3360
		 * could not even figure attr mask
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3361
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3362
		return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3363
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3364
	va_mask = sargp->vap->va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3365
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3366
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3367
	 * dirent's d_ino is always correct value for mounted_on_fileid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3368
	 * mntdfid_set is set once here, but mounted_on_fileid is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3369
	 * set in main dirent processing loop for each dirent.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3370
	 * The mntdfid_set is a simple optimization that lets the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3371
	 * server attr code avoid work when caller is readdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3372
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3373
	sargp->mntdfid_set = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3374
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3375
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3376
	 * Lookup entry only if client asked for any of the following:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3377
	 * a) vattr attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3378
	 * b) vfs attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3379
	 * c) attrs w/per-object scope requested (change, filehandle, etc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3380
	 *    other than mounted_on_fileid (which we can take from dirent)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3381
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3382
	objbits = attrs ? attrs & NFS4_VP_ATTR_MASK : 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3383
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3384
	if (va_mask || sargp->sbp || (objbits & ~FATTR4_MOUNTED_ON_FILEID_MASK))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3385
		*need_to_lookup = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3386
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3387
		*need_to_lookup = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3388
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3389
	if (sargp->sbp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3390
		return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3391
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3392
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3393
	 * If filesystem attrs are requested, get them now from the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3394
	 * directory vp, as most entries will have same filesystem. The only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3395
	 * exception are mounted over entries but we handle
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3396
	 * those as we go (XXX mounted over detection not yet implemented).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3397
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3398
	sargp->vap->va_mask = 0;	/* to avoid VOP_GETATTR */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3399
	status = bitmap4_get_sysattrs(sargp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3400
	sargp->vap->va_mask = va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3401
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3402
	if ((status != NFS4_OK) && sargp->rdattr_error_req) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3403
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3404
		 * Failed to get filesystem attributes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3405
		 * Return a rdattr_error for each entry, but don't fail.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3406
		 * However, don't get any obj-dependent attrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3407
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3408
		sargp->rdattr_error = status;	/* for rdattr_error */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3409
		*need_to_lookup = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3410
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3411
		 * At least get fileid for regular readdir output
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3412
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3413
		sargp->vap->va_mask &= AT_NODEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3414
		status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3415
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3417
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3418
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3419
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3420
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3421
 * readlink: args: CURRENT_FH.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3422
 *	res: status. If success - CURRENT_FH unchanged, return linktext.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3423
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3424
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3425
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3426
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3427
rfs4_op_readlink(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3428
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3429
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3430
	READLINK4res *resp = &resop->nfs_resop4_u.opreadlink;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3431
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3432
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3433
	struct iovec iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3434
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3435
	struct uio uio;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3436
	char *data;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3437
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3438
	/* CURRENT_FH: directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3439
	vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3440
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3441
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3442
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3443
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3444
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3445
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3446
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3447
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3448
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3449
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3450
	if (vp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3451
		*cs->statusp = resp->status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3452
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3453
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3454
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3455
	if (vp->v_type != VLNK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3456
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3457
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3458
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3459
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3460
	va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3461
	error = VOP_GETATTR(vp, &va, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3462
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3463
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3464
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3465
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3466
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3467
	if (MANDLOCK(vp, va.va_mode)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3468
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3469
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3470
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3471
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3472
	data = kmem_alloc(MAXPATHLEN + 1, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3473
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3474
	iov.iov_base = data;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3475
	iov.iov_len = MAXPATHLEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3476
	uio.uio_iov = &iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3477
	uio.uio_iovcnt = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3478
	uio.uio_segflg = UIO_SYSSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3479
	uio.uio_extflg = UIO_COPY_CACHED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3480
	uio.uio_loffset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3481
	uio.uio_resid = MAXPATHLEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3482
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3483
	error = VOP_READLINK(vp, &uio, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3484
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3485
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3486
		kmem_free((caddr_t)data, (uint_t)MAXPATHLEN + 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3487
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3488
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3489
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3490
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3491
	*(data + MAXPATHLEN - uio.uio_resid) = '\0';
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3492
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3493
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3494
	 * treat link name as data
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3495
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3496
	(void) str_to_utf8(data, &resp->link);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3497
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3498
	kmem_free((caddr_t)data, (uint_t)MAXPATHLEN + 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3499
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3500
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3501
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3502
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3503
rfs4_op_readlink_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3504
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3505
	READLINK4res *resp = &resop->nfs_resop4_u.opreadlink;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3506
	utf8string *symlink = &resp->link;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3507
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3508
	if (symlink->utf8string_val) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3509
		UTF8STRING_FREE(*symlink)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3510
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3511
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3512
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3513
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3514
 * release_lockowner:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3515
 *	Release any state associated with the supplied
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3516
 *	lockowner. Note if any lo_state is holding locks we will not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3517
 *	rele that lo_state and thus the lockowner will not be destroyed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3518
 *	A client using lock after the lock owner stateid has been released
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3519
 *	will suffer the consequence of NFS4ERR_BAD_STATEID and would have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3520
 *	to reissue the lock with new_lock_owner set to TRUE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3521
 *	args: lock_owner
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3522
 *	res:  status
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3523
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3524
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3525
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3526
rfs4_op_release_lockowner(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3527
	struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3528
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3529
	RELEASE_LOCKOWNER4args *ap = &argop->nfs_argop4_u.oprelease_lockowner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3530
	RELEASE_LOCKOWNER4res *resp = &resop->nfs_resop4_u.oprelease_lockowner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3531
	rfs4_lockowner_t *lo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3532
	rfs4_openowner_t *oop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3533
	rfs4_state_t *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3534
	rfs4_lo_state_t *lsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3535
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3536
	bool_t create = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3537
	locklist_t *llist;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3538
	sysid_t sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3539
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3540
	/* Make sure there is a clientid around for this request */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3541
	cp = rfs4_findclient_by_id(ap->lock_owner.clientid, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3542
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3543
	if (cp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3544
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3545
			rfs4_check_clientid(&ap->lock_owner.clientid, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3546
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3547
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3548
	rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3549
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3550
	lo = rfs4_findlockowner(&ap->lock_owner, &create);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3551
	if (lo == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3552
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3553
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3554
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3555
	ASSERT(lo->client != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3556
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3557
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3558
	 * Check for EXPIRED client. If so will reap state with in a lease
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3559
	 * period or on next set_clientid_confirm step
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3560
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3561
	if (rfs4_lease_expired(lo->client)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3562
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3563
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3564
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3565
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3566
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3567
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3568
	 * If no sysid has been assigned, then no locks exist; just return.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3569
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3570
	rfs4_dbe_lock(lo->client->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3571
	if (lo->client->sysidt == LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3572
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3573
		rfs4_dbe_unlock(lo->client->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3574
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3575
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3576
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3577
	sysid = lo->client->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3578
	rfs4_dbe_unlock(lo->client->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3579
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3580
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3581
	 * Mark the lockowner invalid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3582
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3583
	rfs4_dbe_hide(lo->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3584
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3585
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3586
	 * sysid-pid pair should now not be used since the lockowner is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3587
	 * invalid. If the client were to instantiate the lockowner again
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3588
	 * it would be assigned a new pid. Thus we can get the list of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3589
	 * current locks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3590
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3591
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3592
	llist = flk_get_active_locks(sysid, lo->pid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3593
	/* If we are still holding locks fail */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3594
	if (llist != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3595
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3596
		*cs->statusp = resp->status = NFS4ERR_LOCKS_HELD;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3597
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3598
		flk_free_locklist(llist);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3599
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3600
		 * We need to unhide the lockowner so the client can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3601
		 * try it again. The bad thing here is if the client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3602
		 * has a logic error that took it here in the first place
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3603
		 * he probably has lost accounting of the locks that it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3604
		 * is holding. So we may have dangling state until the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3605
		 * open owner state is reaped via close. One scenario
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3606
		 * that could possibly occur is that the client has
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3607
		 * sent the unlock request(s) in separate threads
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3608
		 * and has not waited for the replies before sending the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3609
		 * RELEASE_LOCKOWNER request. Presumably, it would expect
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3610
		 * and deal appropriately with NFS4ERR_LOCKS_HELD, by
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3611
		 * reissuing the request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3612
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3613
		rfs4_dbe_unhide(lo->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3614
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3615
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3616
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3617
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3618
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3619
	 * For the corresponding client we need to check each open
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3620
	 * owner for any opens that have lockowner state associated
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3621
	 * with this lockowner.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3622
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3623
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3624
	rfs4_dbe_lock(lo->client->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3625
	for (oop = lo->client->openownerlist.next->oop; oop != NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3626
	    oop = oop->openownerlist.next->oop) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3627
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3628
		rfs4_dbe_lock(oop->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3629
		for (sp = oop->ownerstateids.next->sp; sp != NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3630
		    sp = sp->ownerstateids.next->sp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3631
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3632
			rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3633
			for (lsp = sp->lockownerlist.next->lsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3634
			    lsp != NULL; lsp = lsp->lockownerlist.next->lsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3635
				if (lsp->locker == lo) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3636
					rfs4_dbe_lock(lsp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3637
					rfs4_dbe_invalidate(lsp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3638
					rfs4_dbe_unlock(lsp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3639
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3640
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3641
			rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3642
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3643
		rfs4_dbe_unlock(oop->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3644
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3645
	rfs4_dbe_unlock(lo->client->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3646
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3647
	rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3648
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3649
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3650
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3651
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3652
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3653
 * short utility function to lookup a file and recall the delegation
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3654
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3655
static rfs4_file_t *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3656
rfs4_lookup_and_findfile(vnode_t *dvp, char *nm, vnode_t **vpp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3657
	int *lkup_error, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3658
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3659
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3660
	rfs4_file_t *fp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3661
	bool_t fcreate = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3662
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3663
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3664
	if (vpp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3665
		*vpp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3666
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3667
	if ((error = VOP_LOOKUP(dvp, nm, &vp, NULL, 0, NULL, cr)) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3668
		if (vp->v_type == VREG)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3669
			fp = rfs4_findfile(vp, NULL, &fcreate);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3670
		if (vpp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3671
			*vpp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3672
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3673
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3674
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3675
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3676
	if (lkup_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3677
		*lkup_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3678
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3679
	return (fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3680
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3681
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3682
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3683
 * remove: args: CURRENT_FH: directory; name.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3684
 *	res: status. If success - CURRENT_FH unchanged, return change_info
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3685
 *		for directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3686
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3687
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3688
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3689
rfs4_op_remove(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3690
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3691
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3692
	REMOVE4args *args = &argop->nfs_argop4_u.opremove;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3693
	REMOVE4res *resp = &resop->nfs_resop4_u.opremove;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3694
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3695
	vnode_t *dvp, *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3696
	struct vattr bdva, idva, adva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3697
	char *nm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3698
	uint_t len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3699
	rfs4_file_t *fp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3700
	int in_crit = 0;
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3701
	bslabel_t *clabel;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3702
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3703
	/* CURRENT_FH: directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3704
	dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3705
	if (dvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3706
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3707
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3708
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3709
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3710
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3711
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3712
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3713
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3714
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3715
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3716
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3717
	 * Do not allow to remove anything in this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3718
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3719
	if (vn_ismntpt(dvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3720
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3721
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3722
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3723
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3724
	if (dvp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3725
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3726
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3727
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3728
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3729
	if (!utf8_dir_verify(&args->target)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3730
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3731
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3732
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3733
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3734
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3735
	 * Lookup the file so that we can check if it's a directory
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3736
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3737
	nm = utf8_to_fn(&args->target, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3738
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3739
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3740
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3741
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3742
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3743
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3744
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3745
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3746
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3747
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3748
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3749
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3750
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3751
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3752
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3753
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3754
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3755
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3756
	 * Lookup the file to determine type and while we are see if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3757
	 * there is a file struct around and check for delegation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3758
	 * We don't need to acquire va_seq before this lookup, if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3759
	 * it causes an update, cinfo.before will not match, which will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3760
	 * trigger a cache flush even if atomic is TRUE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3761
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3762
	if (fp = rfs4_lookup_and_findfile(dvp, nm, &vp, &error, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3763
		if (rfs4_check_delegated_byfp(FWRITE, fp, TRUE, TRUE, TRUE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3764
						NULL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3765
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3766
			rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3767
			*cs->statusp = resp->status = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3768
			kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3769
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3770
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3771
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3772
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3773
	/* Didn't find anything to remove */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3774
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3775
		*cs->statusp = resp->status = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3776
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3777
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3778
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3779
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3780
	if (nbl_need_check(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3781
		nbl_start_crit(vp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3782
		in_crit = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3783
		if (nbl_conflict(vp, NBL_REMOVE, 0, 0, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3784
			*cs->statusp = resp->status = NFS4ERR_FILE_OPEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3785
			kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3786
			nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3787
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3788
			if (fp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3789
				rfs4_clear_dont_grant(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3790
				rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3791
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3792
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3793
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3794
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3795
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3796
	/* check label before allowing removal */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3797
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3798
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3799
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3800
		DTRACE_PROBE2(tx__rfs4__log__info__opremove__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3801
		    "got client label from request(1)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3802
		    struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3803
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3804
			if (!do_rfs4_label_check(clabel, vp, EQUALITY_CHECK)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3805
				*cs->statusp = resp->status = NFS4ERR_ACCESS;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3806
				kmem_free(nm, len);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3807
				if (in_crit)
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3808
					nbl_end_crit(vp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3809
				VN_RELE(vp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3810
				if (fp) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3811
					rfs4_clear_dont_grant(fp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3812
					rfs4_file_rele(fp);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3813
				}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3814
				return;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3815
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3816
		}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3817
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3818
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3819
	/* Get dir "before" change value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3820
	bdva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3821
	error = VOP_GETATTR(dvp, &bdva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3822
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3823
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3824
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3825
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3826
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3827
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.before, bdva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3828
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3829
	/* Actually do the REMOVE operation */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3830
	if (vp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3831
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3832
		 * Can't remove a directory that has a mounted-on filesystem.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3833
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3834
		if (vn_ismntpt(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3835
			error = EACCES;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3836
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3837
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3838
			 * System V defines rmdir to return EEXIST,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3839
			 * not * ENOTEMPTY, if the directory is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3840
			 * empty.  A System V NFS server needs to map
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3841
			 * NFS4ERR_EXIST to NFS4ERR_NOTEMPTY to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3842
			 * transmit over the wire.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3843
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3844
			if ((error = VOP_RMDIR(dvp, nm, rootdir, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3845
				== EEXIST)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3846
				error = ENOTEMPTY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3847
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3848
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3849
		if ((error = VOP_REMOVE(dvp, nm, cs->cr)) == 0 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3850
			fp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3851
			struct vattr va;
154
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3852
			vnode_t *tvp;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3853
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3854
			rfs4_dbe_lock(fp->dbe);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3855
			tvp = fp->vp;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3856
			if (tvp)
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3857
				VN_HOLD(tvp);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3858
			rfs4_dbe_unlock(fp->dbe);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3859
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3860
			if (tvp) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3861
				/*
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3862
				 * This is va_seq safe because we are not
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3863
				 * manipulating dvp.
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3864
				 */
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3865
				va.va_mask = AT_NLINK;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3866
				if (!VOP_GETATTR(tvp, &va, 0, cs->cr) &&
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3867
					va.va_nlink == 0) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3868
					/* Remove state on file remove */
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3869
					if (in_crit) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3870
						nbl_end_crit(vp);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3871
						in_crit = 0;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3872
					}
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3873
					rfs4_close_all_state(fp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3874
				}
154
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  3875
				VN_RELE(tvp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3876
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3877
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3878
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3879
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3880
	if (in_crit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3881
		nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3882
	VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3883
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3884
	if (fp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3885
		rfs4_clear_dont_grant(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3886
		rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3887
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3888
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3889
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3890
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3891
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3892
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3893
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3894
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3895
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3896
	 * Get the initial "after" sequence number, if it fails, set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3897
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3898
	idva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3899
	if (VOP_GETATTR(dvp, &idva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3900
		idva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3901
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3902
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3903
	 * Force modified data and metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3904
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3905
	(void) VOP_FSYNC(dvp, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3906
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3907
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3908
	 * Get "after" change value, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3909
	 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3910
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3911
	adva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3912
	if (VOP_GETATTR(dvp, &adva, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3913
		adva.va_ctime = bdva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3914
		adva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3915
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3917
	NFS4_SET_FATTR4_CHANGE(resp->cinfo.after, adva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3918
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3919
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3920
	 * The cinfo.atomic = TRUE only if we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3921
	 * non-zero va_seq's, and it has incremented by exactly one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3922
	 * during the VOP_REMOVE/RMDIR and it didn't change during
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3923
	 * the VOP_FSYNC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3924
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3925
	if (bdva.va_seq && idva.va_seq && adva.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3926
			idva.va_seq == (bdva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3927
			idva.va_seq == adva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3928
		resp->cinfo.atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3929
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3930
		resp->cinfo.atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3931
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3932
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3933
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3934
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3935
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3936
 * rename: args: SAVED_FH: from directory, CURRENT_FH: target directory,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3937
 *		oldname and newname.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3938
 *	res: status. If success - CURRENT_FH unchanged, return change_info
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3939
 *		for both from and target directories.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3940
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3941
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3942
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3943
rfs4_op_rename(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3944
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3945
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3946
	RENAME4args *args = &argop->nfs_argop4_u.oprename;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3947
	RENAME4res *resp = &resop->nfs_resop4_u.oprename;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3948
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3949
	vnode_t *odvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3950
	vnode_t *ndvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3951
	vnode_t *srcvp, *targvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3952
	struct vattr obdva, oidva, oadva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3953
	struct vattr nbdva, nidva, nadva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3954
	char *onm, *nnm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3955
	uint_t olen, nlen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3956
	rfs4_file_t *fp, *sfp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3957
	int in_crit_src, in_crit_targ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3958
	int fp_rele_grant_hold, sfp_rele_grant_hold;
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  3959
	bslabel_t *clabel;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3960
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3961
	fp = sfp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3962
	srcvp = targvp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3963
	in_crit_src = in_crit_targ = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3964
	fp_rele_grant_hold = sfp_rele_grant_hold = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3965
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3966
	/* CURRENT_FH: target directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3967
	ndvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3968
	if (ndvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3969
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3970
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3971
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3972
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3973
	/* SAVED_FH: from directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3974
	odvp = cs->saved_vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3975
	if (odvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3976
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3977
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3978
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3979
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3980
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3981
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3982
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3983
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3984
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3985
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3986
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3987
	 * do not allow to rename objects in this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3988
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3989
	if (vn_ismntpt(odvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3990
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3991
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3992
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3993
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3994
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3995
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3996
	 * do not allow to rename to this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3997
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3998
	if (vn_ismntpt(ndvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3999
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4000
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4001
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4002
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4003
	if (odvp->v_type != VDIR || ndvp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4004
		*cs->statusp = resp->status = NFS4ERR_NOTDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4005
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4006
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4007
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4008
	if (cs->saved_exi != cs->exi) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4009
		*cs->statusp = resp->status = NFS4ERR_XDEV;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4010
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4011
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4012
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4013
	if (!utf8_dir_verify(&args->oldname)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4014
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4015
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4016
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4017
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4018
	if (!utf8_dir_verify(&args->newname)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4019
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4020
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4021
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4022
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4023
	onm = utf8_to_fn(&args->oldname, &olen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4024
	if (onm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4025
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4026
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4027
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4028
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4029
	nnm = utf8_to_fn(&args->newname, &nlen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4030
	if (nnm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4031
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4032
		kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4033
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4034
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4035
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4036
	if (olen > MAXNAMELEN || nlen > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4037
		*cs->statusp = resp->status = NFS4ERR_NAMETOOLONG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4038
		kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4039
		kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4040
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4041
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4042
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4043
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4044
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4045
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4046
		kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4047
		kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4048
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4049
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4050
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4051
	/* check label of the target dir */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4052
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4053
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4054
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4055
		DTRACE_PROBE2(tx__rfs4__log__info__oprename__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4056
		    "got client label from request(1)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4057
		    struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4058
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4059
			if (!do_rfs4_label_check(clabel, ndvp,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4060
			    EQUALITY_CHECK)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4061
				*cs->statusp = resp->status = NFS4ERR_ACCESS;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4062
				return;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4063
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4064
		}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4065
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4066
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4067
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4068
	 * Is the source a file and have a delegation?
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4069
	 * We don't need to acquire va_seq before these lookups, if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4070
	 * it causes an update, cinfo.before will not match, which will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4071
	 * trigger a cache flush even if atomic is TRUE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4072
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4073
	if (sfp = rfs4_lookup_and_findfile(odvp, onm, &srcvp, &error, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4074
		if (rfs4_check_delegated_byfp(FWRITE, sfp, TRUE, TRUE, TRUE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4075
						NULL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4076
			*cs->statusp = resp->status = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4077
			goto err_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4078
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4079
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4080
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4081
	if (srcvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4082
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4083
		kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4084
		kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4085
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4086
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4087
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4088
	sfp_rele_grant_hold = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4089
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4090
	/* Does the destination exist and a file and have a delegation? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4091
	if (fp = rfs4_lookup_and_findfile(ndvp, nnm, &targvp, NULL, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4092
		if (rfs4_check_delegated_byfp(FWRITE, fp, TRUE, TRUE, TRUE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4093
						NULL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4094
			*cs->statusp = resp->status = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4095
			goto err_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4096
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4097
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4098
	fp_rele_grant_hold = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4099
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4100
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4101
	/* Check for NBMAND lock on both source and target */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4102
	if (nbl_need_check(srcvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4103
		nbl_start_crit(srcvp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4104
		in_crit_src = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4105
		if (nbl_conflict(srcvp, NBL_RENAME, 0, 0, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4106
			*cs->statusp = resp->status = NFS4ERR_FILE_OPEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4107
			goto err_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4108
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4109
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4110
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4111
	if (targvp && nbl_need_check(targvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4112
		nbl_start_crit(targvp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4113
		in_crit_targ = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4114
		if (nbl_conflict(targvp, NBL_REMOVE, 0, 0, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4115
			*cs->statusp = resp->status = NFS4ERR_FILE_OPEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4116
			goto err_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4117
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4118
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4119
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4120
	/* Get source "before" change value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4121
	obdva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4122
	error = VOP_GETATTR(odvp, &obdva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4123
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4124
		nbdva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4125
		error = VOP_GETATTR(ndvp, &nbdva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4126
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4127
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4128
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4129
		goto err_out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4130
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4131
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4132
	NFS4_SET_FATTR4_CHANGE(resp->source_cinfo.before, obdva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4133
	NFS4_SET_FATTR4_CHANGE(resp->target_cinfo.before, nbdva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4134
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4135
	if ((error = VOP_RENAME(odvp, onm, ndvp, nnm, cs->cr)) == 0 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4136
		fp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4137
		struct vattr va;
154
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4138
		vnode_t *tvp;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4139
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4140
		rfs4_dbe_lock(fp->dbe);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4141
		tvp = fp->vp;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4142
		if (tvp)
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4143
			VN_HOLD(tvp);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4144
		rfs4_dbe_unlock(fp->dbe);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4145
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4146
		if (tvp) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4147
			va.va_mask = AT_NLINK;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4148
			if (!VOP_GETATTR(tvp, &va, 0, cs->cr) &&
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4149
				va.va_nlink == 0) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4150
				/* The file is gone and so should the state */
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4151
				if (in_crit_targ) {
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4152
					nbl_end_crit(targvp);
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4153
					in_crit_targ = 0;
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4154
				}
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4155
				rfs4_close_all_state(fp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4156
			}
154
7a745d3804a2 6262563 NFSv4 server panic in fop_getattr due to bad call by rfs4_op_remove
shepler
parents: 0
diff changeset
  4157
			VN_RELE(tvp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4158
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4159
	}
1146
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4160
	if (error == 0) {
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4161
		char *tmp;
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4162
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4163
		/* fix the path name for the renamed file */
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4164
		mutex_enter(&srcvp->v_lock);
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4165
		tmp = srcvp->v_path;
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4166
		srcvp->v_path = NULL;
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4167
		mutex_exit(&srcvp->v_lock);
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4168
		vn_setpath(rootdir, ndvp, srcvp, nnm, nlen - 1);
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4169
		if (tmp != NULL)
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4170
			kmem_free(tmp, strlen(tmp) + 1);
c3555cdf52c2 6331154 v_path contains old name after file is renamed
jwahlig
parents: 602
diff changeset
  4171
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4172
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4173
	if (in_crit_src)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4174
		nbl_end_crit(srcvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4175
	if (srcvp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4176
		VN_RELE(srcvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4177
	if (in_crit_targ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4178
		nbl_end_crit(targvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4179
	if (targvp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4180
		VN_RELE(targvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4181
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4182
	if (sfp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4183
		rfs4_clear_dont_grant(sfp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4184
		rfs4_file_rele(sfp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4185
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4186
	if (fp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4187
		rfs4_clear_dont_grant(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4188
		rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4189
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4190
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4191
	kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4192
	kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4193
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4194
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4195
	 * Get the initial "after" sequence number, if it fails, set to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4196
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4197
	oidva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4198
	if (VOP_GETATTR(odvp, &oidva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4199
		oidva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4200
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4201
	nidva.va_mask = AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4202
	if (VOP_GETATTR(ndvp, &nidva, 0, cs->cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4203
		nidva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4204
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4205
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4206
	 * Force modified data and metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4207
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4208
	(void) VOP_FSYNC(odvp, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4209
	(void) VOP_FSYNC(ndvp, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4210
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4211
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4212
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4213
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4214
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4215
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4216
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4217
	 * Get "after" change values, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4218
	 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4219
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4220
	oadva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4221
	if (VOP_GETATTR(odvp, &oadva, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4222
		oadva.va_ctime = obdva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4223
		oadva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4224
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4225
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4226
	nadva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4227
	if (VOP_GETATTR(odvp, &nadva, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4228
		nadva.va_ctime = nbdva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4229
		nadva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4230
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4231
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4232
	NFS4_SET_FATTR4_CHANGE(resp->source_cinfo.after, oadva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4233
	NFS4_SET_FATTR4_CHANGE(resp->target_cinfo.after, nadva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4234
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4235
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4236
	 * The cinfo.atomic = TRUE only if we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4237
	 * non-zero va_seq's, and it has incremented by exactly one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4238
	 * during the VOP_RENAME and it didn't change during the VOP_FSYNC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4239
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4240
	if (obdva.va_seq && oidva.va_seq && oadva.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4241
			oidva.va_seq == (obdva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4242
			oidva.va_seq == oadva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4243
		resp->source_cinfo.atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4244
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4245
		resp->source_cinfo.atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4246
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4247
	if (nbdva.va_seq && nidva.va_seq && nadva.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4248
			nidva.va_seq == (nbdva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4249
			nidva.va_seq == nadva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4250
		resp->target_cinfo.atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4251
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4252
		resp->target_cinfo.atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4253
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4254
#ifdef	VOLATILE_FH_TEST
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4255
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4256
	extern void add_volrnm_fh(struct exportinfo *, vnode_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4257
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4258
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4259
	 * Add the renamed file handle to the volatile rename list
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4260
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4261
	if (cs->exi->exi_export.ex_flags & EX_VOLRNM) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4262
		/* file handles may expire on rename */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4263
		vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4264
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4265
		nnm = utf8_to_fn(&args->newname, &nlen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4266
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4267
		 * Already know that nnm will be a valid string
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4268
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4269
		error = VOP_LOOKUP(ndvp, nnm, &vp, NULL, 0, NULL, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4270
		kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4271
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4272
			add_volrnm_fh(cs->exi, vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4273
			VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4274
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4275
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4276
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4277
#endif	/* VOLATILE_FH_TEST */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4278
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4279
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4280
	return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4281
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4282
err_out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4283
	kmem_free(onm, olen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4284
	kmem_free(nnm, nlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4285
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4286
	if (in_crit_src) nbl_end_crit(srcvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4287
	if (in_crit_targ) nbl_end_crit(targvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4288
	if (targvp) VN_RELE(targvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4289
	if (srcvp) VN_RELE(srcvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4290
	if (sfp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4291
		if (sfp_rele_grant_hold) rfs4_clear_dont_grant(sfp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4292
		rfs4_file_rele(sfp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4293
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4294
	if (fp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4295
		if (fp_rele_grant_hold) rfs4_clear_dont_grant(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4296
		rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4297
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4298
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4299
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4300
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4301
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4302
rfs4_op_renew(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4303
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4304
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4305
	RENEW4args *args = &argop->nfs_argop4_u.oprenew;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4306
	RENEW4res *resp = &resop->nfs_resop4_u.oprenew;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4307
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4308
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4309
	if ((cp = rfs4_findclient_by_id(args->clientid, FALSE)) == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4310
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4311
			rfs4_check_clientid(&args->clientid, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4312
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4313
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4314
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4315
	if (rfs4_lease_expired(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4316
		rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4317
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4318
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4319
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4320
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4321
	rfs4_update_lease(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4322
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4323
	mutex_enter(cp->cbinfo.cb_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4324
	if (cp->cbinfo.cb_notified_of_cb_path_down == FALSE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4325
		cp->cbinfo.cb_notified_of_cb_path_down = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4326
		*cs->statusp = resp->status = NFS4ERR_CB_PATH_DOWN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4327
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4328
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4329
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4330
	mutex_exit(cp->cbinfo.cb_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4331
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4332
	rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4333
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4334
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4335
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4336
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4337
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4338
rfs4_op_restorefh(nfs_argop4 *args, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4339
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4340
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4341
	RESTOREFH4res *resp = &resop->nfs_resop4_u.oprestorefh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4342
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4343
	/* No need to check cs->access - we are not accessing any object */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4344
	if ((cs->saved_vp == NULL) || (cs->saved_fh.nfs_fh4_val == NULL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4345
		*cs->statusp = resp->status = NFS4ERR_RESTOREFH;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4346
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4347
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4348
	if (cs->vp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4349
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4350
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4351
	cs->vp = cs->saved_vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4352
	cs->saved_vp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4353
	cs->exi = cs->saved_exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4354
	nfs_fh4_copy(&cs->saved_fh, &cs->fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4355
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4356
	cs->deleg = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4357
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4358
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4359
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4360
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4361
rfs4_op_savefh(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4362
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4363
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4364
	SAVEFH4res *resp = &resop->nfs_resop4_u.opsavefh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4365
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4366
	/* No need to check cs->access - we are not accessing any object */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4367
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4368
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4369
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4370
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4371
	if (cs->saved_vp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4372
		VN_RELE(cs->saved_vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4373
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4374
	cs->saved_vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4375
	VN_HOLD(cs->saved_vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4376
	cs->saved_exi = cs->exi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4377
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4378
	 * since SAVEFH is fairly rare, don't alloc space for its fh
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4379
	 * unless necessary.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4380
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4381
	if (cs->saved_fh.nfs_fh4_val == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4382
		cs->saved_fh.nfs_fh4_val = kmem_alloc(NFS4_FHSIZE, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4383
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4384
	nfs_fh4_copy(&cs->fh, &cs->saved_fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4385
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4386
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4387
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4388
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4389
 * rfs4_verify_attr is called when nfsv4 Setattr failed, but we wish to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4390
 * return the bitmap of attrs that were set successfully. It is also
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4391
 * called by Verify/Nverify to test the vattr/vfsstat attrs. It should
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4392
 * always be called only after rfs4_do_set_attrs().
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4393
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4394
 * Verify that the attributes are same as the expected ones. sargp->vap
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4395
 * and sargp->sbp contain the input attributes as translated from fattr4.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4396
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4397
 * This function verifies only the attrs that correspond to a vattr or
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4398
 * vfsstat struct. That is because of the extra step needed to get the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4399
 * corresponding system structs. Other attributes have already been set or
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4400
 * verified by do_rfs4_set_attrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4401
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4402
 * Return 0 if all attrs match, -1 if some don't, error if error processing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4403
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4404
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4405
rfs4_verify_attr(struct nfs4_svgetit_arg *sargp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4406
	bitmap4 *resp, struct nfs4_ntov_table *ntovp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4407
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4408
	int error, ret_error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4409
	int i, k;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4410
	uint_t sva_mask = sargp->vap->va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4411
	uint_t vbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4412
	union nfs4_attr_u *na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4413
	uint8_t *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4414
	bool_t getsb = ntovp->vfsstat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4415
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4416
	if (sva_mask != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4417
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4418
		 * Okay to overwrite sargp->vap because we verify based
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4419
		 * on the incoming values.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4420
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4421
		ret_error = VOP_GETATTR(sargp->cs->vp, sargp->vap, 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4422
				sargp->cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4423
		if (ret_error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4424
			if (resp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4425
				return (ret_error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4426
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4427
			 * Must return bitmap of successful attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4428
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4429
			sva_mask = 0;	/* to prevent checking vap later */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4430
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4431
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4432
			 * Some file systems clobber va_mask. it is probably
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4433
			 * wrong of them to do so, nonethless we practice
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4434
			 * defensive coding.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4435
			 * See bug id 4276830.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4436
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4437
			sargp->vap->va_mask = sva_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4438
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4439
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4440
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4441
	if (getsb) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4442
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4443
		 * Now get the superblock and loop on the bitmap, as there is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4444
		 * no simple way of translating from superblock to bitmap4.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4445
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4446
		ret_error = VFS_STATVFS(sargp->cs->vp->v_vfsp, sargp->sbp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4447
		if (ret_error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4448
			if (resp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4449
				goto errout;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4450
			getsb = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4451
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4452
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4453
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4454
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4455
	 * Now loop and verify each attribute which getattr returned
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4456
	 * whether it's the same as the input.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4457
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4458
	if (resp == NULL && !getsb && (sva_mask == 0))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4459
		goto errout;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4460
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4461
	na = ntovp->na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4462
	amap = ntovp->amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4463
	k = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4464
	for (i = 0; i < ntovp->attrcnt; i++, na++, amap++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4465
		k = *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4466
		ASSERT(nfs4_ntov_map[k].nval == k);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4467
		vbit = nfs4_ntov_map[k].vbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4468
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4469
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4470
		 * If vattr attribute but VOP_GETATTR failed, or it's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4471
		 * superblock attribute but VFS_STATVFS failed, skip
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4472
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4473
		if (vbit) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4474
			if ((vbit & sva_mask) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4475
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4476
		} else if (!(getsb && nfs4_ntov_map[k].vfsstat)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4477
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4478
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4479
		error = (*nfs4_ntov_map[k].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4480
				NFS4ATTR_VERIT, sargp, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4481
		if (resp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4482
			if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4483
				ret_error = -1;	/* not all match */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4484
			else	/* update response bitmap */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4485
				*resp |= nfs4_ntov_map[k].fbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4486
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4487
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4488
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4489
			ret_error = -1;	/* not all match */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4490
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4491
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4492
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4493
errout:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4494
	return (ret_error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4495
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4496
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4497
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4498
 * Decode the attribute to be set/verified. If the attr requires a sys op
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4499
 * (VOP_GETATTR, VFS_VFSSTAT), and the request is to verify, then don't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4500
 * call the sv_getit function for it, because the sys op hasn't yet been done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4501
 * Return 0 for success, error code if failed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4502
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4503
 * Note: the decoded arg is not freed here but in nfs4_ntov_table_free.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4504
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4505
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4506
decode_fattr4_attr(nfs4_attr_cmd_t cmd, struct nfs4_svgetit_arg *sargp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4507
	int k, XDR *xdrp, bitmap4 *resp_bval, union nfs4_attr_u *nap)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4508
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4509
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4510
	bool_t set_later;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4511
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4512
	sargp->vap->va_mask |= nfs4_ntov_map[k].vbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4513
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4514
	if ((*nfs4_ntov_map[k].xfunc)(xdrp, nap)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4515
		set_later = nfs4_ntov_map[k].vbit || nfs4_ntov_map[k].vfsstat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4516
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4517
		 * don't verify yet if a vattr or sb dependent attr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4518
		 * because we don't have their sys values yet.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4519
		 * Will be done later.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4520
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4521
		if (! (set_later && (cmd == NFS4ATTR_VERIT))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4522
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4523
			 * ACLs are a special case, since setting the MODE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4524
			 * conflicts with setting the ACL.  We delay setting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4525
			 * the ACL until all other attributes have been set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4526
			 * The ACL gets set in do_rfs4_op_setattr().
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4527
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4528
			if (nfs4_ntov_map[k].fbit != FATTR4_ACL_MASK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4529
				error = (*nfs4_ntov_map[k].sv_getit)(cmd,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4530
				    sargp, nap);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4531
				if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4532
					xdr_free(nfs4_ntov_map[k].xfunc,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4533
					    (caddr_t)nap);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4534
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4535
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4536
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4537
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4538
#ifdef  DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4539
		cmn_err(CE_NOTE, "decode_fattr4_attr: error "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4540
			"decoding attribute %d\n", k);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4541
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4542
		error = EINVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4543
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4544
	if (!error && resp_bval && !set_later) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4545
		*resp_bval |= nfs4_ntov_map[k].fbit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4546
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4547
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4548
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4549
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4550
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4551
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4552
 * Set vattr based on incoming fattr4 attrs - used by setattr.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4553
 * Set response mask. Ignore any values that are not writable vattr attrs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4554
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4555
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4556
do_rfs4_set_attrs(bitmap4 *resp, fattr4 *fattrp, struct compound_state *cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4557
		struct nfs4_svgetit_arg *sargp, struct nfs4_ntov_table *ntovp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4558
		nfs4_attr_cmd_t cmd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4559
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4560
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4561
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4562
	char *attrs = fattrp->attrlist4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4563
	uint32_t attrslen = fattrp->attrlist4_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4564
	XDR xdr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4565
	nfsstat4 status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4566
	vnode_t *vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4567
	union nfs4_attr_u *na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4568
	uint8_t *amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4569
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4570
#ifndef lint
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4571
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4572
	 * Make sure that maximum attribute number can be expressed as an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4573
	 * 8 bit quantity.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4574
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4575
	ASSERT(NFS4_MAXNUM_ATTRS <= (UINT8_MAX + 1));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4576
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4577
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4578
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4579
		if (resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4580
			*resp = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4581
		return (NFS4ERR_NOFILEHANDLE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4582
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4583
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4584
		if (resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4585
			*resp = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4586
		return (NFS4ERR_ACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4587
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4588
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4589
	sargp->op = cmd;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4590
	sargp->cs = cs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4591
	sargp->flag = 0;	/* may be set later */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4592
	sargp->vap->va_mask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4593
	sargp->rdattr_error = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4594
	sargp->rdattr_error_req = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4595
	/* sargp->sbp is set by the caller */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4596
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4597
	xdrmem_create(&xdr, attrs, attrslen, XDR_DECODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4598
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4599
	na = ntovp->na;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4600
	amap = ntovp->amap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4601
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4602
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4603
	 * The following loop iterates on the nfs4_ntov_map checking
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4604
	 * if the fbit is set in the requested bitmap.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4605
	 * If set then we process the arguments using the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4606
	 * rfs4_fattr4 conversion functions to populate the setattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4607
	 * vattr and va_mask. Any settable attrs that are not using vattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4608
	 * will be set in this loop.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4609
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4610
	for (i = 0; i < nfs4_ntov_map_size; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4611
		if (!(fattrp->attrmask & nfs4_ntov_map[i].fbit)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4612
			continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4613
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4614
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4615
		 * If setattr, must be a writable attr.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4616
		 * If verify/nverify, must be a readable attr.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4617
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4618
		if ((error = (*nfs4_ntov_map[i].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4619
				    NFS4ATTR_SUPPORTED, sargp, NULL)) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4620
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4621
			 * Client tries to set/verify an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4622
			 * unsupported attribute, tries to set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4623
			 * a read only attr or verify a write
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4624
			 * only one - error!
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4625
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4626
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4627
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4628
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4629
		 * Decode the attribute to set/verify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4630
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4631
		error = decode_fattr4_attr(cmd, sargp, nfs4_ntov_map[i].nval,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4632
					&xdr, resp ? resp : NULL, na);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4633
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4634
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4635
		*amap++ = (uint8_t)nfs4_ntov_map[i].nval;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4636
		na++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4637
		(ntovp->attrcnt)++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4638
		if (nfs4_ntov_map[i].vfsstat)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4639
			ntovp->vfsstat = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4640
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4641
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4642
	if (error != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4643
		status = (error == ENOTSUP ?	NFS4ERR_ATTRNOTSUPP :
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4644
						puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4645
	/* xdrmem_destroy(&xdrs); */	/* NO-OP */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4646
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4647
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4648
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4649
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4650
do_rfs4_op_setattr(bitmap4 *resp, fattr4 *fattrp, struct compound_state *cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4651
		stateid4 *stateid)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4652
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4653
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4654
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4655
	bool_t trunc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4656
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4657
	nfsstat4 status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4658
	cred_t *cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4659
	vnode_t *vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4660
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4661
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4662
	struct vattr bva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4663
	struct flock64 bf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4664
	int in_crit = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4665
	uint_t saved_mask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4666
	caller_context_t ct;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4667
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4668
	*resp = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4669
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4670
	nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4671
	status = do_rfs4_set_attrs(resp, fattrp, cs, &sarg, &ntov,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4672
			NFS4ATTR_SETIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4673
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4674
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4675
		 * failed set attrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4676
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4677
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4678
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4679
	if ((sarg.vap->va_mask == 0) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4680
	    (! (fattrp->attrmask & FATTR4_ACL_MASK))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4681
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4682
		 * no further work to be done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4683
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4684
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4685
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4686
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4687
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4688
	 * If we got a request to set the ACL and the MODE, only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4689
	 * allow changing VSUID, VSGID, and VSVTX.  Attempting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4690
	 * to change any other bits, along with setting an ACL,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4691
	 * gives NFS4ERR_INVAL.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4692
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4693
	if ((fattrp->attrmask & FATTR4_ACL_MASK) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4694
	    (fattrp->attrmask & FATTR4_MODE_MASK)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4695
		vattr_t va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4696
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4697
		va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4698
		error = VOP_GETATTR(vp, &va, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4699
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4700
			status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4701
			goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4702
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4703
		if ((sarg.vap->va_mode ^ va.va_mode) &
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4704
		    ~(VSUID | VSGID | VSVTX)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4705
			status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4706
			goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4707
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4708
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4709
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4710
	/* Check stateid only if size has been set */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4711
	if (sarg.vap->va_mask & AT_SIZE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4712
		trunc = (sarg.vap->va_size == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4713
		status = rfs4_check_stateid(FWRITE, cs->vp, stateid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4714
			trunc, &cs->deleg, sarg.vap->va_mask & AT_SIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4715
		if (status != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4716
			goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4717
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4718
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4719
	ct.cc_sysid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4720
	ct.cc_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4721
	ct.cc_caller_id = nfs4_srv_caller_id;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4722
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4723
	/* XXX start of possible race with delegations */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4724
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4725
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4726
	 * We need to specially handle size changes because it is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4727
	 * possible for the client to create a file with read-only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4728
	 * modes, but with the file opened for writing. If the client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4729
	 * then tries to set the file size, e.g. ftruncate(3C),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4730
	 * fcntl(F_FREESP), the normal access checking done in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4731
	 * VOP_SETATTR would prevent the client from doing it even though
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4732
	 * it should be allowed to do so.  To get around this, we do the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4733
	 * access checking for ourselves and use VOP_SPACE which doesn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4734
	 * do the access checking.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4735
	 * Also the client should not be allowed to change the file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4736
	 * size if there is a conflicting non-blocking mandatory lock in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4737
	 * the region of the change.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4738
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4739
	if (vp->v_type == VREG && (sarg.vap->va_mask & AT_SIZE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4740
		u_offset_t offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4741
		ssize_t length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4742
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4743
		/*
602
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4744
		 * ufs_setattr clears AT_SIZE from vap->va_mask, but
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4745
		 * before returning, sarg.vap->va_mask is used to
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4746
		 * generate the setattr reply bitmap.  We also clear
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4747
		 * AT_SIZE below before calling VOP_SPACE.  For both
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4748
		 * of these cases, the va_mask needs to be saved here
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4749
		 * and restored after calling VOP_SETATTR.
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4750
		 */
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4751
		saved_mask = sarg.vap->va_mask;
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4752
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4753
		/*
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4754
		 * Check any possible conflict due to NBMAND locks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4755
		 * Get into critical region before VOP_GETATTR, so the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4756
		 * size attribute is valid when checking conflicts.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4757
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4758
		if (nbl_need_check(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4759
			nbl_start_crit(vp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4760
			in_crit = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4761
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4762
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4763
		bva.va_mask = AT_UID|AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4764
		if (error = VOP_GETATTR(vp, &bva, 0, cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4765
			status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4766
			goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4767
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4768
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4769
		if (in_crit) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4770
			if (sarg.vap->va_size < bva.va_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4771
				offset = sarg.vap->va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4772
				length = bva.va_size - sarg.vap->va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4773
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4774
				offset = bva.va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4775
				length = sarg.vap->va_size - bva.va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4776
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4777
			if (nbl_conflict(vp, NBL_WRITE, offset, length, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4778
				status = NFS4ERR_LOCKED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4779
				goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4780
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4781
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4782
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4783
		if (crgetuid(cr) == bva.va_uid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4784
			sarg.vap->va_mask &= ~AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4785
			bf.l_type = F_WRLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4786
			bf.l_whence = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4787
			bf.l_start = (off64_t)sarg.vap->va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4788
			bf.l_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4789
			bf.l_sysid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4790
			bf.l_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4791
			error = VOP_SPACE(vp, F_FREESP, &bf, FWRITE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4792
					(offset_t)sarg.vap->va_size, cr, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4793
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4794
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4795
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4796
	if (!error && sarg.vap->va_mask != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4797
		error = VOP_SETATTR(vp, sarg.vap, sarg.flag, cr, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4798
602
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4799
	/* restore va_mask -- ufs_setattr clears AT_SIZE */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4800
	if (saved_mask & AT_SIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4801
		sarg.vap->va_mask |= AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4802
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4803
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4804
	 * If an ACL was being set, it has been delayed until now,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4805
	 * in order to set the mode (via the VOP_SETATTR() above) first.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4806
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4807
	if ((! error) && (fattrp->attrmask & FATTR4_ACL_MASK)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4808
		int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4809
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4810
		for (i = 0; i < NFS4_MAXNUM_ATTRS; i++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4811
			if (ntov.amap[i] == FATTR4_ACL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4812
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4813
		if (i < NFS4_MAXNUM_ATTRS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4814
			error = (*nfs4_ntov_map[FATTR4_ACL].sv_getit)(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4815
			    NFS4ATTR_SETIT, &sarg, &ntov.na[i]);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4816
			if (error == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4817
				*resp |= FATTR4_ACL_MASK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4818
			} else if (error == ENOTSUP) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4819
				(void) rfs4_verify_attr(&sarg, resp, &ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4820
				status = NFS4ERR_ATTRNOTSUPP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4821
				goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4822
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4823
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4824
			NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4825
			    (CE_NOTE, "do_rfs4_op_setattr: "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4826
			    "unable to find ACL in fattr4"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4827
			error = EINVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4828
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4829
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4830
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4831
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4832
		status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4833
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4834
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4835
		 * Set the response bitmap when setattr failed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4836
		 * If VOP_SETATTR partially succeeded, test by doing a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4837
		 * VOP_GETATTR on the object and comparing the data
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4838
		 * to the setattr arguments.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4839
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4840
		(void) rfs4_verify_attr(&sarg, resp, &ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4841
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4842
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4843
		 * Force modified metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4844
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4845
		(void) VOP_FSYNC(vp, FNODSYNC, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4846
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4847
		 * Set response bitmap
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4848
		 */
602
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4849
		nfs4_vmask_to_nmask_set(sarg.vap->va_mask, resp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4850
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4851
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4852
/* Return early and already have a NFSv4 error */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4853
done:
602
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4854
	/*
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4855
	 * Except for nfs4_vmask_to_nmask_set(), vattr --> fattr
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4856
	 * conversion sets both readable and writeable NFS4 attrs
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4857
	 * for AT_MTIME and AT_ATIME.  The line below masks out
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4858
	 * unrequested attrs from the setattr result bitmap.  This
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4859
	 * is placed after the done: label to catch the ATTRNOTSUP
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4860
	 * case.
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4861
	 */
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4862
	*resp &= fattrp->attrmask;
4ef537c7acd0 6284687 SETATTR(size) with special stateid=0 does not have bitmask set
jasmith
parents: 254
diff changeset
  4863
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4864
	if (in_crit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4865
		nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4866
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4867
	nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4868
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4869
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4870
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4871
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4872
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4873
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4874
rfs4_op_setattr(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4875
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4876
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4877
	SETATTR4args *args = &argop->nfs_argop4_u.opsetattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4878
	SETATTR4res *resp = &resop->nfs_resop4_u.opsetattr;
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4879
	bslabel_t *clabel;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4880
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4881
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4882
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4883
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4884
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4885
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4886
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4887
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4888
	 * do not allow to setattr on this vnode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4889
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4890
	if (vn_ismntpt(cs->vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4891
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4892
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4893
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4894
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4895
	resp->attrsset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4896
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4897
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4898
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4899
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4900
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4901
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4902
	/* check label before setting attributes */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4903
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4904
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4905
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4906
		DTRACE_PROBE2(tx__rfs4__log__info__opsetattr__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4907
		    "got client label from request(1)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4908
		    struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4909
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4910
			if (!do_rfs4_label_check(clabel, cs->vp,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4911
			    EQUALITY_CHECK)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4912
				*cs->statusp = resp->status = NFS4ERR_ACCESS;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4913
				return;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4914
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4915
		}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4916
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  4917
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4918
	*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4919
		do_rfs4_op_setattr(&resp->attrsset, &args->obj_attributes, cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4920
			&args->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4921
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4922
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4923
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4924
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4925
rfs4_op_verify(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4926
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4927
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4928
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4929
	 * verify and nverify are exactly the same, except that nverify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4930
	 * succeeds when some argument changed, and verify succeeds when
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4931
	 * when none changed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4932
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4933
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4934
	VERIFY4args  *args = &argop->nfs_argop4_u.opverify;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4935
	VERIFY4res *resp = &resop->nfs_resop4_u.opverify;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4936
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4937
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4938
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4939
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4940
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4941
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4942
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4943
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4944
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4945
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4946
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4947
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4948
	nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4949
	resp->status = do_rfs4_set_attrs(NULL, &args->obj_attributes, cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4950
				&sarg, &ntov, NFS4ATTR_VERIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4951
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4952
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4953
		 * do_rfs4_set_attrs will try to verify systemwide attrs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4954
		 * so could return -1 for "no match".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4955
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4956
		if (resp->status == -1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4957
			resp->status = NFS4ERR_NOT_SAME;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4958
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4959
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4960
	error = rfs4_verify_attr(&sarg, NULL, &ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4961
	switch (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4962
	case 0:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4963
		resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4964
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4965
	case -1:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4966
		resp->status = NFS4ERR_NOT_SAME;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4967
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4968
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4969
		resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4970
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4971
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4972
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4973
	*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4974
	nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4975
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4976
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4977
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4978
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4979
rfs4_op_nverify(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4980
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4981
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4982
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4983
	 * verify and nverify are exactly the same, except that nverify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4984
	 * succeeds when some argument changed, and verify succeeds when
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4985
	 * when none changed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4986
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4987
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4988
	NVERIFY4args  *args = &argop->nfs_argop4_u.opnverify;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4989
	NVERIFY4res *resp = &resop->nfs_resop4_u.opnverify;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4990
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4991
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4992
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4993
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4994
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4995
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4996
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4997
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4998
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4999
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5000
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5001
	nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5002
	resp->status = do_rfs4_set_attrs(NULL, &args->obj_attributes, cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5003
				&sarg, &ntov, NFS4ATTR_VERIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5004
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5005
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5006
		 * do_rfs4_set_attrs will try to verify systemwide attrs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5007
		 * so could return -1 for "no match".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5008
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5009
		if (resp->status == -1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5010
			resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5011
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5012
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5013
	error = rfs4_verify_attr(&sarg, NULL, &ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5014
	switch (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5015
	case 0:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5016
		resp->status = NFS4ERR_SAME;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5017
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5018
	case -1:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5019
		resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5020
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5021
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5022
		resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5023
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5024
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5025
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5026
	*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5027
	nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5028
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5029
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5030
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5031
 * XXX - This should live in an NFS header file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5032
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5033
#define	MAX_IOVECS	12
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5034
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5035
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5036
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5037
rfs4_op_write(nfs_argop4 *argop, nfs_resop4 *resop, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5038
	struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5039
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5040
	WRITE4args  *args = &argop->nfs_argop4_u.opwrite;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5041
	WRITE4res *resp = &resop->nfs_resop4_u.opwrite;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5042
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5043
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5044
	struct vattr bva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5045
	u_offset_t rlimit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5046
	struct uio uio;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5047
	struct iovec iov[MAX_IOVECS];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5048
	struct iovec *iovp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5049
	int iovcnt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5050
	int ioflag;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5051
	cred_t *savecred, *cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5052
	bool_t *deleg = &cs->deleg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5053
	nfsstat4 stat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5054
	int in_crit = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5055
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5056
	vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5057
	if (vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5058
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5059
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5060
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5061
	if (cs->access == CS_ACCESS_DENIED) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5062
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5063
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5064
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5065
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5066
	cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5067
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5068
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5069
	 * We have to enter the critical region before calling VOP_RWLOCK
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5070
	 * to avoid a deadlock with ufs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5071
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5072
	if (nbl_need_check(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5073
		nbl_start_crit(vp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5074
		in_crit = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5075
		if (nbl_conflict(vp, NBL_WRITE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5076
				args->offset, args->data_len, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5077
			*cs->statusp = resp->status = NFS4ERR_LOCKED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5078
			goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5079
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5080
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5081
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5082
	if ((stat = rfs4_check_stateid(FWRITE, vp, &args->stateid, FALSE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5083
					deleg, TRUE)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5084
		*cs->statusp = resp->status = stat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5085
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5086
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5087
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5088
	bva.va_mask = AT_MODE | AT_UID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5089
	error = VOP_GETATTR(vp, &bva, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5090
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5091
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5092
	 * If we can't get the attributes, then we can't do the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5093
	 * right access checking.  So, we'll fail the request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5094
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5095
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5096
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5097
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5098
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5099
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5100
	if (rdonly4(cs->exi, cs->vp, req)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5101
		*cs->statusp = resp->status = NFS4ERR_ROFS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5102
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5103
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5104
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5105
	if (vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5106
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5107
			((vp->v_type == VDIR) ? NFS4ERR_ISDIR : NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5108
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5109
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5110
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5111
	if (crgetuid(cr) != bva.va_uid &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5112
	    (error = VOP_ACCESS(vp, VWRITE, 0, cr))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5113
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5114
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5115
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5116
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5117
	if (MANDLOCK(vp, bva.va_mode)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5118
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5119
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5120
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5121
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5122
	if (args->data_len == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5123
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5124
		resp->count = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5125
		resp->committed = args->stable;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5126
		resp->writeverf = Write4verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5127
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5128
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5129
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5130
	if (args->mblk != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5131
		mblk_t *m;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5132
		uint_t bytes, round_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5133
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5134
		iovcnt = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5135
		bytes = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5136
		round_len = roundup(args->data_len, BYTES_PER_XDR_UNIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5137
		for (m = args->mblk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5138
		    m != NULL && bytes < round_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5139
		    m = m->b_cont) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5140
			iovcnt++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5141
			bytes += MBLKL(m);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5142
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5143
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5144
		/* should have ended on an mblk boundary */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5145
		if (bytes != round_len) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5146
			printf("bytes=0x%x, round_len=0x%x, req len=0x%x\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5147
			    bytes, round_len, args->data_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5148
			printf("args=%p, args->mblk=%p, m=%p", (void *)args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5149
			    (void *)args->mblk, (void *)m);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5150
			ASSERT(bytes == round_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5151
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5152
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5153
		if (iovcnt <= MAX_IOVECS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5154
			iovp = iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5155
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5156
			iovp = kmem_alloc(sizeof (*iovp) * iovcnt, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5157
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5158
		mblk_to_iov(args->mblk, iovcnt, iovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5159
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5160
		iovcnt = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5161
		iovp = iov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5162
		iovp->iov_base = args->data_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5163
		iovp->iov_len = args->data_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5164
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5165
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5166
	uio.uio_iov = iovp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5167
	uio.uio_iovcnt = iovcnt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5168
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5169
	uio.uio_segflg = UIO_SYSSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5170
	uio.uio_extflg = UIO_COPY_DEFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5171
	uio.uio_loffset = args->offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5172
	uio.uio_resid = args->data_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5173
	uio.uio_llimit = curproc->p_fsz_ctl;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5174
	rlimit = uio.uio_llimit - args->offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5175
	if (rlimit < (u_offset_t)uio.uio_resid)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5176
		uio.uio_resid = (int)rlimit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5177
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5178
	if (args->stable == UNSTABLE4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5179
		ioflag = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5180
	else if (args->stable == FILE_SYNC4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5181
		ioflag = FSYNC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5182
	else if (args->stable == DATA_SYNC4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5183
		ioflag = FDSYNC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5184
	else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5185
		if (iovp != iov)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5186
			kmem_free(iovp, sizeof (*iovp) * iovcnt);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5187
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5188
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5189
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5190
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5191
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5192
	 * We're changing creds because VM may fault and we need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5193
	 * the cred of the current thread to be used if quota
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5194
	 * checking is enabled.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5195
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5196
	savecred = curthread->t_cred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5197
	curthread->t_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5198
	error = do_io(FWRITE, vp, &uio, ioflag, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5199
	curthread->t_cred = savecred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5200
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5201
	if (iovp != iov)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5202
		kmem_free(iovp, sizeof (*iovp) * iovcnt);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5203
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5204
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5205
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5206
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5207
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5208
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5209
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5210
	resp->count = args->data_len - uio.uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5211
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5212
	if (ioflag == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5213
		resp->committed = UNSTABLE4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5214
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5215
		resp->committed = FILE_SYNC4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5216
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5217
	resp->writeverf = Write4verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5218
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5219
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5220
	if (in_crit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5221
		nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5222
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5223
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5224
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5225
/* XXX put in a header file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5226
extern int	sec_svc_getcred(struct svc_req *, cred_t *,  caddr_t *, int *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5227
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5228
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5229
rfs4_compound(COMPOUND4args *args, COMPOUND4res *resp, struct exportinfo *exi,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5230
	struct svc_req *req, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5231
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5232
	uint_t i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5233
	struct compound_state cs;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5234
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5235
	rfs4_init_compound_state(&cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5236
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5237
	 * Form a reply tag by copying over the reqeuest tag.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5238
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5239
	resp->tag.utf8string_val =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5240
				kmem_alloc(args->tag.utf8string_len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5241
	resp->tag.utf8string_len = args->tag.utf8string_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5242
	bcopy(args->tag.utf8string_val, resp->tag.utf8string_val,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5243
					resp->tag.utf8string_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5244
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5245
	cs.statusp = &resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5246
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5247
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5248
	 * XXX for now, minorversion should be zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5249
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5250
	if (args->minorversion != NFS4_MINORVERSION) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5251
		resp->array_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5252
		resp->array = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5253
		resp->status = NFS4ERR_MINOR_VERS_MISMATCH;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5254
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5255
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5256
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5257
	resp->array_len = args->array_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5258
	resp->array = kmem_zalloc(args->array_len * sizeof (nfs_resop4),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5259
		KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5260
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5261
	ASSERT(exi == NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5262
	ASSERT(cr == NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5263
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5264
	cr = crget();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5265
	ASSERT(cr != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5266
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5267
	if (sec_svc_getcred(req, cr, &cs.principal, &cs.nfsflavor) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5268
		crfree(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5269
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5270
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5271
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5272
	cs.basecr = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5273
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5274
	cs.req = req;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5275
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5276
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5277
	 * For now, NFS4 compound processing must be protected by
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5278
	 * exported_lock because it can access more than one exportinfo
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5279
	 * per compound and share/unshare can now change multiple
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5280
	 * exinfo structs.  The NFS2/3 code only refs 1 exportinfo
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5281
	 * per proc (excluding public exinfo), and exi_count design
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5282
	 * is sufficient to protect concurrent execution of NFS2/3
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5283
	 * ops along with unexport.  This lock will be removed as
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5284
	 * part of the NFSv4 phase 2 namespace redesign work.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5285
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5286
	rw_enter(&exported_lock, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5287
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5288
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5289
	 * If this is the first compound we've seen, we need to start all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5290
	 * new instances' grace periods.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5291
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5292
	if (rfs4_seen_first_compound == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5293
		rfs4_grace_start_new();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5294
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5295
		 * This must be set after rfs4_grace_start_new(), otherwise
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5296
		 * another thread could proceed past here before the former
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5297
		 * is finished.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5298
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5299
		rfs4_seen_first_compound = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5300
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5301
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5302
	for (i = 0; i < args->array_len && cs.cont; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5303
		nfs_argop4 *argop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5304
		nfs_resop4 *resop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5305
		uint_t op;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5306
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5307
		argop = &args->array[i];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5308
		resop = &resp->array[i];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5309
		resop->resop = argop->argop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5310
		op = (uint_t)resop->resop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5311
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5312
		if (op < rfsv4disp_cnt) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5313
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5314
			 * Count the individual ops here; NULL and COMPOUND
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5315
			 * are counted in common_dispatch()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5316
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5317
			rfsproccnt_v4_ptr[op].value.ui64++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5318
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5319
			NFS4_DEBUG(rfs4_debug > 1,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5320
				(CE_NOTE, "Executing %s", rfs4_op_string[op]));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5321
			(*rfsv4disptab[op].dis_proc)(argop, resop, req, &cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5322
			NFS4_DEBUG(rfs4_debug > 1,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5323
				(CE_NOTE, "%s returned %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5324
				rfs4_op_string[op], *cs.statusp));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5325
			if (*cs.statusp != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5326
				cs.cont = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5327
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5328
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5329
			 * This is effectively dead code since XDR code
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5330
			 * will have already returned BADXDR if op doesn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5331
			 * decode to legal value.  This only done for a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5332
			 * day when XDR code doesn't verify v4 opcodes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5333
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5334
			op = OP_ILLEGAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5335
			rfsproccnt_v4_ptr[OP_ILLEGAL_IDX].value.ui64++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5336
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5337
			rfs4_op_illegal(argop, resop, req, &cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5338
			cs.cont = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5339
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5340
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5341
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5342
		 * If not at last op, and if we are to stop, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5343
		 * compact the results array.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5344
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5345
		if ((i + 1) < args->array_len && !cs.cont) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5346
			nfs_resop4 *new_res = kmem_alloc(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5347
				(i+1) * sizeof (nfs_resop4), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5348
			bcopy(resp->array,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5349
				new_res, (i+1) * sizeof (nfs_resop4));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5350
			kmem_free(resp->array,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5351
				args->array_len * sizeof (nfs_resop4));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5352
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5353
			resp->array_len =  i + 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5354
			resp->array = new_res;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5355
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5356
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5357
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5358
	rw_exit(&exported_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5360
	if (cs.vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5361
		VN_RELE(cs.vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5362
	if (cs.saved_vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5363
		VN_RELE(cs.saved_vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5364
	if (cs.saved_fh.nfs_fh4_val)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5365
		kmem_free(cs.saved_fh.nfs_fh4_val, NFS4_FHSIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5366
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5367
	if (cs.basecr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5368
		crfree(cs.basecr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5369
	if (cs.cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5370
		crfree(cs.cr);
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5371
	/*
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5372
	 * done with this compound request, free the label
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5373
	 */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5374
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5375
	if (req->rq_label != NULL) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5376
		kmem_free(req->rq_label, sizeof (bslabel_t));
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5377
		req->rq_label = NULL;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5378
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5379
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5380
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5381
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5382
 * XXX because of what appears to be duplicate calls to rfs4_compound_free
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5383
 * XXX zero out the tag and array values. Need to investigate why the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5384
 * XXX calls occur, but at least prevent the panic for now.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5385
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5386
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5387
rfs4_compound_free(COMPOUND4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5388
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5389
	uint_t i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5390
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5391
	if (resp->tag.utf8string_val) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5392
		UTF8STRING_FREE(resp->tag)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5393
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5394
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5395
	for (i = 0; i < resp->array_len; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5396
		nfs_resop4 *resop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5397
		uint_t op;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5398
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5399
		resop = &resp->array[i];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5400
		op = (uint_t)resop->resop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5401
		if (op < rfsv4disp_cnt) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5402
			(*rfsv4disptab[op].dis_resfree)(resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5403
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5404
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5405
	if (resp->array != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5406
		kmem_free(resp->array, resp->array_len * sizeof (nfs_resop4));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5407
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5408
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5409
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5410
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5411
 * Process the value of the compound request rpc flags, as a bit-AND
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5412
 * of the individual per-op flags (idempotent, allowork, publicfh_ok)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5413
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5414
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5415
rfs4_compound_flagproc(COMPOUND4args *args, int *flagp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5416
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5417
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5418
	int flag = RPC_ALL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5419
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5420
	for (i = 0; flag && i < args->array_len; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5421
		uint_t op;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5422
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5423
		op = (uint_t)args->array[i].argop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5424
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5425
		if (op < rfsv4disp_cnt)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5426
			flag &= rfsv4disptab[op].dis_flags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5427
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5428
			flag = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5429
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5430
	*flagp = flag;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5431
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5432
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5433
nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5434
rfs4_client_sysid(rfs4_client_t *cp, sysid_t *sp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5435
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5436
	nfsstat4 e;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5437
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5438
	rfs4_dbe_lock(cp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5439
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5440
	if (cp->sysidt != LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5441
		*sp = cp->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5442
		e = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5443
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5444
	} else if ((cp->sysidt = lm_alloc_sysidt()) != LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5445
		*sp = cp->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5446
		e = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5447
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5448
		NFS4_DEBUG(rfs4_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5449
			"rfs4_client_sysid: allocated 0x%x\n", *sp));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5450
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5451
		e = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5452
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5453
	rfs4_dbe_unlock(cp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5454
	return (e);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5455
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5456
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5457
#if defined(DEBUG) && ! defined(lint)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5458
static void lock_print(char *str, int operation, struct flock64 *flk)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5459
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5460
	char *op, *type;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5461
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5462
	switch (operation) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5463
	case F_GETLK: op = "F_GETLK";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5464
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5465
	case F_SETLK: op = "F_SETLK";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5466
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5467
	default: op = "F_UNKNOWN";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5468
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5469
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5470
	switch (flk->l_type) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5471
	case F_UNLCK: type = "F_UNLCK";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5472
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5473
	case F_RDLCK: type = "F_RDLCK";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5474
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5475
	case F_WRLCK: type = "F_WRLCK";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5476
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5477
	default: type = "F_UNKNOWN";
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5478
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5479
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5480
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5481
	ASSERT(flk->l_whence == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5482
	cmn_err(CE_NOTE, "%s:  %s, type = %s, off = %llx len = %llx pid = %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5483
		str, op, type,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5484
		(longlong_t)flk->l_start,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5485
		flk->l_len ? (longlong_t)flk->l_len : ~0LL,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5486
		flk->l_pid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5487
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5488
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5489
#define	LOCK_PRINT(d, s, t, f) if (d) lock_print(s, t, f)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5490
#else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5491
#define	LOCK_PRINT(d, s, t, f)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5492
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5493
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5494
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5495
static bool_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5496
creds_ok(cred_set_t cr_set, struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5497
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5498
	return (TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5499
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5500
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5501
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5502
 * Look up the pathname using the vp in cs as the directory vnode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5503
 * cs->vp will be the vnode for the file on success
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5504
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5505
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5506
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5507
rfs4_lookup(component4 *component, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5508
	    struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5509
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5510
	char *nm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5511
	uint32_t len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5512
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5513
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5514
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5515
		return (NFS4ERR_NOFILEHANDLE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5516
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5517
	if (cs->vp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5518
		return (NFS4ERR_NOTDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5519
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5520
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5521
	if (!utf8_dir_verify(component))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5522
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5523
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5524
	nm = utf8_to_fn(component, &len, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5525
	if (nm == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5526
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5527
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5528
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5529
	if (len > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5530
		kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5531
		return (NFS4ERR_NAMETOOLONG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5532
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5533
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5534
	status = do_rfs4_op_lookup(nm, len, req, cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5535
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5536
	kmem_free(nm, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5537
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5538
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5539
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5540
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5541
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5542
rfs4_lookupfile(component4 *component, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5543
		struct compound_state *cs, uint32_t access,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5544
		change_info4 *cinfo)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5545
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5546
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5547
	vnode_t *dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5548
	vattr_t bva, ava, fva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5549
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5550
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5551
	/* Get "before" change value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5552
	bva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5553
	error = VOP_GETATTR(dvp, &bva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5554
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5555
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5556
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5557
	/* rfs4_lookup may VN_RELE directory */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5558
	VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5559
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5560
	status = rfs4_lookup(component, req, cs);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5561
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5562
		VN_RELE(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5563
		return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5564
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5565
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5566
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5567
	 * Get "after" change value, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5568
	 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5569
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5570
	ava.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5571
	if (VOP_GETATTR(dvp, &ava, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5572
		ava.va_ctime = bva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5573
		ava.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5574
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5575
	VN_RELE(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5576
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5577
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5578
	 * Validate the file is a file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5579
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5580
	fva.va_mask = AT_TYPE|AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5581
	error = VOP_GETATTR(cs->vp, &fva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5582
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5583
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5584
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5585
	if (fva.va_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5586
		if (fva.va_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5587
			return (NFS4ERR_ISDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5588
		if (fva.va_type == VLNK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5589
			return (NFS4ERR_SYMLINK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5590
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5591
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5592
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5593
	NFS4_SET_FATTR4_CHANGE(cinfo->before, bva.va_ctime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5594
	NFS4_SET_FATTR4_CHANGE(cinfo->after, ava.va_ctime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5595
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5596
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5597
	 * It is undefined if VOP_LOOKUP will change va_seq, so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5598
	 * cinfo.atomic = TRUE only if we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5599
	 * non-zero va_seq's, and they have not changed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5600
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5601
	if (bva.va_seq && ava.va_seq && ava.va_seq == bva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5602
		cinfo->atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5603
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5604
		cinfo->atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5605
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5606
	/* Check for mandatory locking */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5607
	cs->mandlock = MANDLOCK(cs->vp, fva.va_mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5608
	return (check_open_access(access, cs, req));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5609
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5610
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5611
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5612
create_vnode(vnode_t *dvp, char *nm,  vattr_t *vap, createmode4 mode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5613
	    timespec32_t *mtime, cred_t *cr, vnode_t **vpp, bool_t *created)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5614
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5615
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5616
	nfsstat4 status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5617
	vattr_t va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5618
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5619
tryagain:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5621
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5622
	 * The file open mode used is VWRITE.  If the client needs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5623
	 * some other semantic, then it should do the access checking
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5624
	 * itself.  It would have been nice to have the file open mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5625
	 * passed as part of the arguments.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5626
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5627
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5628
	*created = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5629
	error = VOP_CREATE(dvp, nm, vap, EXCL, VWRITE, vpp, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5630
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5631
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5632
		*created = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5633
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5634
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5635
		 * If we got something other than file already exists
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5636
		 * then just return this error.  Otherwise, we got
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5637
		 * EEXIST.  If we were doing a GUARDED create, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5638
		 * just return this error.  Otherwise, we need to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5639
		 * make sure that this wasn't a duplicate of an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5640
		 * exclusive create request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5641
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5642
		 * The assumption is made that a non-exclusive create
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5643
		 * request will never return EEXIST.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5644
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5645
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5646
		if (error != EEXIST || mode == GUARDED4) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5647
			status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5648
			return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5649
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5650
		error = VOP_LOOKUP(dvp, nm, vpp, NULL, 0, NULL, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5651
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5652
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5653
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5654
			 * We couldn't find the file that we thought that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5655
			 * we just created.  So, we'll just try creating
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5656
			 * it again.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5657
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5658
			if (error == ENOENT)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5659
				goto tryagain;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5660
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5661
			status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5662
			return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5663
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5664
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5665
		if (mode == UNCHECKED4) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5666
			/* existing object must be regular file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5667
			if ((*vpp)->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5668
				if ((*vpp)->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5669
					status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5670
				else if ((*vpp)->v_type == VLNK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5671
					status = NFS4ERR_SYMLINK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5672
				else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5673
					status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5674
				VN_RELE(*vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5675
				return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5676
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5677
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5678
			return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5679
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5680
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5681
		/* Check for duplicate request */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5682
		ASSERT(mtime != 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5683
		va.va_mask = AT_MTIME;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5684
		error = VOP_GETATTR(*vpp, &va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5685
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5686
			/* We found the file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5687
			if (va.va_mtime.tv_sec != mtime->tv_sec ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5688
			    va.va_mtime.tv_nsec != mtime->tv_nsec) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5689
				/* but its not our creation */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5690
				VN_RELE(*vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5691
				return (NFS4ERR_EXIST);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5692
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5693
			*created = TRUE; /* retrans of create == created */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5694
			return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5695
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5696
		VN_RELE(*vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5697
		return (NFS4ERR_EXIST);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5698
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5699
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5700
	return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5701
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5702
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5703
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5704
check_open_access(uint32_t access,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5705
		struct compound_state *cs, struct svc_req *req)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5706
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5707
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5708
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5709
	bool_t readonly;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5710
	cred_t *cr = cs->cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5711
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5712
	/* For now we don't allow mandatory locking as per V2/V3 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5713
	if (cs->access == CS_ACCESS_DENIED || cs->mandlock) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5714
		return (NFS4ERR_ACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5715
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5716
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5717
	vp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5718
	ASSERT(cr != NULL && vp->v_type == VREG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5719
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5720
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5721
	 * If the file system is exported read only and we are trying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5722
	 * to open for write, then return NFS4ERR_ROFS
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5723
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5724
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5725
	readonly = rdonly4(cs->exi, cs->vp, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5726
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5727
	if ((access & OPEN4_SHARE_ACCESS_WRITE) && readonly)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5728
		return (NFS4ERR_ROFS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5729
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5730
	if (access & OPEN4_SHARE_ACCESS_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5731
		if ((VOP_ACCESS(vp, VREAD, 0, cr) != 0) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5732
		    (VOP_ACCESS(vp, VEXEC, 0, cr) != 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5733
			return (NFS4ERR_ACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5734
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5735
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5736
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5737
	if (access & OPEN4_SHARE_ACCESS_WRITE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5738
		error = VOP_ACCESS(vp, VWRITE, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5739
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5740
			return (NFS4ERR_ACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5741
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5742
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5743
	return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5744
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5745
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5746
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5747
rfs4_createfile(OPEN4args *args, struct svc_req *req, struct compound_state *cs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5748
		change_info4 *cinfo, bitmap4 *attrset, clientid4 clientid)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5749
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5750
	struct nfs4_svgetit_arg sarg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5751
	struct nfs4_ntov_table ntov;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5752
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5753
	bool_t ntov_table_init = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5754
	struct statvfs64 sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5755
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5756
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5757
	vattr_t bva, ava, iva, cva, *vap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5758
	vnode_t *dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5759
	timespec32_t *mtime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5760
	char *nm = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5761
	uint_t buflen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5762
	bool_t created;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5763
	bool_t setsize = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5764
	len_t reqsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5765
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5766
	bool_t trunc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5767
	caller_context_t ct;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5768
	component4 *component;
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5769
	bslabel_t *clabel;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5770
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5771
	sarg.sbp = &sb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5772
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5773
	dvp = cs->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5774
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5775
	/* Check if the file system is read only */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5776
	if (rdonly4(cs->exi, dvp, req))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5777
		return (NFS4ERR_ROFS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5778
1676
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5779
	/* check the label of including directory */
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5780
	if (is_system_labeled()) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5781
		ASSERT(req->rq_label != NULL);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5782
		clabel = req->rq_label;
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5783
		DTRACE_PROBE2(tx__rfs4__log__info__opremove__clabel, char *,
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5784
		    "got client label from request(1)",
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5785
		    struct svc_req *, req);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5786
		if (!blequal(&l_admin_low->tsl_label, clabel)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5787
			if (!do_rfs4_label_check(clabel, dvp, EQUALITY_CHECK)) {
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5788
				return (NFS4ERR_ACCESS);
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5789
			}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5790
		}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5791
	}
37f4a3e2bd99 PSARC/2002/762 Layered Trusted Solaris
jpk
parents: 1146
diff changeset
  5792
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5793
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5794
	 * Get the last component of path name in nm. cs will reference
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5795
	 * the including directory on success.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5796
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5797
	component = &args->open_claim4_u.file;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5798
	if (!utf8_dir_verify(component))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5799
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5800
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5801
	nm = utf8_to_fn(component, &buflen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5802
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5803
	if (nm == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5804
		return (NFS4ERR_RESOURCE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5805
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5806
	if (buflen > MAXNAMELEN) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5807
		kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5808
		return (NFS4ERR_NAMETOOLONG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5809
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5810
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5811
	bva.va_mask = AT_TYPE|AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5812
	error = VOP_GETATTR(dvp, &bva, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5813
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5814
		kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5815
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5816
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5817
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5818
	if (bva.va_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5819
		kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5820
		return (NFS4ERR_NOTDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5821
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5822
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5823
	NFS4_SET_FATTR4_CHANGE(cinfo->before, bva.va_ctime)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5824
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5825
	switch (args->mode) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5826
	case GUARDED4:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5827
		/*FALLTHROUGH*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5828
	case UNCHECKED4:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5829
		nfs4_ntov_table_init(&ntov);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5830
		ntov_table_init = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5831
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5832
		*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5833
		status = do_rfs4_set_attrs(attrset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5834
					&args->createhow4_u.createattrs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5835
					cs, &sarg, &ntov, NFS4ATTR_SETIT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5836
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5837
		if (status == NFS4_OK && (sarg.vap->va_mask & AT_TYPE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5838
		    sarg.vap->va_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5839
			if (sarg.vap->va_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5840
				status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5841
			else if (sarg.vap->va_type == VLNK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5842
				status = NFS4ERR_SYMLINK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5843
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5844
				status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5845
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5846
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5847
		if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5848
			kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5849
			nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5850
			*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5851
			return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5852
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5853
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5854
		vap = sarg.vap;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5855
		vap->va_type = VREG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5856
		vap->va_mask |= AT_TYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5857
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5858
		if ((vap->va_mask & AT_MODE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5859
			vap->va_mask |= AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5860
			vap->va_mode = (mode_t)0600;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5861
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5862
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5863
		if (vap->va_mask & AT_SIZE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5864
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5865
			/* Disallow create with a non-zero size */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5866
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5867
			if ((reqsize = sarg.vap->va_size) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5868
				kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5869
				nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5870
				*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5871
				return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5872
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5873
			setsize = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5874
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5875
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5876
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5877
	case EXCLUSIVE4:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5878
		/* prohibit EXCL create of named attributes */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5879
		if (dvp->v_flag & V_XATTRDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5880
			kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5881
			*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5882
			return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5883
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5884
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5885
		cva.va_mask = AT_TYPE | AT_MTIME | AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5886
		cva.va_type = VREG;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5887
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5888
		 * Ensure no time overflows. Assumes underlying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5889
		 * filesystem supports at least 32 bits.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5890
		 * Truncate nsec to usec resolution to allow valid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5891
		 * compares even if the underlying filesystem truncates.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5892
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5893
		mtime = (timespec32_t *)&args->createhow4_u.createverf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5894
		cva.va_mtime.tv_sec = mtime->tv_sec % TIME32_MAX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5895
		cva.va_mtime.tv_nsec = (mtime->tv_nsec / 1000) * 1000;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5896
		cva.va_mode = (mode_t)0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5897
		vap = &cva;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5898
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5899
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5900
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5901
	status = create_vnode(dvp, nm, vap, args->mode, mtime,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5902
						cs->cr, &vp, &created);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5903
	kmem_free(nm, buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5904
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5905
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5906
		if (ntov_table_init)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5907
			nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5908
		*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5909
		return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5910
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5911
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5912
	trunc = (setsize && !created);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5913
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5914
	if (args->mode != EXCLUSIVE4) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5915
		bitmap4 createmask = args->createhow4_u.createattrs.attrmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5917
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5918
		 * True verification that object was created with correct
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5919
		 * attrs is impossible.  The attrs could have been changed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5920
		 * immediately after object creation.  If attributes did
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5921
		 * not verify, the only recourse for the server is to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5922
		 * destroy the object.  Maybe if some attrs (like gid)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5923
		 * are set incorrectly, the object should be destroyed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5924
		 * however, seems bad as a default policy.  Do we really
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5925
		 * want to destroy an object over one of the times not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5926
		 * verifying correctly?  For these reasons, the server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5927
		 * currently sets bits in attrset for createattrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5928
		 * that were set; however, no verification is done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5929
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5930
		 * vmask_to_nmask accounts for vattr bits set on create
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5931
		 *	[do_rfs4_set_attrs() only sets resp bits for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5932
		 *	 non-vattr/vfs bits.]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5933
		 * Mask off any bits we set by default so as not to return
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5934
		 * more attrset bits than were requested in createattrs
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5935
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5936
		if (created) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5937
			nfs4_vmask_to_nmask(sarg.vap->va_mask, attrset);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5938
			*attrset &= createmask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5939
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5940
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5941
			 * We did not create the vnode (we tried but it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5942
			 * already existed).  In this case, the only createattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5943
			 * that the spec allows the server to set is size,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5944
			 * and even then, it can only be set if it is 0.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5945
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5946
			*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5947
			if (trunc)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5948
				*attrset = FATTR4_SIZE_MASK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5949
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5950
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5951
	if (ntov_table_init)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5952
		nfs4_ntov_table_free(&ntov, &sarg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5953
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5954
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5955
	 * Get the initial "after" sequence number, if it fails,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5956
	 * set to zero, time to before.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5957
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5958
	iva.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5959
	if (VOP_GETATTR(dvp, &iva, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5960
		iva.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5961
		iva.va_ctime = bva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5962
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5963
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5964
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5965
	 * create_vnode attempts to create the file exclusive,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5966
	 * if it already exists the VOP_CREATE will fail and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5967
	 * may not increase va_seq. It is atomic if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5968
	 * we haven't changed the directory, but if it has changed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5969
	 * we don't know what changed it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5970
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5971
	if (!created) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5972
		if (bva.va_seq && iva.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5973
			bva.va_seq == iva.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5974
			cinfo->atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5975
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5976
			cinfo->atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5977
		NFS4_SET_FATTR4_CHANGE(cinfo->after, iva.va_ctime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5978
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5979
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5980
		 * The entry was created, we need to sync the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5981
		 * directory metadata.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5982
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5983
		(void) VOP_FSYNC(dvp, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5984
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5985
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5986
		 * Get "after" change value, if it fails, simply return the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5987
		 * before value.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5988
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5989
		ava.va_mask = AT_CTIME|AT_SEQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5990
		if (VOP_GETATTR(dvp, &ava, 0, cs->cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5991
			ava.va_ctime = bva.va_ctime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5992
			ava.va_seq = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5993
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5994
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5995
		NFS4_SET_FATTR4_CHANGE(cinfo->after, ava.va_ctime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5996
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5997
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5998
		 * The cinfo->atomic = TRUE only if we have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5999
		 * non-zero va_seq's, and it has incremented by exactly one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6000
		 * during the create_vnode and it didn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6001
		 * change during the VOP_FSYNC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6002
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6003
		if (bva.va_seq && iva.va_seq && ava.va_seq &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6004
				iva.va_seq == (bva.va_seq + 1) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6005
				iva.va_seq == ava.va_seq)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6006
			cinfo->atomic = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6007
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6008
			cinfo->atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6009
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6010
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6011
	/* Check for mandatory locking and that the size gets set. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6012
	cva.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6013
	if (setsize)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6014
		cva.va_mask |= AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6015
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6016
	/* Assume the worst */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6017
	cs->mandlock = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6018
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6019
	if (VOP_GETATTR(vp, &cva, 0, cs->cr) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6020
		cs->mandlock = MANDLOCK(cs->vp, cva.va_mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6021
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6022
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6023
		 * Truncate the file if necessary; this would be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6024
		 * the case for create over an existing file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6025
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6026
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6027
		if (trunc) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6028
			int in_crit = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6029
			rfs4_file_t *fp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6030
			bool_t create = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6031
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6032
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6033
			 * We are writing over an existing file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6034
			 * Check to see if we need to recall a delegation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6035
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6036
			rfs4_hold_deleg_policy();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6037
			if ((fp = rfs4_findfile(vp, NULL, &create)) != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6038
				if (rfs4_check_delegated_byfp(FWRITE, fp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6039
					(reqsize == 0), FALSE, FALSE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6040
							&clientid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6041
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6042
					rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6043
					rfs4_rele_deleg_policy();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6044
					VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6045
					*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6046
					return (NFS4ERR_DELAY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6047
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6048
				rfs4_file_rele(fp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6049
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6050
			rfs4_rele_deleg_policy();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6051
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6052
			if (nbl_need_check(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6053
				in_crit = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6054
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6055
				ASSERT(reqsize == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6056
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6057
				nbl_start_crit(vp, RW_READER);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6058
				if (nbl_conflict(vp, NBL_WRITE, 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6059
						cva.va_size, 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6060
					in_crit = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6061
					nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6062
					VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6063
					*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6064
					return (NFS4ERR_ACCESS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6065
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6066
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6067
			ct.cc_sysid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6068
			ct.cc_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6069
			ct.cc_caller_id = nfs4_srv_caller_id;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6070
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6071
			cva.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6072
			cva.va_size = reqsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6073
			(void) VOP_SETATTR(vp, &cva, 0, cs->cr, &ct);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6074
			if (in_crit)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6075
				nbl_end_crit(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6076
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6077
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6078
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6079
	error = makefh4(&cs->fh, vp, cs->exi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6080
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6081
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6082
	 * Force modified data and metadata out to stable storage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6083
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6084
	(void) VOP_FSYNC(vp, FNODSYNC, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6085
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6086
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6087
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6088
		*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6089
		return (puterrno4(error));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6090
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6091
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6092
	/* if parent dir is attrdir, set namedattr fh flag */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6093
	if (dvp->v_flag & V_XATTRDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6094
		set_fh4_flag(&cs->fh, FH4_NAMEDATTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6095
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6096
	if (cs->vp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6097
		VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6098
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6099
	cs->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6100
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6101
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6102
	 * if we did not create the file, we will need to check
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6103
	 * the access bits on the file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6104
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6105
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6106
	if (!created) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6107
		if (setsize)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6108
			args->share_access |= OPEN4_SHARE_ACCESS_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6109
		status = check_open_access(args->share_access, cs, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6110
		if (status != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6111
			*attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6112
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6113
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6114
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6115
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6116
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6117
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6118
rfs4_do_open(struct compound_state *cs, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6119
		rfs4_openowner_t *oo, delegreq_t deleg,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6120
		uint32_t access, uint32_t deny,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6121
		OPEN4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6122
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6123
	/* XXX Currently not using req  */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6124
	rfs4_state_t *state;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6125
	rfs4_file_t *file;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6126
	bool_t screate = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6127
	bool_t fcreate = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6128
	uint32_t amodes;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6129
	uint32_t dmodes;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6130
	rfs4_deleg_state_t *dsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6131
	struct shrlock shr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6132
	struct shr_locowner shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6133
	sysid_t sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6134
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6135
	int fflags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6136
	int recall = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6137
	int err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6138
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6139
	/* get the file struct and hold a lock on it during initial open */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6140
	file = rfs4_findfile_withlock(cs->vp, &cs->fh, &fcreate);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6141
	if (file == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6142
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6143
			(CE_NOTE, "rfs4_do_open: can't find file"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6144
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6145
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6146
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6147
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6148
	state = rfs4_findstate_by_owner_file(oo, file, &screate);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6149
	if (state == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6150
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6151
			(CE_NOTE, "rfs4_do_open: can't find state"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6152
		resp->status = NFS4ERR_RESOURCE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6153
		/* No need to keep any reference */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6154
		rfs4_file_rele_withunlock(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6155
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6156
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6157
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6158
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6159
	 * Check for conflicts in deny and access before checking for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6160
	 * conflicts in delegation.  We don't want to recall a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6161
	 * delegation based on an open that will eventually fail based
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6162
	 * on shares modes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6163
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6164
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6165
	shr.s_access = (short)access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6166
	shr.s_deny = (short)deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6167
	shr.s_pid = rfs4_dbe_getid(oo->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6168
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6169
	if ((status = rfs4_client_sysid(oo->client, &sysid)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6170
		resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6171
		rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6172
		/* Not a fully formed open; "close" it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6173
		if (screate == TRUE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6174
			rfs4_state_close(state, FALSE, FALSE, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6175
		rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6176
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6177
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6178
	shr.s_sysid = sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6179
	shr_loco.sl_pid = shr.s_pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6180
	shr_loco.sl_id = shr.s_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6181
	shr.s_owner = (caddr_t)&shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6182
	shr.s_own_len = sizeof (shr_loco);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6183
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6184
	fflags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6185
	if (access & OPEN4_SHARE_ACCESS_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6186
		fflags |= FREAD;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6187
	if (access & OPEN4_SHARE_ACCESS_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6188
		fflags |= FWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6189
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6190
	if ((err = vop_shrlock(cs->vp, F_SHARE, &shr, fflags)) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6191
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6192
		resp->status = err == EAGAIN ?
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6193
			NFS4ERR_SHARE_DENIED : puterrno4(err);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6194
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6195
		rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6196
		/* Not a fully formed open; "close" it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6197
		if (screate == TRUE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6198
			rfs4_state_close(state, FALSE, FALSE, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6199
		rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6200
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6201
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6202
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6203
	rfs4_dbe_lock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6204
	rfs4_dbe_lock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6205
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6206
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6207
	 * Calculate the new deny and access mode that this open is adding to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6208
	 * the file for this open owner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6209
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6210
	dmodes = (deny & ~state->share_deny);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6211
	amodes = (access & ~state->share_access);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6212
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6213
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6214
	 * Check to see if this file is delegated and if so, if a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6215
	 * recall needs to be done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6216
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6217
	if (rfs4_check_recall(state, access)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6218
		rfs4_dbe_unlock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6219
		rfs4_dbe_unlock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6220
		rfs4_recall_deleg(file, FALSE, state->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6221
		delay(NFS4_DELEGATION_CONFLICT_DELAY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6222
		rfs4_dbe_lock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6223
		rfs4_dbe_lock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6224
		/* Let's see if the delegation was returned */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6225
		if (rfs4_check_recall(state, access)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6226
			rfs4_dbe_unlock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6227
			rfs4_dbe_unlock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6228
			rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6229
			rfs4_update_lease(state->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6230
			/* recalculate flags to match what was added */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6231
			fflags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6232
			if (amodes & OPEN4_SHARE_ACCESS_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6233
				fflags |= FREAD;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6234
			if (amodes & OPEN4_SHARE_ACCESS_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6235
				fflags |= FWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6236
			(void) vop_shrlock(cs->vp, F_UNSHARE, &shr, fflags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6237
			/* Not a fully formed open; "close" it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6238
			if (screate == TRUE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6239
				rfs4_state_close(state, FALSE, FALSE, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6240
			rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6241
			resp->status = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6242
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6243
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6244
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6245
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6246
	if (dmodes & OPEN4_SHARE_DENY_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6247
		file->deny_read++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6248
	if (dmodes & OPEN4_SHARE_DENY_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6249
		file->deny_write++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6250
	file->share_deny |= deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6251
	state->share_deny |= deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6252
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6253
	if (amodes & OPEN4_SHARE_ACCESS_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6254
		file->access_read++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6255
	if (amodes & OPEN4_SHARE_ACCESS_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6256
		file->access_write++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6257
	file->share_access |= access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6258
	state->share_access |= access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6259
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6260
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6261
	 * Check for delegation here. if the deleg argument is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6262
	 * DELEG_ANY, then this is a reclaim from a client and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6263
	 * we must honor the delegation requested. If necessary we can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6264
	 * set the recall flag.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6265
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6266
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6267
	dsp = rfs4_grant_delegation(deleg, state, &recall);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6268
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6269
	cs->deleg = (file->dinfo->dtype == OPEN_DELEGATE_WRITE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6270
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6271
	next_stateid(&state->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6272
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6273
	resp->stateid = state->stateid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6274
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6275
	rfs4_dbe_unlock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6276
	rfs4_dbe_unlock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6277
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6278
	if (dsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6279
		rfs4_set_deleg_response(dsp, &resp->delegation, NULL, recall);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6280
		rfs4_deleg_state_rele(dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6281
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6282
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6283
	rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6284
	rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6285
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6286
	resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6287
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6288
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6289
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6290
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6291
rfs4_do_opennull(struct compound_state *cs, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6292
		OPEN4args *args, rfs4_openowner_t *oo, OPEN4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6293
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6294
	change_info4 *cinfo = &resp->cinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6295
	bitmap4 *attrset = &resp->attrset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6296
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6297
	if (args->opentype == OPEN4_NOCREATE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6298
		resp->status = rfs4_lookupfile(&args->open_claim4_u.file,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6299
					req, cs, args->share_access, cinfo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6300
	else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6301
		/* inhibit delegation grants during exclusive create */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6302
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6303
		if (args->mode == EXCLUSIVE4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6304
			rfs4_disable_delegation();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6306
		resp->status = rfs4_createfile(args, req, cs, cinfo, attrset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6307
					oo->client->clientid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6308
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6309
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6310
	if (resp->status == NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6311
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6312
		/* cs->vp cs->fh now reference the desired file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6313
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6314
		rfs4_do_open(cs, req, oo, DELEG_ANY, args->share_access,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6315
						args->share_deny, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6316
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6317
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6318
		 * If rfs4_createfile set attrset, we must
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6319
		 * clear this attrset before the response is copied.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6320
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6321
		if (resp->status != NFS4_OK && resp->attrset) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6322
			resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6323
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6324
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6325
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6326
		*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6327
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6328
	if (args->mode == EXCLUSIVE4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6329
		rfs4_enable_delegation();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6330
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6331
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6332
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6333
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6334
rfs4_do_openprev(struct compound_state *cs, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6335
		OPEN4args *args, rfs4_openowner_t *oo, OPEN4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6336
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6337
	change_info4 *cinfo = &resp->cinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6338
	vattr_t va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6339
	vtype_t v_type = cs->vp->v_type;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6340
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6341
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6342
	/* Verify that we have a regular file */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6343
	if (v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6344
		if (v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6345
			resp->status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6346
		else if (v_type == VLNK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6347
			resp->status = NFS4ERR_SYMLINK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6348
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6349
			resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6350
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6351
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6352
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6353
	va.va_mask = AT_MODE|AT_UID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6354
	error = VOP_GETATTR(cs->vp, &va, 0, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6355
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6356
		resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6357
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6358
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6360
	cs->mandlock = MANDLOCK(cs->vp, va.va_mode);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6361
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6362
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6363
	 * Check if we have access to the file, Note the the file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6364
	 * could have originally been open UNCHECKED or GUARDED
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6365
	 * with mode bits that will now fail, but there is nothing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6366
	 * we can really do about that except in the case that the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6367
	 * owner of the file is the one requesting the open.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6368
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6369
	if (crgetuid(cs->cr) != va.va_uid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6370
		resp->status = check_open_access(args->share_access, cs, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6371
		if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6372
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6373
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6374
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6375
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6376
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6377
	 * cinfo on a CLAIM_PREVIOUS is undefined, initialize to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6378
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6379
	cinfo->before = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6380
	cinfo->after = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6381
	cinfo->atomic = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6382
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6383
	rfs4_do_open(cs, req, oo,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6384
		NFS4_DELEG4TYPE2REQTYPE(args->open_claim4_u.delegate_type),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6385
		args->share_access, args->share_deny, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6386
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6387
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6388
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6389
rfs4_do_opendelcur(struct compound_state *cs, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6390
		OPEN4args *args, rfs4_openowner_t *oo, OPEN4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6391
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6392
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6393
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6394
	stateid4 stateid =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6395
			args->open_claim4_u.delegate_cur_info.delegate_stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6396
	rfs4_deleg_state_t *dsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6397
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6398
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6399
	 * Find the state info from the stateid and confirm that the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6400
	 * file is delegated.  If the state openowner is the same as
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6401
	 * the supplied openowner we're done. If not, get the file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6402
	 * info from the found state info. Use that file info to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6403
	 * create the state for this lock owner. Note solaris doen't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6404
	 * really need the pathname to find the file. We may want to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6405
	 * lookup the pathname and make sure that the vp exist and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6406
	 * matches the vp in the file structure. However it is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6407
	 * possible that the pathname nolonger exists (local process
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6408
	 * unlinks the file), so this may not be that useful.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6409
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6410
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6411
	status = rfs4_get_deleg_state(&stateid, &dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6412
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6413
		resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6414
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6415
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6417
	ASSERT(dsp->finfo->dinfo->dtype != OPEN_DELEGATE_NONE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6418
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6419
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6420
	 * New lock owner, create state. Since this was probably called
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6421
	 * in response to a CB_RECALL we set deleg to DELEG_NONE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6422
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6423
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6424
	ASSERT(cs->vp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6425
	VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6426
	VN_HOLD(dsp->finfo->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6427
	cs->vp = dsp->finfo->vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6428
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6429
	if (error = makefh4(&cs->fh, cs->vp, cs->exi)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6430
		rfs4_deleg_state_rele(dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6431
		*cs->statusp = resp->status = puterrno4(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6432
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6433
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6434
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6435
	/* Mark progress for delegation returns */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6436
	dsp->finfo->dinfo->time_lastwrite = gethrestime_sec();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6437
	rfs4_deleg_state_rele(dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6438
	rfs4_do_open(cs, req, oo, DELEG_NONE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6439
				args->share_access, args->share_deny, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6440
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6441
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6442
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6443
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6444
rfs4_do_opendelprev(struct compound_state *cs, struct svc_req *req,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6445
			OPEN4args *args, rfs4_openowner_t *oo, OPEN4res *resp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6446
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6447
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6448
	 * Lookup the pathname, it must already exist since this file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6449
	 * was delegated.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6450
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6451
	 * Find the file and state info for this vp and open owner pair.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6452
	 *	check that they are in fact delegated.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6453
	 *	check that the state access and deny modes are the same.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6454
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6455
	 * Return the delgation possibly seting the recall flag.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6456
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6457
	rfs4_file_t *file;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6458
	rfs4_state_t *state;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6459
	bool_t create = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6460
	bool_t dcreate = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6461
	rfs4_deleg_state_t *dsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6462
	nfsace4 *ace;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6463
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6464
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6465
	/* Note we ignore oflags */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6466
	resp->status = rfs4_lookupfile(&args->open_claim4_u.file_delegate_prev,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6467
				req, cs, args->share_access, &resp->cinfo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6468
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6469
	if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6470
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6471
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6472
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6473
	/* get the file struct and hold a lock on it during initial open */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6474
	file = rfs4_findfile_withlock(cs->vp, NULL, &create);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6475
	if (file == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6476
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6477
			(CE_NOTE, "rfs4_do_opendelprev: can't find file"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6478
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6479
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6480
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6481
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6482
	state = rfs4_findstate_by_owner_file(oo, file, &create);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6483
	if (state == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6484
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6485
			(CE_NOTE, "rfs4_do_opendelprev: can't find state"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6486
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6487
		rfs4_file_rele_withunlock(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6488
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6489
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6490
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6491
	rfs4_dbe_lock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6492
	rfs4_dbe_lock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6493
	if (args->share_access != state->share_access ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6494
			args->share_deny != state->share_deny ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6495
			state->finfo->dinfo->dtype == OPEN_DELEGATE_NONE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6496
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6497
			(CE_NOTE, "rfs4_do_opendelprev: state mixup"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6498
		rfs4_dbe_unlock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6499
		rfs4_dbe_unlock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6500
		rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6501
		rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6502
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6503
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6504
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6505
	rfs4_dbe_unlock(file->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6506
	rfs4_dbe_unlock(state->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6507
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6508
	dsp = rfs4_finddeleg(state, &dcreate);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6509
	if (dsp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6510
		rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6511
		rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6512
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6513
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6514
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6515
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6516
	next_stateid(&state->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6517
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6518
	resp->stateid = state->stateid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6519
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6520
	resp->delegation.delegation_type = dsp->dtype;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6521
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6522
	if (dsp->dtype == OPEN_DELEGATE_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6523
		open_read_delegation4 *rv =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6524
			&resp->delegation.open_delegation4_u.read;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6525
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6526
		rv->stateid = dsp->delegid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6527
		rv->recall = FALSE; /* no policy in place to set to TRUE */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6528
		ace = &rv->permissions;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6529
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6530
		open_write_delegation4 *rv =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6531
			&resp->delegation.open_delegation4_u.write;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6532
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6533
		rv->stateid = dsp->delegid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6534
		rv->recall = FALSE;  /* no policy in place to set to TRUE */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6535
		ace = &rv->permissions;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6536
		rv->space_limit.limitby = NFS_LIMIT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6537
		rv->space_limit.nfs_space_limit4_u.filesize = UINT64_MAX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6538
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6539
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6540
	/* XXX For now */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6541
	ace->type = ACE4_ACCESS_ALLOWED_ACE_TYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6542
	ace->flag = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6543
	ace->access_mask = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6544
	ace->who.utf8string_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6545
	ace->who.utf8string_val = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6546
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6547
	rfs4_deleg_state_rele(dsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6548
	rfs4_state_rele(state);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6549
	rfs4_file_rele(file);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6550
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6551
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6552
typedef enum {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6553
	NFS4_CHKSEQ_OKAY = 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6554
	NFS4_CHKSEQ_REPLAY = 1,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6555
	NFS4_CHKSEQ_BAD = 2
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6556
} rfs4_chkseq_t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6557
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6558
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6559
 * Generic function for sequence number checks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6560
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6561
static rfs4_chkseq_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6562
rfs4_check_seqid(seqid4 seqid, nfs_resop4 *lastop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6563
		seqid4 rqst_seq, nfs_resop4 *resop, bool_t copyres)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6564
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6565
	/* Same sequence ids and matching operations? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6566
	if (seqid == rqst_seq && resop->resop == lastop->resop) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6567
		if (copyres == TRUE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6568
			rfs4_free_reply(resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6569
			rfs4_copy_reply(resop, lastop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6570
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6571
		NFS4_DEBUG(rfs4_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6572
			"Replayed SEQID %d\n", seqid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6573
		return (NFS4_CHKSEQ_REPLAY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6574
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6575
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6576
	/* If the incoming sequence is not the next expected then it is bad */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6577
	if (rqst_seq != seqid + 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6578
		if (rqst_seq == seqid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6579
			NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6580
				(CE_NOTE, "BAD SEQID: Replayed sequence id "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6581
				"but last op was %d current op is %d\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6582
				lastop->resop, resop->resop));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6583
			return (NFS4_CHKSEQ_BAD);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6584
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6585
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6586
			(CE_NOTE, "BAD SEQID: got %u expecting %u\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6587
				rqst_seq, seqid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6588
		return (NFS4_CHKSEQ_BAD);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6589
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6590
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6591
	/* Everything okay -- next expected */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6592
	return (NFS4_CHKSEQ_OKAY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6593
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6594
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6595
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6596
static rfs4_chkseq_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6597
rfs4_check_open_seqid(seqid4 seqid, rfs4_openowner_t *op, nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6598
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6599
	rfs4_chkseq_t rc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6600
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6601
	rfs4_dbe_lock(op->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6602
	rc = rfs4_check_seqid(op->open_seqid, op->reply, seqid, resop, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6603
	rfs4_dbe_unlock(op->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6604
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6605
	if (rc == NFS4_CHKSEQ_OKAY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6606
		rfs4_update_lease(op->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6607
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6608
	return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6609
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6610
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6611
static rfs4_chkseq_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6612
rfs4_check_olo_seqid(seqid4 olo_seqid, rfs4_openowner_t *op,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6613
	nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6614
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6615
	rfs4_chkseq_t rc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6616
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6617
	rfs4_dbe_lock(op->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6618
	rc = rfs4_check_seqid(op->open_seqid, op->reply,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6619
		olo_seqid, resop, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6620
	rfs4_dbe_unlock(op->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6621
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6622
	return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6623
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6624
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6625
static rfs4_chkseq_t
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6626
rfs4_check_lock_seqid(seqid4 seqid, rfs4_lo_state_t *lp, nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6627
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6628
	rfs4_chkseq_t rc = NFS4_CHKSEQ_OKAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6629
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6630
	rfs4_dbe_lock(lp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6631
	if (!lp->skip_seqid_check)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6632
		rc = rfs4_check_seqid(lp->seqid, lp->reply,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6633
			seqid, resop, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6634
	rfs4_dbe_unlock(lp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6635
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6636
	return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6637
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6638
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6639
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6640
rfs4_op_open(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6641
	    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6642
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6643
	OPEN4args *args = &argop->nfs_argop4_u.opopen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6644
	OPEN4res *resp = &resop->nfs_resop4_u.opopen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6645
	open_owner4 *owner = &args->owner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6646
	open_claim_type4 claim = args->claim;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6647
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6648
	rfs4_openowner_t *oo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6649
	bool_t create;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6650
	bool_t replay = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6651
	int can_reclaim;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6652
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6653
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6654
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6655
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6656
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6657
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6658
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6659
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6660
	 * Need to check clientid and lease expiration first based on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6661
	 * error ordering and incrementing sequence id.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6662
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6663
	cp = rfs4_findclient_by_id(owner->clientid, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6664
	if (cp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6665
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6666
			rfs4_check_clientid(&owner->clientid, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6667
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6668
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6669
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6670
	if (rfs4_lease_expired(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6671
		rfs4_client_close(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6672
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6673
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6674
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6675
	can_reclaim = cp->can_reclaim;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6676
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6677
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6678
	 * Find the open_owner for use from this point forward.  Take
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6679
	 * care in updating the sequence id based on the type of error
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6680
	 * being returned.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6681
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6682
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6683
	create = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6684
	oo = rfs4_findopenowner(owner, &create, args->seqid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6685
	if (oo == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6686
		*cs->statusp = resp->status = NFS4ERR_STALE_CLIENTID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6687
		rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6688
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6689
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6690
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6691
	/* Hold off access to the sequence space while the open is done */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6692
	rfs4_sw_enter(&oo->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6693
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6694
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6695
	 * If the open_owner existed before at the server, then check
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6696
	 * the sequence id.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6697
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6698
	if (!create && !oo->postpone_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6699
		switch (rfs4_check_open_seqid(args->seqid, oo, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6700
		case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6701
			if ((args->seqid > oo->open_seqid) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6702
				oo->need_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6703
				rfs4_free_opens(oo, TRUE, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6704
				rfs4_sw_exit(&oo->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6705
				rfs4_openowner_rele(oo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6706
				goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6707
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6708
			resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6709
			goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6710
		case NFS4_CHKSEQ_REPLAY: /* replay of previous request */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6711
			replay = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6712
			goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6713
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6714
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6715
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6716
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6717
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6718
		 * Sequence was ok and open owner exists
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6719
		 * check to see if we have yet to see an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6720
		 * open_confirm.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6721
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6722
		if (oo->need_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6723
			rfs4_free_opens(oo, TRUE, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6724
			rfs4_sw_exit(&oo->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6725
			rfs4_openowner_rele(oo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6726
			goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6727
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6728
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6729
	/* Grace only applies to regular-type OPENs */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6730
	if (rfs4_clnt_in_grace(cp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6731
	    (claim == CLAIM_NULL || claim == CLAIM_DELEGATE_CUR)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6732
		*cs->statusp = resp->status = NFS4ERR_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6733
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6734
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6735
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6736
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6737
	 * If previous state at the server existed then can_reclaim
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6738
	 * will be set. If not reply NFS4ERR_NO_GRACE to the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6739
	 * client.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6740
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6741
	if (rfs4_clnt_in_grace(cp) && claim == CLAIM_PREVIOUS && !can_reclaim) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6742
		*cs->statusp = resp->status = NFS4ERR_NO_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6743
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6744
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6745
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6746
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6747
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6748
	 * Reject the open if the client has missed the grace period
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6749
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6750
	if (!rfs4_clnt_in_grace(cp) && claim == CLAIM_PREVIOUS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6751
		*cs->statusp = resp->status = NFS4ERR_NO_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6752
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6753
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6754
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6755
	/* Couple of up-front bookkeeping items */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6756
	if (oo->need_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6757
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6758
		 * If this is a reclaim OPEN then we should not ask
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6759
		 * for a confirmation of the open_owner per the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6760
		 * protocol specification.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6761
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6762
		if (claim == CLAIM_PREVIOUS)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6763
			oo->need_confirm = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6764
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6765
			resp->rflags |= OPEN4_RESULT_CONFIRM;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6766
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6767
	resp->rflags |= OPEN4_RESULT_LOCKTYPE_POSIX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6768
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6769
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6770
	 * If there is an unshared filesystem mounted on this vnode,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6771
	 * do not allow to open/create in this directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6772
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6773
	if (vn_ismntpt(cs->vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6774
		*cs->statusp = resp->status = NFS4ERR_ACCESS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6775
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6776
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6777
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6778
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6779
	 * access must READ, WRITE, or BOTH.  No access is invalid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6780
	 * deny can be READ, WRITE, BOTH, or NONE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6781
	 * bits not defined for access/deny are invalid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6782
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6783
	if (! (args->share_access & OPEN4_SHARE_ACCESS_BOTH) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6784
	    (args->share_access & ~OPEN4_SHARE_ACCESS_BOTH) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6785
	    (args->share_deny & ~OPEN4_SHARE_DENY_BOTH)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6786
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6787
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6788
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6789
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6790
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6791
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6792
	 * make sure attrset is zero before response is built.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6793
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6794
	resp->attrset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6795
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6796
	switch (claim) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6797
	case CLAIM_NULL:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6798
		rfs4_do_opennull(cs, req, args, oo, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6799
	    break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6800
	case CLAIM_PREVIOUS:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6801
		rfs4_do_openprev(cs, req, args, oo, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6802
	    break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6803
	case CLAIM_DELEGATE_CUR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6804
		rfs4_do_opendelcur(cs, req, args, oo, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6805
	    break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6806
	case CLAIM_DELEGATE_PREV:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6807
		rfs4_do_opendelprev(cs, req, args, oo, resp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6808
	    break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6809
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6810
		resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6811
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6812
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6813
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6814
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6815
	rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6816
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6817
	/* Catch sequence id handling here to make it a little easier */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6818
	switch (resp->status) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6819
	case NFS4ERR_BADXDR:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6820
	case NFS4ERR_BAD_SEQID:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6821
	case NFS4ERR_BAD_STATEID:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6822
	case NFS4ERR_NOFILEHANDLE:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6823
	case NFS4ERR_RESOURCE:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6824
	case NFS4ERR_STALE_CLIENTID:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6825
	case NFS4ERR_STALE_STATEID:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6826
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6827
		 * The protocol states that if any of these errors are
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6828
		 * being returned, the sequence id should not be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6829
		 * incremented.  Any other return requires an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6830
		 * increment.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6831
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6832
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6833
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6834
		/* Always update the lease in this case */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6835
		rfs4_update_lease(oo->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6836
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6837
		/* Regular response - copy the result */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6838
		if (!replay)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6839
			rfs4_update_open_resp(oo, resop, &cs->fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6840
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6841
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6842
		 * REPLAY case: Only if the previous response was OK
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6843
		 * do we copy the filehandle.  If not OK, no
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6844
		 * filehandle to copy.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6845
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6846
		if (replay == TRUE &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6847
		    resp->status == NFS4_OK &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6848
		    oo->reply_fh.nfs_fh4_val) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6849
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6850
			 * If this is a replay, we must restore the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6851
			 * current filehandle/vp to that of what was
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6852
			 * returned originally.  Try our best to do
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6853
			 * it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6854
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6855
			nfs_fh4_fmt_t *fh_fmtp =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6856
				(nfs_fh4_fmt_t *)oo->reply_fh.nfs_fh4_val;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6857
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6858
			cs->exi = checkexport4(&fh_fmtp->fh4_fsid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6859
				(fid_t *)&fh_fmtp->fh4_xlen, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6860
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6861
			if (cs->exi == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6862
				resp->status = NFS4ERR_STALE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6863
				goto finish;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6864
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6865
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6866
			VN_RELE(cs->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6867
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6868
			cs->vp = nfs4_fhtovp(&oo->reply_fh, cs->exi,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6869
				&resp->status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6870
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6871
			if (cs->vp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6872
				goto finish;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6873
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6874
			nfs_fh4_copy(&oo->reply_fh, &cs->fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6875
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6876
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6877
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6878
		 * If this was a replay, no need to update the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6879
		 * sequence id. If the open_owner was not created on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6880
		 * this pass, then update.  The first use of an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6881
		 * open_owner will not bump the sequence id.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6882
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6883
		if (replay == FALSE && !create)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6884
			rfs4_update_open_sequence(oo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6885
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6886
		 * If the client is receiving an error and the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6887
		 * open_owner needs to be confirmed, there is no way
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6888
		 * to notify the client of this fact ignoring the fact
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6889
		 * that the server has no method of returning a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6890
		 * stateid to confirm.  Therefore, the server needs to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6891
		 * mark this open_owner in a way as to avoid the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6892
		 * sequence id checking the next time the client uses
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6893
		 * this open_owner.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6894
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6895
		if (resp->status != NFS4_OK && oo->need_confirm)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6896
			oo->postpone_confirm = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6897
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6898
		 * If OK response then clear the postpone flag and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6899
		 * reset the sequence id to keep in sync with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6900
		 * client.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6901
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6902
		if (resp->status == NFS4_OK && oo->postpone_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6903
			oo->postpone_confirm = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6904
			oo->open_seqid = args->seqid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6905
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6906
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6907
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6908
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6909
finish:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6910
	*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6911
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6912
	rfs4_sw_exit(&oo->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6913
	rfs4_openowner_rele(oo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6914
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6915
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6916
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6917
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6918
rfs4_op_open_confirm(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6919
		    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6920
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6921
	OPEN_CONFIRM4args *args = &argop->nfs_argop4_u.opopen_confirm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6922
	OPEN_CONFIRM4res *resp = &resop->nfs_resop4_u.opopen_confirm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6923
	rfs4_state_t *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6924
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6925
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6926
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6927
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6928
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6929
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6930
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6931
	status = rfs4_get_state(&args->open_stateid, &sp, RFS4_DBS_VALID);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6932
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6933
		*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6934
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6935
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6936
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6937
	/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6938
	if (cs->vp != sp->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6939
		rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6940
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6941
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6942
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6943
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6944
	/* hold off other access to open_owner while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6945
	rfs4_sw_enter(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6946
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6947
	switch (rfs4_check_stateid_seqid(sp, &args->open_stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6948
	case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6949
		if (rfs4_check_open_seqid(args->seqid, sp->owner,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6950
			resop) != 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6951
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6952
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6953
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6954
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6955
		 * If it is the appropriate stateid and determined to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6956
		 * be "OKAY" then this means that the stateid does not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6957
		 * need to be confirmed and the client is in error for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6958
		 * sending an OPEN_CONFIRM.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6959
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6960
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6961
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6962
	case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6963
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6964
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6965
	case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6966
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6967
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6968
	case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6969
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6970
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6971
	case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6972
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6973
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6974
	case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6975
		switch (rfs4_check_open_seqid(args->seqid, sp->owner, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6976
		case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6977
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6978
			 * This is replayed stateid; if seqid matches
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6979
			 * next expected, then client is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6980
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6981
			/* fall through */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6982
		case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6983
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6984
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6985
		case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6986
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6987
			 * Note this case is the duplicate case so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6988
			 * resp->status is already set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6989
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6990
			*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6991
			rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6992
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6993
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6994
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6995
	case NFS4_CHECK_STATEID_UNCONFIRMED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6996
		if (rfs4_check_open_seqid(args->seqid, sp->owner,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6997
			resop) != NFS4_CHKSEQ_OKAY) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6998
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  6999
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7000
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7001
		*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7002
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7003
		next_stateid(&sp->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7004
		resp->open_stateid = sp->stateid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7005
		sp->owner->need_confirm = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7006
		rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7007
		rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7008
		rfs4_update_open_resp(sp->owner, resop, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7009
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7010
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7011
		ASSERT(FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7012
		*cs->statusp = resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7013
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7014
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7015
	rfs4_sw_exit(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7016
	rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7017
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7018
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7019
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7020
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7021
rfs4_op_open_downgrade(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7022
		    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7023
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7024
	OPEN_DOWNGRADE4args *args = &argop->nfs_argop4_u.opopen_downgrade;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7025
	OPEN_DOWNGRADE4res *resp = &resop->nfs_resop4_u.opopen_downgrade;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7026
	uint32_t access = args->share_access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7027
	uint32_t deny = args->share_deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7028
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7029
	rfs4_state_t *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7030
	rfs4_file_t *fp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7031
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7032
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7033
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7034
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7035
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7036
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7037
	status = rfs4_get_state(&args->open_stateid, &sp, RFS4_DBS_VALID);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7038
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7039
		*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7040
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7041
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7042
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7043
	/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7044
	if (cs->vp != sp->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7045
		rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7046
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7047
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7048
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7049
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7050
	/* hold off other access to open_owner while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7051
	rfs4_sw_enter(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7052
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7053
	switch (rfs4_check_stateid_seqid(sp, &args->open_stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7054
	case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7055
		if (rfs4_check_open_seqid(args->seqid, sp->owner,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7056
			resop) != NFS4_CHKSEQ_OKAY) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7057
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7058
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7059
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7060
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7061
	case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7062
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7063
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7064
	case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7065
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7066
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7067
	case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7068
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7069
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7070
	case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7071
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7072
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7073
	case NFS4_CHECK_STATEID_UNCONFIRMED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7074
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7075
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7076
	case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7077
		/* Check the sequence id for the open owner */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7078
		switch (rfs4_check_open_seqid(args->seqid, sp->owner, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7079
		case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7080
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7081
			 * This is replayed stateid; if seqid matches
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7082
			 * next expected, then client is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7083
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7084
			/* fall through */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7085
		case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7086
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7087
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7088
		case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7089
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7090
			 * Note this case is the duplicate case so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7091
			 * resp->status is already set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7092
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7093
			*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7094
			rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7095
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7096
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7097
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7098
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7099
		ASSERT(FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7100
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7101
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7102
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7103
	rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7104
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7105
	 * Check that the new access modes and deny modes are valid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7106
	 * Check that no invalid bits are set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7107
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7108
	if ((access & ~(OPEN4_SHARE_ACCESS_READ | OPEN4_SHARE_ACCESS_WRITE)) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7109
	    (deny & ~(OPEN4_SHARE_DENY_READ | OPEN4_SHARE_DENY_READ))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7110
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7111
		rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7112
		rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7113
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7114
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7115
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7116
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7117
	 * The new modes must be a subset of the current modes and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7118
	 * the access must specify at least one mode. To test that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7119
	 * the new mode is a subset of the current modes we bitwise
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7120
	 * AND them together and check that the result equals the new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7121
	 * mode. For example:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7122
	 * New mode, access == R and current mode, sp->share_access  == RW
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7123
	 * access & sp->share_access == R == access, so the new access mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7124
	 * is valid. Consider access == RW, sp->share_access = R
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7125
	 * access & sp->share_access == R != access, so the new access mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7126
	 * is invalid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7127
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7128
	if ((access & sp->share_access) != access ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7129
	    (deny & sp->share_deny) != deny ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7130
	    (access &
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7131
	    (OPEN4_SHARE_ACCESS_READ | OPEN4_SHARE_ACCESS_WRITE)) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7132
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7133
		rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7134
		rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7135
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7136
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7137
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7138
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7139
	 * Release any share locks associated with this stateID.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7140
	 * Strictly speaking, this violates the spec because the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7141
	 * spec effectively requires that open downgrade be atomic.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7142
	 * At present, fs_shrlock does not have this capability.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7143
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7144
	rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7145
	rfs4_unshare(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7146
	rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7147
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7148
	fp = sp->finfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7149
	rfs4_dbe_lock(fp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7150
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7151
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7152
	 * If the current mode has deny read and the new mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7153
	 * does not, decrement the number of deny read mode bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7154
	 * and if it goes to zero turn off the deny read bit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7155
	 * on the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7156
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7157
	if ((sp->share_deny & OPEN4_SHARE_DENY_READ) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7158
	    (deny & OPEN4_SHARE_DENY_READ) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7159
		fp->deny_read--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7160
		if (fp->deny_read == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7161
			fp->share_deny &= ~OPEN4_SHARE_DENY_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7162
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7163
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7164
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7165
	 * If the current mode has deny write and the new mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7166
	 * does not, decrement the number of deny write mode bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7167
	 * and if it goes to zero turn off the deny write bit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7168
	 * on the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7169
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7170
	if ((sp->share_deny & OPEN4_SHARE_DENY_WRITE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7171
	    (deny & OPEN4_SHARE_DENY_WRITE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7172
		fp->deny_write--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7173
		if (fp->deny_write == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7174
			fp->share_deny &= ~OPEN4_SHARE_DENY_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7175
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7176
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7177
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7178
	 * If the current mode has access read and the new mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7179
	 * does not, decrement the number of access read mode bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7180
	 * and if it goes to zero turn off the access read bit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7181
	 * on the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7182
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7183
	if ((sp->share_access & OPEN4_SHARE_ACCESS_READ) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7184
	    (access & OPEN4_SHARE_ACCESS_READ) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7185
		fp->access_read--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7186
		if (fp->access_read == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7187
			fp->share_access &= ~OPEN4_SHARE_ACCESS_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7188
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7189
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7190
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7191
	 * If the current mode has access write and the new mode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7192
	 * does not, decrement the number of access write mode bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7193
	 * and if it goes to zero turn off the access write bit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7194
	 * on the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7195
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7196
	if ((sp->share_access & OPEN4_SHARE_ACCESS_WRITE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7197
	    (access & OPEN4_SHARE_ACCESS_WRITE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7198
		fp->access_write--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7199
		if (fp->access_write == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7200
			fp->share_deny &= ~OPEN4_SHARE_ACCESS_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7201
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7202
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7203
	/* Set the new access and deny modes */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7204
	sp->share_access = access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7205
	sp->share_deny = deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7206
	/* Check that the file is still accessible */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7207
	ASSERT(fp->share_access);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7208
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7209
	rfs4_dbe_unlock(fp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7210
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7211
	rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7212
	if ((status = rfs4_share(sp)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7213
		*cs->statusp = resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7214
		rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7215
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7216
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7217
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7218
	rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7219
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7220
	/* Update the stateid */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7221
	next_stateid(&sp->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7222
	resp->open_stateid = sp->stateid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7223
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7224
	rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7225
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7226
	*cs->statusp = resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7227
	/* Update the lease */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7228
	rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7229
	/* And the sequence */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7230
	rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7231
	rfs4_update_open_resp(sp->owner, resop, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7232
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7233
end:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7234
	rfs4_sw_exit(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7235
	rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7236
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7237
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7238
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7239
 * The logic behind this function is detailed in the NFSv4 RFC in the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7240
 * SETCLIENTID operation description under IMPLEMENTATION.  Refer to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7241
 * that section for explicit guidance to server behavior for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7242
 * SETCLIENTID.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7243
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7244
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7245
rfs4_op_setclientid(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7246
		    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7247
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7248
	SETCLIENTID4args *args = &argop->nfs_argop4_u.opsetclientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7249
	SETCLIENTID4res *res = &resop->nfs_resop4_u.opsetclientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7250
	rfs4_client_t *cp, *newcp, *cp_confirmed, *cp_unconfirmed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7251
	bool_t create = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7252
	char *addr, *netid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7253
	int len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7254
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7255
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7256
	newcp = cp_confirmed = cp_unconfirmed = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7257
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7258
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7259
	 * In search of an EXISTING client matching the incoming
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7260
	 * request to establish a new client identifier at the server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7261
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7262
	create = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7263
	cp = rfs4_findclient(&args->client, &create, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7264
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7265
	/* Should never happen */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7266
	ASSERT(cp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7267
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7268
	if (cp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7269
		*cs->statusp = res->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7270
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7271
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7272
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7273
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7274
	 * Easiest case. Client identifier is newly created and is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7275
	 * unconfirmed.  Also note that for this case, no other
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7276
	 * entries exist for the client identifier.  Nothing else to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7277
	 * check.  Just setup the response and respond.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7278
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7279
	if (create) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7280
		*cs->statusp = res->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7281
		res->SETCLIENTID4res_u.resok4.clientid = cp->clientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7282
		res->SETCLIENTID4res_u.resok4.setclientid_confirm =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7283
							cp->confirm_verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7284
		/* Setup callback information; CB_NULL confirmation later */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7285
		rfs4_client_setcb(cp, &args->callback, args->callback_ident);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7286
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7287
		rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7288
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7289
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7290
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7291
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7292
	 * An existing, confirmed client may exist but it may not have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7293
	 * been active for at least one lease period.  If so, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7294
	 * "close" the client and create a new client identifier
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7295
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7296
	if (rfs4_lease_expired(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7297
		rfs4_client_close(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7298
		goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7299
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7300
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7301
	if (cp->need_confirm == TRUE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7302
		cp_unconfirmed = cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7303
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7304
		cp_confirmed = cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7306
	cp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7307
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7308
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7309
	 * We have a confirmed client, now check for an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7310
	 * unconfimred entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7311
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7312
	if (cp_confirmed) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7313
		/* If creds don't match then client identifier is inuse */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7314
		if (!creds_ok(cp_confirmed->cr_set, req, cs)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7315
			rfs4_cbinfo_t *cbp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7316
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7317
			 * Some one else has established this client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7318
			 * id. Try and say * who they are. We will use
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7319
			 * the call back address supplied by * the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7320
			 * first client.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7321
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7322
			*cs->statusp = res->status = NFS4ERR_CLID_INUSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7323
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7324
			addr = netid = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7325
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7326
			cbp = &cp_confirmed->cbinfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7327
			if (cbp->cb_callback.cb_location.r_addr &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7328
			    cbp->cb_callback.cb_location.r_netid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7329
				cb_client4 *cbcp = &cbp->cb_callback;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7330
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7331
				len = strlen(cbcp->cb_location.r_addr)+1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7332
				addr = kmem_alloc(len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7333
				bcopy(cbcp->cb_location.r_addr, addr, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7334
				len = strlen(cbcp->cb_location.r_netid)+1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7335
				netid = kmem_alloc(len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7336
				bcopy(cbcp->cb_location.r_netid, netid, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7337
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7338
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7339
			res->SETCLIENTID4res_u.client_using.r_addr = addr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7340
			res->SETCLIENTID4res_u.client_using.r_netid = netid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7341
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7342
			rfs4_client_rele(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7343
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7344
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7345
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7346
		 * Confirmed, creds match, and verifier matches; must
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7347
		 * be an update of the callback info
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7348
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7349
		if (cp_confirmed->nfs_client.verifier ==
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7350
						args->client.verifier) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7351
			/* Setup callback information */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7352
			rfs4_client_setcb(cp_confirmed, &args->callback,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7353
						args->callback_ident);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7354
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7355
			/* everything okay -- move ahead */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7356
			*cs->statusp = res->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7357
			res->SETCLIENTID4res_u.resok4.clientid =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7358
				cp_confirmed->clientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7359
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7360
			/* update the confirm_verifier and return it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7361
			rfs4_client_scv_next(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7362
			res->SETCLIENTID4res_u.resok4.setclientid_confirm =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7363
						cp_confirmed->confirm_verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7364
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7365
			rfs4_client_rele(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7366
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7367
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7368
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7369
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7370
		 * Creds match but the verifier doesn't.  Must search
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7371
		 * for an unconfirmed client that would be replaced by
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7372
		 * this request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7373
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7374
		create = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7375
		cp_unconfirmed = rfs4_findclient(&args->client, &create,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7376
						cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7377
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7378
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7379
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7380
	 * At this point, we have taken care of the brand new client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7381
	 * struct, INUSE case, update of an existing, and confirmed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7382
	 * client struct.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7383
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7384
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7385
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7386
	 * check to see if things have changed while we originally
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7387
	 * picked up the client struct.  If they have, then return and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7388
	 * retry the processing of this SETCLIENTID request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7389
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7390
	if (cp_unconfirmed) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7391
		rfs4_dbe_lock(cp_unconfirmed->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7392
		if (!cp_unconfirmed->need_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7393
			rfs4_dbe_unlock(cp_unconfirmed->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7394
			rfs4_client_rele(cp_unconfirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7395
			if (cp_confirmed)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7396
				rfs4_client_rele(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7397
			goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7398
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7399
		/* do away with the old unconfirmed one */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7400
		rfs4_dbe_invalidate(cp_unconfirmed->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7401
		rfs4_dbe_unlock(cp_unconfirmed->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7402
		rfs4_client_rele(cp_unconfirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7403
		cp_unconfirmed = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7404
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7405
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7406
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7407
	 * This search will temporarily hide the confirmed client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7408
	 * struct while a new client struct is created as the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7409
	 * unconfirmed one.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7410
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7411
	create = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7412
	newcp = rfs4_findclient(&args->client, &create, cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7413
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7414
	ASSERT(newcp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7415
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7416
	if (newcp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7417
		*cs->statusp = res->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7418
		rfs4_client_rele(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7419
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7420
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7421
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7422
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7423
	 * If one was not created, then a similar request must be in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7424
	 * process so release and start over with this one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7425
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7426
	if (create != TRUE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7427
		rfs4_client_rele(newcp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7428
		if (cp_confirmed)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7429
			rfs4_client_rele(cp_confirmed);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7430
		goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7431
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7432
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7433
	*cs->statusp = res->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7434
	res->SETCLIENTID4res_u.resok4.clientid = newcp->clientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7435
	res->SETCLIENTID4res_u.resok4.setclientid_confirm =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7436
							newcp->confirm_verf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7437
	/* Setup callback information; CB_NULL confirmation later */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7438
	rfs4_client_setcb(newcp, &args->callback,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7439
				args->callback_ident);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7440
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7441
	newcp->cp_confirmed = cp_confirmed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7442
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7443
	rfs4_client_rele(newcp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7444
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7445
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7446
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7447
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7448
rfs4_op_setclientid_confirm(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7449
			    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7450
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7451
	SETCLIENTID_CONFIRM4args *args =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7452
		&argop->nfs_argop4_u.opsetclientid_confirm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7453
	SETCLIENTID_CONFIRM4res *res =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7454
		&resop->nfs_resop4_u.opsetclientid_confirm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7455
	rfs4_client_t *cp, *cptoclose = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7456
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7457
	*cs->statusp = res->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7458
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7459
	cp = rfs4_findclient_by_id(args->clientid, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7460
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7461
	if (cp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7462
		*cs->statusp = res->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7463
			rfs4_check_clientid(&args->clientid, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7464
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7465
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7466
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7467
	if (!creds_ok(cp, req, cs)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7468
		*cs->statusp = res->status = NFS4ERR_CLID_INUSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7469
		rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7470
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7471
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7472
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7473
	/* If the verifier doesn't match, the record doesn't match */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7474
	if (cp->confirm_verf != args->setclientid_confirm) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7475
		*cs->statusp = res->status = NFS4ERR_STALE_CLIENTID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7476
		rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7477
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7478
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7479
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7480
	rfs4_dbe_lock(cp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7481
	cp->need_confirm = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7482
	if (cp->cp_confirmed) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7483
		cptoclose = cp->cp_confirmed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7484
		cptoclose->ss_remove = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7485
		cp->cp_confirmed = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7486
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7487
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7488
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7489
	 * Record clientid in stable storage
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7490
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7491
	rfs4_ss_clid(cp, req);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7492
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7493
	rfs4_dbe_unlock(cp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7494
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7495
	if (cptoclose)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7496
		/* don't need to rele, client_close does it */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7497
		rfs4_client_close(cptoclose);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7498
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7499
	/* If needed, initiate CB_NULL call for callback path */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7500
	rfs4_deleg_cb_check(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7501
	rfs4_update_lease(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7502
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7503
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7504
	 * Update the client's associated server instance, if it's changed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7505
	 * since the client was created.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7506
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7507
	if (rfs4_servinst(cp) != rfs4_cur_servinst)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7508
		rfs4_servinst_assign(cp, rfs4_cur_servinst);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7509
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7510
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7511
	 * Check to see if client can perform reclaims
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7512
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7513
	rfs4_ss_chkclid(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7514
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7515
	rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7516
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7517
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7518
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7519
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7520
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7521
rfs4_op_close(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7522
	    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7523
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7524
	/* XXX Currently not using req arg */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7525
	CLOSE4args *args = &argop->nfs_argop4_u.opclose;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7526
	CLOSE4res *resp = &resop->nfs_resop4_u.opclose;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7527
	rfs4_state_t *sp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7528
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7529
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7530
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7531
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7532
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7533
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7534
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7535
	status = rfs4_get_state(&args->open_stateid, &sp, RFS4_DBS_INVALID);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7536
	if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7537
		*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7538
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7539
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7540
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7541
	/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7542
	if (cs->vp != sp->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7543
		rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7544
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7545
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7546
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7547
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7548
	/* hold off other access to open_owner while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7549
	rfs4_sw_enter(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7550
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7551
	switch (rfs4_check_stateid_seqid(sp, &args->open_stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7552
	case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7553
		if (rfs4_check_open_seqid(args->seqid, sp->owner,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7554
			resop) != NFS4_CHKSEQ_OKAY) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7555
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7556
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7557
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7558
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7559
	case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7560
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7561
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7562
	case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7563
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7564
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7565
	case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7566
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7567
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7568
	case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7569
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7570
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7571
	case NFS4_CHECK_STATEID_UNCONFIRMED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7572
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7573
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7574
	case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7575
		/* Check the sequence id for the open owner */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7576
		switch (rfs4_check_open_seqid(args->seqid, sp->owner, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7577
		case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7578
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7579
			 * This is replayed stateid; if seqid matches
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7580
			 * next expected, then client is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7581
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7582
			/* FALL THROUGH */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7583
		case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7584
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7585
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7586
		case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7587
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7588
			 * Note this case is the duplicate case so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7589
			 * resp->status is already set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7590
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7591
			*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7592
			rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7593
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7594
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7595
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7596
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7597
		ASSERT(FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7598
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7599
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7600
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7601
	rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7602
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7603
	/* Update the stateid. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7604
	next_stateid(&sp->stateid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7605
	resp->open_stateid = sp->stateid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7606
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7607
	rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7608
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7609
	rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7610
	rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7611
	rfs4_update_open_resp(sp->owner, resop, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7612
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7613
	rfs4_state_close(sp, FALSE, FALSE, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7614
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7615
	*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7616
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7617
end:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7618
	rfs4_sw_exit(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7619
	rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7620
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7621
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7622
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7623
 * Manage the counts on the file struct and close all file locks
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7624
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7625
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7626
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7627
rfs4_release_share_lock_state(rfs4_state_t *sp, cred_t *cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7628
	bool_t close_of_client)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7629
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7630
	rfs4_file_t *fp = sp->finfo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7631
	rfs4_lo_state_t *lsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7632
	struct shrlock shr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7633
	struct shr_locowner shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7634
	int fflags, s_access, s_deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7635
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7636
	fflags = s_access = s_deny = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7637
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7638
	 * Decrement the count for each access and deny bit that this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7639
	 * state has contributed to the file. If the file counts go to zero
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7640
	 * clear the appropriate bit in the appropriate mask.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7641
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7642
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7643
	if (sp->share_access & OPEN4_SHARE_ACCESS_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7644
		fp->access_read--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7645
		fflags |= FREAD;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7646
		s_access |= F_RDACC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7647
		if (fp->access_read == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7648
			fp->share_access &= ~OPEN4_SHARE_ACCESS_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7649
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7650
	if (sp->share_access & OPEN4_SHARE_ACCESS_WRITE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7651
		fp->access_write--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7652
		fflags |= FWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7653
		s_access |= F_WRACC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7654
		if (fp->access_write == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7655
			fp->share_access &= ~OPEN4_SHARE_ACCESS_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7656
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7657
	if (sp->share_deny & OPEN4_SHARE_DENY_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7658
		fp->deny_read--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7659
		s_deny |= F_RDDNY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7660
		if (fp->deny_read == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7661
			fp->share_deny &= ~OPEN4_SHARE_DENY_READ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7662
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7663
	if (sp->share_deny & OPEN4_SHARE_DENY_WRITE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7664
		fp->deny_write--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7665
		s_deny |= F_WRDNY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7666
		if (fp->deny_write == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7667
			fp->share_deny &= ~OPEN4_SHARE_DENY_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7668
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7669
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7670
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7671
	 * If this call is part of the larger closing down of client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7672
	 * state then it is just easier to release all locks
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7673
	 * associated with this client instead of going through each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7674
	 * individual file and cleaning locks there.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7675
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7676
	if (close_of_client) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7677
		if (sp->owner->client->unlksys_completed == FALSE &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7678
		    sp->lockownerlist.next->lsp != NULL &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7679
			sp->owner->client->sysidt != LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7680
			/* Is the PxFS kernel module loaded? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7681
			if (lm_remove_file_locks != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7682
				int new_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7683
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7684
				/* Encode the cluster nodeid in new sysid */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7685
				new_sysid = sp->owner->client->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7686
				lm_set_nlmid_flk(&new_sysid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7687
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7688
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7689
				 * This PxFS routine removes file locks for a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7690
				 * client over all nodes of a cluster.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7691
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7692
				NFS4_DEBUG(rfs4_debug, (CE_NOTE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7693
				    "lm_remove_file_locks(sysid=0x%x)\n",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7694
				    new_sysid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7695
				(*lm_remove_file_locks)(new_sysid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7696
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7697
				struct flock64 flk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7698
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7699
				/* Release all locks for this client */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7700
				flk.l_type = F_UNLKSYS;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7701
				flk.l_whence = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7702
				flk.l_start = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7703
				flk.l_len = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7704
				flk.l_sysid = sp->owner->client->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7705
				flk.l_pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7706
				(void) VOP_FRLOCK(sp->finfo->vp, F_SETLK, &flk,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7707
				    F_REMOTELOCK | FREAD | FWRITE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7708
				    (u_offset_t)0, NULL, CRED());
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7709
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7710
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7711
			sp->owner->client->unlksys_completed = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7712
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7713
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7714
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7715
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7716
	 * Release all locks on this file by this lock owner or at
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7717
	 * least mark the locks as having been released
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7718
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7719
	for (lsp = sp->lockownerlist.next->lsp; lsp != NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7720
		lsp = lsp->lockownerlist.next->lsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7721
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7722
		lsp->locks_cleaned = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7723
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7724
		/* Was this already taken care of above? */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7725
		if (!close_of_client &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7726
		    sp->owner->client->sysidt != LM_NOSYSID)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7727
			(void) cleanlocks(sp->finfo->vp, lsp->locker->pid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7728
				lsp->locker->client->sysidt);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7729
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7730
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7731
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7732
	 * Release any shrlocks associated with this open state ID.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7733
	 * This must be done before the rfs4_state gets marked closed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7734
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7735
	if (sp->owner->client->sysidt != LM_NOSYSID) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7736
		shr.s_access = s_access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7737
		shr.s_deny = s_deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7738
		shr.s_pid = rfs4_dbe_getid(sp->owner->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7739
		shr.s_sysid = sp->owner->client->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7740
		shr_loco.sl_pid = shr.s_pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7741
		shr_loco.sl_id = shr.s_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7742
		shr.s_owner = (caddr_t)&shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7743
		shr.s_own_len = sizeof (shr_loco);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7744
		(void) vop_shrlock(sp->finfo->vp, F_UNSHARE, &shr, fflags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7745
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7746
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7747
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7748
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7749
 * lock_denied: Fill in a LOCK4deneid structure given an flock64 structure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7750
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7751
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7752
lock_denied(LOCK4denied *dp, struct flock64 *flk)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7753
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7754
	rfs4_lockowner_t *lo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7755
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7756
	uint32_t len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7757
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7758
	lo = rfs4_findlockowner_by_pid(flk->l_pid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7759
	if (lo != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7760
		cp = lo->client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7761
		if (rfs4_lease_expired(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7762
			rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7763
			rfs4_dbe_hold(cp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7764
			rfs4_client_close(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7765
			return (NFS4ERR_EXPIRED);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7766
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7767
		dp->owner.clientid = lo->owner.clientid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7768
		len = lo->owner.owner_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7769
		dp->owner.owner_val = kmem_alloc(len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7770
		bcopy(lo->owner.owner_val, dp->owner.owner_val, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7771
		dp->owner.owner_len = len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7772
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7773
		goto finish;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7774
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7775
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7776
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7777
	 * Its not a NFS4 lock. We take advantage that the upper 32 bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7778
	 * of the client id contain the boot time for a NFS4 lock. So we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7779
	 * fabricate and identity by setting clientid to the sysid, and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7780
	 * the lock owner to the pid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7781
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7782
	dp->owner.clientid = flk->l_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7783
	len = sizeof (pid_t);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7784
	dp->owner.owner_len = len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7785
	dp->owner.owner_val = kmem_alloc(len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7786
	bcopy(&flk->l_pid, dp->owner.owner_val, len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7787
finish:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7788
	dp->offset = flk->l_start;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7789
	dp->length = flk->l_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7790
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7791
	if (flk->l_type == F_RDLCK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7792
		dp->locktype = READ_LT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7793
	else if (flk->l_type == F_WRLCK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7794
		dp->locktype = WRITE_LT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7795
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7796
		return (NFS4ERR_INVAL);	/* no mapping from POSIX ltype to v4 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7797
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7798
	return (NFS4_OK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7799
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7800
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7801
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7802
setlock(vnode_t *vp, struct flock64 *flock, int flag, cred_t *cred)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7803
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7804
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7805
	struct flock64 flk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7806
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7807
	clock_t delaytime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7808
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7809
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7810
	delaytime = MSEC_TO_TICK_ROUNDUP(rfs4_lock_delay);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7811
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7812
	for (i = 0; i < rfs4_maxlock_tries; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7813
		LOCK_PRINT(rfs4_debug, "setlock", F_SETLK, flock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7814
		error = VOP_FRLOCK(vp, F_SETLK,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7815
				flock, flag, (u_offset_t)0, NULL, cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7816
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7817
		if (error != EAGAIN && error != EACCES)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7818
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7819
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7820
		if (i < rfs4_maxlock_tries - 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7821
			delay(delaytime);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7822
			delaytime *= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7823
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7824
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7825
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7826
	if (error == EAGAIN || error == EACCES) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7827
		/* Get the owner of the lock */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7828
		flk = *flock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7829
		LOCK_PRINT(rfs4_debug, "setlock", F_GETLK, &flk);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7830
		if (VOP_FRLOCK(vp, F_GETLK,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7831
			    &flk,  flag, (u_offset_t)0, NULL, cred) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7832
			if (flk.l_type == F_UNLCK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7833
				/* No longer locked, retry */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7834
				goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7835
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7836
			*flock = flk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7837
			LOCK_PRINT(rfs4_debug, "setlock(blocking lock)",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7838
				F_GETLK, &flk);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7839
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7840
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7841
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7842
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7843
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7844
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7845
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7846
static nfsstat4
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7847
rfs4_do_lock(rfs4_lo_state_t *lp, nfs_lock_type4 locktype,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7848
	    seqid4 seqid, offset4 offset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7849
	    length4 length, cred_t *cred, nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7850
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7851
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7852
	rfs4_lockowner_t *lo = lp->locker;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7853
	rfs4_state_t *sp = lp->state;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7854
	struct flock64 flock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7855
	int16_t ltype;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7856
	int flag;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7857
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7858
	sysid_t sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7859
	LOCK4res *lres;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7860
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7861
	if (rfs4_lease_expired(lo->client)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7862
		return (NFS4ERR_EXPIRED);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7863
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7864
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7865
	if ((status = rfs4_client_sysid(lo->client, &sysid)) != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7866
		return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7867
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7868
	/* Check for zero length. To lock to end of file use all ones for V4 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7869
	if (length == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7870
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7871
	else if (length == (length4)(~0))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7872
		length = 0;		/* Posix to end of file  */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7873
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7874
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7875
	rfs4_dbe_lock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7876
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7877
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7878
	if (resop->resop != OP_LOCKU) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7879
		switch (locktype) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7880
		case READ_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7881
		case READW_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7882
			if ((sp->share_access
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7883
			    & OPEN4_SHARE_ACCESS_READ) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7884
				rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7885
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7886
				return (NFS4ERR_OPENMODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7887
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7888
			ltype = F_RDLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7889
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7890
		case WRITE_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7891
		case WRITEW_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7892
			if ((sp->share_access
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7893
			    & OPEN4_SHARE_ACCESS_WRITE) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7894
				rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7895
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7896
				return (NFS4ERR_OPENMODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7897
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7898
			ltype = F_WRLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7899
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7900
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7901
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7902
		ltype = F_UNLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7903
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7904
	flock.l_type = ltype;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7905
	flock.l_whence = 0;		/* SEEK_SET */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7906
	flock.l_start = offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7907
	flock.l_len = length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7908
	flock.l_sysid = sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7909
	flock.l_pid = lp->locker->pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7910
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7911
	/* Note that length4 is uint64_t but l_len and l_start are off64_t */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7912
	if (flock.l_len < 0 || flock.l_start < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7913
		rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7914
		return (NFS4ERR_INVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7915
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7917
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7918
	 * N.B. FREAD has the same value as OPEN4_SHARE_ACCESS_READ and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7919
	 * FWRITE has the same value as OPEN4_SHARE_ACCESS_WRITE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7920
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7921
	flag = (int)sp->share_access | F_REMOTELOCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7922
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7923
	error = setlock(sp->finfo->vp, &flock, flag, cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7924
	if (error == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7925
		rfs4_dbe_lock(lp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7926
		next_stateid(&lp->lockid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7927
		rfs4_dbe_unlock(lp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7928
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7929
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7930
	rfs4_dbe_unlock(sp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7931
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7932
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7933
	 * N.B. We map error values to nfsv4 errors. This is differrent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7934
	 * than puterrno4 routine.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7935
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7936
	switch (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7937
	case 0:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7938
		status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7939
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7940
	case EAGAIN:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7941
	case EACCES:		/* Old value */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7942
		/* Can only get here if op is OP_LOCK */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7943
		ASSERT(resop->resop == OP_LOCK);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7944
		lres = &resop->nfs_resop4_u.oplock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7945
		status = NFS4ERR_DENIED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7946
		if (lock_denied(&lres->LOCK4res_u.denied, &flock)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7947
			== NFS4ERR_EXPIRED)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7948
			goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7949
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7950
	case ENOLCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7951
		status = NFS4ERR_DELAY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7952
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7953
	case EOVERFLOW:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7954
		status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7955
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7956
	case EINVAL:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7957
		status = NFS4ERR_NOTSUPP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7958
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7959
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7960
		cmn_err(CE_WARN, "rfs4_do_lock: unexpected errno (%d)",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7961
			error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7962
		status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7963
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7964
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7965
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7966
	return (status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7967
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7968
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7969
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7970
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7971
rfs4_op_lock(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7972
	    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7973
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7974
	/* XXX Currently not using req arg */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7975
	LOCK4args *args = &argop->nfs_argop4_u.oplock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7976
	LOCK4res *resp = &resop->nfs_resop4_u.oplock;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7977
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7978
	stateid4 *stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7979
	rfs4_lockowner_t *lo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7980
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7981
	rfs4_state_t *sp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7982
	rfs4_lo_state_t *lsp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7983
	bool_t ls_sw_held = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7984
	bool_t create = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7985
	bool_t lcreate = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7986
	bool_t dup_lock = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7987
	int rc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7988
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7989
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7990
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7991
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7992
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7993
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7994
	if (args->locker.new_lock_owner) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7995
		/* Create a new lockowner for this instance */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7996
		open_to_lock_owner4 *olo = &args->locker.locker4_u.open_owner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7997
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7998
		NFS4_DEBUG(rfs4_debug, (CE_NOTE, "Creating new lock owner"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  7999
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8000
		stateid = &olo->open_stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8001
		status = rfs4_get_state(stateid, &sp, RFS4_DBS_VALID);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8002
		if (status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8003
			NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8004
				(CE_NOTE, "Get state failed in lock %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8005
				status));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8006
			*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8007
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8008
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8009
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8010
		/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8011
		if (cs->vp != sp->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8012
			rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8013
			*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8014
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8015
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8016
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8017
		/* hold off other access to open_owner while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8018
		rfs4_sw_enter(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8019
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8020
		switch (rc = rfs4_check_stateid_seqid(sp, stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8021
		case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8022
			*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8023
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8024
		case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8025
			*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8026
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8027
		case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8028
			*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8029
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8030
		case NFS4_CHECK_STATEID_UNCONFIRMED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8031
			*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8032
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8033
		case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8034
			*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8035
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8036
		case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8037
		case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8038
			switch (rfs4_check_olo_seqid(olo->open_seqid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8039
				sp->owner, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8040
			case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8041
				if (rc == NFS4_CHECK_STATEID_OKAY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8042
					break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8043
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8044
				 * This is replayed stateid; if seqid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8045
				 * matches next expected, then client
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8046
				 * is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8047
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8048
				/* FALLTHROUGH */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8049
			case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8050
				*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8051
					NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8052
				goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8053
			case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8054
				/* This is a duplicate LOCK request */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8055
				dup_lock = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8056
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8057
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8058
				 * For a duplicate we do not want to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8059
				 * create a new lockowner as it should
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8060
				 * already exist.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8061
				 * Turn off the lockowner create flag.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8062
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8063
				lcreate = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8064
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8065
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8066
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8067
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8068
		lo = rfs4_findlockowner(&olo->lock_owner, &lcreate);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8069
		if (lo == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8070
			NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8071
				(CE_NOTE, "rfs4_op_lock: no lock owner"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8072
			*cs->statusp = resp->status = NFS4ERR_RESOURCE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8073
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8074
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8075
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8076
		lsp = rfs4_findlo_state_by_owner(lo, sp, &create);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8077
		if (lsp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8078
			rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8079
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8080
			 * Only update theh open_seqid if this is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8081
			 * a duplicate request
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8082
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8083
			if (dup_lock == FALSE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8084
				rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8085
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8086
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8087
			NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8088
				(CE_NOTE, "rfs4_op_lock: no state"));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8089
			*cs->statusp = resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8090
			rfs4_update_open_resp(sp->owner, resop, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8091
			rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8092
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8093
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8094
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8095
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8096
		 * This is the new_lock_owner branch and the client is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8097
		 * supposed to be associating a new lock_owner with
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8098
		 * the open file at this point.  If we find that a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8099
		 * lock_owner/state association already exists and a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8100
		 * successful LOCK request was returned to the client,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8101
		 * an error is returned to the client since this is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8102
		 * not appropriate.  The client should be using the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8103
		 * existing lock_owner branch.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8104
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8105
		if (dup_lock == FALSE && create == FALSE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8106
			if (lsp->lock_completed == TRUE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8107
				*cs->statusp =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8108
					resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8109
				rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8110
				goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8111
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8112
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8113
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8114
		rfs4_update_lease(sp->owner->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8115
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8116
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8117
		 * Only update theh open_seqid if this is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8118
		 * a duplicate request
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8119
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8120
		if (dup_lock == FALSE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8121
			rfs4_update_open_sequence(sp->owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8122
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8123
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8124
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8125
		 * If this is a duplicate lock request, just copy the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8126
		 * previously saved reply and return.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8127
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8128
		if (dup_lock == TRUE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8129
			/* verify that lock_seqid's match */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8130
			if (lsp->seqid != olo->lock_seqid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8131
				NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8132
				(CE_NOTE, "rfs4_op_lock: Dup-Lock seqid bad"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8133
				"lsp->seqid=%d old->seqid=%d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8134
				lsp->seqid, olo->lock_seqid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8135
				*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8136
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8137
				rfs4_copy_reply(resop, lsp->reply);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8138
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8139
				 * Make sure to copy the just
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8140
				 * retrieved reply status into the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8141
				 * overall compound status
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8142
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8143
				*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8144
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8145
			rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8146
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8147
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8148
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8149
		rfs4_dbe_lock(lsp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8150
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8151
		/* Make sure to update the lock sequence id */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8152
		lsp->seqid = olo->lock_seqid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8153
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8154
		NFS4_DEBUG(rfs4_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8155
			(CE_NOTE, "Lock seqid established as %d", lsp->seqid));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8156
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8157
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8158
		 * This is used to signify the newly created lockowner
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8159
		 * stateid and its sequence number.  The checks for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8160
		 * sequence number and increment don't occur on the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8161
		 * very first lock request for a lockowner.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8162
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8163
		lsp->skip_seqid_check = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8164
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8165
		/* hold off other access to lsp while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8166
		rfs4_sw_enter(&lsp->ls_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8167
		ls_sw_held = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8168
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8169
		rfs4_dbe_unlock(lsp->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8170
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8171
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8172
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8173
		stateid = &args->locker.locker4_u.lock_owner.lock_stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8174
		/* get lsp and hold the lock on the underlying file struct */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8175
		if ((status = rfs4_get_lo_state(stateid, &lsp, TRUE))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8176
		    != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8177
			*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8178
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8179
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8180
		create = FALSE;	/* We didn't create lsp */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8181
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8182
		/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8183
		if (cs->vp != lsp->state->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8184
			rfs4_lo_state_rele(lsp, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8185
			*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8186
			return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8187
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8188
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8189
		/* hold off other access to lsp while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8190
		rfs4_sw_enter(&lsp->ls_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8191
		ls_sw_held = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8192
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8193
		switch (rfs4_check_lo_stateid_seqid(lsp, stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8194
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8195
		 * The stateid looks like it was okay (expected to be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8196
		 * the next one)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8197
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8198
		case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8199
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8200
			 * The sequence id is now checked.  Determine
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8201
			 * if this is a replay or if it is in the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8202
			 * expected (next) sequence.  In the case of a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8203
			 * replay, there are two replay conditions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8204
			 * that may occur.  The first is the normal
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8205
			 * condition where a LOCK is done with a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8206
			 * NFS4_OK response and the stateid is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8207
			 * updated.  That case is handled below when
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8208
			 * the stateid is identified as a REPLAY.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8209
			 * second is the case where an error is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8210
			 * returned, like NFS4ERR_DENIED, and the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8211
			 * sequence number is updated but the stateid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8212
			 * is not updated.  This second case is dealt
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8213
			 * with here.  So it may seem odd that the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8214
			 * stateid is okay but the sequence id is a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8215
			 * replay but it is okay.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8216
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8217
			switch (rfs4_check_lock_seqid(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8218
				args->locker.locker4_u.lock_owner.lock_seqid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8219
				lsp, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8220
			case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8221
				if (resp->status != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8222
					/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8223
					 * Here is our replay and need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8224
					 * to verify that the last
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8225
					 * response was an error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8226
					 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8227
					*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8228
					goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8229
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8230
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8231
				 * This is done since the sequence id
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8232
				 * looked like a replay but it didn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8233
				 * pass our check so a BAD_SEQID is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8234
				 * returned as a result.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8235
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8236
				/*FALLTHROUGH*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8237
			case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8238
				*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8239
					NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8240
				goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8241
			case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8242
				/* Everything looks okay move ahead */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8243
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8244
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8245
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8246
		case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8247
			*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8248
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8249
		case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8250
			*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8251
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8252
		case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8253
			*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8254
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8255
		case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8256
			*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8257
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8258
		case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8259
			switch (rfs4_check_lock_seqid(
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8260
				args->locker.locker4_u.lock_owner.lock_seqid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8261
				lsp, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8262
			case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8263
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8264
				 * This is a replayed stateid; if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8265
				 * seqid matches the next expected,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8266
				 * then client is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8267
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8268
			case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8269
				*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8270
					NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8271
				goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8272
			case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8273
				rfs4_update_lease(lsp->locker->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8274
				*cs->statusp = status = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8275
				goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8276
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8277
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8278
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8279
			ASSERT(FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8280
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8281
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8282
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8283
		rfs4_update_lock_sequence(lsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8284
		rfs4_update_lease(lsp->locker->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8285
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8286
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8287
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8288
	 * NFS4 only allows locking on regular files, so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8289
	 * verify type of object.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8290
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8291
	if (cs->vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8292
		if (cs->vp->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8293
			status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8294
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8295
			status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8296
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8297
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8298
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8299
	cp = lsp->state->owner->client;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8300
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8301
	if (rfs4_clnt_in_grace(cp) && !args->reclaim) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8302
		status = NFS4ERR_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8303
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8304
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8306
	if (rfs4_clnt_in_grace(cp) && args->reclaim && !cp->can_reclaim) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8307
		status = NFS4ERR_NO_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8308
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8309
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8310
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8311
	if (!rfs4_clnt_in_grace(cp) && args->reclaim) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8312
		status = NFS4ERR_NO_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8313
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8314
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8315
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8316
	if (lsp->state->finfo->dinfo->dtype == OPEN_DELEGATE_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8317
		cs->deleg = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8318
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8319
	status = rfs4_do_lock(lsp, args->locktype,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8320
				args->locker.locker4_u.lock_owner.lock_seqid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8321
				args->offset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8322
				args->length, cs->cr, resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8323
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8324
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8325
	lsp->skip_seqid_check = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8326
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8327
	*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8328
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8329
	if (status == NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8330
		resp->LOCK4res_u.lock_stateid = lsp->lockid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8331
		lsp->lock_completed = TRUE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8332
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8333
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8334
	 * Only update the "OPEN" response here if this was a new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8335
	 * lock_owner
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8336
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8337
	if (sp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8338
		rfs4_update_open_resp(sp->owner, resop, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8339
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8340
	rfs4_update_lock_resp(lsp, resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8341
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8342
end:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8343
	if (lsp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8344
		if (ls_sw_held)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8345
			rfs4_sw_exit(&lsp->ls_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8346
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8347
		 * If an sp obtained, then the lsp does not represent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8348
		 * a lock on the file struct.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8349
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8350
		if (sp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8351
			rfs4_lo_state_rele(lsp, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8352
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8353
			rfs4_lo_state_rele(lsp, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8354
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8355
	if (sp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8356
		rfs4_sw_exit(&sp->owner->oo_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8357
		rfs4_state_rele(sp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8358
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8359
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8360
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8361
/* free function for LOCK/LOCKT */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8362
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8363
lock_denied_free(nfs_resop4 *resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8364
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8365
	LOCK4denied *dp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8366
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8367
	switch (resop->resop) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8368
	case OP_LOCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8369
		if (resop->nfs_resop4_u.oplock.status == NFS4ERR_DENIED)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8370
			dp = &resop->nfs_resop4_u.oplock.LOCK4res_u.denied;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8371
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8372
	case OP_LOCKT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8373
		if (resop->nfs_resop4_u.oplockt.status == NFS4ERR_DENIED)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8374
			dp = &resop->nfs_resop4_u.oplockt.denied;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8375
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8376
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8377
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8378
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8379
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8380
	if (dp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8381
		kmem_free(dp->owner.owner_val, dp->owner.owner_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8382
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8383
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8384
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8385
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8386
rfs4_op_locku(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8387
	    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8388
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8389
	/* XXX Currently not using req arg */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8390
	LOCKU4args *args = &argop->nfs_argop4_u.oplocku;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8391
	LOCKU4res *resp = &resop->nfs_resop4_u.oplocku;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8392
	nfsstat4 status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8393
	stateid4 *stateid = &args->lock_stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8394
	rfs4_lo_state_t *lsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8395
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8396
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8397
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8398
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8399
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8400
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8401
	if ((status = rfs4_get_lo_state(stateid, &lsp, TRUE)) != NFS4_OK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8402
		*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8403
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8404
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8405
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8406
	/* Ensure specified filehandle matches */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8407
	if (cs->vp != lsp->state->finfo->vp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8408
		rfs4_lo_state_rele(lsp, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8409
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8410
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8411
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8412
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8413
	/* hold off other access to lsp while we tinker */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8414
	rfs4_sw_enter(&lsp->ls_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8415
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8416
	switch (rfs4_check_lo_stateid_seqid(lsp, stateid)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8417
	case NFS4_CHECK_STATEID_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8418
		if (rfs4_check_lock_seqid(args->seqid, lsp, resop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8419
		    != NFS4_CHKSEQ_OKAY) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8420
			*cs->statusp = resp->status = NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8421
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8422
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8423
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8424
	case NFS4_CHECK_STATEID_OLD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8425
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8426
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8427
	case NFS4_CHECK_STATEID_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8428
		*cs->statusp = resp->status = NFS4ERR_BAD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8429
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8430
	case NFS4_CHECK_STATEID_EXPIRED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8431
		*cs->statusp = resp->status = NFS4ERR_EXPIRED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8432
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8433
	case NFS4_CHECK_STATEID_CLOSED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8434
		*cs->statusp = resp->status = NFS4ERR_OLD_STATEID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8435
		goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8436
	case NFS4_CHECK_STATEID_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8437
		switch (rfs4_check_lock_seqid(args->seqid, lsp, resop)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8438
		case NFS4_CHKSEQ_OKAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8439
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8440
				 * This is a replayed stateid; if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8441
				 * seqid matches the next expected,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8442
				 * then client is using wrong seqid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8443
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8444
		case NFS4_CHKSEQ_BAD:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8445
			*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8446
				NFS4ERR_BAD_SEQID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8447
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8448
		case NFS4_CHKSEQ_REPLAY:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8449
			rfs4_update_lease(lsp->locker->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8450
			*cs->statusp = status = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8451
			goto end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8452
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8453
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8454
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8455
		ASSERT(FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8456
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8457
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8458
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8459
	rfs4_update_lock_sequence(lsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8460
	rfs4_update_lease(lsp->locker->client);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8461
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8462
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8463
	 * NFS4 only allows locking on regular files, so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8464
	 * verify type of object.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8465
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8466
	if (cs->vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8467
		if (cs->vp->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8468
			status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8469
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8470
			status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8471
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8472
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8473
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8474
	if (rfs4_clnt_in_grace(lsp->state->owner->client)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8475
		status = NFS4ERR_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8476
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8477
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8478
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8479
	status = rfs4_do_lock(lsp, args->locktype,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8480
			    args->seqid, args->offset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8481
			    args->length, cs->cr, resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8482
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8483
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8484
	*cs->statusp = resp->status = status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8485
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8486
	if (status == NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8487
		resp->lock_stateid = lsp->lockid.stateid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8488
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8489
	rfs4_update_lock_resp(lsp, resop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8490
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8491
end:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8492
	rfs4_sw_exit(&lsp->ls_sw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8493
	rfs4_lo_state_rele(lsp, TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8494
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8495
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8496
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8497
 * LOCKT is a best effort routine, the client can not be guaranteed that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8498
 * the status return is still in effect by the time the reply is received.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8499
 * They are numerous race conditions in this routine, but we are not required
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8500
 * and can not be accurate.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8501
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8502
/*ARGSUSED*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8503
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8504
rfs4_op_lockt(nfs_argop4 *argop, nfs_resop4 *resop,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8505
	    struct svc_req *req, struct compound_state *cs)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8506
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8507
	LOCKT4args *args = &argop->nfs_argop4_u.oplockt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8508
	LOCKT4res *resp = &resop->nfs_resop4_u.oplockt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8509
	rfs4_lockowner_t *lo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8510
	rfs4_client_t *cp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8511
	bool_t create = FALSE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8512
	struct flock64 flk;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8513
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8514
	int flag = FREAD | FWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8515
	int ltype;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8516
	length4 posix_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8517
	sysid_t sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8518
	pid_t pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8519
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8520
	if (cs->vp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8521
		*cs->statusp = resp->status = NFS4ERR_NOFILEHANDLE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8522
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8523
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8524
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8525
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8526
	 * NFS4 only allows locking on regular files, so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8527
	 * verify type of object.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8528
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8529
	if (cs->vp->v_type != VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8530
		if (cs->vp->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8531
			*cs->statusp = resp->status = NFS4ERR_ISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8532
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8533
			*cs->statusp = resp->status =  NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8534
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8535
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8536
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8537
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8538
	 * Check out the clientid to ensure the server knows about it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8539
	 * so that we correctly inform the client of a server reboot.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8540
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8541
	if ((cp = rfs4_findclient_by_id(args->owner.clientid, FALSE))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8542
	    == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8543
		*cs->statusp = resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8544
			rfs4_check_clientid(&args->owner.clientid, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8545
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8546
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8547
	if (rfs4_lease_expired(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8548
		rfs4_client_close(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8549
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8550
		 * Protocol doesn't allow returning NFS4ERR_STALE as
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8551
		 * other operations do on this check so STALE_CLIENTID
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8552
		 * is returned instead
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8553
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8554
		*cs->statusp = resp->status = NFS4ERR_STALE_CLIENTID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8555
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8556
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8557
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8558
	if (rfs4_clnt_in_grace(cp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8559
		*cs->statusp = resp->status = NFS4ERR_GRACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8560
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8561
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8562
	rfs4_client_rele(cp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8563
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8564
	resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8565
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8566
	switch (args->locktype) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8567
	case READ_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8568
	case READW_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8569
		ltype = F_RDLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8570
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8571
	case WRITE_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8572
	case WRITEW_LT:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8573
		ltype = F_WRLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8574
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8575
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8576
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8577
	posix_length = args->length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8578
	/* Check for zero length. To lock to end of file use all ones for V4 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8579
	if (posix_length == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8580
		*cs->statusp = resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8581
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8582
	} else if (posix_length == (length4)(~0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8583
		posix_length = 0;	/* Posix to end of file  */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8584
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8585
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8586
	/* Find or create a lockowner */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8587
	lo = rfs4_findlockowner(&args->owner, &create);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8588
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8589
	if (lo) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8590
		pid = lo->pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8591
		if ((resp->status =
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8592
			rfs4_client_sysid(lo->client, &sysid)) != NFS4_OK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8593
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8594
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8595
		pid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8596
		sysid = lockt_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8597
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8598
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8599
	flk.l_type = ltype;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8600
	flk.l_whence = 0;		/* SEEK_SET */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8601
	flk.l_start = args->offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8602
	flk.l_len = posix_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8603
	flk.l_sysid = sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8604
	flk.l_pid = pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8605
	flag |= F_REMOTELOCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8606
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8607
	LOCK_PRINT(rfs4_debug, "rfs4_op_lockt", F_GETLK, &flk);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8608
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8609
	/* Note that length4 is uint64_t but l_len and l_start are off64_t */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8610
	if (flk.l_len < 0 || flk.l_start < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8611
		resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8612
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8613
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8614
	error = VOP_FRLOCK(cs->vp, F_GETLK, &flk, flag, (u_offset_t)0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8615
	    NULL, cs->cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8616
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8617
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8618
	 * N.B. We map error values to nfsv4 errors. This is differrent
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8619
	 * than puterrno4 routine.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8620
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8621
	switch (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8622
	case 0:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8623
		if (flk.l_type == F_UNLCK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8624
			resp->status = NFS4_OK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8625
		else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8626
			if (lock_denied(&resp->denied, &flk) == NFS4ERR_EXPIRED)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8627
				goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8628
			resp->status = NFS4ERR_DENIED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8629
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8630
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8631
	case EOVERFLOW:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8632
		resp->status = NFS4ERR_INVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8633
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8634
	case EINVAL:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8635
		resp->status = NFS4ERR_NOTSUPP;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8636
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8637
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8638
		cmn_err(CE_WARN, "rfs4_op_lockt: unexpected errno (%d)",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8639
			error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8640
		resp->status = NFS4ERR_SERVERFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8641
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8642
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8643
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8644
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8645
	if (lo)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8646
		rfs4_lockowner_rele(lo);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8647
	*cs->statusp = resp->status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8648
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8649
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8650
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8651
vop_shrlock(vnode_t *vp, int cmd, struct shrlock *sp, int fflags)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8652
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8653
	int err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8654
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8655
	if (cmd == F_UNSHARE && sp->s_deny == 0 && sp->s_access == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8656
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8657
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8658
	err = VOP_SHRLOCK(vp, cmd, sp, fflags, CRED());
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8659
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8660
	NFS4_DEBUG(rfs4_shrlock_debug,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8661
		(CE_NOTE, "rfs4_shrlock %s vp=%p acc=%d dny=%d sysid=%d "
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8662
		"pid=%d err=%d\n", cmd == F_SHARE ? "SHARE" : "UNSHR",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8663
		(void *) vp, sp->s_access, sp->s_deny, sp->s_sysid, sp->s_pid,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8664
		err));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8665
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8666
	return (err);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8667
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8668
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8669
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8670
rfs4_shrlock(rfs4_state_t *sp, int cmd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8671
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8672
	struct shrlock shr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8673
	struct shr_locowner shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8674
	int fflags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8675
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8676
	fflags = shr.s_access = shr.s_deny = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8677
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8678
	if (sp->share_access & OPEN4_SHARE_ACCESS_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8679
		fflags |= FREAD;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8680
		shr.s_access |= F_RDACC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8681
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8682
	if (sp->share_access & OPEN4_SHARE_ACCESS_WRITE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8683
		fflags |= FWRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8684
		shr.s_access |= F_WRACC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8685
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8686
	if (sp->share_deny & OPEN4_SHARE_DENY_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8687
		shr.s_deny |= F_RDDNY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8688
	if (sp->share_deny & OPEN4_SHARE_DENY_WRITE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8689
		shr.s_deny |= F_WRDNY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8690
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8691
	shr.s_pid = rfs4_dbe_getid(sp->owner->dbe);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8692
	shr.s_sysid = sp->owner->client->sysidt;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8693
	shr_loco.sl_pid = shr.s_pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8694
	shr_loco.sl_id = shr.s_sysid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8695
	shr.s_owner = (caddr_t)&shr_loco;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8696
	shr.s_own_len = sizeof (shr_loco);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8697
	return (vop_shrlock(sp->finfo->vp, cmd, &shr, fflags));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8698
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8699
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8700
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8701
rfs4_share(rfs4_state_t *sp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8702
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8703
	return (rfs4_shrlock(sp, F_SHARE));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8704
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8705
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8706
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8707
rfs4_unshare(rfs4_state_t *sp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8708
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8709
	(void) rfs4_shrlock(sp, F_UNSHARE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  8710
}