xref: /OK3568_Linux_fs/kernel/fs/nfsd/nfs2acl.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Process version 2 NFSACL requests.
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Copyright (C) 2002-2003 Andreas Gruenbacher <agruen@suse.de>
6*4882a593Smuzhiyun  */
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include "nfsd.h"
9*4882a593Smuzhiyun /* FIXME: nfsacl.h is a broken header */
10*4882a593Smuzhiyun #include <linux/nfsacl.h>
11*4882a593Smuzhiyun #include <linux/gfp.h>
12*4882a593Smuzhiyun #include "cache.h"
13*4882a593Smuzhiyun #include "xdr3.h"
14*4882a593Smuzhiyun #include "vfs.h"
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun #define NFSDDBG_FACILITY		NFSDDBG_PROC
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun /*
19*4882a593Smuzhiyun  * NULL call.
20*4882a593Smuzhiyun  */
21*4882a593Smuzhiyun static __be32
nfsacld_proc_null(struct svc_rqst * rqstp)22*4882a593Smuzhiyun nfsacld_proc_null(struct svc_rqst *rqstp)
23*4882a593Smuzhiyun {
24*4882a593Smuzhiyun 	return rpc_success;
25*4882a593Smuzhiyun }
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun /*
28*4882a593Smuzhiyun  * Get the Access and/or Default ACL of a file.
29*4882a593Smuzhiyun  */
nfsacld_proc_getacl(struct svc_rqst * rqstp)30*4882a593Smuzhiyun static __be32 nfsacld_proc_getacl(struct svc_rqst *rqstp)
31*4882a593Smuzhiyun {
32*4882a593Smuzhiyun 	struct nfsd3_getaclargs *argp = rqstp->rq_argp;
33*4882a593Smuzhiyun 	struct nfsd3_getaclres *resp = rqstp->rq_resp;
34*4882a593Smuzhiyun 	struct posix_acl *acl;
35*4882a593Smuzhiyun 	struct inode *inode;
36*4882a593Smuzhiyun 	svc_fh *fh;
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun 	dprintk("nfsd: GETACL(2acl)   %s\n", SVCFH_fmt(&argp->fh));
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun 	fh = fh_copy(&resp->fh, &argp->fh);
41*4882a593Smuzhiyun 	resp->status = fh_verify(rqstp, &resp->fh, 0, NFSD_MAY_NOP);
42*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
43*4882a593Smuzhiyun 		goto out;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 	inode = d_inode(fh->fh_dentry);
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	if (argp->mask & ~NFS_ACL_MASK) {
48*4882a593Smuzhiyun 		resp->status = nfserr_inval;
49*4882a593Smuzhiyun 		goto out;
50*4882a593Smuzhiyun 	}
51*4882a593Smuzhiyun 	resp->mask = argp->mask;
52*4882a593Smuzhiyun 
53*4882a593Smuzhiyun 	resp->status = fh_getattr(fh, &resp->stat);
54*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
55*4882a593Smuzhiyun 		goto out;
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	if (resp->mask & (NFS_ACL|NFS_ACLCNT)) {
58*4882a593Smuzhiyun 		acl = get_acl(inode, ACL_TYPE_ACCESS);
59*4882a593Smuzhiyun 		if (acl == NULL) {
60*4882a593Smuzhiyun 			/* Solaris returns the inode's minimum ACL. */
61*4882a593Smuzhiyun 			acl = posix_acl_from_mode(inode->i_mode, GFP_KERNEL);
62*4882a593Smuzhiyun 		}
63*4882a593Smuzhiyun 		if (IS_ERR(acl)) {
64*4882a593Smuzhiyun 			resp->status = nfserrno(PTR_ERR(acl));
65*4882a593Smuzhiyun 			goto fail;
66*4882a593Smuzhiyun 		}
67*4882a593Smuzhiyun 		resp->acl_access = acl;
68*4882a593Smuzhiyun 	}
69*4882a593Smuzhiyun 	if (resp->mask & (NFS_DFACL|NFS_DFACLCNT)) {
70*4882a593Smuzhiyun 		/* Check how Solaris handles requests for the Default ACL
71*4882a593Smuzhiyun 		   of a non-directory! */
72*4882a593Smuzhiyun 		acl = get_acl(inode, ACL_TYPE_DEFAULT);
73*4882a593Smuzhiyun 		if (IS_ERR(acl)) {
74*4882a593Smuzhiyun 			resp->status = nfserrno(PTR_ERR(acl));
75*4882a593Smuzhiyun 			goto fail;
76*4882a593Smuzhiyun 		}
77*4882a593Smuzhiyun 		resp->acl_default = acl;
78*4882a593Smuzhiyun 	}
79*4882a593Smuzhiyun 
80*4882a593Smuzhiyun 	/* resp->acl_{access,default} are released in nfssvc_release_getacl. */
81*4882a593Smuzhiyun out:
82*4882a593Smuzhiyun 	return rpc_success;
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun fail:
85*4882a593Smuzhiyun 	posix_acl_release(resp->acl_access);
86*4882a593Smuzhiyun 	posix_acl_release(resp->acl_default);
87*4882a593Smuzhiyun 	goto out;
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun /*
91*4882a593Smuzhiyun  * Set the Access and/or Default ACL of a file.
92*4882a593Smuzhiyun  */
nfsacld_proc_setacl(struct svc_rqst * rqstp)93*4882a593Smuzhiyun static __be32 nfsacld_proc_setacl(struct svc_rqst *rqstp)
94*4882a593Smuzhiyun {
95*4882a593Smuzhiyun 	struct nfsd3_setaclargs *argp = rqstp->rq_argp;
96*4882a593Smuzhiyun 	struct nfsd_attrstat *resp = rqstp->rq_resp;
97*4882a593Smuzhiyun 	struct inode *inode;
98*4882a593Smuzhiyun 	svc_fh *fh;
99*4882a593Smuzhiyun 	int error;
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	dprintk("nfsd: SETACL(2acl)   %s\n", SVCFH_fmt(&argp->fh));
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun 	fh = fh_copy(&resp->fh, &argp->fh);
104*4882a593Smuzhiyun 	resp->status = fh_verify(rqstp, &resp->fh, 0, NFSD_MAY_SATTR);
105*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
106*4882a593Smuzhiyun 		goto out;
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 	inode = d_inode(fh->fh_dentry);
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun 	error = fh_want_write(fh);
111*4882a593Smuzhiyun 	if (error)
112*4882a593Smuzhiyun 		goto out_errno;
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	fh_lock(fh);
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun 	error = set_posix_acl(inode, ACL_TYPE_ACCESS, argp->acl_access);
117*4882a593Smuzhiyun 	if (error)
118*4882a593Smuzhiyun 		goto out_drop_lock;
119*4882a593Smuzhiyun 	error = set_posix_acl(inode, ACL_TYPE_DEFAULT, argp->acl_default);
120*4882a593Smuzhiyun 	if (error)
121*4882a593Smuzhiyun 		goto out_drop_lock;
122*4882a593Smuzhiyun 
123*4882a593Smuzhiyun 	fh_unlock(fh);
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	fh_drop_write(fh);
126*4882a593Smuzhiyun 
127*4882a593Smuzhiyun 	resp->status = fh_getattr(fh, &resp->stat);
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun out:
130*4882a593Smuzhiyun 	/* argp->acl_{access,default} may have been allocated in
131*4882a593Smuzhiyun 	   nfssvc_decode_setaclargs. */
132*4882a593Smuzhiyun 	posix_acl_release(argp->acl_access);
133*4882a593Smuzhiyun 	posix_acl_release(argp->acl_default);
134*4882a593Smuzhiyun 	return rpc_success;
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun out_drop_lock:
137*4882a593Smuzhiyun 	fh_unlock(fh);
138*4882a593Smuzhiyun 	fh_drop_write(fh);
139*4882a593Smuzhiyun out_errno:
140*4882a593Smuzhiyun 	resp->status = nfserrno(error);
141*4882a593Smuzhiyun 	goto out;
142*4882a593Smuzhiyun }
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun /*
145*4882a593Smuzhiyun  * Check file attributes
146*4882a593Smuzhiyun  */
nfsacld_proc_getattr(struct svc_rqst * rqstp)147*4882a593Smuzhiyun static __be32 nfsacld_proc_getattr(struct svc_rqst *rqstp)
148*4882a593Smuzhiyun {
149*4882a593Smuzhiyun 	struct nfsd_fhandle *argp = rqstp->rq_argp;
150*4882a593Smuzhiyun 	struct nfsd_attrstat *resp = rqstp->rq_resp;
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 	dprintk("nfsd: GETATTR  %s\n", SVCFH_fmt(&argp->fh));
153*4882a593Smuzhiyun 
154*4882a593Smuzhiyun 	fh_copy(&resp->fh, &argp->fh);
155*4882a593Smuzhiyun 	resp->status = fh_verify(rqstp, &resp->fh, 0, NFSD_MAY_NOP);
156*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
157*4882a593Smuzhiyun 		goto out;
158*4882a593Smuzhiyun 	resp->status = fh_getattr(&resp->fh, &resp->stat);
159*4882a593Smuzhiyun out:
160*4882a593Smuzhiyun 	return rpc_success;
161*4882a593Smuzhiyun }
162*4882a593Smuzhiyun 
163*4882a593Smuzhiyun /*
164*4882a593Smuzhiyun  * Check file access
165*4882a593Smuzhiyun  */
nfsacld_proc_access(struct svc_rqst * rqstp)166*4882a593Smuzhiyun static __be32 nfsacld_proc_access(struct svc_rqst *rqstp)
167*4882a593Smuzhiyun {
168*4882a593Smuzhiyun 	struct nfsd3_accessargs *argp = rqstp->rq_argp;
169*4882a593Smuzhiyun 	struct nfsd3_accessres *resp = rqstp->rq_resp;
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 	dprintk("nfsd: ACCESS(2acl)   %s 0x%x\n",
172*4882a593Smuzhiyun 			SVCFH_fmt(&argp->fh),
173*4882a593Smuzhiyun 			argp->access);
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun 	fh_copy(&resp->fh, &argp->fh);
176*4882a593Smuzhiyun 	resp->access = argp->access;
177*4882a593Smuzhiyun 	resp->status = nfsd_access(rqstp, &resp->fh, &resp->access, NULL);
178*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
179*4882a593Smuzhiyun 		goto out;
180*4882a593Smuzhiyun 	resp->status = fh_getattr(&resp->fh, &resp->stat);
181*4882a593Smuzhiyun out:
182*4882a593Smuzhiyun 	return rpc_success;
183*4882a593Smuzhiyun }
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun /*
186*4882a593Smuzhiyun  * XDR decode functions
187*4882a593Smuzhiyun  */
nfsaclsvc_decode_voidarg(struct svc_rqst * rqstp,__be32 * p)188*4882a593Smuzhiyun static int nfsaclsvc_decode_voidarg(struct svc_rqst *rqstp, __be32 *p)
189*4882a593Smuzhiyun {
190*4882a593Smuzhiyun 	return 1;
191*4882a593Smuzhiyun }
192*4882a593Smuzhiyun 
nfsaclsvc_decode_getaclargs(struct svc_rqst * rqstp,__be32 * p)193*4882a593Smuzhiyun static int nfsaclsvc_decode_getaclargs(struct svc_rqst *rqstp, __be32 *p)
194*4882a593Smuzhiyun {
195*4882a593Smuzhiyun 	struct nfsd3_getaclargs *argp = rqstp->rq_argp;
196*4882a593Smuzhiyun 
197*4882a593Smuzhiyun 	p = nfs2svc_decode_fh(p, &argp->fh);
198*4882a593Smuzhiyun 	if (!p)
199*4882a593Smuzhiyun 		return 0;
200*4882a593Smuzhiyun 	argp->mask = ntohl(*p); p++;
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun 	return xdr_argsize_check(rqstp, p);
203*4882a593Smuzhiyun }
204*4882a593Smuzhiyun 
205*4882a593Smuzhiyun 
nfsaclsvc_decode_setaclargs(struct svc_rqst * rqstp,__be32 * p)206*4882a593Smuzhiyun static int nfsaclsvc_decode_setaclargs(struct svc_rqst *rqstp, __be32 *p)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun 	struct nfsd3_setaclargs *argp = rqstp->rq_argp;
209*4882a593Smuzhiyun 	struct kvec *head = rqstp->rq_arg.head;
210*4882a593Smuzhiyun 	unsigned int base;
211*4882a593Smuzhiyun 	int n;
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	p = nfs2svc_decode_fh(p, &argp->fh);
214*4882a593Smuzhiyun 	if (!p)
215*4882a593Smuzhiyun 		return 0;
216*4882a593Smuzhiyun 	argp->mask = ntohl(*p++);
217*4882a593Smuzhiyun 	if (argp->mask & ~NFS_ACL_MASK ||
218*4882a593Smuzhiyun 	    !xdr_argsize_check(rqstp, p))
219*4882a593Smuzhiyun 		return 0;
220*4882a593Smuzhiyun 
221*4882a593Smuzhiyun 	base = (char *)p - (char *)head->iov_base;
222*4882a593Smuzhiyun 	n = nfsacl_decode(&rqstp->rq_arg, base, NULL,
223*4882a593Smuzhiyun 			  (argp->mask & NFS_ACL) ?
224*4882a593Smuzhiyun 			  &argp->acl_access : NULL);
225*4882a593Smuzhiyun 	if (n > 0)
226*4882a593Smuzhiyun 		n = nfsacl_decode(&rqstp->rq_arg, base + n, NULL,
227*4882a593Smuzhiyun 				  (argp->mask & NFS_DFACL) ?
228*4882a593Smuzhiyun 				  &argp->acl_default : NULL);
229*4882a593Smuzhiyun 	return (n > 0);
230*4882a593Smuzhiyun }
231*4882a593Smuzhiyun 
nfsaclsvc_decode_fhandleargs(struct svc_rqst * rqstp,__be32 * p)232*4882a593Smuzhiyun static int nfsaclsvc_decode_fhandleargs(struct svc_rqst *rqstp, __be32 *p)
233*4882a593Smuzhiyun {
234*4882a593Smuzhiyun 	struct nfsd_fhandle *argp = rqstp->rq_argp;
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 	p = nfs2svc_decode_fh(p, &argp->fh);
237*4882a593Smuzhiyun 	if (!p)
238*4882a593Smuzhiyun 		return 0;
239*4882a593Smuzhiyun 	return xdr_argsize_check(rqstp, p);
240*4882a593Smuzhiyun }
241*4882a593Smuzhiyun 
nfsaclsvc_decode_accessargs(struct svc_rqst * rqstp,__be32 * p)242*4882a593Smuzhiyun static int nfsaclsvc_decode_accessargs(struct svc_rqst *rqstp, __be32 *p)
243*4882a593Smuzhiyun {
244*4882a593Smuzhiyun 	struct nfsd3_accessargs *argp = rqstp->rq_argp;
245*4882a593Smuzhiyun 
246*4882a593Smuzhiyun 	p = nfs2svc_decode_fh(p, &argp->fh);
247*4882a593Smuzhiyun 	if (!p)
248*4882a593Smuzhiyun 		return 0;
249*4882a593Smuzhiyun 	argp->access = ntohl(*p++);
250*4882a593Smuzhiyun 
251*4882a593Smuzhiyun 	return xdr_argsize_check(rqstp, p);
252*4882a593Smuzhiyun }
253*4882a593Smuzhiyun 
254*4882a593Smuzhiyun /*
255*4882a593Smuzhiyun  * XDR encode functions
256*4882a593Smuzhiyun  */
257*4882a593Smuzhiyun 
258*4882a593Smuzhiyun /*
259*4882a593Smuzhiyun  * There must be an encoding function for void results so svc_process
260*4882a593Smuzhiyun  * will work properly.
261*4882a593Smuzhiyun  */
nfsaclsvc_encode_voidres(struct svc_rqst * rqstp,__be32 * p)262*4882a593Smuzhiyun static int nfsaclsvc_encode_voidres(struct svc_rqst *rqstp, __be32 *p)
263*4882a593Smuzhiyun {
264*4882a593Smuzhiyun 	return xdr_ressize_check(rqstp, p);
265*4882a593Smuzhiyun }
266*4882a593Smuzhiyun 
267*4882a593Smuzhiyun /* GETACL */
nfsaclsvc_encode_getaclres(struct svc_rqst * rqstp,__be32 * p)268*4882a593Smuzhiyun static int nfsaclsvc_encode_getaclres(struct svc_rqst *rqstp, __be32 *p)
269*4882a593Smuzhiyun {
270*4882a593Smuzhiyun 	struct nfsd3_getaclres *resp = rqstp->rq_resp;
271*4882a593Smuzhiyun 	struct dentry *dentry = resp->fh.fh_dentry;
272*4882a593Smuzhiyun 	struct inode *inode;
273*4882a593Smuzhiyun 	struct kvec *head = rqstp->rq_res.head;
274*4882a593Smuzhiyun 	unsigned int base;
275*4882a593Smuzhiyun 	int n;
276*4882a593Smuzhiyun 	int w;
277*4882a593Smuzhiyun 
278*4882a593Smuzhiyun 	*p++ = resp->status;
279*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
280*4882a593Smuzhiyun 		return xdr_ressize_check(rqstp, p);
281*4882a593Smuzhiyun 
282*4882a593Smuzhiyun 	/*
283*4882a593Smuzhiyun 	 * Since this is version 2, the check for nfserr in
284*4882a593Smuzhiyun 	 * nfsd_dispatch actually ensures the following cannot happen.
285*4882a593Smuzhiyun 	 * However, it seems fragile to depend on that.
286*4882a593Smuzhiyun 	 */
287*4882a593Smuzhiyun 	if (dentry == NULL || d_really_is_negative(dentry))
288*4882a593Smuzhiyun 		return 0;
289*4882a593Smuzhiyun 	inode = d_inode(dentry);
290*4882a593Smuzhiyun 
291*4882a593Smuzhiyun 	p = nfs2svc_encode_fattr(rqstp, p, &resp->fh, &resp->stat);
292*4882a593Smuzhiyun 	*p++ = htonl(resp->mask);
293*4882a593Smuzhiyun 	if (!xdr_ressize_check(rqstp, p))
294*4882a593Smuzhiyun 		return 0;
295*4882a593Smuzhiyun 	base = (char *)p - (char *)head->iov_base;
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun 	rqstp->rq_res.page_len = w = nfsacl_size(
298*4882a593Smuzhiyun 		(resp->mask & NFS_ACL)   ? resp->acl_access  : NULL,
299*4882a593Smuzhiyun 		(resp->mask & NFS_DFACL) ? resp->acl_default : NULL);
300*4882a593Smuzhiyun 	while (w > 0) {
301*4882a593Smuzhiyun 		if (!*(rqstp->rq_next_page++))
302*4882a593Smuzhiyun 			return 0;
303*4882a593Smuzhiyun 		w -= PAGE_SIZE;
304*4882a593Smuzhiyun 	}
305*4882a593Smuzhiyun 
306*4882a593Smuzhiyun 	n = nfsacl_encode(&rqstp->rq_res, base, inode,
307*4882a593Smuzhiyun 			  resp->acl_access,
308*4882a593Smuzhiyun 			  resp->mask & NFS_ACL, 0);
309*4882a593Smuzhiyun 	if (n > 0)
310*4882a593Smuzhiyun 		n = nfsacl_encode(&rqstp->rq_res, base + n, inode,
311*4882a593Smuzhiyun 				  resp->acl_default,
312*4882a593Smuzhiyun 				  resp->mask & NFS_DFACL,
313*4882a593Smuzhiyun 				  NFS_ACL_DEFAULT);
314*4882a593Smuzhiyun 	return (n > 0);
315*4882a593Smuzhiyun }
316*4882a593Smuzhiyun 
nfsaclsvc_encode_attrstatres(struct svc_rqst * rqstp,__be32 * p)317*4882a593Smuzhiyun static int nfsaclsvc_encode_attrstatres(struct svc_rqst *rqstp, __be32 *p)
318*4882a593Smuzhiyun {
319*4882a593Smuzhiyun 	struct nfsd_attrstat *resp = rqstp->rq_resp;
320*4882a593Smuzhiyun 
321*4882a593Smuzhiyun 	*p++ = resp->status;
322*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
323*4882a593Smuzhiyun 		goto out;
324*4882a593Smuzhiyun 
325*4882a593Smuzhiyun 	p = nfs2svc_encode_fattr(rqstp, p, &resp->fh, &resp->stat);
326*4882a593Smuzhiyun out:
327*4882a593Smuzhiyun 	return xdr_ressize_check(rqstp, p);
328*4882a593Smuzhiyun }
329*4882a593Smuzhiyun 
330*4882a593Smuzhiyun /* ACCESS */
nfsaclsvc_encode_accessres(struct svc_rqst * rqstp,__be32 * p)331*4882a593Smuzhiyun static int nfsaclsvc_encode_accessres(struct svc_rqst *rqstp, __be32 *p)
332*4882a593Smuzhiyun {
333*4882a593Smuzhiyun 	struct nfsd3_accessres *resp = rqstp->rq_resp;
334*4882a593Smuzhiyun 
335*4882a593Smuzhiyun 	*p++ = resp->status;
336*4882a593Smuzhiyun 	if (resp->status != nfs_ok)
337*4882a593Smuzhiyun 		goto out;
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun 	p = nfs2svc_encode_fattr(rqstp, p, &resp->fh, &resp->stat);
340*4882a593Smuzhiyun 	*p++ = htonl(resp->access);
341*4882a593Smuzhiyun out:
342*4882a593Smuzhiyun 	return xdr_ressize_check(rqstp, p);
343*4882a593Smuzhiyun }
344*4882a593Smuzhiyun 
345*4882a593Smuzhiyun /*
346*4882a593Smuzhiyun  * XDR release functions
347*4882a593Smuzhiyun  */
nfsaclsvc_release_getacl(struct svc_rqst * rqstp)348*4882a593Smuzhiyun static void nfsaclsvc_release_getacl(struct svc_rqst *rqstp)
349*4882a593Smuzhiyun {
350*4882a593Smuzhiyun 	struct nfsd3_getaclres *resp = rqstp->rq_resp;
351*4882a593Smuzhiyun 
352*4882a593Smuzhiyun 	fh_put(&resp->fh);
353*4882a593Smuzhiyun 	posix_acl_release(resp->acl_access);
354*4882a593Smuzhiyun 	posix_acl_release(resp->acl_default);
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun 
nfsaclsvc_release_attrstat(struct svc_rqst * rqstp)357*4882a593Smuzhiyun static void nfsaclsvc_release_attrstat(struct svc_rqst *rqstp)
358*4882a593Smuzhiyun {
359*4882a593Smuzhiyun 	struct nfsd_attrstat *resp = rqstp->rq_resp;
360*4882a593Smuzhiyun 
361*4882a593Smuzhiyun 	fh_put(&resp->fh);
362*4882a593Smuzhiyun }
363*4882a593Smuzhiyun 
nfsaclsvc_release_access(struct svc_rqst * rqstp)364*4882a593Smuzhiyun static void nfsaclsvc_release_access(struct svc_rqst *rqstp)
365*4882a593Smuzhiyun {
366*4882a593Smuzhiyun 	struct nfsd3_accessres *resp = rqstp->rq_resp;
367*4882a593Smuzhiyun 
368*4882a593Smuzhiyun 	fh_put(&resp->fh);
369*4882a593Smuzhiyun }
370*4882a593Smuzhiyun 
371*4882a593Smuzhiyun struct nfsd3_voidargs { int dummy; };
372*4882a593Smuzhiyun 
373*4882a593Smuzhiyun #define ST 1		/* status*/
374*4882a593Smuzhiyun #define AT 21		/* attributes */
375*4882a593Smuzhiyun #define pAT (1+AT)	/* post attributes - conditional */
376*4882a593Smuzhiyun #define ACL (1+NFS_ACL_MAX_ENTRIES*3)  /* Access Control List */
377*4882a593Smuzhiyun 
378*4882a593Smuzhiyun static const struct svc_procedure nfsd_acl_procedures2[5] = {
379*4882a593Smuzhiyun 	[ACLPROC2_NULL] = {
380*4882a593Smuzhiyun 		.pc_func = nfsacld_proc_null,
381*4882a593Smuzhiyun 		.pc_decode = nfsaclsvc_decode_voidarg,
382*4882a593Smuzhiyun 		.pc_encode = nfsaclsvc_encode_voidres,
383*4882a593Smuzhiyun 		.pc_argsize = sizeof(struct nfsd3_voidargs),
384*4882a593Smuzhiyun 		.pc_ressize = sizeof(struct nfsd3_voidargs),
385*4882a593Smuzhiyun 		.pc_cachetype = RC_NOCACHE,
386*4882a593Smuzhiyun 		.pc_xdrressize = ST,
387*4882a593Smuzhiyun 	},
388*4882a593Smuzhiyun 	[ACLPROC2_GETACL] = {
389*4882a593Smuzhiyun 		.pc_func = nfsacld_proc_getacl,
390*4882a593Smuzhiyun 		.pc_decode = nfsaclsvc_decode_getaclargs,
391*4882a593Smuzhiyun 		.pc_encode = nfsaclsvc_encode_getaclres,
392*4882a593Smuzhiyun 		.pc_release = nfsaclsvc_release_getacl,
393*4882a593Smuzhiyun 		.pc_argsize = sizeof(struct nfsd3_getaclargs),
394*4882a593Smuzhiyun 		.pc_ressize = sizeof(struct nfsd3_getaclres),
395*4882a593Smuzhiyun 		.pc_cachetype = RC_NOCACHE,
396*4882a593Smuzhiyun 		.pc_xdrressize = ST+1+2*(1+ACL),
397*4882a593Smuzhiyun 	},
398*4882a593Smuzhiyun 	[ACLPROC2_SETACL] = {
399*4882a593Smuzhiyun 		.pc_func = nfsacld_proc_setacl,
400*4882a593Smuzhiyun 		.pc_decode = nfsaclsvc_decode_setaclargs,
401*4882a593Smuzhiyun 		.pc_encode = nfsaclsvc_encode_attrstatres,
402*4882a593Smuzhiyun 		.pc_release = nfsaclsvc_release_attrstat,
403*4882a593Smuzhiyun 		.pc_argsize = sizeof(struct nfsd3_setaclargs),
404*4882a593Smuzhiyun 		.pc_ressize = sizeof(struct nfsd_attrstat),
405*4882a593Smuzhiyun 		.pc_cachetype = RC_NOCACHE,
406*4882a593Smuzhiyun 		.pc_xdrressize = ST+AT,
407*4882a593Smuzhiyun 	},
408*4882a593Smuzhiyun 	[ACLPROC2_GETATTR] = {
409*4882a593Smuzhiyun 		.pc_func = nfsacld_proc_getattr,
410*4882a593Smuzhiyun 		.pc_decode = nfsaclsvc_decode_fhandleargs,
411*4882a593Smuzhiyun 		.pc_encode = nfsaclsvc_encode_attrstatres,
412*4882a593Smuzhiyun 		.pc_release = nfsaclsvc_release_attrstat,
413*4882a593Smuzhiyun 		.pc_argsize = sizeof(struct nfsd_fhandle),
414*4882a593Smuzhiyun 		.pc_ressize = sizeof(struct nfsd_attrstat),
415*4882a593Smuzhiyun 		.pc_cachetype = RC_NOCACHE,
416*4882a593Smuzhiyun 		.pc_xdrressize = ST+AT,
417*4882a593Smuzhiyun 	},
418*4882a593Smuzhiyun 	[ACLPROC2_ACCESS] = {
419*4882a593Smuzhiyun 		.pc_func = nfsacld_proc_access,
420*4882a593Smuzhiyun 		.pc_decode = nfsaclsvc_decode_accessargs,
421*4882a593Smuzhiyun 		.pc_encode = nfsaclsvc_encode_accessres,
422*4882a593Smuzhiyun 		.pc_release = nfsaclsvc_release_access,
423*4882a593Smuzhiyun 		.pc_argsize = sizeof(struct nfsd3_accessargs),
424*4882a593Smuzhiyun 		.pc_ressize = sizeof(struct nfsd3_accessres),
425*4882a593Smuzhiyun 		.pc_cachetype = RC_NOCACHE,
426*4882a593Smuzhiyun 		.pc_xdrressize = ST+AT+1,
427*4882a593Smuzhiyun 	},
428*4882a593Smuzhiyun };
429*4882a593Smuzhiyun 
430*4882a593Smuzhiyun static unsigned int nfsd_acl_count2[ARRAY_SIZE(nfsd_acl_procedures2)];
431*4882a593Smuzhiyun const struct svc_version nfsd_acl_version2 = {
432*4882a593Smuzhiyun 	.vs_vers	= 2,
433*4882a593Smuzhiyun 	.vs_nproc	= 5,
434*4882a593Smuzhiyun 	.vs_proc	= nfsd_acl_procedures2,
435*4882a593Smuzhiyun 	.vs_count	= nfsd_acl_count2,
436*4882a593Smuzhiyun 	.vs_dispatch	= nfsd_dispatch,
437*4882a593Smuzhiyun 	.vs_xdrsize	= NFS3_SVC_XDRSIZE,
438*4882a593Smuzhiyun };
439