usr/src/uts/common/fs/nfs/nfs_vnops.c
author Casper H.S. Dik <Casper.Dik@Sun.COM>
Wed, 28 Apr 2010 10:01:37 +0200
changeset 12273 63678502e95e
parent 10847 37439331af3e
permissions -rw-r--r--
PSARC 2009/377 In-kernel pfexec implementation. PSARC 2009/378 Basic File Privileges PSARC 2010/072 RBAC update: user attrs from profiles 4912090 pfzsh(1) should exist 4912093 pfbash(1) should exist 4912096 pftcsh(1) should exist 6440298 Expand the basic privilege set in order to restrict file access 6859862 Move pfexec into the kernel 6919171 cred_t sidesteps kmem_debug; we need to be able to detect bad hold/free when they occur 6923721 The new SYS_SMB privilege is not backward compatible 6937562 autofs doesn't remove its door when the zone shuts down 6937727 Zones stuck on deathrow; netstack_zone keeps a credential reference to the zone 6940159 Implement PSARC 2010/072
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
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
     5
 * Common Development and Distribution License (the "License").
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
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
/*
12273
63678502e95e PSARC 2009/377 In-kernel pfexec implementation.
Casper H.S. Dik <Casper.Dik@Sun.COM>
parents: 10847
diff changeset
    22
 * Copyright (c) 1990, 2010, Oracle and/or its affiliates. All rights reserved.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 *	Copyright (c) 1983,1984,1985,1986,1987,1988,1989 AT&T.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 *	All rights reserved.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
#include <sys/param.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
#include <sys/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
#include <sys/systm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
#include <sys/cred.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
#include <sys/time.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#include <sys/vnode.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
#include <sys/vfs.h>
3898
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
    35
#include <sys/vfs_opreg.h>
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include <sys/file.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <sys/filio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <sys/uio.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <sys/buf.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include <sys/mman.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
#include <sys/pathname.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#include <sys/dirent.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
#include <sys/debug.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
#include <sys/vmsystm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#include <sys/fcntl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
#include <sys/flock.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
#include <sys/swap.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
#include <sys/errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
#include <sys/strsubr.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
#include <sys/sysmacros.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
#include <sys/kmem.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
#include <sys/cmn_err.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
#include <sys/pathconf.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
#include <sys/utsname.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
#include <sys/dnlc.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
#include <sys/acl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
#include <sys/atomic.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
#include <sys/policy.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
#include <sys/sdt.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
#include <rpc/types.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
#include <rpc/auth.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
#include <rpc/clnt.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#include <nfs/nfs.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
#include <nfs/nfs_clnt.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
#include <nfs/rnode.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
#include <nfs/nfs_acl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
#include <nfs/lm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
#include <vm/hat.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
#include <vm/as.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
#include <vm/page.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
#include <vm/pvn.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
#include <vm/seg.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
#include <vm/seg_map.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
#include <vm/seg_kpm.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
#include <vm/seg_vn.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
#include <fs/fs_subr.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
#include <sys/ddi.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
static int	nfs_rdwrlbn(vnode_t *, page_t *, u_offset_t, size_t, int,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
			cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
static int	nfswrite(vnode_t *, caddr_t, uint_t, int, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
static int	nfsread(vnode_t *, caddr_t, uint_t, int, size_t *, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
static int	nfssetattr(vnode_t *, struct vattr *, int, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
static int	nfslookup_dnlc(vnode_t *, char *, vnode_t **, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
static int	nfslookup_otw(vnode_t *, char *, vnode_t **, cred_t *, int);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
    91
static int	nfsrename(vnode_t *, char *, vnode_t *, char *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
    92
			caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
static int	nfsreaddir(vnode_t *, rddir_cache *, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
static int	nfs_bio(struct buf *, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
static int	nfs_getapage(vnode_t *, u_offset_t, size_t, uint_t *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
			page_t *[], size_t, struct seg *, caddr_t,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
			enum seg_rw, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
static void	nfs_readahead(vnode_t *, u_offset_t, caddr_t, struct seg *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
			cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
static int	nfs_sync_putapage(vnode_t *, page_t *, u_offset_t, size_t,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
			int, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
static int	nfs_sync_pageio(vnode_t *, page_t *, u_offset_t, size_t,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
			int, cred_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
static void	nfs_delmap_callback(struct as *, void *, uint_t);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
 * Error flags used to pass information about certain special errors
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
 * which need to be handled specially.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
#define	NFS_EOF			-98
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
 * These are the vnode ops routines which implement the vnode interface to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
 * the networked file system.  These routines just take their parameters,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
 * make them look networkish by putting the right info into interface structs,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
 * and then calling the appropriate remote routine(s) to do the work.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
 * Note on directory name lookup cacheing:  If we detect a stale fhandle,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
 * we purge the directory cache relative to that vnode.  This way, the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
 * user won't get burned by the cache repeatedly.  See <nfs/rnode.h> for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
 * more details on rnode locking.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   124
static int	nfs_open(vnode_t **, int, cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   125
static int	nfs_close(vnode_t *, int, int, offset_t, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   126
			caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
static int	nfs_read(vnode_t *, struct uio *, int, cred_t *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
			caller_context_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
static int	nfs_write(vnode_t *, struct uio *, int, cred_t *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
			caller_context_t *);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   131
static int	nfs_ioctl(vnode_t *, int, intptr_t, int, cred_t *, int *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   132
			caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   133
static int	nfs_getattr(vnode_t *, struct vattr *, int, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   134
			caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
static int	nfs_setattr(vnode_t *, struct vattr *, int, cred_t *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
			caller_context_t *);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   137
static int	nfs_access(vnode_t *, int, int, cred_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
static int	nfs_accessx(void *, int, cred_t *);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   139
static int	nfs_readlink(vnode_t *, struct uio *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   140
			caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   141
static int	nfs_fsync(vnode_t *, int, cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   142
static void	nfs_inactive(vnode_t *, cred_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   143
static int	nfs_lookup(vnode_t *, char *, vnode_t **, struct pathname *,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   144
			int, vnode_t *, cred_t *, caller_context_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   145
			int *, pathname_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   146
static int	nfs_create(vnode_t *, char *, struct vattr *, enum vcexcl,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   147
			int, vnode_t **, cred_t *, int, caller_context_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   148
			vsecattr_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   149
static int	nfs_remove(vnode_t *, char *, cred_t *, caller_context_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   150
			int);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   151
static int	nfs_link(vnode_t *, vnode_t *, char *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   152
			caller_context_t *, int);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   153
static int	nfs_rename(vnode_t *, char *, vnode_t *, char *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   154
			caller_context_t *, int);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   155
static int	nfs_mkdir(vnode_t *, char *, struct vattr *, vnode_t **,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   156
			cred_t *, caller_context_t *, int, vsecattr_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   157
static int	nfs_rmdir(vnode_t *, char *, vnode_t *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   158
			caller_context_t *, int);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
static int	nfs_symlink(vnode_t *, char *, struct vattr *, char *,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   160
			cred_t *, caller_context_t *, int);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   161
static int	nfs_readdir(vnode_t *, struct uio *, cred_t *, int *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   162
			caller_context_t *, int);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   163
static int	nfs_fid(vnode_t *, fid_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   164
static int	nfs_rwlock(vnode_t *, int, caller_context_t *);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   165
static void	nfs_rwunlock(vnode_t *, int, caller_context_t *);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   166
static int	nfs_seek(vnode_t *, offset_t, offset_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   167
static int	nfs_getpage(vnode_t *, offset_t, size_t, uint_t *,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   168
			page_t *[], size_t, struct seg *, caddr_t,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   169
			enum seg_rw, cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   170
static int	nfs_putpage(vnode_t *, offset_t, size_t, int, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   171
			caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   172
static int	nfs_map(vnode_t *, offset_t, struct as *, caddr_t *, size_t,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   173
			uchar_t, uchar_t, uint_t, cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   174
static int	nfs_addmap(vnode_t *, offset_t, struct as *, caddr_t, size_t,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   175
			uchar_t, uchar_t, uint_t, cred_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   176
static int	nfs_frlock(vnode_t *, int, struct flock64 *, int, offset_t,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   177
			struct flk_callback *, cred_t *, caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   178
static int	nfs_space(vnode_t *, int, struct flock64 *, int, offset_t,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   179
			cred_t *, caller_context_t *);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   180
static int	nfs_realvp(vnode_t *, vnode_t **, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   181
static int	nfs_delmap(vnode_t *, offset_t, struct as *, caddr_t, size_t,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   182
			uint_t, uint_t, uint_t, cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   183
static int	nfs_pathconf(vnode_t *, int, ulong_t *, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   184
			caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   185
static int	nfs_pageio(vnode_t *, page_t *, u_offset_t, size_t, int,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   186
			cred_t *, caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   187
static int	nfs_setsecattr(vnode_t *, vsecattr_t *, int, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   188
			caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   189
static int	nfs_getsecattr(vnode_t *, vsecattr_t *, int, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   190
			caller_context_t *);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   191
static int	nfs_shrlock(vnode_t *, int, struct shrlock *, int, cred_t *,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   192
			caller_context_t *);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   193
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   194
struct vnodeops *nfs_vnodeops;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   195
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   196
const fs_operation_def_t nfs_vnodeops_template[] = {
3898
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   197
	VOPNAME_OPEN,		{ .vop_open = nfs_open },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   198
	VOPNAME_CLOSE,		{ .vop_close = nfs_close },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   199
	VOPNAME_READ,		{ .vop_read = nfs_read },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   200
	VOPNAME_WRITE,		{ .vop_write = nfs_write },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   201
	VOPNAME_IOCTL,		{ .vop_ioctl = nfs_ioctl },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   202
	VOPNAME_GETATTR,	{ .vop_getattr = nfs_getattr },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   203
	VOPNAME_SETATTR,	{ .vop_setattr = nfs_setattr },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   204
	VOPNAME_ACCESS,		{ .vop_access = nfs_access },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   205
	VOPNAME_LOOKUP,		{ .vop_lookup = nfs_lookup },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   206
	VOPNAME_CREATE,		{ .vop_create = nfs_create },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   207
	VOPNAME_REMOVE,		{ .vop_remove = nfs_remove },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   208
	VOPNAME_LINK,		{ .vop_link = nfs_link },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   209
	VOPNAME_RENAME,		{ .vop_rename = nfs_rename },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   210
	VOPNAME_MKDIR,		{ .vop_mkdir = nfs_mkdir },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   211
	VOPNAME_RMDIR,		{ .vop_rmdir = nfs_rmdir },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   212
	VOPNAME_READDIR,	{ .vop_readdir = nfs_readdir },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   213
	VOPNAME_SYMLINK,	{ .vop_symlink = nfs_symlink },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   214
	VOPNAME_READLINK,	{ .vop_readlink = nfs_readlink },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   215
	VOPNAME_FSYNC,		{ .vop_fsync = nfs_fsync },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   216
	VOPNAME_INACTIVE,	{ .vop_inactive = nfs_inactive },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   217
	VOPNAME_FID,		{ .vop_fid = nfs_fid },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   218
	VOPNAME_RWLOCK,		{ .vop_rwlock = nfs_rwlock },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   219
	VOPNAME_RWUNLOCK,	{ .vop_rwunlock = nfs_rwunlock },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   220
	VOPNAME_SEEK,		{ .vop_seek = nfs_seek },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   221
	VOPNAME_FRLOCK,		{ .vop_frlock = nfs_frlock },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   222
	VOPNAME_SPACE,		{ .vop_space = nfs_space },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   223
	VOPNAME_REALVP,		{ .vop_realvp = nfs_realvp },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   224
	VOPNAME_GETPAGE,	{ .vop_getpage = nfs_getpage },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   225
	VOPNAME_PUTPAGE,	{ .vop_putpage = nfs_putpage },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   226
	VOPNAME_MAP,		{ .vop_map = nfs_map },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   227
	VOPNAME_ADDMAP,		{ .vop_addmap = nfs_addmap },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   228
	VOPNAME_DELMAP,		{ .vop_delmap = nfs_delmap },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   229
	VOPNAME_DUMP,		{ .vop_dump = nfs_dump },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   230
	VOPNAME_PATHCONF,	{ .vop_pathconf = nfs_pathconf },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   231
	VOPNAME_PAGEIO,		{ .vop_pageio = nfs_pageio },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   232
	VOPNAME_SETSECATTR,	{ .vop_setsecattr = nfs_setsecattr },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   233
	VOPNAME_GETSECATTR,	{ .vop_getsecattr = nfs_getsecattr },
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   234
	VOPNAME_SHRLOCK,	{ .vop_shrlock = nfs_shrlock },
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
   235
	VOPNAME_VNEVENT, 	{ .vop_vnevent = fs_vnevent_support },
3898
c788126f2a20 PSARC/2007/124 Strong Type-Checking for VFS Operation Registration Mechanism
rsb
parents: 1841
diff changeset
   236
	NULL,			NULL
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   237
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   238
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   239
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   240
 * XXX:  This is referenced in modstubs.s
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   241
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   242
struct vnodeops *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   243
nfs_getvnodeops(void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   244
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   245
	return (nfs_vnodeops);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   246
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   247
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   248
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   249
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   250
nfs_open(vnode_t **vpp, int flag, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   251
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   252
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   253
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   254
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   255
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   256
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   257
	vp = *vpp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   258
	rp = VTOR(vp);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   259
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   260
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   261
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   262
	if (rp->r_cred == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   263
		crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   264
		rp->r_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   265
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   266
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   267
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   268
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   269
	 * If there is no cached data or if close-to-open
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   270
	 * consistency checking is turned off, we can avoid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   271
	 * the over the wire getattr.  Otherwise, if the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   272
	 * file system is mounted readonly, then just verify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   273
	 * the caches are up to date using the normal mechanism.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   274
	 * Else, if the file is not mmap'd, then just mark
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   275
	 * the attributes as timed out.  They will be refreshed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   276
	 * and the caches validated prior to being used.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   277
	 * Else, the file system is mounted writeable so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   278
	 * force an over the wire GETATTR in order to ensure
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   279
	 * that all cached data is valid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   280
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   281
	if (vp->v_count > 1 ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   282
	    ((vn_has_cached_data(vp) || HAVE_RDDIR_CACHE(rp)) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   283
	    !(VTOMI(vp)->mi_flags & MI_NOCTO))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   284
		if (vn_is_readonly(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   285
			error = nfs_validate_caches(vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   286
		else if (rp->r_mapcnt == 0 && vp->v_count == 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   287
			PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   288
			error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   289
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   290
			va.va_mask = AT_ALL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   291
			error = nfs_getattr_otw(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   292
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   293
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   294
		error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   295
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   296
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   297
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   298
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   299
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   300
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   301
nfs_close(vnode_t *vp, int flag, int count, offset_t offset, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   302
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   303
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   304
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   305
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   306
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   307
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   308
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   309
	 * zone_enter(2) prevents processes from changing zones with NFS files
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   310
	 * open; if we happen to get here from the wrong zone we can't do
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   311
	 * anything over the wire.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   312
	 */
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   313
	if (VTOMI(vp)->mi_zone != nfs_zone()) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   314
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   315
		 * We could attempt to clean up locks, except we're sure
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   316
		 * that the current process didn't acquire any locks on
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   317
		 * the file: any attempt to lock a file belong to another zone
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   318
		 * will fail, and one can't lock an NFS file and then change
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   319
		 * zones, as that fails too.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   320
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   321
		 * Returning an error here is the sane thing to do.  A
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   322
		 * subsequent call to VN_RELE() which translates to a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   323
		 * nfs_inactive() will clean up state: if the zone of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   324
		 * vnode's origin is still alive and kicking, an async worker
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   325
		 * thread will handle the request (from the correct zone), and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   326
		 * everything (minus the final nfs_getattr_otw() call) should
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   327
		 * be OK. If the zone is going away nfs_async_inactive() will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   328
		 * throw away cached pages inline.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   329
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   330
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   331
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   332
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   333
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   334
	 * If we are using local locking for this filesystem, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   335
	 * release all of the SYSV style record locks.  Otherwise,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   336
	 * we are doing network locking and we need to release all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   337
	 * of the network locks.  All of the locks held by this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   338
	 * process on this file are released no matter what the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   339
	 * incoming reference count is.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   340
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   341
	if (VTOMI(vp)->mi_flags & MI_LLOCK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   342
		cleanlocks(vp, ttoproc(curthread)->p_pid, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   343
		cleanshares(vp, ttoproc(curthread)->p_pid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   344
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   345
		nfs_lockrelease(vp, flag, offset, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   346
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   347
	if (count > 1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   348
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   349
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   350
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   351
	 * If the file has been `unlinked', then purge the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   352
	 * DNLC so that this vnode will get reycled quicker
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   353
	 * and the .nfs* file on the server will get removed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   354
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   355
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   356
	if (rp->r_unldvp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   357
		dnlc_purge_vp(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   358
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   359
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   360
	 * If the file was open for write and there are pages,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   361
	 * then if the file system was mounted using the "no-close-
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   362
	 *	to-open" semantics, then start an asynchronous flush
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   363
	 *	of the all of the pages in the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   364
	 * else the file system was not mounted using the "no-close-
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   365
	 *	to-open" semantics, then do a synchronous flush and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   366
	 *	commit of all of the dirty and uncommitted pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   367
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   368
	 * The asynchronous flush of the pages in the "nocto" path
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   369
	 * mostly just associates a cred pointer with the rnode so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   370
	 * writes which happen later will have a better chance of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   371
	 * working.  It also starts the data being written to the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   372
	 * server, but without unnecessarily delaying the application.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   373
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   374
	if ((flag & FWRITE) && vn_has_cached_data(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   375
		if ((VTOMI(vp)->mi_flags & MI_NOCTO)) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   376
			error = nfs_putpage(vp, (offset_t)0, 0, B_ASYNC,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   377
			    cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   378
			if (error == EAGAIN)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   379
				error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   380
		} else
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   381
			error = nfs_putpage(vp, (offset_t)0, 0, 0, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   382
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   383
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   384
			error = rp->r_error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   385
			rp->r_error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   386
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   387
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   388
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   389
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   390
		error = rp->r_error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   391
		rp->r_error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   392
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   393
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   394
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   395
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   396
	 * If RWRITEATTR is set, then issue an over the wire GETATTR to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   397
	 * refresh the attribute cache with a set of attributes which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   398
	 * weren't returned from a WRITE.  This will enable the close-
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   399
	 * to-open processing to work.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   400
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   401
	if (rp->r_flags & RWRITEATTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   402
		(void) nfs_getattr_otw(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   403
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   404
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   405
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   406
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   407
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   408
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   409
nfs_read(vnode_t *vp, struct uio *uiop, int ioflag, cred_t *cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   410
	caller_context_t *ct)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   411
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   412
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   413
	u_offset_t off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   414
	offset_t diff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   415
	int on;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   416
	size_t n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   417
	caddr_t base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   418
	uint_t flags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   419
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   420
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   421
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   422
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   423
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   424
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   425
	if (nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   426
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   427
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   428
	ASSERT(nfs_rw_lock_held(&rp->r_rwlock, RW_READER));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   429
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   430
	if (vp->v_type != VREG)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   431
		return (EISDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   432
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   433
	if (uiop->uio_resid == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   434
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   435
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   436
	if (uiop->uio_loffset > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   437
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   438
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   439
	if (uiop->uio_loffset < 0 ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   440
	    uiop->uio_loffset + uiop->uio_resid > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   441
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   442
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   443
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   444
	 * Bypass VM if caching has been disabled (e.g., locking) or if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   445
	 * using client-side direct I/O and the file is not mmap'd and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   446
	 * there are no cached pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   447
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   448
	if ((vp->v_flag & VNOCACHE) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   449
	    (((rp->r_flags & RDIRECTIO) || (mi->mi_flags & MI_DIRECTIO)) &&
7689
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
   450
	    rp->r_mapcnt == 0 && rp->r_inmap == 0 &&
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
   451
	    !vn_has_cached_data(vp))) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   452
		size_t bufsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   453
		size_t resid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   454
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   455
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   456
		 * Let's try to do read in as large a chunk as we can
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   457
		 * (Filesystem (NFS client) bsize if possible/needed).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   458
		 * For V3, this is 32K and for V2, this is 8K.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   459
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   460
		bufsize = MIN(uiop->uio_resid, VTOMI(vp)->mi_curread);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   461
		base = kmem_alloc(bufsize, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   462
		do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   463
			n = MIN(uiop->uio_resid, bufsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   464
			error = nfsread(vp, base, uiop->uio_offset, n,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   465
			    &resid, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   466
			if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   467
				n -= resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   468
				error = uiomove(base, n, UIO_READ, uiop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   469
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   470
		} while (!error && uiop->uio_resid > 0 && n > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   471
		kmem_free(base, bufsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   472
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   473
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   474
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   475
	error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   476
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   477
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   478
		off = uiop->uio_loffset & MAXBMASK; /* mapping offset */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   479
		on = uiop->uio_loffset & MAXBOFFSET; /* Relative offset */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   480
		n = MIN(MAXBSIZE - on, uiop->uio_resid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   481
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   482
		error = nfs_validate_caches(vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   483
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   484
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   485
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   486
		mutex_enter(&rp->r_statelock);
5486
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   487
		while (rp->r_flags & RINCACHEPURGE) {
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   488
			if (!cv_wait_sig(&rp->r_cv, &rp->r_statelock)) {
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   489
				mutex_exit(&rp->r_statelock);
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   490
				return (EINTR);
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   491
			}
72816cc23019 6599124 NFS client read() returns zero-filled data
vv149972
parents: 5331
diff changeset
   492
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   493
		diff = rp->r_size - uiop->uio_loffset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   494
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   495
		if (diff <= 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   496
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   497
		if (diff < n)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   498
			n = (size_t)diff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   499
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   500
		if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   501
			/*
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   502
			 * Copy data.
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   503
			 */
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   504
			error = vpm_data_copy(vp, off + on, n, uiop,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   505
			    1, NULL, 0, S_READ);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   506
		} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   507
			base = segmap_getmapflt(segkmap, vp, off + on, n,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   508
			    1, S_READ);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   509
			error = uiomove(base + on, n, UIO_READ, uiop);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   510
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   511
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   512
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   513
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   514
			 * If read a whole block or read to eof,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   515
			 * won't need this buffer again soon.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   516
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   517
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   518
			if (n + on == MAXBSIZE ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   519
			    uiop->uio_loffset == rp->r_size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   520
				flags = SM_DONTNEED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   521
			else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   522
				flags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   523
			mutex_exit(&rp->r_statelock);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   524
			if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   525
				error = vpm_sync_pages(vp, off, n, flags);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   526
			} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   527
				error = segmap_release(segkmap, base, flags);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   528
			}
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   529
		} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   530
			if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   531
				(void) vpm_sync_pages(vp, off, n, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   532
			} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   533
				(void) segmap_release(segkmap, base, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   534
			}
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   535
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   536
	} while (!error && uiop->uio_resid > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   537
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   538
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   539
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   540
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   541
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   542
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   543
nfs_write(vnode_t *vp, struct uio *uiop, int ioflag, cred_t *cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   544
	caller_context_t *ct)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   545
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   546
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   547
	u_offset_t off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   548
	caddr_t base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   549
	uint_t flags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   550
	int remainder;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   551
	size_t n;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   552
	int on;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   553
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   554
	int resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   555
	offset_t offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   556
	rlim_t limit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   557
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   558
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   559
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   560
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   561
	mi = VTOMI(vp);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   562
	if (nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   563
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   564
	if (vp->v_type != VREG)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   565
		return (EISDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   566
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   567
	if (uiop->uio_resid == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   568
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   569
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   570
	if (ioflag & FAPPEND) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   571
		struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   572
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   573
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   574
		 * Must serialize if appending.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   575
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   576
		if (nfs_rw_lock_held(&rp->r_rwlock, RW_READER)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   577
			nfs_rw_exit(&rp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   578
			if (nfs_rw_enter_sig(&rp->r_rwlock, RW_WRITER,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   579
			    INTR(vp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   580
				return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   581
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   582
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   583
		va.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   584
		error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   585
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   586
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   587
		uiop->uio_loffset = va.va_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   588
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   589
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   590
	if (uiop->uio_loffset > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   591
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   592
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   593
	offset = uiop->uio_loffset + uiop->uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   594
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   595
	if (uiop->uio_loffset < 0 || offset > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   596
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   597
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   598
	if (uiop->uio_llimit > (rlim64_t)MAXOFF32_T) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   599
		limit = MAXOFF32_T;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   600
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   601
		limit = (rlim_t)uiop->uio_llimit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   602
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   603
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   604
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   605
	 * Check to make sure that the process will not exceed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   606
	 * its limit on file size.  It is okay to write up to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   607
	 * the limit, but not beyond.  Thus, the write which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   608
	 * reaches the limit will be short and the next write
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   609
	 * will return an error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   610
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   611
	remainder = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   612
	if (offset > limit) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   613
		remainder = offset - limit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   614
		uiop->uio_resid = limit - uiop->uio_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   615
		if (uiop->uio_resid <= 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   616
			proc_t *p = ttoproc(curthread);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   617
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   618
			uiop->uio_resid += remainder;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   619
			mutex_enter(&p->p_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   620
			(void) rctl_action(rctlproc_legacy[RLIMIT_FSIZE],
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   621
			    p->p_rctls, p, RCA_UNSAFE_SIGINFO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   622
			mutex_exit(&p->p_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   623
			return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   624
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   625
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   626
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   627
	if (nfs_rw_enter_sig(&rp->r_lkserlock, RW_READER, INTR(vp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   628
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   629
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   630
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   631
	 * Bypass VM if caching has been disabled (e.g., locking) or if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   632
	 * using client-side direct I/O and the file is not mmap'd and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   633
	 * there are no cached pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   634
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   635
	if ((vp->v_flag & VNOCACHE) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   636
	    (((rp->r_flags & RDIRECTIO) || (mi->mi_flags & MI_DIRECTIO)) &&
7689
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
   637
	    rp->r_mapcnt == 0 && rp->r_inmap == 0 &&
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
   638
	    !vn_has_cached_data(vp))) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   639
		size_t bufsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   640
		int count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   641
		uint_t org_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   642
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   643
nfs_fwrite:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   644
		if (rp->r_flags & RSTALE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   645
			resid = uiop->uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   646
			offset = uiop->uio_loffset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   647
			error = rp->r_error;
9776
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   648
			/*
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   649
			 * A close may have cleared r_error, if so,
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   650
			 * propagate ESTALE error return properly
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   651
			 */
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   652
			if (error == 0)
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   653
				error = ESTALE;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   654
			goto bottom;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   655
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   656
		bufsize = MIN(uiop->uio_resid, mi->mi_curwrite);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   657
		base = kmem_alloc(bufsize, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   658
		do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   659
			resid = uiop->uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   660
			offset = uiop->uio_loffset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   661
			count = MIN(uiop->uio_resid, bufsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   662
			org_offset = uiop->uio_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   663
			error = uiomove(base, count, UIO_WRITE, uiop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   664
			if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   665
				error = nfswrite(vp, base, org_offset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   666
				    count, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   667
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   668
		} while (!error && uiop->uio_resid > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   669
		kmem_free(base, bufsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   670
		goto bottom;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   671
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   672
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   673
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   674
		off = uiop->uio_loffset & MAXBMASK; /* mapping offset */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   675
		on = uiop->uio_loffset & MAXBOFFSET; /* Relative offset */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   676
		n = MIN(MAXBSIZE - on, uiop->uio_resid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   677
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   678
		resid = uiop->uio_resid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   679
		offset = uiop->uio_loffset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   680
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   681
		if (rp->r_flags & RSTALE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   682
			error = rp->r_error;
9776
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   683
			/*
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   684
			 * A close may have cleared r_error, if so,
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   685
			 * propagate ESTALE error return properly
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   686
			 */
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   687
			if (error == 0)
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
   688
				error = ESTALE;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   689
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   690
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   691
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   692
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   693
		 * Don't create dirty pages faster than they
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   694
		 * can be cleaned so that the system doesn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   695
		 * get imbalanced.  If the async queue is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   696
		 * maxed out, then wait for it to drain before
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   697
		 * creating more dirty pages.  Also, wait for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   698
		 * any threads doing pagewalks in the vop_getattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   699
		 * entry points so that they don't block for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   700
		 * long periods.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   701
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   702
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   703
		while ((mi->mi_max_threads != 0 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   704
		    rp->r_awcount > 2 * mi->mi_max_threads) ||
10847
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   705
		    rp->r_gcount > 0) {
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   706
			if (INTR(vp)) {
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   707
				klwp_t *lwp = ttolwp(curthread);
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   708
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   709
				if (lwp != NULL)
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   710
					lwp->lwp_nostop++;
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   711
				if (!cv_wait_sig(&rp->r_cv, &rp->r_statelock)) {
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   712
					mutex_exit(&rp->r_statelock);
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   713
					if (lwp != NULL)
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   714
						lwp->lwp_nostop--;
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   715
					error = EINTR;
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   716
					goto bottom;
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   717
				}
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   718
				if (lwp != NULL)
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   719
					lwp->lwp_nostop--;
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   720
			} else
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   721
				cv_wait(&rp->r_cv, &rp->r_statelock);
37439331af3e 6577647 NFS writes aren't interruptible if there are too many outstanding writes
Marcel Telka <Marcel.Telka@Sun.COM>
parents: 9776
diff changeset
   722
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   723
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   724
8059
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   725
		/*
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   726
		 * Touch the page and fault it in if it is not in core
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   727
		 * before segmap_getmapflt or vpm_data_copy can lock it.
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   728
		 * This is to avoid the deadlock if the buffer is mapped
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   729
		 * to the same file through mmap which we want to write.
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   730
		 */
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   731
		uio_prefaultpages((long)n, uiop);
92e2183c5863 1246893 mmap and write to the same file deadlocks.
Donghai Qiao <Donghai.Qiao@Sun.COM>
parents: 7689
diff changeset
   732
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   733
		if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   734
			/*
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   735
			 * It will use kpm mappings, so no need to
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   736
			 * pass an address.
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   737
			 */
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   738
			error = writerp(rp, NULL, n, uiop, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   739
		} else  {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   740
			if (segmap_kpm) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   741
				int pon = uiop->uio_loffset & PAGEOFFSET;
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   742
				size_t pn = MIN(PAGESIZE - pon,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   743
				    uiop->uio_resid);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   744
				int pagecreate;
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   745
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   746
				mutex_enter(&rp->r_statelock);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   747
				pagecreate = (pon == 0) && (pn == PAGESIZE ||
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   748
				    uiop->uio_loffset + pn >= rp->r_size);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   749
				mutex_exit(&rp->r_statelock);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   750
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   751
				base = segmap_getmapflt(segkmap, vp, off + on,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   752
				    pn, !pagecreate, S_WRITE);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   753
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   754
				error = writerp(rp, base + pon, n, uiop,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   755
				    pagecreate);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   756
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   757
			} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   758
				base = segmap_getmapflt(segkmap, vp, off + on,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   759
				    n, 0, S_READ);
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   760
				error = writerp(rp, base + on, n, uiop, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   761
			}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   762
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   763
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   764
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   765
			if (mi->mi_flags & MI_NOAC)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   766
				flags = SM_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   767
			else if (n + on == MAXBSIZE || IS_SWAPVP(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   768
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   769
				 * Have written a whole block.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   770
				 * Start an asynchronous write
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   771
				 * and mark the buffer to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   772
				 * indicate that it won't be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   773
				 * needed again soon.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   774
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   775
				flags = SM_WRITE | SM_ASYNC | SM_DONTNEED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   776
			} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   777
				flags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   778
			if ((ioflag & (FSYNC|FDSYNC)) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   779
			    (rp->r_flags & ROUTOFSPACE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   780
				flags &= ~SM_ASYNC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   781
				flags |= SM_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   782
			}
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   783
			if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   784
				error = vpm_sync_pages(vp, off, n, flags);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   785
			} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   786
				error = segmap_release(segkmap, base, flags);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   787
			}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   788
		} else {
1841
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   789
			if (vpm_enable) {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   790
				(void) vpm_sync_pages(vp, off, n, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   791
			} else {
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   792
				(void) segmap_release(segkmap, base, 0);
9d7ebafcda38 6256083 Need a lightweight file page mapping mechanism to substitute segmap
praks
parents: 766
diff changeset
   793
			}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   794
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   795
			 * In the event that we got an access error while
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   796
			 * faulting in a page for a write-only file just
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   797
			 * force a write.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   798
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   799
			if (error == EACCES)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   800
				goto nfs_fwrite;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   801
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   802
	} while (!error && uiop->uio_resid > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   803
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   804
bottom:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   805
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   806
		uiop->uio_resid = resid + remainder;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   807
		uiop->uio_loffset = offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   808
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   809
		uiop->uio_resid += remainder;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   810
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   811
	nfs_rw_exit(&rp->r_lkserlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   812
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   813
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   814
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   815
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   816
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   817
 * Flags are composed of {B_ASYNC, B_INVAL, B_FREE, B_DONTNEED}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   818
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   819
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   820
nfs_rdwrlbn(vnode_t *vp, page_t *pp, u_offset_t off, size_t len,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   821
	int flags, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   822
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   823
	struct buf *bp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   824
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   825
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   826
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   827
	bp = pageio_setup(pp, len, vp, flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   828
	ASSERT(bp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   829
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   830
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   831
	 * pageio_setup should have set b_addr to 0.  This
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   832
	 * is correct since we want to do I/O on a page
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   833
	 * boundary.  bp_mapin will use this addr to calculate
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   834
	 * an offset, and then set b_addr to the kernel virtual
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   835
	 * address it allocated for us.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   836
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   837
	ASSERT(bp->b_un.b_addr == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   838
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   839
	bp->b_edev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   840
	bp->b_dev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   841
	bp->b_lblkno = lbtodb(off);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   842
	bp->b_file = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   843
	bp->b_offset = (offset_t)off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   844
	bp_mapin(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   845
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   846
	error = nfs_bio(bp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   847
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   848
	bp_mapout(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   849
	pageio_done(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   850
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   851
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   852
}
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
 * Write to file.  Writes to remote server in largest size
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   856
 * chunks that the server can handle.  Write is synchronous.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   857
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   858
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   859
nfswrite(vnode_t *vp, caddr_t base, uint_t offset, int count, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   860
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   861
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   862
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   863
	struct nfswriteargs wa;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   864
	struct nfsattrstat ns;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   865
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   866
	int tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   867
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   868
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   869
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   870
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   871
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   872
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   873
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   874
	ASSERT(nfs_zone() == mi->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   875
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   876
	wa.wa_args = &wa.wa_args_buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   877
	wa.wa_fhandle = *VTOFH(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   878
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   879
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   880
		tsize = MIN(mi->mi_curwrite, count);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   881
		wa.wa_data = base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   882
		wa.wa_begoff = offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   883
		wa.wa_totcount = tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   884
		wa.wa_count = tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   885
		wa.wa_offset = offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   886
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   887
		if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   888
			mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   889
			kstat_runq_enter(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   890
			mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   891
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   892
		wa.wa_mblk = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   893
		do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   894
			error = rfs2call(mi, RFS_WRITE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   895
			    xdr_writeargs, (caddr_t)&wa,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   896
			    xdr_attrstat, (caddr_t)&ns, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   897
			    &douprintf, &ns.ns_status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   898
		} while (error == ENFS_TRYAGAIN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   899
		if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   900
			mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   901
			kstat_runq_exit(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   902
			mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   903
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   904
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   905
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   906
			error = geterrno(ns.ns_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   907
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   908
			 * Can't check for stale fhandle and purge caches
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   909
			 * here because pages are held by nfs_getpage.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   910
			 * Just mark the attribute cache as timed out
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   911
			 * and set RWRITEATTR to indicate that the file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   912
			 * was modified with a WRITE operation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   913
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   914
			if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   915
				count -= tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   916
				base += tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   917
				offset += tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   918
				if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   919
					mutex_enter(&mi->mi_lock);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   920
					KSTAT_IO_PTR(mi->mi_io_kstats)->
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   921
					    writes++;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   922
					KSTAT_IO_PTR(mi->mi_io_kstats)->
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
   923
					    nwritten += tsize;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   924
					mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   925
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   926
				lwp_stat_update(LWP_STAT_OUBLK, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   927
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   928
				PURGE_ATTRCACHE_LOCKED(rp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   929
				rp->r_flags |= RWRITEATTR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   930
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   931
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   932
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   933
	} while (!error && count);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   934
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   935
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   936
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   937
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   938
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   939
 * Read from a file.  Reads data in largest chunks our interface can handle.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   940
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   941
static int
7387
0b3a92e31fd8 PSARC 2007/347 NFS/RDMA - Transport Version Update
Robert Gordon <Robert.Gordon@Sun.COM>
parents: 7067
diff changeset
   942
nfsread(vnode_t *vp, caddr_t base, uint_t offset,
0b3a92e31fd8 PSARC 2007/347 NFS/RDMA - Transport Version Update
Robert Gordon <Robert.Gordon@Sun.COM>
parents: 7067
diff changeset
   943
    int count, size_t *residp, cred_t *cr)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   944
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   945
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   946
	struct nfsreadargs ra;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   947
	struct nfsrdresult rr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   948
	int tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   949
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   950
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   951
	failinfo_t fi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   952
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   953
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   954
	hrtime_t t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   955
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   956
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   957
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   958
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
   959
	ASSERT(nfs_zone() == mi->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   960
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   961
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   962
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   963
	ra.ra_fhandle = *VTOFH(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   964
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   965
	fi.vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   966
	fi.fhp = (caddr_t)&ra.ra_fhandle;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   967
	fi.copyproc = nfscopyfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   968
	fi.lookupproc = nfslookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   969
	fi.xattrdirproc = acl_getxattrdir2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   970
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   971
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   972
		if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   973
			mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   974
			kstat_runq_enter(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   975
			mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   976
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   977
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   978
		do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   979
			tsize = MIN(mi->mi_curread, count);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   980
			rr.rr_data = base;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   981
			ra.ra_offset = offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   982
			ra.ra_totcount = tsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   983
			ra.ra_count = tsize;
7387
0b3a92e31fd8 PSARC 2007/347 NFS/RDMA - Transport Version Update
Robert Gordon <Robert.Gordon@Sun.COM>
parents: 7067
diff changeset
   984
			ra.ra_data = base;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   985
			t = gethrtime();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   986
			error = rfs2call(mi, RFS_READ,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   987
			    xdr_readargs, (caddr_t)&ra,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   988
			    xdr_rdresult, (caddr_t)&rr, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   989
			    &douprintf, &rr.rr_status, 0, &fi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   990
		} while (error == ENFS_TRYAGAIN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   991
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   992
		if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   993
			mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   994
			kstat_runq_exit(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   995
			mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   996
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   997
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   998
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   999
			error = geterrno(rr.rr_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1000
			if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1001
				count -= rr.rr_count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1002
				base += rr.rr_count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1003
				offset += rr.rr_count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1004
				if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1005
					mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1006
					KSTAT_IO_PTR(mi->mi_io_kstats)->reads++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1007
					KSTAT_IO_PTR(mi->mi_io_kstats)->nread +=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1008
					    rr.rr_count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1009
					mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1010
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1011
				lwp_stat_update(LWP_STAT_INBLK, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1012
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1013
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1014
	} while (!error && count && rr.rr_count == tsize);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1015
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1016
	*residp = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1017
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1018
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1019
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1020
		 * Since no error occurred, we have the current
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1021
		 * attributes and we need to do a cache check and then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1022
		 * potentially update the cached attributes.  We can't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1023
		 * use the normal attribute check and cache mechanisms
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1024
		 * because they might cause a cache flush which would
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1025
		 * deadlock.  Instead, we just check the cache to see
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1026
		 * if the attributes have changed.  If it is, then we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1027
		 * just mark the attributes as out of date.  The next
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1028
		 * time that the attributes are checked, they will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1029
		 * out of date, new attributes will be fetched, and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1030
		 * the page cache will be flushed.  If the attributes
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1031
		 * weren't changed, then we just update the cached
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1032
		 * attributes with these attributes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1033
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1034
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1035
		 * If NFS_ACL is supported on the server, then the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1036
		 * attributes returned by server may have minimal
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1037
		 * permissions sometimes denying access to users having
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1038
		 * proper access.  To get the proper attributes, mark
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1039
		 * the attributes as expired so that they will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1040
		 * regotten via the NFS_ACL GETATTR2 procedure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1041
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1042
		error = nattr_to_vattr(vp, &rr.rr_attr, &va);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1043
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1044
		if (error || !CACHE_VALID(rp, va.va_mtime, va.va_size) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1045
		    (mi->mi_flags & MI_ACL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1046
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1047
			PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1048
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1049
			if (rp->r_mtime <= t) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1050
				nfs_attrcache_va(vp, &va);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1051
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1052
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1053
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1054
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1055
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1056
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1057
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1058
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1059
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1060
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1061
nfs_ioctl(vnode_t *vp, int cmd, intptr_t arg, int flag, cred_t *cr, int *rvalp,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1062
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1063
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1064
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1065
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1066
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1067
	switch (cmd) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1068
		case _FIODIRECTIO:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1069
			return (nfs_directio(vp, (int)arg, cr));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1070
		default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1071
			return (ENOTTY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1072
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1073
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1074
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1075
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1076
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1077
nfs_getattr(vnode_t *vp, struct vattr *vap, int flags, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1078
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1079
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1080
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1081
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1082
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1083
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1084
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1085
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1086
	 * If it has been specified that the return value will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1087
	 * just be used as a hint, and we are only being asked
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1088
	 * for size, fsid or rdevid, then return the client's
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1089
	 * notion of these values without checking to make sure
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1090
	 * that the attribute cache is up to date.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1091
	 * The whole point is to avoid an over the wire GETATTR
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1092
	 * call.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1093
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1094
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1095
	if (flags & ATTR_HINT) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1096
		if (vap->va_mask ==
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1097
		    (vap->va_mask & (AT_SIZE | AT_FSID | AT_RDEV))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1098
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1099
			if (vap->va_mask | AT_SIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1100
				vap->va_size = rp->r_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1101
			if (vap->va_mask | AT_FSID)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1102
				vap->va_fsid = rp->r_attr.va_fsid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1103
			if (vap->va_mask | AT_RDEV)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1104
				vap->va_rdev = rp->r_attr.va_rdev;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1105
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1106
			return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1107
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1108
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1109
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1110
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1111
	 * Only need to flush pages if asking for the mtime
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1112
	 * and if there any dirty pages or any outstanding
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1113
	 * asynchronous (write) requests for this file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1114
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1115
	if (vap->va_mask & AT_MTIME) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1116
		if (vn_has_cached_data(vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1117
		    ((rp->r_flags & RDIRTY) || rp->r_awcount > 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1118
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1119
			rp->r_gcount++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1120
			mutex_exit(&rp->r_statelock);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1121
			error = nfs_putpage(vp, (offset_t)0, 0, 0, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1122
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1123
			if (error && (error == ENOSPC || error == EDQUOT)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1124
				if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1125
					rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1126
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1127
			if (--rp->r_gcount == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1128
				cv_broadcast(&rp->r_cv);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1129
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1130
		}
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
	return (nfsgetattr(vp, vap, cr));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1134
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1135
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1136
/*ARGSUSED4*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1137
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1138
nfs_setattr(vnode_t *vp, struct vattr *vap, int flags, cred_t *cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1139
		caller_context_t *ct)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1140
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1141
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1142
	uint_t mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1143
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1144
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1145
	mask = vap->va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1146
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1147
	if (mask & AT_NOSET)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1148
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1149
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1150
	if ((mask & AT_SIZE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1151
	    vap->va_type == VREG &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1152
	    vap->va_size > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1153
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1154
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1155
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1156
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1157
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1158
	va.va_mask = AT_UID | AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1159
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1160
	error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1161
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1162
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1163
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1164
	error = secpolicy_vnode_setattr(cr, vp, vap, &va, flags, nfs_accessx,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1165
	    vp);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1166
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1167
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1168
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1169
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1170
	return (nfssetattr(vp, vap, flags, cr));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1171
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1172
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1173
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1174
nfssetattr(vnode_t *vp, struct vattr *vap, int flags, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1175
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1176
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1177
	uint_t mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1178
	struct nfssaargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1179
	struct nfsattrstat ns;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1180
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1181
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1182
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1183
	mode_t omode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1184
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1185
	vsecattr_t *vsp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1186
	hrtime_t t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1187
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1188
	mask = vap->va_mask;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1189
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1190
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1191
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1192
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1193
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1194
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1195
	 * Only need to flush pages if there are any pages and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1196
	 * if the file is marked as dirty in some fashion.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1197
	 * file must be flushed so that we can accurately
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1198
	 * determine the size of the file and the cached data
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1199
	 * after the SETATTR returns.  A file is considered to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1200
	 * be dirty if it is either marked with RDIRTY, has
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1201
	 * outstanding i/o's active, or is mmap'd.  In this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1202
	 * last case, we can't tell whether there are dirty
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1203
	 * pages, so we flush just to be sure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1204
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1205
	if (vn_has_cached_data(vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1206
	    ((rp->r_flags & RDIRTY) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1207
	    rp->r_count > 0 ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1208
	    rp->r_mapcnt > 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1209
		ASSERT(vp->v_type != VCHR);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1210
		error = nfs_putpage(vp, (offset_t)0, 0, 0, cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1211
		if (error && (error == ENOSPC || error == EDQUOT)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1212
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1213
			if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1214
				rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1215
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1216
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1217
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1218
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1219
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1220
	 * If the system call was utime(2) or utimes(2) and the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1221
	 * application did not specify the times, then set the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1222
	 * mtime nanosecond field to 1 billion.  This will get
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1223
	 * translated from 1 billion nanoseconds to 1 million
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1224
	 * microseconds in the over the wire request.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1225
	 * server will use 1 million in the microsecond field
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1226
	 * to tell whether both the mtime and atime should be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1227
	 * set to the server's current time.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1228
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1229
	 * This is an overload of the protocol and should be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1230
	 * documented in the NFS Version 2 protocol specification.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1231
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1232
	if ((mask & AT_MTIME) && !(flags & ATTR_UTIME)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1233
		vap->va_mtime.tv_nsec = 1000000000;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1234
		if (NFS_TIME_T_OK(vap->va_mtime.tv_sec) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1235
		    NFS_TIME_T_OK(vap->va_atime.tv_sec)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1236
			error = vattr_to_sattr(vap, &args.saa_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1237
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1238
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1239
			 * Use server times. vap time values will not be used.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1240
			 * To ensure no time overflow, make sure vap has
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1241
			 * valid values, but retain the original values.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1242
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1243
			timestruc_t	mtime = vap->va_mtime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1244
			timestruc_t	atime = vap->va_atime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1245
			time_t		now;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1246
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1247
			now = gethrestime_sec();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1248
			if (NFS_TIME_T_OK(now)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1249
				/* Just in case server does not know of this */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1250
				vap->va_mtime.tv_sec = now;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1251
				vap->va_atime.tv_sec = now;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1252
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1253
				vap->va_mtime.tv_sec = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1254
				vap->va_atime.tv_sec = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1255
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1256
			error = vattr_to_sattr(vap, &args.saa_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1257
			/* set vap times back on */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1258
			vap->va_mtime = mtime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1259
			vap->va_atime = atime;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1260
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1261
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1262
		/* Either do not set times or use the client specified times */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1263
		error = vattr_to_sattr(vap, &args.saa_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1264
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1265
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1266
		/* req time field(s) overflow - return immediately */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1267
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1268
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1269
	args.saa_fh = *VTOFH(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1270
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1271
	va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1272
	error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1273
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1274
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1275
	omode = va.va_mode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1276
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1277
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1278
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1279
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1280
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1281
	t = gethrtime();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1282
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1283
	error = rfs2call(mi, RFS_SETATTR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1284
	    xdr_saargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1285
	    xdr_attrstat, (caddr_t)&ns, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1286
	    &douprintf, &ns.ns_status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1287
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1288
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1289
	 * Purge the access cache and ACL cache if changing either the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1290
	 * owner of the file, the group owner, or the mode.  These may
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1291
	 * change the access permissions of the file, so purge old
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1292
	 * information and start over again.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1293
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1294
	if ((mask & (AT_UID | AT_GID | AT_MODE)) && (mi->mi_flags & MI_ACL)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1295
		(void) nfs_access_purge_rp(rp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1296
		if (rp->r_secattr != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1297
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1298
			vsp = rp->r_secattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1299
			rp->r_secattr = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1300
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1301
			if (vsp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1302
				nfs_acl_free(vsp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1303
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1304
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1306
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1307
		error = geterrno(ns.ns_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1308
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1309
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1310
			 * If changing the size of the file, invalidate
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1311
			 * any local cached data which is no longer part
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1312
			 * of the file.  We also possibly invalidate the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1313
			 * last page in the file.  We could use
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1314
			 * pvn_vpzero(), but this would mark the page as
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1315
			 * modified and require it to be written back to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1316
			 * the server for no particularly good reason.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1317
			 * This way, if we access it, then we bring it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1318
			 * back in.  A read should be cheaper than a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1319
			 * write.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1320
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1321
			if (mask & AT_SIZE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1322
				nfs_invalidate_pages(vp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1323
				    (vap->va_size & PAGEMASK), cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1324
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1325
			(void) nfs_cache_fattr(vp, &ns.ns_attr, &va, t, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1326
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1327
			 * If NFS_ACL is supported on the server, then the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1328
			 * attributes returned by server may have minimal
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1329
			 * permissions sometimes denying access to users having
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1330
			 * proper access.  To get the proper attributes, mark
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1331
			 * the attributes as expired so that they will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1332
			 * regotten via the NFS_ACL GETATTR2 procedure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1333
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1334
			if (mi->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1335
				PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1336
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1337
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1338
			 * This next check attempts to deal with NFS
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1339
			 * servers which can not handle increasing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1340
			 * the size of the file via setattr.  Most
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1341
			 * of these servers do not return an error,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1342
			 * but do not change the size of the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1343
			 * Hence, this check and then attempt to set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1344
			 * the file size by writing 1 byte at the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1345
			 * offset of the end of the file that we need.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1346
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1347
			if ((mask & AT_SIZE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1348
			    ns.ns_attr.na_size < (uint32_t)vap->va_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1349
				char zb = '\0';
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1350
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1351
				error = nfswrite(vp, &zb,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1352
				    vap->va_size - sizeof (zb),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1353
				    sizeof (zb), cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1354
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1355
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1356
			 * Some servers will change the mode to clear the setuid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1357
			 * and setgid bits when changing the uid or gid.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1358
			 * client needs to compensate appropriately.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1359
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1360
			if (mask & (AT_UID | AT_GID)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1361
				int terror;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1362
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1363
				va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1364
				terror = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1365
				if (!terror &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1366
				    (((mask & AT_MODE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1367
				    va.va_mode != vap->va_mode) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1368
				    (!(mask & AT_MODE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1369
				    va.va_mode != omode))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1370
					va.va_mask = AT_MODE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1371
					if (mask & AT_MODE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1372
						va.va_mode = vap->va_mode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1373
					else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1374
						va.va_mode = omode;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1375
					(void) nfssetattr(vp, &va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1376
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1377
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1378
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1379
			PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1380
			PURGE_STALE_FH(error, vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1381
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1382
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1383
		PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1384
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1385
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1386
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1387
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1388
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1389
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1390
nfs_accessx(void *vp, int mode, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1391
{
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1392
	ASSERT(nfs_zone() == VTOMI((vnode_t *)vp)->mi_zone);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1393
	return (nfs_access(vp, mode, 0, cr, NULL));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1394
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1395
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1396
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1397
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1398
nfs_access(vnode_t *vp, int mode, int flags, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1399
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1400
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1401
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1402
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1403
	int shift = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1404
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1405
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1406
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1407
	if (nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1408
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1409
	if (mi->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1410
		error = acl_access2(vp, mode, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1411
		if (mi->mi_flags & MI_ACL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1412
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1413
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1414
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1415
	va.va_mask = AT_MODE | AT_UID | AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1416
	error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1417
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1418
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1419
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1420
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1421
	 * Disallow write attempts on read-only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1422
	 * file systems, unless the file is a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1423
	 * device node.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1424
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1425
	if ((mode & VWRITE) && vn_is_readonly(vp) && !IS_DEVVP(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1426
		return (EROFS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1427
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1428
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1429
	 * Disallow attempts to access mandatory lock files.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1430
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1431
	if ((mode & (VWRITE | VREAD | VEXEC)) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1432
	    MANDLOCK(vp, va.va_mode))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1433
		return (EACCES);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1434
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1435
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1436
	 * Access check is based on only
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1437
	 * one of owner, group, public.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1438
	 * If not owner, then check group.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1439
	 * If not a member of the group,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1440
	 * then check public access.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1441
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1442
	if (crgetuid(cr) != va.va_uid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1443
		shift += 3;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1444
		if (!groupmember(va.va_gid, cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1445
			shift += 3;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1446
	}
12273
63678502e95e PSARC 2009/377 In-kernel pfexec implementation.
Casper H.S. Dik <Casper.Dik@Sun.COM>
parents: 10847
diff changeset
  1447
63678502e95e PSARC 2009/377 In-kernel pfexec implementation.
Casper H.S. Dik <Casper.Dik@Sun.COM>
parents: 10847
diff changeset
  1448
	return (secpolicy_vnode_access2(cr, vp, va.va_uid,
63678502e95e PSARC 2009/377 In-kernel pfexec implementation.
Casper H.S. Dik <Casper.Dik@Sun.COM>
parents: 10847
diff changeset
  1449
	    va.va_mode << shift, mode));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1450
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1451
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1452
static int nfs_do_symlink_cache = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1453
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1454
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1455
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1456
nfs_readlink(vnode_t *vp, struct uio *uiop, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1457
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1458
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1459
	struct nfsrdlnres rl;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1460
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1461
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1462
	failinfo_t fi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1463
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1464
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1465
	 * We want to be consistent with UFS semantics so we will return
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1466
	 * EINVAL instead of ENXIO. This violates the XNFS spec and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1467
	 * the RFC 1094, which are wrong any way. BUGID 1138002.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1468
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1469
	if (vp->v_type != VLNK)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1470
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1471
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1472
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1473
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1474
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1475
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1476
	if (nfs_do_symlink_cache && rp->r_symlink.contents != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1477
		error = nfs_validate_caches(vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1478
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1479
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1480
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1481
		if (rp->r_symlink.contents != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1482
			error = uiomove(rp->r_symlink.contents,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1483
			    rp->r_symlink.len, UIO_READ, uiop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1484
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1485
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1486
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1487
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1488
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1489
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1490
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1491
	rl.rl_data = kmem_alloc(NFS_MAXPATHLEN, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1492
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1493
	fi.vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1494
	fi.fhp = NULL;		/* no need to update, filehandle not copied */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1495
	fi.copyproc = nfscopyfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1496
	fi.lookupproc = nfslookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1497
	fi.xattrdirproc = acl_getxattrdir2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1498
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1499
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1500
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1501
	error = rfs2call(VTOMI(vp), RFS_READLINK,
7387
0b3a92e31fd8 PSARC 2007/347 NFS/RDMA - Transport Version Update
Robert Gordon <Robert.Gordon@Sun.COM>
parents: 7067
diff changeset
  1502
	    xdr_readlink, (caddr_t)VTOFH(vp),
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1503
	    xdr_rdlnres, (caddr_t)&rl, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1504
	    &douprintf, &rl.rl_status, 0, &fi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1505
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1506
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1507
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1508
		kmem_free((void *)rl.rl_data, NFS_MAXPATHLEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1509
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1510
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1511
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1512
	error = geterrno(rl.rl_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1513
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1514
		error = uiomove(rl.rl_data, (int)rl.rl_count, UIO_READ, uiop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1515
		if (nfs_do_symlink_cache && rp->r_symlink.contents == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1516
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1517
			if (rp->r_symlink.contents == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1518
				rp->r_symlink.contents = rl.rl_data;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1519
				rp->r_symlink.len = (int)rl.rl_count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1520
				rp->r_symlink.size = NFS_MAXPATHLEN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1521
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1522
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1523
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1524
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1525
				kmem_free((void *)rl.rl_data,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1526
				    NFS_MAXPATHLEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1527
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1528
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1529
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1530
			kmem_free((void *)rl.rl_data, NFS_MAXPATHLEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1531
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1532
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1533
		PURGE_STALE_FH(error, vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1534
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1535
		kmem_free((void *)rl.rl_data, NFS_MAXPATHLEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1536
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1537
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1538
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1539
	 * Conform to UFS semantics (see comment above)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1540
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1541
	return (error == ENXIO ? EINVAL : error);
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
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1545
 * Flush local dirty pages to stable storage on the server.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1546
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1547
 * If FNODSYNC is specified, then there is nothing to do because
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1548
 * metadata changes are not cached on the client before being
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1549
 * sent to the server.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1550
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1551
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1552
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1553
nfs_fsync(vnode_t *vp, int syncflag, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1554
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1555
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1556
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1557
	if ((syncflag & FNODSYNC) || IS_SWAPVP(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1558
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1559
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1560
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1561
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1562
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1563
	error = nfs_putpage(vp, (offset_t)0, 0, 0, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1564
	if (!error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1565
		error = VTOR(vp)->r_error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1566
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1567
}
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
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1571
 * Weirdness: if the file was removed or the target of a rename
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1572
 * operation while it was open, it got renamed instead.  Here we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1573
 * remove the renamed file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1574
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1575
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1576
static void
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1577
nfs_inactive(vnode_t *vp, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1578
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1579
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1580
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1581
	ASSERT(vp != DNLC_NO_VNODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1582
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1583
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1584
	 * If this is coming from the wrong zone, we let someone in the right
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1585
	 * zone take care of it asynchronously.  We can get here due to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1586
	 * VN_RELE() being called from pageout() or fsflush().  This call may
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1587
	 * potentially turn into an expensive no-op if, for instance, v_count
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1588
	 * gets incremented in the meantime, but it's still correct.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1589
	 */
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1590
	if (nfs_zone() != VTOMI(vp)->mi_zone) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1591
		nfs_async_inactive(vp, cr, nfs_inactive);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1592
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1593
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1594
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1595
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1596
redo:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1597
	if (rp->r_unldvp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1598
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1599
		 * Save the vnode pointer for the directory where the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1600
		 * unlinked-open file got renamed, then set it to NULL
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1601
		 * to prevent another thread from getting here before
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1602
		 * we're done with the remove.  While we have the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1603
		 * statelock, make local copies of the pertinent rnode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1604
		 * fields.  If we weren't to do this in an atomic way, the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1605
		 * the unl* fields could become inconsistent with respect
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1606
		 * to each other due to a race condition between this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1607
		 * code and nfs_remove().  See bug report 1034328.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1608
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1609
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1610
		if (rp->r_unldvp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1611
			vnode_t *unldvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1612
			char *unlname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1613
			cred_t *unlcred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1614
			struct nfsdiropargs da;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1615
			enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1616
			int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1617
			int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1618
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1619
			unldvp = rp->r_unldvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1620
			rp->r_unldvp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1621
			unlname = rp->r_unlname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1622
			rp->r_unlname = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1623
			unlcred = rp->r_unlcred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1624
			rp->r_unlcred = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1625
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1626
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1627
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1628
			 * If there are any dirty pages left, then flush
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1629
			 * them.  This is unfortunate because they just
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1630
			 * may get thrown away during the remove operation,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1631
			 * but we have to do this for correctness.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1632
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1633
			if (vn_has_cached_data(vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1634
			    ((rp->r_flags & RDIRTY) || rp->r_count > 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1635
				ASSERT(vp->v_type != VCHR);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1636
				error = nfs_putpage(vp, (offset_t)0, 0, 0,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1637
				    cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1638
				if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1639
					mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1640
					if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1641
						rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1642
					mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1643
				}
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
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1647
			 * Do the remove operation on the renamed file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1648
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1649
			setdiropargs(&da, unlname, unldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1650
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1651
			douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1652
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1653
			(void) rfs2call(VTOMI(unldvp), RFS_REMOVE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1654
			    xdr_diropargs, (caddr_t)&da,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1655
			    xdr_enum, (caddr_t)&status, unlcred,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1656
			    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1657
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1658
			if (HAVE_RDDIR_CACHE(VTOR(unldvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1659
				nfs_purge_rddir_cache(unldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1660
			PURGE_ATTRCACHE(unldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1661
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1662
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1663
			 * Release stuff held for the remove
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1664
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1665
			VN_RELE(unldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1666
			kmem_free(unlname, MAXNAMELEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1667
			crfree(unlcred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1668
			goto redo;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1669
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1670
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1671
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1672
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1673
	rp_addfree(rp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1674
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1675
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1676
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1677
 * Remote file system operations having to do with directory manipulation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1678
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1679
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1680
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1681
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1682
nfs_lookup(vnode_t *dvp, char *nm, vnode_t **vpp, struct pathname *pnp,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1683
	int flags, vnode_t *rdir, cred_t *cr, caller_context_t *ct,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1684
	int *direntflags, pathname_t *realpnp)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1685
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1686
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1687
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1688
	vnode_t *avp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1689
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1690
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1691
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1692
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1693
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1694
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1695
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1696
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1697
	 * Are we looking up extended attributes?  If so, "dvp" is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1698
	 * the file or directory for which we want attributes, and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1699
	 * we need a lookup of the hidden attribute directory
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1700
	 * before we lookup the rest of the path.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1701
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1702
	if (flags & LOOKUP_XATTR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1703
		bool_t cflag = ((flags & CREATE_XATTR_DIR) != 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1704
		mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1705
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1706
		mi = VTOMI(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1707
		if (!(mi->mi_flags & MI_EXTATTR))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1708
			return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1709
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1710
		if (nfs_rw_enter_sig(&drp->r_rwlock, RW_READER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1711
			return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1712
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1713
		(void) nfslookup_dnlc(dvp, XATTR_DIR_NAME, &avp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1714
		if (avp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1715
			error = acl_getxattrdir2(dvp, &avp, cflag, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1716
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1717
			error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1718
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1719
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1720
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1721
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1722
			if (mi->mi_flags & MI_EXTATTR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1723
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1724
			return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1725
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1726
		dvp = avp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1727
		drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1728
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1729
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1730
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_READER, INTR(dvp))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1731
		error = EINTR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1732
		goto out;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1733
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1734
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1735
	error = nfslookup(dvp, nm, vpp, pnp, flags, rdir, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1736
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1737
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1738
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1739
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1740
	 * If vnode is a device, create special vnode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1741
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1742
	if (!error && IS_DEVVP(*vpp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1743
		vp = *vpp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1744
		*vpp = specvp(vp, vp->v_rdev, vp->v_type, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1745
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1746
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1747
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1748
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1749
	if (avp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1750
		VN_RELE(avp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1751
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1752
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1753
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1754
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1755
static int nfs_lookup_neg_cache = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1756
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1757
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1758
static int nfs_lookup_dnlc_hits = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1759
static int nfs_lookup_dnlc_misses = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1760
static int nfs_lookup_dnlc_neg_hits = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1761
static int nfs_lookup_dnlc_disappears = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1762
static int nfs_lookup_dnlc_lookups = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1763
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1764
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1765
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1766
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1767
nfslookup(vnode_t *dvp, char *nm, vnode_t **vpp, struct pathname *pnp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1768
	int flags, vnode_t *rdir, cred_t *cr, int rfscall_flags)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1769
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1770
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1771
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1772
	ASSERT(nfs_zone() == VTOMI(dvp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1773
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1774
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1775
	 * If lookup is for "", just return dvp.  Don't need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1776
	 * to send it over the wire, look it up in the dnlc,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1777
	 * or perform any access checks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1778
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1779
	if (*nm == '\0') {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1780
		VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1781
		*vpp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1782
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1783
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1784
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1785
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1786
	 * Can't do lookups in non-directories.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1787
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1788
	if (dvp->v_type != VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1789
		return (ENOTDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1790
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1791
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1792
	 * If we're called with RFSCALL_SOFT, it's important that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1793
	 * the only rfscall is one we make directly; if we permit
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1794
	 * an access call because we're looking up "." or validating
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1795
	 * a dnlc hit, we'll deadlock because that rfscall will not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1796
	 * have the RFSCALL_SOFT set.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1797
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1798
	if (rfscall_flags & RFSCALL_SOFT)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1799
		goto callit;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1800
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1801
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1802
	 * If lookup is for ".", just return dvp.  Don't need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1803
	 * to send it over the wire or look it up in the dnlc,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1804
	 * just need to check access.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1805
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1806
	if (strcmp(nm, ".") == 0) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1807
		error = nfs_access(dvp, VEXEC, 0, cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1808
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1809
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1810
		VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1811
		*vpp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1812
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1813
	}
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
	 * Lookup this name in the DNLC.  If there was a valid entry,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1817
	 * then return the results of the lookup.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1818
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1819
	error = nfslookup_dnlc(dvp, nm, vpp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1820
	if (error || *vpp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1821
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1822
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1823
callit:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1824
	error = nfslookup_otw(dvp, nm, vpp, cr, rfscall_flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1825
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1826
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1827
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1828
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1829
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1830
nfslookup_dnlc(vnode_t *dvp, char *nm, vnode_t **vpp, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1831
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1832
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1833
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1834
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1835
	ASSERT(*nm != '\0');
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1836
	ASSERT(nfs_zone() == VTOMI(dvp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1837
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1838
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1839
	 * Lookup this name in the DNLC.  If successful, then validate
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1840
	 * the caches and then recheck the DNLC.  The DNLC is rechecked
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1841
	 * just in case this entry got invalidated during the call
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1842
	 * to nfs_validate_caches.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1843
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1844
	 * An assumption is being made that it is safe to say that a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1845
	 * file exists which may not on the server.  Any operations to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1846
	 * the server will fail with ESTALE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1847
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1848
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1849
	nfs_lookup_dnlc_lookups++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1850
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1851
	vp = dnlc_lookup(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1852
	if (vp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1853
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1854
		if (vp == DNLC_NO_VNODE && !vn_is_readonly(dvp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1855
			PURGE_ATTRCACHE(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1856
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1857
		error = nfs_validate_caches(dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1858
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1859
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1860
		vp = dnlc_lookup(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1861
		if (vp != NULL) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1862
			error = nfs_access(dvp, VEXEC, 0, cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1863
			if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1864
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1865
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1866
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1867
			if (vp == DNLC_NO_VNODE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1868
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1869
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1870
				nfs_lookup_dnlc_neg_hits++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1871
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1872
				return (ENOENT);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1873
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1874
			*vpp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1875
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1876
			nfs_lookup_dnlc_hits++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1877
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1878
			return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1879
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1880
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1881
		nfs_lookup_dnlc_disappears++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1882
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1883
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1884
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1885
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1886
		nfs_lookup_dnlc_misses++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1887
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1888
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1889
	*vpp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1890
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1891
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1892
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1893
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1894
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1895
nfslookup_otw(vnode_t *dvp, char *nm, vnode_t **vpp, cred_t *cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1896
	int rfscall_flags)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1897
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1898
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1899
	struct nfsdiropargs da;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1900
	struct nfsdiropres dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1901
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1902
	failinfo_t fi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1903
	hrtime_t t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1904
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1905
	ASSERT(*nm != '\0');
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1906
	ASSERT(dvp->v_type == VDIR);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1907
	ASSERT(nfs_zone() == VTOMI(dvp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1908
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1909
	setdiropargs(&da, nm, dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1910
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1911
	fi.vp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1912
	fi.fhp = NULL;		/* no need to update, filehandle not copied */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1913
	fi.copyproc = nfscopyfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1914
	fi.lookupproc = nfslookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1915
	fi.xattrdirproc = acl_getxattrdir2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1917
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1918
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1919
	t = gethrtime();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1920
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1921
	error = rfs2call(VTOMI(dvp), RFS_LOOKUP,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1922
	    xdr_diropargs, (caddr_t)&da,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1923
	    xdr_diropres, (caddr_t)&dr, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1924
	    &douprintf, &dr.dr_status, rfscall_flags, &fi);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1925
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1926
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1927
		error = geterrno(dr.dr_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1928
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1929
			*vpp = makenfsnode(&dr.dr_fhandle, &dr.dr_attr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1930
			    dvp->v_vfsp, t, cr, VTOR(dvp)->r_path, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1931
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1932
			 * If NFS_ACL is supported on the server, then the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1933
			 * attributes returned by server may have minimal
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1934
			 * permissions sometimes denying access to users having
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1935
			 * proper access.  To get the proper attributes, mark
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1936
			 * the attributes as expired so that they will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1937
			 * regotten via the NFS_ACL GETATTR2 procedure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1938
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1939
			if (VTOMI(*vpp)->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1940
				PURGE_ATTRCACHE(*vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1941
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1942
			if (!(rfscall_flags & RFSCALL_SOFT))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1943
				dnlc_update(dvp, nm, *vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1944
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1945
			PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1946
			if (error == ENOENT && nfs_lookup_neg_cache)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1947
				dnlc_enter(dvp, nm, DNLC_NO_VNODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1948
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1949
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1950
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1951
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1952
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1953
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1954
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1955
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1956
nfs_create(vnode_t *dvp, char *nm, struct vattr *va, enum vcexcl exclusive,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1957
	int mode, vnode_t **vpp, cred_t *cr, int lfaware, caller_context_t *ct,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1958
	vsecattr_t *vsecp)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1959
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1960
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1961
	struct nfscreatargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1962
	struct nfsdiropres dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1963
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1964
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1965
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1966
	struct vattr vattr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1967
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1968
	vnode_t *tempvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1969
	hrtime_t t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1970
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1971
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1972
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  1973
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1974
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1975
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_WRITER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1976
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1977
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1978
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1979
	 * We make a copy of the attributes because the caller does not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1980
	 * expect us to change what va points to.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1981
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1982
	vattr = *va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1983
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1984
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1985
	 * If the pathname is "", just use dvp.  Don't need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1986
	 * to send it over the wire, look it up in the dnlc,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1987
	 * or perform any access checks.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1988
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1989
	if (*nm == '\0') {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1990
		error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1991
		VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1992
		vp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1993
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1994
	 * If the pathname is ".", just use dvp.  Don't need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1995
	 * to send it over the wire or look it up in the dnlc,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1996
	 * just need to check access.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1997
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  1998
	} else if (strcmp(nm, ".") == 0) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  1999
		error = nfs_access(dvp, VEXEC, 0, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2000
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2001
			nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2002
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2003
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2004
		VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2005
		vp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2006
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2007
	 * We need to go over the wire, just to be sure whether the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2008
	 * file exists or not.  Using the DNLC can be dangerous in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2009
	 * this case when making a decision regarding existence.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2010
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2011
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2012
		error = nfslookup_otw(dvp, nm, &vp, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2013
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2014
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2015
		if (exclusive == EXCL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2016
			error = EEXIST;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2017
		else if (vp->v_type == VDIR && (mode & VWRITE))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2018
			error = EISDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2019
		else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2020
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2021
			 * If vnode is a device, create special vnode.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2022
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2023
			if (IS_DEVVP(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2024
				tempvp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2025
				vp = specvp(vp, vp->v_rdev, vp->v_type, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2026
				VN_RELE(tempvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2027
			}
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2028
			if (!(error = VOP_ACCESS(vp, mode, 0, cr, ct))) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2029
				if ((vattr.va_mask & AT_SIZE) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2030
				    vp->v_type == VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2031
					vattr.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2032
					error = nfssetattr(vp, &vattr, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2033
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2034
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2035
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2036
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2037
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2038
			VN_RELE(vp);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2039
		} else {
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2040
			/*
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2041
			 * existing file got truncated, notify.
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2042
			 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2043
			vnevent_create(vp, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2044
			*vpp = vp;
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2045
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2046
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2047
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2048
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2049
	ASSERT(vattr.va_mask & AT_TYPE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2050
	if (vattr.va_type == VREG) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2051
		ASSERT(vattr.va_mask & AT_MODE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2052
		if (MANDMODE(vattr.va_mode)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2053
			nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2054
			return (EACCES);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2055
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2056
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2057
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2058
	dnlc_remove(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2059
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2060
	setdiropargs(&args.ca_da, nm, dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2061
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2062
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2063
	 * Decide what the group-id of the created file should be.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2064
	 * Set it in attribute list as advisory...then do a setattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2065
	 * if the server didn't get it right the first time.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2066
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2067
	error = setdirgid(dvp, &vattr.va_gid, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2068
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2069
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2070
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2071
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2072
	vattr.va_mask |= AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2073
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2074
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2075
	 * This is a completely gross hack to make mknod
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2076
	 * work over the wire until we can wack the protocol
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2077
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2078
#define	IFCHR		0020000		/* character special */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2079
#define	IFBLK		0060000		/* block special */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2080
#define	IFSOCK		0140000		/* socket */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2081
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2082
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2083
	 * dev_t is uint_t in 5.x and short in 4.x. Both 4.x
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2084
	 * supports 8 bit majors. 5.x supports 14 bit majors. 5.x supports 18
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2085
	 * bits in the minor number where 4.x supports 8 bits.  If the 5.x
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2086
	 * minor/major numbers <= 8 bits long, compress the device
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2087
	 * number before sending it. Otherwise, the 4.x server will not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2088
	 * create the device with the correct device number and nothing can be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2089
	 * done about this.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2090
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2091
	if (vattr.va_type == VCHR || vattr.va_type == VBLK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2092
		dev_t d = vattr.va_rdev;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2093
		dev32_t dev32;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2094
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2095
		if (vattr.va_type == VCHR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2096
			vattr.va_mode |= IFCHR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2097
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2098
			vattr.va_mode |= IFBLK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2099
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2100
		(void) cmpldev(&dev32, d);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2101
		if (dev32 & ~((SO4_MAXMAJ << L_BITSMINOR32) | SO4_MAXMIN))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2102
			vattr.va_size = (u_offset_t)dev32;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2103
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2104
			vattr.va_size = (u_offset_t)nfsv2_cmpdev(d);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2105
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2106
		vattr.va_mask |= AT_MODE|AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2107
	} else if (vattr.va_type == VFIFO) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2108
		vattr.va_mode |= IFCHR;		/* xtra kludge for namedpipe */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2109
		vattr.va_size = (u_offset_t)NFS_FIFO_DEV;	/* blech */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2110
		vattr.va_mask |= AT_MODE|AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2111
	} else if (vattr.va_type == VSOCK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2112
		vattr.va_mode |= IFSOCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2113
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2114
		 * To avoid triggering bugs in the servers set AT_SIZE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2115
		 * (all other RFS_CREATE calls set this).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2116
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2117
		vattr.va_size = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2118
		vattr.va_mask |= AT_MODE|AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2119
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2120
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2121
	args.ca_sa = &args.ca_sa_buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2122
	error = vattr_to_sattr(&vattr, args.ca_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2123
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2124
		/* req time field(s) overflow - return immediately */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2125
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2126
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2127
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2128
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2129
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2130
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2131
	t = gethrtime();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2132
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2133
	error = rfs2call(VTOMI(dvp), RFS_CREATE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2134
	    xdr_creatargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2135
	    xdr_diropres, (caddr_t)&dr, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2136
	    &douprintf, &dr.dr_status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2137
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2138
	PURGE_ATTRCACHE(dvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2139
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2140
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2141
		error = geterrno(dr.dr_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2142
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2143
			if (HAVE_RDDIR_CACHE(drp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2144
				nfs_purge_rddir_cache(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2145
			vp = makenfsnode(&dr.dr_fhandle, &dr.dr_attr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2146
			    dvp->v_vfsp, t, cr, NULL, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2147
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2148
			 * If NFS_ACL is supported on the server, then the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2149
			 * attributes returned by server may have minimal
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2150
			 * permissions sometimes denying access to users having
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2151
			 * proper access.  To get the proper attributes, mark
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2152
			 * the attributes as expired so that they will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2153
			 * regotten via the NFS_ACL GETATTR2 procedure.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2154
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2155
			if (VTOMI(vp)->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2156
				PURGE_ATTRCACHE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2157
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2158
			dnlc_update(dvp, nm, vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2159
			rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2160
			if (vattr.va_size == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2161
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2162
				rp->r_size = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2163
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2164
				if (vn_has_cached_data(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2165
					ASSERT(vp->v_type != VCHR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2166
					nfs_invalidate_pages(vp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2167
					    (u_offset_t)0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2168
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2169
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2170
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2171
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2172
			 * Make sure the gid was set correctly.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2173
			 * If not, try to set it (but don't lose
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2174
			 * any sleep over it).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2175
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2176
			if (vattr.va_gid != rp->r_attr.va_gid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2177
				vattr.va_mask = AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2178
				(void) nfssetattr(vp, &vattr, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2179
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2180
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2181
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2182
			 * If vnode is a device create special vnode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2183
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2184
			if (IS_DEVVP(vp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2185
				*vpp = specvp(vp, vp->v_rdev, vp->v_type, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2186
				VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2187
			} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2188
				*vpp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2189
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2190
			PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2191
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2192
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2193
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2194
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2195
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2196
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2197
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2198
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2199
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2200
 * Weirdness: if the vnode to be removed is open
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2201
 * we rename it instead of removing it and nfs_inactive
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2202
 * will remove the new name.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2203
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2204
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2205
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2206
nfs_remove(vnode_t *dvp, char *nm, cred_t *cr, caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2207
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2208
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2209
	struct nfsdiropargs da;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2210
	enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2211
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2212
	char *tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2213
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2214
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2215
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2216
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2217
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2218
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2219
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2220
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_WRITER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2221
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2222
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2223
	error = nfslookup(dvp, nm, &vp, NULL, 0, NULL, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2224
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2225
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2226
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2227
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2228
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2229
	if (vp->v_type == VDIR && secpolicy_fs_linkdir(cr, dvp->v_vfsp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2230
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2231
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2232
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2233
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2234
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2235
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2236
	 * First just remove the entry from the name cache, as it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2237
	 * is most likely the only entry for this vp.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2238
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2239
	dnlc_remove(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2240
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2241
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2242
	 * If the file has a v_count > 1 then there may be more than one
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2243
	 * entry in the name cache due multiple links or an open file,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2244
	 * but we don't have the real reference count so flush all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2245
	 * possible entries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2246
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2247
	if (vp->v_count > 1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2248
		dnlc_purge_vp(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2249
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2250
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2251
	 * Now we have the real reference count on the vnode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2252
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2253
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2254
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2255
	if (vp->v_count > 1 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2256
	    (rp->r_unldvp == NULL || strcmp(nm, rp->r_unlname) == 0)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2257
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2258
		tmpname = newname();
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2259
		error = nfsrename(dvp, nm, dvp, tmpname, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2260
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2261
			kmem_free(tmpname, MAXNAMELEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2262
		else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2263
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2264
			if (rp->r_unldvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2265
				VN_HOLD(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2266
				rp->r_unldvp = dvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2267
				if (rp->r_unlcred != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2268
					crfree(rp->r_unlcred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2269
				crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2270
				rp->r_unlcred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2271
				rp->r_unlname = tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2272
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2273
				kmem_free(rp->r_unlname, MAXNAMELEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2274
				rp->r_unlname = tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2275
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2276
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2277
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2278
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2279
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2280
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2281
		 * We need to flush any dirty pages which happen to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2282
		 * be hanging around before removing the file.  This
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2283
		 * shouldn't happen very often and mostly on file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2284
		 * systems mounted "nocto".
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2285
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2286
		if (vn_has_cached_data(vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2287
		    ((rp->r_flags & RDIRTY) || rp->r_count > 0)) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2288
			error = nfs_putpage(vp, (offset_t)0, 0, 0, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2289
			if (error && (error == ENOSPC || error == EDQUOT)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2290
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2291
				if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2292
					rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2293
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2294
			}
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
		setdiropargs(&da, nm, dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2298
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2299
		douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2300
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2301
		error = rfs2call(VTOMI(dvp), RFS_REMOVE,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2302
		    xdr_diropargs, (caddr_t)&da,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2303
		    xdr_enum, (caddr_t)&status, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2304
		    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2305
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2306
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2307
		 * The xattr dir may be gone after last attr is removed,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2308
		 * so flush it from dnlc.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2309
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2310
		if (dvp->v_flag & V_XATTRDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2311
			dnlc_purge_vp(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2312
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2313
		PURGE_ATTRCACHE(dvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2314
		PURGE_ATTRCACHE(vp);	/* link count changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2315
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2316
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2317
			error = geterrno(status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2318
			if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2319
				if (HAVE_RDDIR_CACHE(drp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2320
					nfs_purge_rddir_cache(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2321
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2322
				PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2323
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2324
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2325
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2326
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2327
	if (error == 0) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2328
		vnevent_remove(vp, dvp, nm, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2329
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2330
	VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2331
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2332
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2333
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2334
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2335
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2336
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2337
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2338
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2339
nfs_link(vnode_t *tdvp, vnode_t *svp, char *tnm, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2340
	caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2341
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2342
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2343
	struct nfslinkargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2344
	enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2345
	vnode_t *realvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2346
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2347
	rnode_t *tdrp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2348
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2349
	if (nfs_zone() != VTOMI(tdvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2350
		return (EPERM);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2351
	if (VOP_REALVP(svp, &realvp, ct) == 0)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2352
		svp = realvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2353
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2354
	args.la_from = VTOFH(svp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2355
	setdiropargs(&args.la_to, tnm, tdvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2356
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2357
	tdrp = VTOR(tdvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2358
	if (nfs_rw_enter_sig(&tdrp->r_rwlock, RW_WRITER, INTR(tdvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2359
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2360
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2361
	dnlc_remove(tdvp, tnm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2362
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2363
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2364
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2365
	error = rfs2call(VTOMI(svp), RFS_LINK,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2366
	    xdr_linkargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2367
	    xdr_enum, (caddr_t)&status, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2368
	    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2369
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2370
	PURGE_ATTRCACHE(tdvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2371
	PURGE_ATTRCACHE(svp);	/* link count changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2372
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2373
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2374
		error = geterrno(status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2375
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2376
			if (HAVE_RDDIR_CACHE(tdrp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2377
				nfs_purge_rddir_cache(tdvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2378
		}
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
	nfs_rw_exit(&tdrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2382
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2383
	if (!error) {
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2384
		/*
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2385
		 * Notify the source file of this link operation.
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2386
		 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2387
		vnevent_link(svp, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2388
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2389
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2390
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2391
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2392
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2393
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2394
nfs_rename(vnode_t *odvp, char *onm, vnode_t *ndvp, char *nnm, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2395
	caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2396
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2397
	vnode_t *realvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2398
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2399
	if (nfs_zone() != VTOMI(odvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2400
		return (EPERM);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2401
	if (VOP_REALVP(ndvp, &realvp, ct) == 0)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2402
		ndvp = realvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2403
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2404
	return (nfsrename(odvp, onm, ndvp, nnm, cr, ct));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2405
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2406
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2407
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2408
 * nfsrename does the real work of renaming in NFS Version 2.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2409
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2410
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2411
nfsrename(vnode_t *odvp, char *onm, vnode_t *ndvp, char *nnm, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2412
    caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2413
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2414
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2415
	enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2416
	struct nfsrnmargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2417
	int douprintf;
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2418
	vnode_t *nvp = NULL;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2419
	vnode_t *ovp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2420
	char *tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2421
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2422
	rnode_t *odrp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2423
	rnode_t *ndrp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2424
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2425
	ASSERT(nfs_zone() == VTOMI(odvp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2426
	if (strcmp(onm, ".") == 0 || strcmp(onm, "..") == 0 ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2427
	    strcmp(nnm, ".") == 0 || strcmp(nnm, "..") == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2428
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2429
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2430
	odrp = VTOR(odvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2431
	ndrp = VTOR(ndvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2432
	if ((intptr_t)odrp < (intptr_t)ndrp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2433
		if (nfs_rw_enter_sig(&odrp->r_rwlock, RW_WRITER, INTR(odvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2434
			return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2435
		if (nfs_rw_enter_sig(&ndrp->r_rwlock, RW_WRITER, INTR(ndvp))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2436
			nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2437
			return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2438
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2439
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2440
		if (nfs_rw_enter_sig(&ndrp->r_rwlock, RW_WRITER, INTR(ndvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2441
			return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2442
		if (nfs_rw_enter_sig(&odrp->r_rwlock, RW_WRITER, INTR(odvp))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2443
			nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2444
			return (EINTR);
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
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2448
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2449
	 * Lookup the target file.  If it exists, it needs to be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2450
	 * checked to see whether it is a mount point and whether
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2451
	 * it is active (open).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2452
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2453
	error = nfslookup(ndvp, nnm, &nvp, NULL, 0, NULL, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2454
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2455
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2456
		 * If this file has been mounted on, then just
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2457
		 * return busy because renaming to it would remove
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2458
		 * the mounted file system from the name space.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2459
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2460
		if (vn_mountedvfs(nvp) != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2461
			VN_RELE(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2462
			nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2463
			nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2464
			return (EBUSY);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2465
		}
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
		 * Purge the name cache of all references to this vnode
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2469
		 * so that we can check the reference count to infer
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2470
		 * whether it is active or not.
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
		 * First just remove the entry from the name cache, as it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2474
		 * is most likely the only entry for this vp.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2475
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2476
		dnlc_remove(ndvp, nnm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2477
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2478
		 * If the file has a v_count > 1 then there may be more
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2479
		 * than one entry in the name cache due multiple links
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2480
		 * or an open file, but we don't have the real reference
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2481
		 * count so flush all possible entries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2482
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2483
		if (nvp->v_count > 1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2484
			dnlc_purge_vp(nvp);
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
		 * If the vnode is active and is not a directory,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2488
		 * arrange to rename it to a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2489
		 * temporary file so that it will continue to be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2490
		 * accessible.  This implements the "unlink-open-file"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2491
		 * semantics for the target of a rename operation.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2492
		 * Before doing this though, make sure that the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2493
		 * source and target files are not already the same.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2494
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2495
		if (nvp->v_count > 1 && nvp->v_type != VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2496
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2497
			 * Lookup the source name.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2498
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2499
			error = nfslookup(odvp, onm, &ovp, NULL, 0, NULL,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2500
			    cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2501
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2502
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2503
			 * The source name *should* already exist.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2504
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2505
			if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2506
				VN_RELE(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2507
				nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2508
				nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2509
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2510
			}
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
			 * Compare the two vnodes.  If they are the same,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2514
			 * just release all held vnodes and return success.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2515
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2516
			if (ovp == nvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2517
				VN_RELE(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2518
				VN_RELE(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2519
				nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2520
				nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2521
				return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2522
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2523
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2524
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2525
			 * Can't mix and match directories and non-
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2526
			 * directories in rename operations.  We already
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2527
			 * know that the target is not a directory.  If
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2528
			 * the source is a directory, return an error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2529
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2530
			if (ovp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2531
				VN_RELE(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2532
				VN_RELE(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2533
				nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2534
				nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2535
				return (ENOTDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2536
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2537
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2538
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2539
			 * The target file exists, is not the same as
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2540
			 * the source file, and is active.  Link it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2541
			 * to a temporary filename to avoid having
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2542
			 * the server removing the file completely.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2543
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2544
			tmpname = newname();
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2545
			error = nfs_link(ndvp, nvp, tmpname, cr, NULL, 0);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2546
			if (error == EOPNOTSUPP) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2547
				error = nfs_rename(ndvp, nnm, ndvp, tmpname,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2548
				    cr, NULL, 0);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2549
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2550
			if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2551
				kmem_free(tmpname, MAXNAMELEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2552
				VN_RELE(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2553
				VN_RELE(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2554
				nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2555
				nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2556
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2557
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2558
			rp = VTOR(nvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2559
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2560
			if (rp->r_unldvp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2561
				VN_HOLD(ndvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2562
				rp->r_unldvp = ndvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2563
				if (rp->r_unlcred != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2564
					crfree(rp->r_unlcred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2565
				crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2566
				rp->r_unlcred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2567
				rp->r_unlname = tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2568
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2569
				kmem_free(rp->r_unlname, MAXNAMELEN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2570
				rp->r_unlname = tmpname;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2571
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2572
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2573
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2574
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2575
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2576
	if (ovp == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2577
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2578
		 * When renaming directories to be a subdirectory of a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2579
		 * different parent, the dnlc entry for ".." will no
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2580
		 * longer be valid, so it must be removed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2581
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2582
		 * We do a lookup here to determine whether we are renaming
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2583
		 * a directory and we need to check if we are renaming
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2584
		 * an unlinked file.  This might have already been done
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2585
		 * in previous code, so we check ovp == NULL to avoid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2586
		 * doing it twice.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2587
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2588
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2589
		error = nfslookup(odvp, onm, &ovp, NULL, 0, NULL, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2590
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2591
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2592
		 * The source name *should* already exist.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2593
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2594
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2595
			nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2596
			nfs_rw_exit(&ndrp->r_rwlock);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2597
			if (nvp) {
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2598
				VN_RELE(nvp);
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2599
			}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2600
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2601
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2602
		ASSERT(ovp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2603
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2604
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2605
	dnlc_remove(odvp, onm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2606
	dnlc_remove(ndvp, nnm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2607
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2608
	setdiropargs(&args.rna_from, onm, odvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2609
	setdiropargs(&args.rna_to, nnm, ndvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2610
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2611
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2612
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2613
	error = rfs2call(VTOMI(odvp), RFS_RENAME,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2614
	    xdr_rnmargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2615
	    xdr_enum, (caddr_t)&status, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2616
	    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2617
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2618
	PURGE_ATTRCACHE(odvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2619
	PURGE_ATTRCACHE(ndvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2621
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2622
		error = geterrno(status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2623
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2624
			if (HAVE_RDDIR_CACHE(odrp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2625
				nfs_purge_rddir_cache(odvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2626
			if (HAVE_RDDIR_CACHE(ndrp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2627
				nfs_purge_rddir_cache(ndvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2628
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2629
			 * when renaming directories to be a subdirectory of a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2630
			 * different parent, the dnlc entry for ".." will no
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2631
			 * longer be valid, so it must be removed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2632
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2633
			rp = VTOR(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2634
			if (ndvp != odvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2635
				if (ovp->v_type == VDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2636
					dnlc_remove(ovp, "..");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2637
					if (HAVE_RDDIR_CACHE(rp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2638
						nfs_purge_rddir_cache(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2639
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2640
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2641
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2642
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2643
			 * If we are renaming the unlinked file, update the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2644
			 * r_unldvp and r_unlname as needed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2645
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2646
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2647
			if (rp->r_unldvp != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2648
				if (strcmp(rp->r_unlname, onm) == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2649
					(void) strncpy(rp->r_unlname,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2650
					    nnm, MAXNAMELEN);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2651
					rp->r_unlname[MAXNAMELEN - 1] = '\0';
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2652
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2653
					if (ndvp != rp->r_unldvp) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2654
						VN_RELE(rp->r_unldvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2655
						rp->r_unldvp = ndvp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2656
						VN_HOLD(ndvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2657
					}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2658
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2659
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2660
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2661
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2662
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2663
			 * System V defines rename to return EEXIST, not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2664
			 * ENOTEMPTY if the target directory is not empty.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2665
			 * Over the wire, the error is NFSERR_ENOTEMPTY
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2666
			 * which geterrno maps to ENOTEMPTY.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2667
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2668
			if (error == ENOTEMPTY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2669
				error = EEXIST;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2670
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2671
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2672
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2673
	if (error == 0) {
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2674
		if (nvp)
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2675
			vnevent_rename_dest(nvp, ndvp, nnm, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2676
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2677
		if (odvp != ndvp)
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2678
			vnevent_rename_dest_dir(ndvp, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2679
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2680
		ASSERT(ovp != NULL);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2681
		vnevent_rename_src(ovp, odvp, onm, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2682
	}
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2683
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2684
	if (nvp) {
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2685
		VN_RELE(nvp);
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2686
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2687
	VN_RELE(ovp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2688
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2689
	nfs_rw_exit(&odrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2690
	nfs_rw_exit(&ndrp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2691
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2692
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2693
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2694
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2695
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2696
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2697
nfs_mkdir(vnode_t *dvp, char *nm, struct vattr *va, vnode_t **vpp, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2698
	caller_context_t *ct, int flags, vsecattr_t *vsecp)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2699
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2700
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2701
	struct nfscreatargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2702
	struct nfsdiropres dr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2703
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2704
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2705
	hrtime_t t;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2706
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2707
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2708
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2709
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2710
	setdiropargs(&args.ca_da, nm, dvp);
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
	 * Decide what the group-id and set-gid bit of the created directory
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2714
	 * should be.  May have to do a setattr to get the gid right.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2715
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2716
	error = setdirgid(dvp, &va->va_gid, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2717
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2718
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2719
	error = setdirmode(dvp, &va->va_mode, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2720
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2721
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2722
	va->va_mask |= AT_MODE|AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2723
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2724
	args.ca_sa = &args.ca_sa_buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2725
	error = vattr_to_sattr(va, args.ca_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2726
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2727
		/* req time field(s) overflow - return immediately */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2728
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2729
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2730
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2731
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2732
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_WRITER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2733
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2734
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2735
	dnlc_remove(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2736
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2737
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2738
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2739
	t = gethrtime();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2740
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2741
	error = rfs2call(VTOMI(dvp), RFS_MKDIR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2742
	    xdr_creatargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2743
	    xdr_diropres, (caddr_t)&dr, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2744
	    &douprintf, &dr.dr_status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2745
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2746
	PURGE_ATTRCACHE(dvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2747
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2748
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2749
		error = geterrno(dr.dr_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2750
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2751
			if (HAVE_RDDIR_CACHE(drp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2752
				nfs_purge_rddir_cache(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2753
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2754
			 * The attributes returned by RFS_MKDIR can not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2755
			 * be depended upon, so mark the attribute cache
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2756
			 * as purged.  A subsequent GETATTR will get the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2757
			 * correct attributes from the server.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2758
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2759
			*vpp = makenfsnode(&dr.dr_fhandle, &dr.dr_attr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2760
			    dvp->v_vfsp, t, cr, NULL, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2761
			PURGE_ATTRCACHE(*vpp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2762
			dnlc_update(dvp, nm, *vpp);
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
			 * Make sure the gid was set correctly.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2766
			 * If not, try to set it (but don't lose
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2767
			 * any sleep over it).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2768
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2769
			if (va->va_gid != VTOR(*vpp)->r_attr.va_gid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2770
				va->va_mask = AT_GID;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2771
				(void) nfssetattr(*vpp, va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2772
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2773
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2774
			PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2775
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2776
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2777
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2778
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2779
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2780
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2781
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2782
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2783
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2784
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2785
nfs_rmdir(vnode_t *dvp, char *nm, vnode_t *cdir, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2786
	caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2787
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2788
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2789
	enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2790
	struct nfsdiropargs da;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2791
	vnode_t *vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2792
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2793
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2794
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2795
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2796
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2797
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2798
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_WRITER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2799
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2800
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2801
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2802
	 * Attempt to prevent a rmdir(".") from succeeding.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2803
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2804
	error = nfslookup(dvp, nm, &vp, NULL, 0, NULL, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2805
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2806
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2807
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2808
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2809
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2810
	if (vp == cdir) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2811
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2812
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2813
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2814
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2815
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2816
	setdiropargs(&da, nm, dvp);
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
	 * First just remove the entry from the name cache, as it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2820
	 * is most likely an entry for this vp.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2821
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2822
	dnlc_remove(dvp, nm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2823
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2824
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2825
	 * If there vnode reference count is greater than one, then
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2826
	 * there may be additional references in the DNLC which will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2827
	 * need to be purged.  First, trying removing the entry for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2828
	 * the parent directory and see if that removes the additional
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2829
	 * reference(s).  If that doesn't do it, then use dnlc_purge_vp
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2830
	 * to completely remove any references to the directory which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2831
	 * might still exist in the DNLC.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2832
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2833
	if (vp->v_count > 1) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2834
		dnlc_remove(vp, "..");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2835
		if (vp->v_count > 1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2836
			dnlc_purge_vp(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2837
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2838
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2839
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2840
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2841
	error = rfs2call(VTOMI(dvp), RFS_RMDIR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2842
	    xdr_diropargs, (caddr_t)&da,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2843
	    xdr_enum, (caddr_t)&status, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2844
	    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2845
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2846
	PURGE_ATTRCACHE(dvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2847
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2848
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2849
		VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2850
		nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2851
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2852
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2853
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2854
	error = geterrno(status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2855
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2856
		if (HAVE_RDDIR_CACHE(drp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2857
			nfs_purge_rddir_cache(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2858
		if (HAVE_RDDIR_CACHE(VTOR(vp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2859
			nfs_purge_rddir_cache(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2860
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2861
		PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2862
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2863
		 * System V defines rmdir to return EEXIST, not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2864
		 * ENOTEMPTY if the directory is not empty.  Over
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2865
		 * the wire, the error is NFSERR_ENOTEMPTY which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2866
		 * geterrno maps to ENOTEMPTY.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2867
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2868
		if (error == ENOTEMPTY)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2869
			error = EEXIST;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2870
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2871
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2872
	if (error == 0) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2873
		vnevent_rmdir(vp, dvp, nm, ct);
4863
7b14ad153d91 PSARC/2007/027 File Events Notification API
praks
parents: 4813
diff changeset
  2874
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2875
	VN_RELE(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2876
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2877
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2878
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2879
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2880
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2881
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2882
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2883
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2884
nfs_symlink(vnode_t *dvp, char *lnm, struct vattr *tva, char *tnm, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2885
	caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2886
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2887
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2888
	struct nfsslargs args;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2889
	enum nfsstat status;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2890
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2891
	rnode_t *drp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2892
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2893
	if (nfs_zone() != VTOMI(dvp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2894
		return (EPERM);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2895
	setdiropargs(&args.sla_from, lnm, dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2896
	args.sla_sa = &args.sla_sa_buf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2897
	error = vattr_to_sattr(tva, args.sla_sa);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2898
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2899
		/* req time field(s) overflow - return immediately */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2900
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2901
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2902
	args.sla_tnm = tnm;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2903
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2904
	drp = VTOR(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2905
	if (nfs_rw_enter_sig(&drp->r_rwlock, RW_WRITER, INTR(dvp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2906
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2907
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2908
	dnlc_remove(dvp, lnm);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2909
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2910
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2911
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2912
	error = rfs2call(VTOMI(dvp), RFS_SYMLINK,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2913
	    xdr_slargs, (caddr_t)&args,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2914
	    xdr_enum, (caddr_t)&status, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2915
	    &douprintf, &status, 0, NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2916
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2917
	PURGE_ATTRCACHE(dvp);	/* mod time changed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2918
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2919
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2920
		error = geterrno(status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2921
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2922
			if (HAVE_RDDIR_CACHE(drp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2923
				nfs_purge_rddir_cache(dvp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2924
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2925
			PURGE_STALE_FH(error, dvp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2926
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2927
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2928
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2929
	nfs_rw_exit(&drp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2930
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2931
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2932
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2933
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2934
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2935
static int nfs_readdir_cache_hits = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2936
static int nfs_readdir_cache_shorts = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2937
static int nfs_readdir_cache_waits = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2938
static int nfs_readdir_cache_misses = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2939
static int nfs_readdir_readahead = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2940
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2941
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2942
static int nfs_shrinkreaddir = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2943
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2944
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2945
 * Read directory entries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2946
 * There are some weird things to look out for here.  The uio_offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2947
 * field is either 0 or it is the offset returned from a previous
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2948
 * readdir.  It is an opaque value used by the server to find the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2949
 * correct directory block to read. The count field is the number
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2950
 * of blocks to read on the server.  This is advisory only, the server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2951
 * may return only one block's worth of entries.  Entries may be compressed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2952
 * on the server.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2953
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2954
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2955
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2956
nfs_readdir(vnode_t *vp, struct uio *uiop, cred_t *cr, int *eofp,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  2957
	caller_context_t *ct, int flags)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2958
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2959
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2960
	size_t count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2961
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2962
	rddir_cache *rdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2963
	rddir_cache *nrdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2964
	rddir_cache *rrdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2965
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2966
	int missed;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2967
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2968
	rddir_cache srdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2969
	avl_index_t where;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2970
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2971
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2972
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2973
	ASSERT(nfs_rw_lock_held(&rp->r_rwlock, RW_READER));
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  2974
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2975
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2976
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2977
	 * Make sure that the directory cache is valid.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2978
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2979
	if (HAVE_RDDIR_CACHE(rp)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2980
		if (nfs_disable_rddir_cache) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2981
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2982
			 * Setting nfs_disable_rddir_cache in /etc/system
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2983
			 * allows interoperability with servers that do not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2984
			 * properly update the attributes of directories.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2985
			 * Any cached information gets purged before an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2986
			 * access is made to it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2987
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2988
			nfs_purge_rddir_cache(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2989
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2990
			error = nfs_validate_caches(vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2991
			if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2992
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2993
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2994
	}
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
	 * UGLINESS: SunOS 3.2 servers apparently cannot always handle an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2998
	 * RFS_READDIR request with rda_count set to more than 0x400. So
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  2999
	 * we reduce the request size here purely for compatibility.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3000
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3001
	 * In general, this is no longer required.  However, if a server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3002
	 * is discovered which can not handle requests larger than 1024,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3003
	 * nfs_shrinkreaddir can be set to 1 to enable this backwards
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3004
	 * compatibility.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3005
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3006
	 * In any case, the request size is limited to NFS_MAXDATA bytes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3007
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3008
	count = MIN(uiop->uio_iov->iov_len,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3009
	    nfs_shrinkreaddir ? 0x400 : NFS_MAXDATA);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3010
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3011
	nrdc = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3012
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3013
	missed = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3014
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3015
top:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3016
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3017
	 * Short circuit last readdir which always returns 0 bytes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3018
	 * This can be done after the directory has been read through
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3019
	 * completely at least once.  This will set r_direof which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3020
	 * can be used to find the value of the last cookie.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3021
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3022
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3023
	if (rp->r_direof != NULL &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3024
	    uiop->uio_offset == rp->r_direof->nfs_ncookie) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3025
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3026
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3027
		nfs_readdir_cache_shorts++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3028
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3029
		if (eofp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3030
			*eofp = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3031
		if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3032
			rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3033
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3034
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3035
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3036
	 * Look for a cache entry.  Cache entries are identified
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3037
	 * by the NFS cookie value and the byte count requested.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3038
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3039
	srdc.nfs_cookie = uiop->uio_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3040
	srdc.buflen = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3041
	rdc = avl_find(&rp->r_dir, &srdc, &where);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3042
	if (rdc != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3043
		rddir_cache_hold(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3044
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3045
		 * If the cache entry is in the process of being
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3046
		 * filled in, wait until this completes.  The
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3047
		 * RDDIRWAIT bit is set to indicate that someone
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3048
		 * is waiting and then the thread currently
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3049
		 * filling the entry is done, it should do a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3050
		 * cv_broadcast to wakeup all of the threads
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3051
		 * waiting for it to finish.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3052
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3053
		if (rdc->flags & RDDIR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3054
			nfs_rw_exit(&rp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3055
			rdc->flags |= RDDIRWAIT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3056
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3057
			nfs_readdir_cache_waits++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3058
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3059
			if (!cv_wait_sig(&rdc->cv, &rp->r_statelock)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3060
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3061
				 * We got interrupted, probably
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3062
				 * the user typed ^C or an alarm
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3063
				 * fired.  We free the new entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3064
				 * if we allocated one.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3065
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3066
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3067
				(void) nfs_rw_enter_sig(&rp->r_rwlock,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3068
				    RW_READER, FALSE);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3069
				rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3070
				if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3071
					rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3072
				return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3073
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3074
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3075
			(void) nfs_rw_enter_sig(&rp->r_rwlock,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3076
			    RW_READER, FALSE);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3077
			rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3078
			goto top;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3079
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3080
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3081
		 * Check to see if a readdir is required to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3082
		 * fill the entry.  If so, mark this entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3083
		 * as being filled, remove our reference,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3084
		 * and branch to the code to fill the entry.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3085
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3086
		if (rdc->flags & RDDIRREQ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3087
			rdc->flags &= ~RDDIRREQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3088
			rdc->flags |= RDDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3089
			if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3090
				rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3091
			nrdc = rdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3092
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3093
			goto bottom;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3094
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3095
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3096
		if (!missed)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3097
			nfs_readdir_cache_hits++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3098
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3099
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3100
		 * If an error occurred while attempting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3101
		 * to fill the cache entry, just return it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3102
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3103
		if (rdc->error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3104
			error = rdc->error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3105
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3106
			rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3107
			if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3108
				rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3109
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3110
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3111
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3112
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3113
		 * The cache entry is complete and good,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3114
		 * copyout the dirent structs to the calling
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3115
		 * thread.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3116
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3117
		error = uiomove(rdc->entries, rdc->entlen, UIO_READ, uiop);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3118
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3119
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3120
		 * If no error occurred during the copyout,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3121
		 * update the offset in the uio struct to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3122
		 * contain the value of the next cookie
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3123
		 * and set the eof value appropriately.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3124
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3125
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3126
			uiop->uio_offset = rdc->nfs_ncookie;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3127
			if (eofp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3128
				*eofp = rdc->eof;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3129
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3130
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3131
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3132
		 * Decide whether to do readahead.  Don't if
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3133
		 * have already read to the end of directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3134
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3135
		if (rdc->eof) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3136
			rp->r_direof = rdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3137
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3138
			rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3139
			if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3140
				rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3141
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3142
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3143
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3144
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3145
		 * Check to see whether we found an entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3146
		 * for the readahead.  If so, we don't need
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3147
		 * to do anything further, so free the new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3148
		 * entry if one was allocated.  Otherwise,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3149
		 * allocate a new entry, add it to the cache,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3150
		 * and then initiate an asynchronous readdir
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3151
		 * operation to fill it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3152
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3153
		srdc.nfs_cookie = rdc->nfs_ncookie;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3154
		srdc.buflen = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3155
		rrdc = avl_find(&rp->r_dir, &srdc, &where);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3156
		if (rrdc != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3157
			if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3158
				rddir_cache_rele(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3159
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3160
			if (nrdc != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3161
				rrdc = nrdc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3162
			else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3163
				rrdc = rddir_cache_alloc(KM_NOSLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3164
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3165
			if (rrdc != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3166
				rrdc->nfs_cookie = rdc->nfs_ncookie;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3167
				rrdc->buflen = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3168
				avl_insert(&rp->r_dir, rrdc, where);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3169
				rddir_cache_hold(rrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3170
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3171
				rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3172
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3173
				nfs_readdir_readahead++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3174
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3175
				nfs_async_readdir(vp, rrdc, cr, nfsreaddir);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3176
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3177
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3178
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3179
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3180
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3181
		rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3182
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3183
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3184
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3185
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3186
	 * Didn't find an entry in the cache.  Construct a new empty
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3187
	 * entry and link it into the cache.  Other processes attempting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3188
	 * to access this entry will need to wait until it is filled in.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3189
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3190
	 * Since kmem_alloc may block, another pass through the cache
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3191
	 * will need to be taken to make sure that another process
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3192
	 * hasn't already added an entry to the cache for this request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3193
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3194
	if (nrdc == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3195
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3196
		nrdc = rddir_cache_alloc(KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3197
		nrdc->nfs_cookie = uiop->uio_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3198
		nrdc->buflen = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3199
		goto top;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3200
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3201
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3202
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3203
	 * Add this entry to the cache.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3204
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3205
	avl_insert(&rp->r_dir, nrdc, where);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3206
	rddir_cache_hold(nrdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3207
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3208
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3209
bottom:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3210
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3211
	missed = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3212
	nfs_readdir_cache_misses++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3213
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3214
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3215
	 * Do the readdir.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3216
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3217
	error = nfsreaddir(vp, nrdc, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3218
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3219
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3220
	 * If this operation failed, just return the error which occurred.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3221
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3222
	if (error != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3223
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3224
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3225
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3226
	 * Since the RPC operation will have taken sometime and blocked
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3227
	 * this process, another pass through the cache will need to be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3228
	 * taken to find the correct cache entry.  It is possible that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3229
	 * the correct cache entry will not be there (although one was
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3230
	 * added) because the directory changed during the RPC operation
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3231
	 * and the readdir cache was flushed.  In this case, just start
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3232
	 * over.  It is hoped that this will not happen too often... :-)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3233
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3234
	nrdc = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3235
	goto top;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3236
	/* NOTREACHED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3237
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3238
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3239
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3240
nfsreaddir(vnode_t *vp, rddir_cache *rdc, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3241
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3242
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3243
	struct nfsrddirargs rda;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3244
	struct nfsrddirres rd;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3245
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3246
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3247
	uint_t count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3248
	int douprintf;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3249
	failinfo_t fi, *fip;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3250
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  3251
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3252
	count = rdc->buflen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3253
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3254
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3255
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3256
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3257
	rda.rda_fh = *VTOFH(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3258
	rda.rda_offset = rdc->nfs_cookie;
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
	 * NFS client failover support
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3262
	 * suppress failover unless we have a zero cookie
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3263
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3264
	if (rdc->nfs_cookie == (off_t)0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3265
		fi.vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3266
		fi.fhp = (caddr_t)&rda.rda_fh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3267
		fi.copyproc = nfscopyfh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3268
		fi.lookupproc = nfslookup;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3269
		fi.xattrdirproc = acl_getxattrdir2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3270
		fip = &fi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3271
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3272
		fip = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3273
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3274
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3275
	rd.rd_entries = kmem_alloc(rdc->buflen, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3276
	rd.rd_size = count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3277
	rd.rd_offset = rda.rda_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3278
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3279
	douprintf = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3280
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3281
	if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3282
		mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3283
		kstat_runq_enter(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3284
		mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3285
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3286
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3287
	do {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3288
		rda.rda_count = MIN(count, mi->mi_curread);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3289
		error = rfs2call(mi, RFS_READDIR,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3290
		    xdr_rddirargs, (caddr_t)&rda,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3291
		    xdr_getrddirres, (caddr_t)&rd, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3292
		    &douprintf, &rd.rd_status, 0, fip);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3293
	} while (error == ENFS_TRYAGAIN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3294
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3295
	if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3296
		mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3297
		kstat_runq_exit(KSTAT_IO_PTR(mi->mi_io_kstats));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3298
		mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3299
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3300
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3301
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3302
	 * Since we are actually doing a READDIR RPC, we must have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3303
	 * exclusive access to the cache entry being filled.  Thus,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3304
	 * it is safe to update all fields except for the flags
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3305
	 * field.  The r_statelock in the rnode must be held to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3306
	 * prevent two different threads from simultaneously
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3307
	 * attempting to update the flags field.  This can happen
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3308
	 * if we are turning off RDDIR and the other thread is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3309
	 * trying to set RDDIRWAIT.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3310
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3311
	ASSERT(rdc->flags & RDDIR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3312
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3313
		error = geterrno(rd.rd_status);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3314
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3315
			rdc->nfs_ncookie = rd.rd_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3316
			rdc->eof = rd.rd_eof ? 1 : 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3317
			rdc->entlen = rd.rd_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3318
			ASSERT(rdc->entlen <= rdc->buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3319
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3320
			rdc->entries = rddir_cache_buf_alloc(rdc->buflen,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3321
			    KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3322
#else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3323
			rdc->entries = kmem_alloc(rdc->buflen, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3324
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3325
			bcopy(rd.rd_entries, rdc->entries, rdc->entlen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3326
			rdc->error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3327
			if (mi->mi_io_kstats) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3328
				mutex_enter(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3329
				KSTAT_IO_PTR(mi->mi_io_kstats)->reads++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3330
				KSTAT_IO_PTR(mi->mi_io_kstats)->nread +=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3331
				    rd.rd_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3332
				mutex_exit(&mi->mi_lock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3333
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3334
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3335
			PURGE_STALE_FH(error, vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3336
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3337
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3338
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3339
		rdc->entries = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3340
		rdc->error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3341
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3342
	kmem_free(rd.rd_entries, rdc->buflen);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3343
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3344
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3345
	rdc->flags &= ~RDDIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3346
	if (rdc->flags & RDDIRWAIT) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3347
		rdc->flags &= ~RDDIRWAIT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3348
		cv_broadcast(&rdc->cv);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3349
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3350
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3351
		rdc->flags |= RDDIRREQ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3352
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3353
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3354
	rddir_cache_rele(rdc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3355
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3356
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3357
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3358
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3359
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3360
static int nfs_bio_do_stop = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3361
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3362
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3363
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3364
nfs_bio(struct buf *bp, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3365
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3366
	rnode_t *rp = VTOR(bp->b_vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3367
	int count;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3368
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3369
	cred_t *cred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3370
	uint_t offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3371
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3372
	DTRACE_IO1(start, struct buf *, bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3373
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  3374
	ASSERT(nfs_zone() == VTOMI(bp->b_vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3375
	offset = dbtob(bp->b_blkno);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3376
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3377
	if (bp->b_flags & B_READ) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3378
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3379
		if (rp->r_cred != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3380
			cred = rp->r_cred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3381
			crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3382
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3383
			rp->r_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3384
			crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3385
			cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3386
			crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3387
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3388
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3389
	read_again:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3390
		error = bp->b_error = nfsread(bp->b_vp, bp->b_un.b_addr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3391
		    offset, bp->b_bcount, &bp->b_resid, cred);
7387
0b3a92e31fd8 PSARC 2007/347 NFS/RDMA - Transport Version Update
Robert Gordon <Robert.Gordon@Sun.COM>
parents: 7067
diff changeset
  3392
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3393
		crfree(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3394
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3395
			if (bp->b_resid) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3396
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3397
				 * Didn't get it all because we hit EOF,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3398
				 * zero all the memory beyond the EOF.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3399
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3400
				/* bzero(rdaddr + */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3401
				bzero(bp->b_un.b_addr +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3402
				    bp->b_bcount - bp->b_resid, bp->b_resid);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3403
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3404
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3405
			if (bp->b_resid == bp->b_bcount &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3406
			    offset >= rp->r_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3407
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3408
				 * We didn't read anything at all as we are
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3409
				 * past EOF.  Return an error indicator back
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3410
				 * but don't destroy the pages (yet).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3411
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3412
				error = NFS_EOF;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3413
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3414
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3415
		} else if (error == EACCES) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3416
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3417
			if (cred != cr) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3418
				if (rp->r_cred != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3419
					crfree(rp->r_cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3420
				rp->r_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3421
				crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3422
				cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3423
				crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3424
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3425
				goto read_again;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3426
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3427
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3428
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3429
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3430
		if (!(rp->r_flags & RSTALE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3431
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3432
			if (rp->r_cred != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3433
				cred = rp->r_cred;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3434
				crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3435
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3436
				rp->r_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3437
				crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3438
				cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3439
				crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3440
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3441
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3442
		write_again:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3443
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3444
			count = MIN(bp->b_bcount, rp->r_size - offset);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3445
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3446
			if (count < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3447
				cmn_err(CE_PANIC, "nfs_bio: write count < 0");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3448
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3449
			if (count == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3450
				zcmn_err(getzoneid(), CE_WARN,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3451
				    "nfs_bio: zero length write at %d",
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3452
				    offset);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3453
				nfs_printfhandle(&rp->r_fh);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3454
				if (nfs_bio_do_stop)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3455
					debug_enter("nfs_bio");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3456
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3457
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3458
			error = nfswrite(bp->b_vp, bp->b_un.b_addr, offset,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3459
			    count, cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3460
			if (error == EACCES) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3461
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3462
				if (cred != cr) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3463
					if (rp->r_cred != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3464
						crfree(rp->r_cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3465
					rp->r_cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3466
					crhold(cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3467
					crfree(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3468
					cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3469
					crhold(cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3470
					mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3471
					goto write_again;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3472
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3473
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3474
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3475
			bp->b_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3476
			if (error && error != EINTR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3477
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3478
				 * Don't print EDQUOT errors on the console.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3479
				 * Don't print asynchronous EACCES errors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3480
				 * Don't print EFBIG errors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3481
				 * Print all other write errors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3482
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3483
				if (error != EDQUOT && error != EFBIG &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3484
				    (error != EACCES ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3485
				    !(bp->b_flags & B_ASYNC)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3486
					nfs_write_error(bp->b_vp, error, cred);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3487
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3488
				 * Update r_error and r_flags as appropriate.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3489
				 * If the error was ESTALE, then mark the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3490
				 * rnode as not being writeable and save
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3491
				 * the error status.  Otherwise, save any
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3492
				 * errors which occur from asynchronous
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3493
				 * page invalidations.  Any errors occurring
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3494
				 * from other operations should be saved
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3495
				 * by the caller.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3496
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3497
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3498
				if (error == ESTALE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3499
					rp->r_flags |= RSTALE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3500
					if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3501
						rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3502
				} else if (!rp->r_error &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3503
				    (bp->b_flags &
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3504
				    (B_INVAL|B_FORCE|B_ASYNC)) ==
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3505
				    (B_INVAL|B_FORCE|B_ASYNC)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3506
					rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3507
				}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3508
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3509
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3510
			crfree(cred);
9776
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3511
		} else {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3512
			error = rp->r_error;
9776
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3513
			/*
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3514
			 * A close may have cleared r_error, if so,
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3515
			 * propagate ESTALE error return properly
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3516
			 */
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3517
			if (error == 0)
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3518
				error = ESTALE;
8dc7c135dda4 6843151 nfs[3/4]_write() can fail but return error 0
batschul <Frank.Batschulat@Sun.COM>
parents: 8059
diff changeset
  3519
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3520
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3521
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3522
	if (error != 0 && error != NFS_EOF)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3523
		bp->b_flags |= B_ERROR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3524
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3525
	DTRACE_IO1(done, struct buf *, bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3526
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3527
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3528
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3529
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3530
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3531
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3532
nfs_fid(vnode_t *vp, fid_t *fidp, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3533
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3534
	struct nfs_fid *fp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3535
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3536
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3537
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3538
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3539
	if (fidp->fid_len < (sizeof (struct nfs_fid) - sizeof (short))) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3540
		fidp->fid_len = sizeof (struct nfs_fid) - sizeof (short);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3541
		return (ENOSPC);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3542
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3543
	fp = (struct nfs_fid *)fidp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3544
	fp->nf_pad = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3545
	fp->nf_len = sizeof (struct nfs_fid) - sizeof (short);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3546
	bcopy(rp->r_fh.fh_buf, fp->nf_data, NFS_FHSIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3547
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3548
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3549
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3550
/* ARGSUSED2 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3551
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3552
nfs_rwlock(vnode_t *vp, int write_lock, caller_context_t *ctp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3553
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3554
	rnode_t *rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3555
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3556
	if (!write_lock) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3557
		(void) nfs_rw_enter_sig(&rp->r_rwlock, RW_READER, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3558
		return (V_WRITELOCK_FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3559
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3560
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3561
	if ((rp->r_flags & RDIRECTIO) || (VTOMI(vp)->mi_flags & MI_DIRECTIO)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3562
		(void) nfs_rw_enter_sig(&rp->r_rwlock, RW_READER, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3563
		if (rp->r_mapcnt == 0 && !vn_has_cached_data(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3564
			return (V_WRITELOCK_FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3565
		nfs_rw_exit(&rp->r_rwlock);
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
	(void) nfs_rw_enter_sig(&rp->r_rwlock, RW_WRITER, FALSE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3569
	return (V_WRITELOCK_TRUE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3570
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3571
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3572
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3573
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3574
nfs_rwunlock(vnode_t *vp, int write_lock, caller_context_t *ctp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3575
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3576
	rnode_t *rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3577
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3578
	nfs_rw_exit(&rp->r_rwlock);
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
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3582
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3583
nfs_seek(vnode_t *vp, offset_t ooff, offset_t *noffp, caller_context_t *ct)
0
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
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3587
	 * Because we stuff the readdir cookie into the offset field
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3588
	 * someone may attempt to do an lseek with the cookie which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3589
	 * we want to succeed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3590
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3591
	if (vp->v_type == VDIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3592
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3593
	if (*noffp < 0 || *noffp > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3594
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3595
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3596
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3597
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3598
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3599
 * number of NFS_MAXDATA blocks to read ahead
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3600
 * optimized for 100 base-T.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3601
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3602
static int nfs_nra = 4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3603
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3604
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3605
static int nfs_lostpage = 0;	/* number of times we lost original page */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3606
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3607
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3608
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3609
 * Return all the pages from [off..off+len) in file
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3610
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3611
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3612
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3613
nfs_getpage(vnode_t *vp, offset_t off, size_t len, uint_t *protp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3614
	page_t *pl[], size_t plsz, struct seg *seg, caddr_t addr,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3615
	enum seg_rw rw, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3616
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3617
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3618
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3619
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3620
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3621
	if (vp->v_flag & VNOMAP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3622
		return (ENOSYS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3623
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3624
	ASSERT(off <= MAXOFF32_T);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  3625
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3626
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3627
	if (protp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3628
		*protp = PROT_ALL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3629
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3630
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3631
	 * Now valididate that the caches are up to date.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3632
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3633
	error = nfs_validate_caches(vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3634
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3635
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3636
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3637
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3638
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3639
retry:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3640
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3641
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3642
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3643
	 * Don't create dirty pages faster than they
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3644
	 * can be cleaned so that the system doesn't
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3645
	 * get imbalanced.  If the async queue is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3646
	 * maxed out, then wait for it to drain before
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3647
	 * creating more dirty pages.  Also, wait for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3648
	 * any threads doing pagewalks in the vop_getattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3649
	 * entry points so that they don't block for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3650
	 * long periods.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3651
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3652
	if (rw == S_CREATE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3653
		while ((mi->mi_max_threads != 0 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3654
		    rp->r_awcount > 2 * mi->mi_max_threads) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3655
		    rp->r_gcount > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3656
			cv_wait(&rp->r_cv, &rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3657
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3658
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3659
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3660
	 * If we are getting called as a side effect of an nfs_write()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3661
	 * operation the local file size might not be extended yet.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3662
	 * In this case we want to be able to return pages of zeroes.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3663
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3664
	if (off + len > rp->r_size + PAGEOFFSET && seg != segkmap) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3665
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3666
		return (EFAULT);		/* beyond EOF */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3667
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3668
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3669
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3670
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3671
	if (len <= PAGESIZE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3672
		error = nfs_getapage(vp, off, len, protp, pl, plsz,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3673
		    seg, addr, rw, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3674
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3675
		error = pvn_getpages(nfs_getapage, vp, off, len, protp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3676
		    pl, plsz, seg, addr, rw, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3677
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3678
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3679
	switch (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3680
	case NFS_EOF:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3681
		nfs_purge_caches(vp, NFS_NOPURGE_DNLC, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3682
		goto retry;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3683
	case ESTALE:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3684
		PURGE_STALE_FH(error, vp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3685
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3686
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3687
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3688
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3689
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3690
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3691
 * Called from pvn_getpages or nfs_getpage to get a particular page.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3692
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3693
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3694
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3695
nfs_getapage(vnode_t *vp, u_offset_t off, size_t len, uint_t *protp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3696
	page_t *pl[], size_t plsz, struct seg *seg, caddr_t addr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3697
	enum seg_rw rw, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3698
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3699
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3700
	uint_t bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3701
	struct buf *bp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3702
	page_t *pp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3703
	u_offset_t lbn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3704
	u_offset_t io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3705
	u_offset_t blkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3706
	u_offset_t rablkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3707
	size_t io_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3708
	uint_t blksize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3709
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3710
	int readahead;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3711
	int readahead_issued = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3712
	int ra_window; /* readahead window */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3713
	page_t *pagefound;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3714
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  3715
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3716
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3717
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3718
	bsize = MAX(vp->v_vfsp->vfs_bsize, PAGESIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3719
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3720
reread:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3721
	bp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3722
	pp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3723
	pagefound = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3724
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3725
	if (pl != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3726
		pl[0] = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3727
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3728
	error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3729
	lbn = off / bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3730
	blkoff = lbn * bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3731
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3732
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3733
	 * Queueing up the readahead before doing the synchronous read
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3734
	 * results in a significant increase in read throughput because
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3735
	 * of the increased parallelism between the async threads and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3736
	 * the process context.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3737
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3738
	if ((off & ((vp->v_vfsp->vfs_bsize) - 1)) == 0 &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3739
	    rw != S_CREATE &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3740
	    !(vp->v_flag & VNOCACHE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3741
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3742
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3743
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3744
		 * Calculate the number of readaheads to do.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3745
		 * a) No readaheads at offset = 0.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3746
		 * b) Do maximum(nfs_nra) readaheads when the readahead
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3747
		 *    window is closed.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3748
		 * c) Do readaheads between 1 to (nfs_nra - 1) depending
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3749
		 *    upon how far the readahead window is open or close.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3750
		 * d) No readaheads if rp->r_nextr is not within the scope
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3751
		 *    of the readahead window (random i/o).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3752
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3753
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3754
		if (off == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3755
			readahead = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3756
		else if (blkoff == rp->r_nextr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3757
			readahead = nfs_nra;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3758
		else if (rp->r_nextr > blkoff &&
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3759
		    ((ra_window = (rp->r_nextr - blkoff) / bsize)
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3760
		    <= (nfs_nra - 1)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3761
			readahead = nfs_nra - ra_window;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3762
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3763
			readahead = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3764
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3765
		rablkoff = rp->r_nextr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3766
		while (readahead > 0 && rablkoff + bsize < rp->r_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3767
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3768
			if (nfs_async_readahead(vp, rablkoff + bsize,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3769
			    addr + (rablkoff + bsize - off), seg, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3770
			    nfs_readahead) < 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3771
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3772
				break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3773
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3774
			readahead--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3775
			rablkoff += bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3776
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3777
			 * Indicate that we did a readahead so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3778
			 * readahead offset is not updated
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3779
			 * by the synchronous read below.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3780
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3781
			readahead_issued = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3782
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3783
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3784
			 * set readahead offset to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3785
			 * offset of last async readahead
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3786
			 * request.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3787
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3788
			rp->r_nextr = rablkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3789
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3790
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3791
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3792
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3793
again:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3794
	if ((pagefound = page_exists(vp, off)) == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3795
		if (pl == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3796
			(void) nfs_async_readahead(vp, blkoff, addr, seg, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3797
			    nfs_readahead);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3798
		} else if (rw == S_CREATE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3799
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3800
			 * Block for this page is not allocated, or the offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3801
			 * is beyond the current allocation size, or we're
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3802
			 * allocating a swap slot and the page was not found,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3803
			 * so allocate it and return a zero page.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3804
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3805
			if ((pp = page_create_va(vp, off,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3806
			    PAGESIZE, PG_WAIT, seg, addr)) == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3807
				cmn_err(CE_PANIC, "nfs_getapage: page_create");
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3808
			io_len = PAGESIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3809
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3810
			rp->r_nextr = off + PAGESIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3811
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3812
		} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3813
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3814
			 * Need to go to server to get a BLOCK, exception to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3815
			 * that being while reading at offset = 0 or doing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3816
			 * random i/o, in that case read only a PAGE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3817
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3818
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3819
			if (blkoff < rp->r_size &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3820
			    blkoff + bsize >= rp->r_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3821
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3822
				 * If only a block or less is left in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3823
				 * the file, read all that is remaining.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3824
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3825
				if (rp->r_size <= off) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3826
					/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3827
					 * Trying to access beyond EOF,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3828
					 * set up to get at least one page.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3829
					 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3830
					blksize = off + PAGESIZE - blkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3831
				} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3832
					blksize = rp->r_size - blkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3833
			} else if ((off == 0) ||
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3834
			    (off != rp->r_nextr && !readahead_issued)) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3835
				blksize = PAGESIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3836
				blkoff = off; /* block = page here */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3837
			} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3838
				blksize = bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3839
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3840
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3841
			pp = pvn_read_kluster(vp, off, seg, addr, &io_off,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3842
			    &io_len, blkoff, blksize, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3843
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3844
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3845
			 * Some other thread has entered the page,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3846
			 * so just use it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3847
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3848
			if (pp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3849
				goto again;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3850
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3851
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3852
			 * Now round the request size up to page boundaries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3853
			 * This ensures that the entire page will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3854
			 * initialized to zeroes if EOF is encountered.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3855
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3856
			io_len = ptob(btopr(io_len));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3857
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3858
			bp = pageio_setup(pp, io_len, vp, B_READ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3859
			ASSERT(bp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3860
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3861
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3862
			 * pageio_setup should have set b_addr to 0.  This
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3863
			 * is correct since we want to do I/O on a page
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3864
			 * boundary.  bp_mapin will use this addr to calculate
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3865
			 * an offset, and then set b_addr to the kernel virtual
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3866
			 * address it allocated for us.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3867
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3868
			ASSERT(bp->b_un.b_addr == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3869
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3870
			bp->b_edev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3871
			bp->b_dev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3872
			bp->b_lblkno = lbtodb(io_off);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3873
			bp->b_file = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3874
			bp->b_offset = (offset_t)off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3875
			bp_mapin(bp);
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
			 * If doing a write beyond what we believe is EOF,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3879
			 * don't bother trying to read the pages from the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3880
			 * server, we'll just zero the pages here.  We
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3881
			 * don't check that the rw flag is S_WRITE here
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3882
			 * because some implementations may attempt a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3883
			 * read access to the buffer before copying data.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3884
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3885
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3886
			if (io_off >= rp->r_size && seg == segkmap) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3887
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3888
				bzero(bp->b_un.b_addr, io_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3889
			} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3890
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3891
				error = nfs_bio(bp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3892
			}
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
			 * Unmap the buffer before freeing it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3896
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3897
			bp_mapout(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3898
			pageio_done(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3899
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3900
			if (error == NFS_EOF) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3901
				/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3902
				 * If doing a write system call just return
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3903
				 * zeroed pages, else user tried to get pages
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3904
				 * beyond EOF, return error.  We don't check
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3905
				 * that the rw flag is S_WRITE here because
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3906
				 * some implementations may attempt a read
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3907
				 * access to the buffer before copying data.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3908
				 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3909
				if (seg == segkmap)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3910
					error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3911
				else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3912
					error = EFAULT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3913
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3914
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3915
			if (!readahead_issued && !error) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3916
				mutex_enter(&rp->r_statelock);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3917
				rp->r_nextr = io_off + io_len;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  3918
				mutex_exit(&rp->r_statelock);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3919
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3920
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3921
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3922
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3923
out:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3924
	if (pl == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3925
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3926
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3927
	if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3928
		if (pp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3929
			pvn_read_done(pp, B_ERROR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3930
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3931
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3932
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3933
	if (pagefound) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3934
		se_t se = (rw == S_CREATE ? SE_EXCL : SE_SHARED);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3935
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3936
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3937
		 * Page exists in the cache, acquire the appropriate lock.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3938
		 * If this fails, start all over again.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3939
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3940
		if ((pp = page_lookup(vp, off, se)) == NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3941
#ifdef DEBUG
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3942
			nfs_lostpage++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3943
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3944
			goto reread;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3945
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3946
		pl[0] = pp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3947
		pl[1] = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3948
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3949
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3950
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3951
	if (pp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3952
		pvn_plist_init(pp, pl, plsz, off, io_len, rw);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3953
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3954
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3955
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3956
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3957
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3958
nfs_readahead(vnode_t *vp, u_offset_t blkoff, caddr_t addr, struct seg *seg,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3959
	cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3960
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3961
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3962
	page_t *pp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3963
	u_offset_t io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3964
	size_t io_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3965
	struct buf *bp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3966
	uint_t bsize, blksize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3967
	rnode_t *rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3968
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  3969
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3970
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3971
	bsize = MAX(vp->v_vfsp->vfs_bsize, PAGESIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3972
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3973
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3974
	if (blkoff < rp->r_size && blkoff + bsize > rp->r_size) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3975
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3976
		 * If less than a block left in file read less
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3977
		 * than a block.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3978
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3979
		blksize = rp->r_size - blkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3980
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3981
		blksize = bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3982
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3983
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3984
	pp = pvn_read_kluster(vp, blkoff, segkmap, addr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3985
	    &io_off, &io_len, blkoff, blksize, 1);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3986
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3987
	 * The isra flag passed to the kluster function is 1, we may have
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3988
	 * gotten a return value of NULL for a variety of reasons (# of free
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3989
	 * pages < minfree, someone entered the page on the vnode etc). In all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3990
	 * cases, we want to punt on the readahead.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3991
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3992
	if (pp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3993
		return;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3994
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3995
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3996
	 * Now round the request size up to page boundaries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3997
	 * This ensures that the entire page will be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3998
	 * initialized to zeroes if EOF is encountered.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  3999
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4000
	io_len = ptob(btopr(io_len));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4001
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4002
	bp = pageio_setup(pp, io_len, vp, B_READ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4003
	ASSERT(bp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4004
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4005
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4006
	 * pageio_setup should have set b_addr to 0.  This is correct since
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4007
	 * we want to do I/O on a page boundary. bp_mapin() will use this addr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4008
	 * to calculate an offset, and then set b_addr to the kernel virtual
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4009
	 * address it allocated for us.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4010
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4011
	ASSERT(bp->b_un.b_addr == 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4012
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4013
	bp->b_edev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4014
	bp->b_dev = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4015
	bp->b_lblkno = lbtodb(io_off);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4016
	bp->b_file = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4017
	bp->b_offset = (offset_t)blkoff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4018
	bp_mapin(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4019
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4020
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4021
	 * If doing a write beyond what we believe is EOF, don't bother trying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4022
	 * to read the pages from the server, we'll just zero the pages here.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4023
	 * We don't check that the rw flag is S_WRITE here because some
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4024
	 * implementations may attempt a read access to the buffer before
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4025
	 * copying data.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4026
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4027
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4028
	if (io_off >= rp->r_size && seg == segkmap) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4029
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4030
		bzero(bp->b_un.b_addr, io_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4031
		error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4032
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4033
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4034
		error = nfs_bio(bp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4035
		if (error == NFS_EOF)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4036
			error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4037
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4038
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4039
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4040
	 * Unmap the buffer before freeing it.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4041
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4042
	bp_mapout(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4043
	pageio_done(bp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4044
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4045
	pvn_read_done(pp, error ? B_READ | B_ERROR : B_READ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4046
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4047
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4048
	 * In case of error set readahead offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4049
	 * to the lowest offset.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4050
	 * pvn_read_done() calls VN_DISPOSE to destroy the pages
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4051
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4052
	if (error && rp->r_nextr > io_off) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4053
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4054
		if (rp->r_nextr > io_off)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4055
			rp->r_nextr = io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4056
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4057
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4058
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4059
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4060
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4061
 * Flags are composed of {B_INVAL, B_FREE, B_DONTNEED, B_FORCE}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4062
 * If len == 0, do from off to EOF.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4063
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4064
 * The normal cases should be len == 0 && off == 0 (entire vp list),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4065
 * len == MAXBSIZE (from segmap_release actions), and len == PAGESIZE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4066
 * (from pageout).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4067
 */
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4068
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4069
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4070
nfs_putpage(vnode_t *vp, offset_t off, size_t len, int flags, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4071
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4072
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4073
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4074
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4075
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4076
	ASSERT(cr != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4077
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4078
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4079
	 * XXX - Why should this check be made here?
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4080
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4081
	if (vp->v_flag & VNOMAP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4082
		return (ENOSYS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4083
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4084
	if (len == 0 && !(flags & B_INVAL) && vn_is_readonly(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4085
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4086
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4087
	if (!(flags & B_ASYNC) && nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4088
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4089
	ASSERT(off <= MAXOFF32_T);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4090
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4091
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4092
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4093
	rp->r_count++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4094
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4095
	error = nfs_putpages(vp, off, len, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4096
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4097
	rp->r_count--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4098
	cv_broadcast(&rp->r_cv);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4099
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4100
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4101
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4102
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4103
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4104
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4105
 * Write out a single page, possibly klustering adjacent dirty pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4106
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4107
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4108
nfs_putapage(vnode_t *vp, page_t *pp, u_offset_t *offp, size_t *lenp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4109
	int flags, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4110
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4111
	u_offset_t io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4112
	u_offset_t lbn_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4113
	u_offset_t lbn;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4114
	size_t io_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4115
	uint_t bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4116
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4117
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4118
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4119
	ASSERT(!vn_is_readonly(vp));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4120
	ASSERT(pp != NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4121
	ASSERT(cr != NULL);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4122
	ASSERT((flags & B_ASYNC) || nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4123
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4124
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4125
	ASSERT(rp->r_count > 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4126
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4127
	ASSERT(pp->p_offset <= MAXOFF32_T);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4128
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4129
	bsize = MAX(vp->v_vfsp->vfs_bsize, PAGESIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4130
	lbn = pp->p_offset / bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4131
	lbn_off = lbn * bsize;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4132
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4133
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4134
	 * Find a kluster that fits in one block, or in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4135
	 * one page if pages are bigger than blocks.  If
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4136
	 * there is less file space allocated than a whole
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4137
	 * page, we'll shorten the i/o request below.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4138
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4139
	pp = pvn_write_kluster(vp, pp, &io_off, &io_len, lbn_off,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4140
	    roundup(bsize, PAGESIZE), flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4141
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4142
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4143
	 * pvn_write_kluster shouldn't have returned a page with offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4144
	 * behind the original page we were given.  Verify that.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4145
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4146
	ASSERT((pp->p_offset / bsize) >= lbn);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4147
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4148
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4149
	 * Now pp will have the list of kept dirty pages marked for
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4150
	 * write back.  It will also handle invalidation and freeing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4151
	 * of pages that are not dirty.  Check for page length rounding
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4152
	 * problems.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4153
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4154
	if (io_off + io_len > lbn_off + bsize) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4155
		ASSERT((io_off + io_len) - (lbn_off + bsize) < PAGESIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4156
		io_len = lbn_off + bsize - io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4157
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4158
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4159
	 * The RMODINPROGRESS flag makes sure that nfs(3)_bio() sees a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4160
	 * consistent value of r_size. RMODINPROGRESS is set in writerp().
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4161
	 * When RMODINPROGRESS is set it indicates that a uiomove() is in
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4162
	 * progress and the r_size has not been made consistent with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4163
	 * new size of the file. When the uiomove() completes the r_size is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4164
	 * updated and the RMODINPROGRESS flag is cleared.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4165
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4166
	 * The RMODINPROGRESS flag makes sure that nfs(3)_bio() sees a
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4167
	 * consistent value of r_size. Without this handshaking, it is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4168
	 * possible that nfs(3)_bio() picks  up the old value of r_size
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4169
	 * before the uiomove() in writerp() completes. This will result
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4170
	 * in the write through nfs(3)_bio() being dropped.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4171
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4172
	 * More precisely, there is a window between the time the uiomove()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4173
	 * completes and the time the r_size is updated. If a VOP_PUTPAGE()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4174
	 * operation intervenes in this window, the page will be picked up,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4175
	 * because it is dirty (it will be unlocked, unless it was
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4176
	 * pagecreate'd). When the page is picked up as dirty, the dirty
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4177
	 * bit is reset (pvn_getdirty()). In nfs(3)write(), r_size is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4178
	 * checked. This will still be the old size. Therefore the page will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4179
	 * not be written out. When segmap_release() calls VOP_PUTPAGE(),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4180
	 * the page will be found to be clean and the write will be dropped.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4181
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4182
	if (rp->r_flags & RMODINPROGRESS) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4183
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4184
		if ((rp->r_flags & RMODINPROGRESS) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4185
		    rp->r_modaddr + MAXBSIZE > io_off &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4186
		    rp->r_modaddr < io_off + io_len) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4187
			page_t *plist;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4188
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4189
			 * A write is in progress for this region of the file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4190
			 * If we did not detect RMODINPROGRESS here then this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4191
			 * path through nfs_putapage() would eventually go to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4192
			 * nfs(3)_bio() and may not write out all of the data
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4193
			 * in the pages. We end up losing data. So we decide
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4194
			 * to set the modified bit on each page in the page
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4195
			 * list and mark the rnode with RDIRTY. This write
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4196
			 * will be restarted at some later time.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4197
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4198
			plist = pp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4199
			while (plist != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4200
				pp = plist;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4201
				page_sub(&plist, pp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4202
				hat_setmod(pp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4203
				page_io_unlock(pp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4204
				page_unlock(pp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4205
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4206
			rp->r_flags |= RDIRTY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4207
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4208
			if (offp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4209
				*offp = io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4210
			if (lenp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4211
				*lenp = io_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4212
			return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4213
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4214
		mutex_exit(&rp->r_statelock);
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
	if (flags & B_ASYNC) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4218
		error = nfs_async_putapage(vp, pp, io_off, io_len, flags, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4219
		    nfs_sync_putapage);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4220
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4221
		error = nfs_sync_putapage(vp, pp, io_off, io_len, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4222
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4223
	if (offp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4224
		*offp = io_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4225
	if (lenp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4226
		*lenp = io_len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4227
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4228
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4229
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4230
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4231
nfs_sync_putapage(vnode_t *vp, page_t *pp, u_offset_t io_off, size_t io_len,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4232
	int flags, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4233
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4234
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4235
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4236
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4237
	flags |= B_WRITE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4238
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4239
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4240
	error = nfs_rdwrlbn(vp, pp, io_off, io_len, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4241
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4242
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4243
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4244
	if ((error == ENOSPC || error == EDQUOT || error == EACCES) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4245
	    (flags & (B_INVAL|B_FORCE)) != (B_INVAL|B_FORCE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4246
		if (!(rp->r_flags & ROUTOFSPACE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4247
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4248
			rp->r_flags |= ROUTOFSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4249
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4250
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4251
		flags |= B_ERROR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4252
		pvn_write_done(pp, flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4253
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4254
		 * If this was not an async thread, then try again to
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4255
		 * write out the pages, but this time, also destroy
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4256
		 * them whether or not the write is successful.  This
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4257
		 * will prevent memory from filling up with these
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4258
		 * pages and destroying them is the only alternative
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4259
		 * if they can't be written out.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4260
		 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4261
		 * Don't do this if this is an async thread because
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4262
		 * when the pages are unlocked in pvn_write_done,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4263
		 * some other thread could have come along, locked
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4264
		 * them, and queued for an async thread.  It would be
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4265
		 * possible for all of the async threads to be tied
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4266
		 * up waiting to lock the pages again and they would
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4267
		 * all already be locked and waiting for an async
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4268
		 * thread to handle them.  Deadlock.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4269
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4270
		if (!(flags & B_ASYNC)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4271
			error = nfs_putpage(vp, io_off, io_len,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4272
			    B_INVAL | B_FORCE, cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4273
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4274
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4275
		if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4276
			flags |= B_ERROR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4277
		else if (rp->r_flags & ROUTOFSPACE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4278
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4279
			rp->r_flags &= ~ROUTOFSPACE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4280
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4281
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4282
		pvn_write_done(pp, flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4283
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4284
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4285
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4286
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4287
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4288
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4289
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4290
nfs_map(vnode_t *vp, offset_t off, struct as *as, caddr_t *addrp,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4291
	size_t len, uchar_t prot, uchar_t maxprot, uint_t flags, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4292
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4293
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4294
	struct segvn_crargs vn_a;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4295
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4296
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4297
	struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4298
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4299
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4300
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4301
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4302
	if (vp->v_flag & VNOMAP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4303
		return (ENOSYS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4304
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4305
	if (off > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4306
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4307
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4308
	if (off < 0 || off + len < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4309
		return (ENXIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4310
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4311
	if (vp->v_type != VREG)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4312
		return (ENODEV);
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 there is cached data and if close-to-open consistency
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4316
	 * checking is not turned off and if the file system is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4317
	 * mounted readonly, then force an over the wire getattr.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4318
	 * Otherwise, just invoke nfsgetattr to get a copy of the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4319
	 * attributes.  The attribute cache will be used unless it
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4320
	 * is timed out and if it is, then an over the wire getattr
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4321
	 * will be issued.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4322
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4323
	va.va_mask = AT_ALL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4324
	if (vn_has_cached_data(vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4325
	    !(VTOMI(vp)->mi_flags & MI_NOCTO) && !vn_is_readonly(vp))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4326
		error = nfs_getattr_otw(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4327
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4328
		error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4329
	if (error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4330
		return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4331
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4332
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4333
	 * Check to see if the vnode is currently marked as not cachable.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4334
	 * This means portions of the file are locked (through VOP_FRLOCK).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4335
	 * In this case the map request must be refused.  We use
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4336
	 * rp->r_lkserlock to avoid a race with concurrent lock requests.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4337
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4338
	rp = VTOR(vp);
7689
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4339
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4340
	/*
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4341
	 * Atomically increment r_inmap after acquiring r_rwlock. The
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4342
	 * idea here is to acquire r_rwlock to block read/write and
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4343
	 * not to protect r_inmap. r_inmap will inform nfs_read/write()
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4344
	 * that we are in nfs_map(). Now, r_rwlock is acquired in order
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4345
	 * and we can prevent the deadlock that would have occurred
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4346
	 * when nfs_addmap() would have acquired it out of order.
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4347
	 *
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4348
	 * Since we are not protecting r_inmap by any lock, we do not
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4349
	 * hold any lock when we decrement it. We atomically decrement
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4350
	 * r_inmap after we release r_lkserlock.
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4351
	 */
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4352
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4353
	if (nfs_rw_enter_sig(&rp->r_rwlock, RW_WRITER, INTR(vp)))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4354
		return (EINTR);
7689
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4355
	atomic_add_int(&rp->r_inmap, 1);
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4356
	nfs_rw_exit(&rp->r_rwlock);
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4357
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4358
	if (nfs_rw_enter_sig(&rp->r_lkserlock, RW_READER, INTR(vp))) {
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4359
		atomic_add_int(&rp->r_inmap, -1);
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4360
		return (EINTR);
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4361
	}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4362
	if (vp->v_flag & VNOCACHE) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4363
		error = EAGAIN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4364
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4365
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4366
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4367
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4368
	 * Don't allow concurrent locks and mapping if mandatory locking is
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4369
	 * enabled.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4370
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4371
	if ((flk_has_remote_locks(vp) || lm_has_sleep(vp)) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4372
	    MANDLOCK(vp, va.va_mode)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4373
		error = EAGAIN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4374
		goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4375
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4376
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4377
	as_rangelock(as);
6036
c98c367c32cb PSARC 2006/123 Mmap compatibility
mec
parents: 5486
diff changeset
  4378
	error = choose_addr(as, addrp, len, off, ADDR_VACALIGN, flags);
c98c367c32cb PSARC 2006/123 Mmap compatibility
mec
parents: 5486
diff changeset
  4379
	if (error != 0) {
c98c367c32cb PSARC 2006/123 Mmap compatibility
mec
parents: 5486
diff changeset
  4380
		as_rangeunlock(as);
c98c367c32cb PSARC 2006/123 Mmap compatibility
mec
parents: 5486
diff changeset
  4381
		goto done;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4382
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4383
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4384
	vn_a.vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4385
	vn_a.offset = off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4386
	vn_a.type = (flags & MAP_TYPE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4387
	vn_a.prot = (uchar_t)prot;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4388
	vn_a.maxprot = (uchar_t)maxprot;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4389
	vn_a.flags = (flags & ~MAP_TYPE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4390
	vn_a.cred = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4391
	vn_a.amp = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4392
	vn_a.szc = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4393
	vn_a.lgrp_mem_policy_flags = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4394
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4395
	error = as_map(as, *addrp, len, segvn_create, &vn_a);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4396
	as_rangeunlock(as);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4397
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4398
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4399
	nfs_rw_exit(&rp->r_lkserlock);
7689
fb0773b77db2 6503547 deadlock between utilities accessing /proc and processes using nfs
Deepak Honnalli <Deepak.Honnalli@Sun.COM>
parents: 7387
diff changeset
  4400
	atomic_add_int(&rp->r_inmap, -1);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4401
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4402
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4403
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4404
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4405
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4406
nfs_addmap(vnode_t *vp, offset_t off, struct as *as, caddr_t addr,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4407
	size_t len, uchar_t prot, uchar_t maxprot, uint_t flags, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4408
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4409
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4410
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4411
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4412
	if (vp->v_flag & VNOMAP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4413
		return (ENOSYS);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4414
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4415
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4416
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4417
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4418
	atomic_add_long((ulong_t *)&rp->r_mapcnt, btopr(len));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4419
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4420
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4421
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4422
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4423
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4424
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4425
nfs_frlock(vnode_t *vp, int cmd, struct flock64 *bfp, int flag, offset_t offset,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4426
	struct flk_callback *flk_cbp, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4427
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4428
	netobj lm_fh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4429
	int rc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4430
	u_offset_t start, end;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4431
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4432
	int error = 0, intr = INTR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4433
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4434
	/* check for valid cmd parameter */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4435
	if (cmd != F_GETLK && cmd != F_SETLK && cmd != F_SETLKW)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4436
		return (EINVAL);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4437
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4438
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4439
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4440
	/* Verify l_type. */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4441
	switch (bfp->l_type) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4442
	case F_RDLCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4443
		if (cmd != F_GETLK && !(flag & FREAD))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4444
			return (EBADF);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4445
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4446
	case F_WRLCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4447
		if (cmd != F_GETLK && !(flag & FWRITE))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4448
			return (EBADF);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4449
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4450
	case F_UNLCK:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4451
		intr = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4452
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4453
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4454
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4455
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4456
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4457
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4458
	/* check the validity of the lock range */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4459
	if (rc = flk_convert_lock_data(vp, bfp, &start, &end, offset))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4460
		return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4461
	if (rc = flk_check_lock_data(start, end, MAXOFF32_T))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4462
		return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4463
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4464
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4465
	 * If the filesystem is mounted using local locking, pass the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4466
	 * request off to the local locking code.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4467
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4468
	if (VTOMI(vp)->mi_flags & MI_LLOCK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4469
		if (offset > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4470
			return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4471
		if (cmd == F_SETLK || cmd == F_SETLKW) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4472
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4473
			 * For complete safety, we should be holding
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4474
			 * r_lkserlock.  However, we can't call
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4475
			 * lm_safelock and then fs_frlock while
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4476
			 * holding r_lkserlock, so just invoke
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4477
			 * lm_safelock and expect that this will
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4478
			 * catch enough of the cases.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4479
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4480
			if (!lm_safelock(vp, bfp, cr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4481
				return (EAGAIN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4482
		}
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4483
		return (fs_frlock(vp, cmd, bfp, flag, offset, flk_cbp, cr, ct));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4484
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4485
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4486
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4487
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4488
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4489
	 * Check whether the given lock request can proceed, given the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4490
	 * current file mappings.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4491
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4492
	if (nfs_rw_enter_sig(&rp->r_lkserlock, RW_WRITER, intr))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4493
		return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4494
	if (cmd == F_SETLK || cmd == F_SETLKW) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4495
		if (!lm_safelock(vp, bfp, cr)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4496
			rc = EAGAIN;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4497
			goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4498
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4499
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4500
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4501
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4502
	 * Flush the cache after waiting for async I/O to finish.  For new
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4503
	 * locks, this is so that the process gets the latest bits from the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4504
	 * server.  For unlocks, this is so that other clients see the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4505
	 * latest bits once the file has been unlocked.  If currently dirty
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4506
	 * pages can't be flushed, then don't allow a lock to be set.  But
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4507
	 * allow unlocks to succeed, to avoid having orphan locks on the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4508
	 * server.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4509
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4510
	if (cmd != F_GETLK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4511
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4512
		while (rp->r_count > 0) {
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4513
			if (intr) {
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4514
				klwp_t *lwp = ttolwp(curthread);
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4515
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4516
				if (lwp != NULL)
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4517
					lwp->lwp_nostop++;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4518
				if (cv_wait_sig(&rp->r_cv, &rp->r_statelock)
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4519
				    == 0) {
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4520
					if (lwp != NULL)
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4521
						lwp->lwp_nostop--;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4522
					rc = EINTR;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4523
					break;
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4524
				}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4525
				if (lwp != NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4526
					lwp->lwp_nostop--;
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4527
			} else
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4528
			cv_wait(&rp->r_cv, &rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4529
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4530
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4531
		if (rc != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4532
			goto done;
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4533
		error = nfs_putpage(vp, (offset_t)0, 0, B_INVAL, cr, ct);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4534
		if (error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4535
			if (error == ENOSPC || error == EDQUOT) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4536
				mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4537
				if (!rp->r_error)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4538
					rp->r_error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4539
				mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4540
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4541
			if (bfp->l_type != F_UNLCK) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4542
				rc = ENOLCK;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4543
				goto done;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4544
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4545
		}
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
	lm_fh.n_len = sizeof (fhandle_t);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4549
	lm_fh.n_bytes = (char *)VTOFH(vp);
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
	 * Call the lock manager to do the real work of contacting
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4553
	 * the server and obtaining the lock.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4554
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4555
	rc = lm_frlock(vp, cmd, bfp, flag, offset, cr, &lm_fh, flk_cbp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4556
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4557
	if (rc == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4558
		nfs_lockcompletion(vp, cmd);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4559
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4560
done:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4561
	nfs_rw_exit(&rp->r_lkserlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4562
	return (rc);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4563
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4564
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4565
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4566
 * Free storage space associated with the specified vnode.  The portion
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4567
 * to be freed is specified by bfp->l_start and bfp->l_len (already
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4568
 * normalized to a "whence" of 0).
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4569
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4570
 * This is an experimental facility whose continued existence is not
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4571
 * guaranteed.  Currently, we only support the special case
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4572
 * of l_len == 0, meaning free to end of file.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4573
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4574
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4575
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4576
nfs_space(vnode_t *vp, int cmd, struct flock64 *bfp, int flag,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4577
	offset_t offset, cred_t *cr, caller_context_t *ct)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4578
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4579
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4580
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4581
	ASSERT(vp->v_type == VREG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4582
	if (cmd != F_FREESP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4583
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4584
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4585
	if (offset > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4586
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4587
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4588
	if ((bfp->l_start > MAXOFF32_T) || (bfp->l_end > MAXOFF32_T) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4589
	    (bfp->l_len > MAXOFF32_T))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4590
		return (EFBIG);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4591
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4592
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4593
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4594
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4595
	error = convoff(vp, bfp, 0, offset);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4596
	if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4597
		ASSERT(bfp->l_start >= 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4598
		if (bfp->l_len == 0) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4599
			struct vattr va;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4600
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4601
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4602
			 * ftruncate should not change the ctime and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4603
			 * mtime if we truncate the file to its
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4604
			 * previous size.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4605
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4606
			va.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4607
			error = nfsgetattr(vp, &va, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4608
			if (error || va.va_size == bfp->l_start)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4609
				return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4610
			va.va_mask = AT_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4611
			va.va_size = bfp->l_start;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4612
			error = nfssetattr(vp, &va, 0, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4613
		} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4614
			error = EINVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4615
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4616
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4617
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4618
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4619
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4620
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4621
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4622
nfs_realvp(vnode_t *vp, vnode_t **vpp, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4623
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4624
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4625
	return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4626
}
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
 * Setup and add an address space callback to do the work of the delmap call.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4630
 * The callback will (and must be) deleted in the actual callback function.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4631
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4632
 * This is done in order to take care of the problem that we have with holding
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4633
 * the address space's a_lock for a long period of time (e.g. if the NFS server
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4634
 * is down).  Callbacks will be executed in the address space code while the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4635
 * a_lock is not held.	Holding the address space's a_lock causes things such
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4636
 * as ps and fork to hang because they are trying to acquire this lock as well.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4637
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4638
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4639
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4640
nfs_delmap(vnode_t *vp, offset_t off, struct as *as, caddr_t addr,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4641
	size_t len, uint_t prot, uint_t maxprot, uint_t flags, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4642
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4643
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4644
	int			caller_found;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4645
	int			error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4646
	rnode_t			*rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4647
	nfs_delmap_args_t	*dmapp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4648
	nfs_delmapcall_t	*delmap_call;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4649
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4650
	if (vp->v_flag & VNOMAP)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4651
		return (ENOSYS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4652
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4653
	 * A process may not change zones if it has NFS pages mmap'ed
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4654
	 * in, so we can't legitimately get here from the wrong zone.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4655
	 */
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4656
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4657
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4658
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4659
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4660
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4661
	 * The way that the address space of this process deletes its mapping
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4662
	 * of this file is via the following call chains:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4663
	 * - as_free()->SEGOP_UNMAP()/segvn_unmap()->VOP_DELMAP()/nfs_delmap()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4664
	 * - as_unmap()->SEGOP_UNMAP()/segvn_unmap()->VOP_DELMAP()/nfs_delmap()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4665
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4666
	 * With the use of address space callbacks we are allowed to drop the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4667
	 * address space lock, a_lock, while executing the NFS operations that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4668
	 * need to go over the wire.  Returning EAGAIN to the caller of this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4669
	 * function is what drives the execution of the callback that we add
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4670
	 * below.  The callback will be executed by the address space code
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4671
	 * after dropping the a_lock.  When the callback is finished, since
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4672
	 * we dropped the a_lock, it must be re-acquired and segvn_unmap()
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4673
	 * is called again on the same segment to finish the rest of the work
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4674
	 * that needs to happen during unmapping.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4675
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4676
	 * This action of calling back into the segment driver causes
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4677
	 * nfs_delmap() to get called again, but since the callback was
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4678
	 * already executed at this point, it already did the work and there
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4679
	 * is nothing left for us to do.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4680
	 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4681
	 * To Summarize:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4682
	 * - The first time nfs_delmap is called by the current thread is when
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4683
	 * we add the caller associated with this delmap to the delmap caller
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4684
	 * list, add the callback, and return EAGAIN.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4685
	 * - The second time in this call chain when nfs_delmap is called we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4686
	 * will find this caller in the delmap caller list and realize there
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4687
	 * is no more work to do thus removing this caller from the list and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4688
	 * returning the error that was set in the callback execution.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4689
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4690
	caller_found = nfs_find_and_delete_delmapcall(rp, &error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4691
	if (caller_found) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4692
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4693
		 * 'error' is from the actual delmap operations.  To avoid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4694
		 * hangs, we need to handle the return of EAGAIN differently
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4695
		 * since this is what drives the callback execution.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4696
		 * In this case, we don't want to return EAGAIN and do the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4697
		 * callback execution because there are none to execute.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4698
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4699
		if (error == EAGAIN)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4700
			return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4701
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4702
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4703
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4704
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4705
	/* current caller was not in the list */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4706
	delmap_call = nfs_init_delmapcall();
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4707
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4708
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4709
	list_insert_tail(&rp->r_indelmap, delmap_call);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4710
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4711
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4712
	dmapp = kmem_alloc(sizeof (nfs_delmap_args_t), KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4713
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4714
	dmapp->vp = vp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4715
	dmapp->off = off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4716
	dmapp->addr = addr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4717
	dmapp->len = len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4718
	dmapp->prot = prot;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4719
	dmapp->maxprot = maxprot;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4720
	dmapp->flags = flags;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4721
	dmapp->cr = cr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4722
	dmapp->caller = delmap_call;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4723
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4724
	error = as_add_callback(as, nfs_delmap_callback, dmapp,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4725
	    AS_UNMAP_EVENT, addr, len, KM_SLEEP);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4726
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4727
	return (error ? error : EAGAIN);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4728
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4729
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4730
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4731
 * Remove some pages from an mmap'd vnode.  Just update the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4732
 * count of pages.  If doing close-to-open, then flush all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4733
 * of the pages associated with this file.  Otherwise, start
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4734
 * an asynchronous page flush to write out any dirty pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4735
 * This will also associate a credential with the rnode which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4736
 * can be used to write the pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4737
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4738
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4739
static void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4740
nfs_delmap_callback(struct as *as, void *arg, uint_t event)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4741
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4742
	int			error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4743
	rnode_t			*rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4744
	mntinfo_t		*mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4745
	nfs_delmap_args_t	*dmapp = (nfs_delmap_args_t *)arg;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4746
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4747
	rp = VTOR(dmapp->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4748
	mi = VTOMI(dmapp->vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4749
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4750
	atomic_add_long((ulong_t *)&rp->r_mapcnt, -btopr(dmapp->len));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4751
	ASSERT(rp->r_mapcnt >= 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4752
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4753
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4754
	 * Initiate a page flush if there are pages, the file system
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4755
	 * was not mounted readonly, the segment was mapped shared, and
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4756
	 * the pages themselves were writeable.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4757
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4758
	if (vn_has_cached_data(dmapp->vp) && !vn_is_readonly(dmapp->vp) &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4759
	    dmapp->flags == MAP_SHARED && (dmapp->maxprot & PROT_WRITE)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4760
		mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4761
		rp->r_flags |= RDIRTY;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4762
		mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4763
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4764
		 * If this is a cross-zone access a sync putpage won't work, so
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4765
		 * the best we can do is try an async putpage.  That seems
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4766
		 * better than something more draconian such as discarding the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4767
		 * dirty pages.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4768
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4769
		if ((mi->mi_flags & MI_NOCTO) ||
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4770
		    nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4771
			error = nfs_putpage(dmapp->vp, dmapp->off, dmapp->len,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4772
			    B_ASYNC, dmapp->cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4773
		else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4774
			error = nfs_putpage(dmapp->vp, dmapp->off, dmapp->len,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4775
			    0, dmapp->cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4776
		if (!error) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4777
			mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4778
			error = rp->r_error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4779
			rp->r_error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4780
			mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4781
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4782
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4783
		error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4784
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4785
	if ((rp->r_flags & RDIRECTIO) || (mi->mi_flags & MI_DIRECTIO))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4786
		(void) nfs_putpage(dmapp->vp, dmapp->off, dmapp->len,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4787
		    B_INVAL, dmapp->cr, NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4788
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4789
	dmapp->caller->error = error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4790
	(void) as_delete_callback(as, arg);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4791
	kmem_free(dmapp, sizeof (nfs_delmap_args_t));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4792
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4793
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4794
/* ARGSUSED */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4795
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4796
nfs_pathconf(vnode_t *vp, int cmd, ulong_t *valp, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4797
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4798
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4799
	int error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4800
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4801
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4802
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4803
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4804
	 * This looks a little weird because it's written in a general
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4805
	 * manner but we make little use of cases.  If cntl() ever gets
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4806
	 * widely used, the outer switch will make more sense.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4807
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4808
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4809
	switch (cmd) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4810
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4811
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4812
	 * Large file spec - need to base answer new query with
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4813
	 * hardcoded constant based on the protocol.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4814
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4815
	case _PC_FILESIZEBITS:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4816
		*valp = 32;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4817
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4818
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4819
	case _PC_LINK_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4820
	case _PC_NAME_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4821
	case _PC_PATH_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4822
	case _PC_SYMLINK_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4823
	case _PC_CHOWN_RESTRICTED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4824
	case _PC_NO_TRUNC: {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4825
		mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4826
		struct pathcnf *pc;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4827
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4828
		if ((mi = VTOMI(vp)) == NULL || (pc = mi->mi_pathconf) == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4829
			return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4830
		error = _PC_ISSET(cmd, pc->pc_mask);    /* error or bool */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4831
		switch (cmd) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4832
		case _PC_LINK_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4833
			*valp = pc->pc_link_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4834
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4835
		case _PC_NAME_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4836
			*valp = pc->pc_name_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4837
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4838
		case _PC_PATH_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4839
		case _PC_SYMLINK_MAX:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4840
			*valp = pc->pc_path_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4841
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4842
		case _PC_CHOWN_RESTRICTED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4843
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4844
			 * if we got here, error is really a boolean which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4845
			 * indicates whether cmd is set or not.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4846
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4847
			*valp = error ? 1 : 0;	/* see above */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4848
			error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4849
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4850
		case _PC_NO_TRUNC:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4851
			/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4852
			 * if we got here, error is really a boolean which
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4853
			 * indicates whether cmd is set or not.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4854
			 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4855
			*valp = error ? 1 : 0;	/* see above */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4856
			error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4857
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4858
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4859
		return (error ? EINVAL : 0);
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4860
		}
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4861
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4862
	case _PC_XATTR_EXISTS:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4863
		*valp = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4864
		if (vp->v_vfsp->vfs_flag & VFS_XATTR) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4865
			vnode_t *avp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4866
			rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4867
			mntinfo_t *mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4868
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4869
			if (!(mi->mi_flags & MI_EXTATTR))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4870
				return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4871
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4872
			rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4873
			if (nfs_rw_enter_sig(&rp->r_rwlock, RW_READER,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4874
			    INTR(vp)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4875
				return (EINTR);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4876
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4877
			error = nfslookup_dnlc(vp, XATTR_DIR_NAME, &avp, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4878
			if (error || avp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4879
				error = acl_getxattrdir2(vp, &avp, 0, cr, 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4880
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4881
			nfs_rw_exit(&rp->r_rwlock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4882
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4883
			if (error == 0 && avp != NULL) {
7067
09d6b97e1874 6683575 Revisit handling of CREATE_XATTR_DIR
marks
parents: 6036
diff changeset
  4884
				error = do_xattr_exists_check(avp, valp, cr);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4885
				VN_RELE(avp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4886
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4887
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4888
		return (error ? EINVAL : 0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4889
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4890
	case _PC_ACL_ENABLED:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4891
		*valp = _ACL_ACLENT_ENABLED;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4892
		return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4893
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4894
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4895
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4896
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4897
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4898
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4899
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4900
 * Called by async thread to do synchronous pageio. Do the i/o, wait
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4901
 * for it to complete, and cleanup the page list when done.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4902
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4903
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4904
nfs_sync_pageio(vnode_t *vp, page_t *pp, u_offset_t io_off, size_t io_len,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4905
	int flags, cred_t *cr)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4906
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4907
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4908
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4909
	ASSERT(nfs_zone() == VTOMI(vp)->mi_zone);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4910
	error = nfs_rdwrlbn(vp, pp, io_off, io_len, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4911
	if (flags & B_READ)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4912
		pvn_read_done(pp, (error ? B_ERROR : 0) | flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4913
	else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4914
		pvn_write_done(pp, (error ? B_ERROR : 0) | flags);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4915
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4916
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4917
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4918
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4919
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4920
nfs_pageio(vnode_t *vp, page_t *pp, u_offset_t io_off, size_t io_len,
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4921
	int flags, cred_t *cr, caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4922
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4923
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4924
	rnode_t *rp;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4925
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4926
	if (pp == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4927
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4928
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4929
	if (io_off > MAXOFF32_T)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4930
		return (EFBIG);
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4931
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4932
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4933
	rp = VTOR(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4934
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4935
	rp->r_count++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4936
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4937
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4938
	if (flags & B_ASYNC) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4939
		error = nfs_async_pageio(vp, pp, io_off, io_len, flags, cr,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4940
		    nfs_sync_pageio);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4941
	} else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4942
		error = nfs_rdwrlbn(vp, pp, io_off, io_len, flags, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4943
	mutex_enter(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4944
	rp->r_count--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4945
	cv_broadcast(&rp->r_cv);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4946
	mutex_exit(&rp->r_statelock);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4947
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4948
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4949
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4950
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4951
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4952
nfs_setsecattr(vnode_t *vp, vsecattr_t *vsecattr, int flag, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4953
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4954
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4955
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4956
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4957
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4958
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4959
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4960
	if (nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4961
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4962
	if (mi->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4963
		error = acl_setacl2(vp, vsecattr, flag, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4964
		if (mi->mi_flags & MI_ACL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4965
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4966
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4967
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4968
	return (ENOSYS);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4969
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4970
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4971
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4972
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4973
nfs_getsecattr(vnode_t *vp, vsecattr_t *vsecattr, int flag, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4974
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4975
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4976
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4977
	mntinfo_t *mi;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4978
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4979
	mi = VTOMI(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4980
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  4981
	if (nfs_zone() != mi->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4982
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4983
	if (mi->mi_flags & MI_ACL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4984
		error = acl_getacl2(vp, vsecattr, flag, cr);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4985
		if (mi->mi_flags & MI_ACL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4986
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4987
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4988
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4989
	return (fs_fab_acl(vp, vsecattr, flag, cr, ct));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4990
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4991
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4992
/* ARGSUSED */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4993
static int
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4994
nfs_shrlock(vnode_t *vp, int cmd, struct shrlock *shr, int flag, cred_t *cr,
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  4995
	caller_context_t *ct)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4996
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4997
	int error;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4998
	struct shrlock nshr;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  4999
	struct nfs_owner nfs_owner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5000
	netobj lm_fh;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5001
766
c521de78a32f PSARC 2005/474 Zones Upgrade (Ashanti and Zulu)
carlsonj
parents: 0
diff changeset
  5002
	if (nfs_zone() != VTOMI(vp)->mi_zone)
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5003
		return (EIO);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5004
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5005
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5006
	 * check for valid cmd parameter
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5007
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5008
	if (cmd != F_SHARE && cmd != F_UNSHARE && cmd != F_HASREMOTELOCKS)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5009
		return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5010
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5011
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5012
	 * Check access permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5013
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5014
	if (cmd == F_SHARE &&
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5015
	    (((shr->s_access & F_RDACC) && !(flag & FREAD)) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5016
	    ((shr->s_access & F_WRACC) && !(flag & FWRITE))))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5017
		return (EBADF);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5018
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5019
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5020
	 * If the filesystem is mounted using local locking, pass the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5021
	 * request off to the local share code.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5022
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5023
	if (VTOMI(vp)->mi_flags & MI_LLOCK)
5331
3047ad28a67b PSARC/2007/218 caller_context_t in all VOPs
amw
parents: 4863
diff changeset
  5024
		return (fs_shrlock(vp, cmd, shr, flag, cr, ct));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5025
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5026
	switch (cmd) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5027
	case F_SHARE:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5028
	case F_UNSHARE:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5029
		lm_fh.n_len = sizeof (fhandle_t);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5030
		lm_fh.n_bytes = (char *)VTOFH(vp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5031
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5032
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5033
		 * If passed an owner that is too large to fit in an
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5034
		 * nfs_owner it is likely a recursive call from the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5035
		 * lock manager client and pass it straight through.  If
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5036
		 * it is not a nfs_owner then simply return an error.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5037
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5038
		if (shr->s_own_len > sizeof (nfs_owner.lowner)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5039
			if (((struct nfs_owner *)shr->s_owner)->magic !=
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5040
			    NFS_OWNER_MAGIC)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5041
				return (EINVAL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5042
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5043
			if (error = lm_shrlock(vp, cmd, shr, flag, &lm_fh)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5044
				error = set_errno(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5045
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5046
			return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5047
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5048
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5049
		 * Remote share reservations owner is a combination of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5050
		 * a magic number, hostname, and the local owner
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5051
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5052
		bzero(&nfs_owner, sizeof (nfs_owner));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5053
		nfs_owner.magic = NFS_OWNER_MAGIC;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5054
		(void) strncpy(nfs_owner.hname, uts_nodename(),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5055
		    sizeof (nfs_owner.hname));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5056
		bcopy(shr->s_owner, nfs_owner.lowner, shr->s_own_len);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5057
		nshr.s_access = shr->s_access;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5058
		nshr.s_deny = shr->s_deny;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5059
		nshr.s_sysid = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5060
		nshr.s_pid = ttoproc(curthread)->p_pid;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5061
		nshr.s_own_len = sizeof (nfs_owner);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5062
		nshr.s_owner = (caddr_t)&nfs_owner;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5063
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5064
		if (error = lm_shrlock(vp, cmd, &nshr, flag, &lm_fh)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5065
			error = set_errno(error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5066
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5067
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5068
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5069
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5070
	case F_HASREMOTELOCKS:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5071
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5072
		 * NFS client can't store remote locks itself
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5073
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5074
		shr->s_access = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5075
		error = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5076
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5077
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5078
	default:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5079
		error = EINVAL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5080
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5081
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5082
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5083
	return (error);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
  5084
}