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