blob: 88ec38c2d83cc0d3989d02258444b21325b6abd2 [file] [log] [blame]
David Howells08e0e7c2007-04-26 15:55:03 -07001/* AFS File Server client stubs
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 *
David Howells08e0e7c2007-04-26 15:55:03 -07003 * Copyright (C) 2002, 2007 Red Hat, Inc. All Rights Reserved.
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 * Written by David Howells (dhowells@redhat.com)
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version
9 * 2 of the License, or (at your option) any later version.
10 */
11
12#include <linux/init.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090013#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070014#include <linux/sched.h>
David Howells08e0e7c2007-04-26 15:55:03 -070015#include <linux/circ_buf.h>
Jeff Laytona01179e2017-12-11 06:35:11 -050016#include <linux/iversion.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070017#include "internal.h"
David Howells08e0e7c2007-04-26 15:55:03 -070018#include "afs_fs.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070019
David Howells025db802017-11-02 15:27:51 +000020static const struct afs_fid afs_zero_fid;
21
Linus Torvalds1da177e2005-04-16 15:20:36 -070022/*
David Howells6db3ac32017-03-16 16:27:44 +000023 * We need somewhere to discard into in case the server helpfully returns more
24 * than we asked for in FS.FetchData{,64}.
25 */
26static u8 afs_discard_buffer[64];
27
David Howellsd2ddc772017-11-02 15:27:50 +000028static inline void afs_use_fs_server(struct afs_call *call, struct afs_cb_interest *cbi)
David Howellsc435ee32017-11-02 15:27:49 +000029{
David Howellsd2ddc772017-11-02 15:27:50 +000030 call->cbi = afs_get_cb_interest(cbi);
David Howellsc435ee32017-11-02 15:27:49 +000031}
32
David Howells6db3ac32017-03-16 16:27:44 +000033/*
David Howells260a9802007-04-26 15:59:35 -070034 * decode an AFSFid block
35 */
36static void xdr_decode_AFSFid(const __be32 **_bp, struct afs_fid *fid)
37{
38 const __be32 *bp = *_bp;
39
40 fid->vid = ntohl(*bp++);
41 fid->vnode = ntohl(*bp++);
42 fid->unique = ntohl(*bp++);
43 *_bp = bp;
44}
45
46/*
David Howells08e0e7c2007-04-26 15:55:03 -070047 * decode an AFSFetchStatus block
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 */
David Howells08e0e7c2007-04-26 15:55:03 -070049static void xdr_decode_AFSFetchStatus(const __be32 **_bp,
David Howells260a9802007-04-26 15:59:35 -070050 struct afs_file_status *status,
David Howells31143d52007-05-09 02:33:46 -070051 struct afs_vnode *vnode,
52 afs_dataversion_t *store_version)
Linus Torvalds1da177e2005-04-16 15:20:36 -070053{
David Howells31143d52007-05-09 02:33:46 -070054 afs_dataversion_t expected_version;
David Howells08e0e7c2007-04-26 15:55:03 -070055 const __be32 *bp = *_bp;
56 umode_t mode;
David Howells260a9802007-04-26 15:59:35 -070057 u64 data_version, size;
David Howellsc435ee32017-11-02 15:27:49 +000058 bool changed = false;
Eric W. Biedermana0a53862012-02-07 16:20:48 -080059 kuid_t owner;
60 kgid_t group;
David Howells08e0e7c2007-04-26 15:55:03 -070061
David Howellsd2ddc772017-11-02 15:27:50 +000062 if (vnode)
63 write_seqlock(&vnode->cb_lock);
David Howellsc435ee32017-11-02 15:27:49 +000064
David Howells08e0e7c2007-04-26 15:55:03 -070065#define EXTRACT(DST) \
66 do { \
67 u32 x = ntohl(*bp++); \
David Howellsc435ee32017-11-02 15:27:49 +000068 if (DST != x) \
69 changed |= true; \
David Howells08e0e7c2007-04-26 15:55:03 -070070 DST = x; \
71 } while (0)
72
David Howells260a9802007-04-26 15:59:35 -070073 status->if_version = ntohl(*bp++);
74 EXTRACT(status->type);
75 EXTRACT(status->nlink);
76 size = ntohl(*bp++);
David Howells08e0e7c2007-04-26 15:55:03 -070077 data_version = ntohl(*bp++);
David Howells260a9802007-04-26 15:59:35 -070078 EXTRACT(status->author);
Eric W. Biedermana0a53862012-02-07 16:20:48 -080079 owner = make_kuid(&init_user_ns, ntohl(*bp++));
80 changed |= !uid_eq(owner, status->owner);
81 status->owner = owner;
David Howells260a9802007-04-26 15:59:35 -070082 EXTRACT(status->caller_access); /* call ticket dependent */
83 EXTRACT(status->anon_access);
84 EXTRACT(status->mode);
David Howellsbe080a62017-11-02 15:27:49 +000085 bp++; /* parent.vnode */
86 bp++; /* parent.unique */
David Howells08e0e7c2007-04-26 15:55:03 -070087 bp++; /* seg size */
David Howells260a9802007-04-26 15:59:35 -070088 status->mtime_client = ntohl(*bp++);
89 status->mtime_server = ntohl(*bp++);
Eric W. Biedermana0a53862012-02-07 16:20:48 -080090 group = make_kgid(&init_user_ns, ntohl(*bp++));
91 changed |= !gid_eq(group, status->group);
92 status->group = group;
David Howells08e0e7c2007-04-26 15:55:03 -070093 bp++; /* sync counter */
94 data_version |= (u64) ntohl(*bp++) << 32;
David Howellse8d6c552007-07-15 23:40:12 -070095 EXTRACT(status->lock_count);
David Howells260a9802007-04-26 15:59:35 -070096 size |= (u64) ntohl(*bp++) << 32;
97 bp++; /* spare 4 */
David Howells08e0e7c2007-04-26 15:55:03 -070098 *_bp = bp;
99
David Howells260a9802007-04-26 15:59:35 -0700100 if (size != status->size) {
101 status->size = size;
102 changed |= true;
David Howells08e0e7c2007-04-26 15:55:03 -0700103 }
David Howells260a9802007-04-26 15:59:35 -0700104 status->mode &= S_IALLUGO;
David Howells08e0e7c2007-04-26 15:55:03 -0700105
106 _debug("vnode time %lx, %lx",
David Howells260a9802007-04-26 15:59:35 -0700107 status->mtime_client, status->mtime_server);
David Howells08e0e7c2007-04-26 15:55:03 -0700108
David Howells260a9802007-04-26 15:59:35 -0700109 if (vnode) {
David Howells260a9802007-04-26 15:59:35 -0700110 if (changed && !test_bit(AFS_VNODE_UNSET, &vnode->flags)) {
111 _debug("vnode changed");
112 i_size_write(&vnode->vfs_inode, size);
113 vnode->vfs_inode.i_uid = status->owner;
114 vnode->vfs_inode.i_gid = status->group;
David Howellsd6e43f72011-06-14 00:45:44 +0100115 vnode->vfs_inode.i_generation = vnode->fid.unique;
Miklos Szeredibfe86842011-10-28 14:13:29 +0200116 set_nlink(&vnode->vfs_inode, status->nlink);
David Howells260a9802007-04-26 15:59:35 -0700117
118 mode = vnode->vfs_inode.i_mode;
119 mode &= ~S_IALLUGO;
120 mode |= status->mode;
121 barrier();
122 vnode->vfs_inode.i_mode = mode;
123 }
124
Marc Dionneab94f5d2017-03-16 16:27:47 +0000125 vnode->vfs_inode.i_ctime.tv_sec = status->mtime_client;
David Howells260a9802007-04-26 15:59:35 -0700126 vnode->vfs_inode.i_mtime = vnode->vfs_inode.i_ctime;
127 vnode->vfs_inode.i_atime = vnode->vfs_inode.i_ctime;
Jeff Laytona01179e2017-12-11 06:35:11 -0500128 inode_set_iversion_raw(&vnode->vfs_inode, data_version);
David Howells260a9802007-04-26 15:59:35 -0700129 }
130
David Howells31143d52007-05-09 02:33:46 -0700131 expected_version = status->data_version;
132 if (store_version)
133 expected_version = *store_version;
134
135 if (expected_version != data_version) {
David Howells260a9802007-04-26 15:59:35 -0700136 status->data_version = data_version;
137 if (vnode && !test_bit(AFS_VNODE_UNSET, &vnode->flags)) {
138 _debug("vnode modified %llx on {%x:%u}",
David S. Millerba3e0e12007-04-26 16:06:22 -0700139 (unsigned long long) data_version,
140 vnode->fid.vid, vnode->fid.vnode);
David Howellsc435ee32017-11-02 15:27:49 +0000141 set_bit(AFS_VNODE_DIR_MODIFIED, &vnode->flags);
David Howells260a9802007-04-26 15:59:35 -0700142 set_bit(AFS_VNODE_ZAP_DATA, &vnode->flags);
143 }
David Howells31143d52007-05-09 02:33:46 -0700144 } else if (store_version) {
145 status->data_version = data_version;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700146 }
David Howellsc435ee32017-11-02 15:27:49 +0000147
David Howellsd2ddc772017-11-02 15:27:50 +0000148 if (vnode)
149 write_sequnlock(&vnode->cb_lock);
David Howellsec268152007-04-26 15:49:28 -0700150}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700151
Linus Torvalds1da177e2005-04-16 15:20:36 -0700152/*
David Howells08e0e7c2007-04-26 15:55:03 -0700153 * decode an AFSCallBack block
Linus Torvalds1da177e2005-04-16 15:20:36 -0700154 */
David Howellsc435ee32017-11-02 15:27:49 +0000155static void xdr_decode_AFSCallBack(struct afs_call *call,
156 struct afs_vnode *vnode,
157 const __be32 **_bp)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700158{
David Howellsd2ddc772017-11-02 15:27:50 +0000159 struct afs_cb_interest *old, *cbi = call->cbi;
David Howells08e0e7c2007-04-26 15:55:03 -0700160 const __be32 *bp = *_bp;
David Howellsc435ee32017-11-02 15:27:49 +0000161 u32 cb_expiry;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700162
David Howellsc435ee32017-11-02 15:27:49 +0000163 write_seqlock(&vnode->cb_lock);
164
David Howellsd2ddc772017-11-02 15:27:50 +0000165 if (call->cb_break == (vnode->cb_break + cbi->server->cb_s_break)) {
David Howellsc435ee32017-11-02 15:27:49 +0000166 vnode->cb_version = ntohl(*bp++);
167 cb_expiry = ntohl(*bp++);
168 vnode->cb_type = ntohl(*bp++);
169 vnode->cb_expires_at = cb_expiry + ktime_get_real_seconds();
David Howellsd2ddc772017-11-02 15:27:50 +0000170 old = vnode->cb_interest;
171 if (old != call->cbi) {
172 vnode->cb_interest = cbi;
173 cbi = old;
174 }
David Howellsc435ee32017-11-02 15:27:49 +0000175 set_bit(AFS_VNODE_CB_PROMISED, &vnode->flags);
176 } else {
177 bp += 3;
178 }
179
180 write_sequnlock(&vnode->cb_lock);
David Howellsd2ddc772017-11-02 15:27:50 +0000181 call->cbi = cbi;
David Howells08e0e7c2007-04-26 15:55:03 -0700182 *_bp = bp;
David Howellsec268152007-04-26 15:49:28 -0700183}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700184
David Howells260a9802007-04-26 15:59:35 -0700185static void xdr_decode_AFSCallBack_raw(const __be32 **_bp,
186 struct afs_callback *cb)
187{
188 const __be32 *bp = *_bp;
189
190 cb->version = ntohl(*bp++);
191 cb->expiry = ntohl(*bp++);
192 cb->type = ntohl(*bp++);
193 *_bp = bp;
194}
195
Linus Torvalds1da177e2005-04-16 15:20:36 -0700196/*
David Howells08e0e7c2007-04-26 15:55:03 -0700197 * decode an AFSVolSync block
Linus Torvalds1da177e2005-04-16 15:20:36 -0700198 */
David Howells08e0e7c2007-04-26 15:55:03 -0700199static void xdr_decode_AFSVolSync(const __be32 **_bp,
200 struct afs_volsync *volsync)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700201{
David Howells08e0e7c2007-04-26 15:55:03 -0700202 const __be32 *bp = *_bp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700203
David Howells08e0e7c2007-04-26 15:55:03 -0700204 volsync->creation = ntohl(*bp++);
205 bp++; /* spare2 */
206 bp++; /* spare3 */
207 bp++; /* spare4 */
208 bp++; /* spare5 */
209 bp++; /* spare6 */
210 *_bp = bp;
David Howellsec268152007-04-26 15:49:28 -0700211}
David Howells08e0e7c2007-04-26 15:55:03 -0700212
213/*
David Howells31143d52007-05-09 02:33:46 -0700214 * encode the requested attributes into an AFSStoreStatus block
215 */
216static void xdr_encode_AFS_StoreStatus(__be32 **_bp, struct iattr *attr)
217{
218 __be32 *bp = *_bp;
219 u32 mask = 0, mtime = 0, owner = 0, group = 0, mode = 0;
220
221 mask = 0;
222 if (attr->ia_valid & ATTR_MTIME) {
223 mask |= AFS_SET_MTIME;
224 mtime = attr->ia_mtime.tv_sec;
225 }
226
227 if (attr->ia_valid & ATTR_UID) {
228 mask |= AFS_SET_OWNER;
Eric W. Biedermana0a53862012-02-07 16:20:48 -0800229 owner = from_kuid(&init_user_ns, attr->ia_uid);
David Howells31143d52007-05-09 02:33:46 -0700230 }
231
232 if (attr->ia_valid & ATTR_GID) {
233 mask |= AFS_SET_GROUP;
Eric W. Biedermana0a53862012-02-07 16:20:48 -0800234 group = from_kgid(&init_user_ns, attr->ia_gid);
David Howells31143d52007-05-09 02:33:46 -0700235 }
236
237 if (attr->ia_valid & ATTR_MODE) {
238 mask |= AFS_SET_MODE;
239 mode = attr->ia_mode & S_IALLUGO;
240 }
241
242 *bp++ = htonl(mask);
243 *bp++ = htonl(mtime);
244 *bp++ = htonl(owner);
245 *bp++ = htonl(group);
246 *bp++ = htonl(mode);
247 *bp++ = 0; /* segment size */
248 *_bp = bp;
249}
250
251/*
David Howells45222b92007-05-10 22:22:20 -0700252 * decode an AFSFetchVolumeStatus block
253 */
254static void xdr_decode_AFSFetchVolumeStatus(const __be32 **_bp,
255 struct afs_volume_status *vs)
256{
257 const __be32 *bp = *_bp;
258
259 vs->vid = ntohl(*bp++);
260 vs->parent_id = ntohl(*bp++);
261 vs->online = ntohl(*bp++);
262 vs->in_service = ntohl(*bp++);
263 vs->blessed = ntohl(*bp++);
264 vs->needs_salvage = ntohl(*bp++);
265 vs->type = ntohl(*bp++);
266 vs->min_quota = ntohl(*bp++);
267 vs->max_quota = ntohl(*bp++);
268 vs->blocks_in_use = ntohl(*bp++);
269 vs->part_blocks_avail = ntohl(*bp++);
270 vs->part_max_blocks = ntohl(*bp++);
271 *_bp = bp;
272}
273
274/*
David Howells08e0e7c2007-04-26 15:55:03 -0700275 * deliver reply data to an FS.FetchStatus
276 */
David Howellsd0016482016-08-30 20:42:14 +0100277static int afs_deliver_fs_fetch_status(struct afs_call *call)
David Howells08e0e7c2007-04-26 15:55:03 -0700278{
David Howells97e30432017-11-02 15:27:48 +0000279 struct afs_vnode *vnode = call->reply[0];
David Howells08e0e7c2007-04-26 15:55:03 -0700280 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100281 int ret;
David Howells08e0e7c2007-04-26 15:55:03 -0700282
David Howellsd0016482016-08-30 20:42:14 +0100283 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100284 if (ret < 0)
285 return ret;
David Howells08e0e7c2007-04-26 15:55:03 -0700286
David Howellsc435ee32017-11-02 15:27:49 +0000287 _enter("{%x:%u}", vnode->fid.vid, vnode->fid.vnode);
288
David Howells08e0e7c2007-04-26 15:55:03 -0700289 /* unmarshall the reply once we've received all of it */
290 bp = call->buffer;
David Howells31143d52007-05-09 02:33:46 -0700291 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
David Howellsc435ee32017-11-02 15:27:49 +0000292 xdr_decode_AFSCallBack(call, vnode, &bp);
David Howells97e30432017-11-02 15:27:48 +0000293 if (call->reply[1])
294 xdr_decode_AFSVolSync(&bp, call->reply[1]);
David Howells08e0e7c2007-04-26 15:55:03 -0700295
296 _leave(" = 0 [done]");
297 return 0;
298}
299
300/*
301 * FS.FetchStatus operation type
302 */
303static const struct afs_call_type afs_RXFSFetchStatus = {
David Howells00d3b7a2007-04-26 15:57:07 -0700304 .name = "FS.FetchStatus",
David Howells025db802017-11-02 15:27:51 +0000305 .op = afs_FS_FetchStatus,
David Howells08e0e7c2007-04-26 15:55:03 -0700306 .deliver = afs_deliver_fs_fetch_status,
David Howells08e0e7c2007-04-26 15:55:03 -0700307 .destructor = afs_flat_call_destructor,
308};
Linus Torvalds1da177e2005-04-16 15:20:36 -0700309
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310/*
311 * fetch the status information for a file
312 */
David Howellsd2ddc772017-11-02 15:27:50 +0000313int afs_fs_fetch_file_status(struct afs_fs_cursor *fc, struct afs_volsync *volsync)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700314{
David Howellsd2ddc772017-11-02 15:27:50 +0000315 struct afs_vnode *vnode = fc->vnode;
David Howells08e0e7c2007-04-26 15:55:03 -0700316 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000317 struct afs_net *net = afs_v2net(vnode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 __be32 *bp;
319
David Howells416351f2007-05-09 02:33:45 -0700320 _enter(",%x,{%x:%u},,",
David Howellsd2ddc772017-11-02 15:27:50 +0000321 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700322
David Howellsf044c882017-11-02 15:27:45 +0000323 call = afs_alloc_flat_call(net, &afs_RXFSFetchStatus, 16, (21 + 3 + 6) * 4);
David Howellsd2ddc772017-11-02 15:27:50 +0000324 if (!call) {
325 fc->ac.error = -ENOMEM;
David Howells08e0e7c2007-04-26 15:55:03 -0700326 return -ENOMEM;
David Howellsd2ddc772017-11-02 15:27:50 +0000327 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700328
David Howellsd2ddc772017-11-02 15:27:50 +0000329 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000330 call->reply[0] = vnode;
331 call->reply[1] = volsync;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700332
333 /* marshall the parameters */
David Howells08e0e7c2007-04-26 15:55:03 -0700334 bp = call->request;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700335 bp[0] = htonl(FSFETCHSTATUS);
336 bp[1] = htonl(vnode->fid.vid);
337 bp[2] = htonl(vnode->fid.vnode);
338 bp[3] = htonl(vnode->fid.unique);
339
David Howellsd2ddc772017-11-02 15:27:50 +0000340 call->cb_break = fc->cb_break;
341 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000342 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000343 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsec268152007-04-26 15:49:28 -0700344}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345
Linus Torvalds1da177e2005-04-16 15:20:36 -0700346/*
David Howells08e0e7c2007-04-26 15:55:03 -0700347 * deliver reply data to an FS.FetchData
Linus Torvalds1da177e2005-04-16 15:20:36 -0700348 */
David Howellsd0016482016-08-30 20:42:14 +0100349static int afs_deliver_fs_fetch_data(struct afs_call *call)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700350{
David Howells97e30432017-11-02 15:27:48 +0000351 struct afs_vnode *vnode = call->reply[0];
352 struct afs_read *req = call->reply[2];
David Howells08e0e7c2007-04-26 15:55:03 -0700353 const __be32 *bp;
David Howells196ee9c2017-01-05 10:38:34 +0000354 unsigned int size;
David Howells08e0e7c2007-04-26 15:55:03 -0700355 void *buffer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700356 int ret;
David Howells08e0e7c2007-04-26 15:55:03 -0700357
David Howells6a0e3992017-03-16 16:27:46 +0000358 _enter("{%u,%zu/%u;%llu/%llu}",
David Howells196ee9c2017-01-05 10:38:34 +0000359 call->unmarshall, call->offset, call->count,
360 req->remain, req->actual_len);
David Howells08e0e7c2007-04-26 15:55:03 -0700361
362 switch (call->unmarshall) {
363 case 0:
David Howells196ee9c2017-01-05 10:38:34 +0000364 req->actual_len = 0;
David Howells08e0e7c2007-04-26 15:55:03 -0700365 call->offset = 0;
366 call->unmarshall++;
David Howellsb9b1f8d2007-05-10 03:15:21 -0700367 if (call->operation_ID != FSFETCHDATA64) {
368 call->unmarshall++;
369 goto no_msw;
370 }
David Howells08e0e7c2007-04-26 15:55:03 -0700371
David Howellsb9b1f8d2007-05-10 03:15:21 -0700372 /* extract the upper part of the returned data length of an
373 * FSFETCHDATA64 op (which should always be 0 using this
374 * client) */
David Howells08e0e7c2007-04-26 15:55:03 -0700375 case 1:
David Howellsb9b1f8d2007-05-10 03:15:21 -0700376 _debug("extract data length (MSW)");
David Howellsd0016482016-08-30 20:42:14 +0100377 ret = afs_extract_data(call, &call->tmp, 4, true);
David Howells372ee162016-08-03 14:11:40 +0100378 if (ret < 0)
379 return ret;
David Howellsb9b1f8d2007-05-10 03:15:21 -0700380
David Howells196ee9c2017-01-05 10:38:34 +0000381 req->actual_len = ntohl(call->tmp);
382 req->actual_len <<= 32;
David Howellsb9b1f8d2007-05-10 03:15:21 -0700383 call->offset = 0;
384 call->unmarshall++;
385
386 no_msw:
387 /* extract the returned data length */
388 case 2:
David Howells08e0e7c2007-04-26 15:55:03 -0700389 _debug("extract data length");
David Howellsd0016482016-08-30 20:42:14 +0100390 ret = afs_extract_data(call, &call->tmp, 4, true);
David Howells372ee162016-08-03 14:11:40 +0100391 if (ret < 0)
392 return ret;
David Howells08e0e7c2007-04-26 15:55:03 -0700393
David Howells196ee9c2017-01-05 10:38:34 +0000394 req->actual_len |= ntohl(call->tmp);
395 _debug("DATA length: %llu", req->actual_len);
David Howells196ee9c2017-01-05 10:38:34 +0000396
397 req->remain = req->actual_len;
398 call->offset = req->pos & (PAGE_SIZE - 1);
399 req->index = 0;
400 if (req->actual_len == 0)
401 goto no_more_data;
David Howells08e0e7c2007-04-26 15:55:03 -0700402 call->unmarshall++;
403
David Howells196ee9c2017-01-05 10:38:34 +0000404 begin_page:
David Howells6db3ac32017-03-16 16:27:44 +0000405 ASSERTCMP(req->index, <, req->nr_pages);
David Howells196ee9c2017-01-05 10:38:34 +0000406 if (req->remain > PAGE_SIZE - call->offset)
407 size = PAGE_SIZE - call->offset;
408 else
409 size = req->remain;
410 call->count = call->offset + size;
411 ASSERTCMP(call->count, <=, PAGE_SIZE);
412 req->remain -= size;
413
David Howells08e0e7c2007-04-26 15:55:03 -0700414 /* extract the returned data */
David Howellsb9b1f8d2007-05-10 03:15:21 -0700415 case 3:
David Howells6a0e3992017-03-16 16:27:46 +0000416 _debug("extract data %llu/%llu %zu/%u",
David Howells196ee9c2017-01-05 10:38:34 +0000417 req->remain, req->actual_len, call->offset, call->count);
418
419 buffer = kmap(req->pages[req->index]);
420 ret = afs_extract_data(call, buffer, call->count, true);
421 kunmap(req->pages[req->index]);
422 if (ret < 0)
423 return ret;
424 if (call->offset == PAGE_SIZE) {
425 if (req->page_done)
426 req->page_done(call, req);
David Howells29f06982017-03-16 16:27:46 +0000427 req->index++;
David Howells196ee9c2017-01-05 10:38:34 +0000428 if (req->remain > 0) {
David Howells196ee9c2017-01-05 10:38:34 +0000429 call->offset = 0;
David Howellse8e581a2017-03-16 16:27:44 +0000430 if (req->index >= req->nr_pages) {
431 call->unmarshall = 4;
David Howells6db3ac32017-03-16 16:27:44 +0000432 goto begin_discard;
David Howellse8e581a2017-03-16 16:27:44 +0000433 }
David Howells196ee9c2017-01-05 10:38:34 +0000434 goto begin_page;
435 }
David Howells08e0e7c2007-04-26 15:55:03 -0700436 }
David Howells6db3ac32017-03-16 16:27:44 +0000437 goto no_more_data;
438
439 /* Discard any excess data the server gave us */
440 begin_discard:
441 case 4:
David Howells6a0e3992017-03-16 16:27:46 +0000442 size = min_t(loff_t, sizeof(afs_discard_buffer), req->remain);
David Howells6db3ac32017-03-16 16:27:44 +0000443 call->count = size;
David Howells6a0e3992017-03-16 16:27:46 +0000444 _debug("extract discard %llu/%llu %zu/%u",
David Howells6db3ac32017-03-16 16:27:44 +0000445 req->remain, req->actual_len, call->offset, call->count);
446
447 call->offset = 0;
448 ret = afs_extract_data(call, afs_discard_buffer, call->count, true);
449 req->remain -= call->offset;
450 if (ret < 0)
451 return ret;
452 if (req->remain > 0)
453 goto begin_discard;
David Howells08e0e7c2007-04-26 15:55:03 -0700454
David Howells196ee9c2017-01-05 10:38:34 +0000455 no_more_data:
David Howells08e0e7c2007-04-26 15:55:03 -0700456 call->offset = 0;
David Howells6db3ac32017-03-16 16:27:44 +0000457 call->unmarshall = 5;
David Howells08e0e7c2007-04-26 15:55:03 -0700458
459 /* extract the metadata */
David Howells6db3ac32017-03-16 16:27:44 +0000460 case 5:
David Howellsd0016482016-08-30 20:42:14 +0100461 ret = afs_extract_data(call, call->buffer,
462 (21 + 3 + 6) * 4, false);
David Howells372ee162016-08-03 14:11:40 +0100463 if (ret < 0)
464 return ret;
David Howells08e0e7c2007-04-26 15:55:03 -0700465
466 bp = call->buffer;
David Howells31143d52007-05-09 02:33:46 -0700467 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
David Howellsc435ee32017-11-02 15:27:49 +0000468 xdr_decode_AFSCallBack(call, vnode, &bp);
David Howells97e30432017-11-02 15:27:48 +0000469 if (call->reply[1])
470 xdr_decode_AFSVolSync(&bp, call->reply[1]);
David Howells08e0e7c2007-04-26 15:55:03 -0700471
472 call->offset = 0;
473 call->unmarshall++;
474
David Howells6db3ac32017-03-16 16:27:44 +0000475 case 6:
David Howells08e0e7c2007-04-26 15:55:03 -0700476 break;
477 }
478
David Howells6db3ac32017-03-16 16:27:44 +0000479 for (; req->index < req->nr_pages; req->index++) {
480 if (call->count < PAGE_SIZE)
481 zero_user_segment(req->pages[req->index],
482 call->count, PAGE_SIZE);
David Howells196ee9c2017-01-05 10:38:34 +0000483 if (req->page_done)
484 req->page_done(call, req);
David Howells6db3ac32017-03-16 16:27:44 +0000485 call->count = 0;
David Howells416351f2007-05-09 02:33:45 -0700486 }
487
David Howells08e0e7c2007-04-26 15:55:03 -0700488 _leave(" = 0 [done]");
489 return 0;
490}
491
David Howells196ee9c2017-01-05 10:38:34 +0000492static void afs_fetch_data_destructor(struct afs_call *call)
493{
David Howells97e30432017-11-02 15:27:48 +0000494 struct afs_read *req = call->reply[2];
David Howells196ee9c2017-01-05 10:38:34 +0000495
496 afs_put_read(req);
497 afs_flat_call_destructor(call);
498}
499
David Howells08e0e7c2007-04-26 15:55:03 -0700500/*
501 * FS.FetchData operation type
502 */
503static const struct afs_call_type afs_RXFSFetchData = {
David Howells00d3b7a2007-04-26 15:57:07 -0700504 .name = "FS.FetchData",
David Howells025db802017-11-02 15:27:51 +0000505 .op = afs_FS_FetchData,
David Howells08e0e7c2007-04-26 15:55:03 -0700506 .deliver = afs_deliver_fs_fetch_data,
David Howells196ee9c2017-01-05 10:38:34 +0000507 .destructor = afs_fetch_data_destructor,
David Howells08e0e7c2007-04-26 15:55:03 -0700508};
509
David Howellsb9b1f8d2007-05-10 03:15:21 -0700510static const struct afs_call_type afs_RXFSFetchData64 = {
511 .name = "FS.FetchData64",
David Howells025db802017-11-02 15:27:51 +0000512 .op = afs_FS_FetchData64,
David Howellsb9b1f8d2007-05-10 03:15:21 -0700513 .deliver = afs_deliver_fs_fetch_data,
David Howells196ee9c2017-01-05 10:38:34 +0000514 .destructor = afs_fetch_data_destructor,
David Howellsb9b1f8d2007-05-10 03:15:21 -0700515};
516
517/*
518 * fetch data from a very large file
519 */
David Howellsd2ddc772017-11-02 15:27:50 +0000520static int afs_fs_fetch_data64(struct afs_fs_cursor *fc, struct afs_read *req)
David Howellsb9b1f8d2007-05-10 03:15:21 -0700521{
David Howellsd2ddc772017-11-02 15:27:50 +0000522 struct afs_vnode *vnode = fc->vnode;
David Howellsb9b1f8d2007-05-10 03:15:21 -0700523 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000524 struct afs_net *net = afs_v2net(vnode);
David Howellsb9b1f8d2007-05-10 03:15:21 -0700525 __be32 *bp;
526
527 _enter("");
528
David Howellsf044c882017-11-02 15:27:45 +0000529 call = afs_alloc_flat_call(net, &afs_RXFSFetchData64, 32, (21 + 3 + 6) * 4);
David Howellsb9b1f8d2007-05-10 03:15:21 -0700530 if (!call)
531 return -ENOMEM;
532
David Howellsd2ddc772017-11-02 15:27:50 +0000533 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000534 call->reply[0] = vnode;
535 call->reply[1] = NULL; /* volsync */
536 call->reply[2] = req;
David Howellsb9b1f8d2007-05-10 03:15:21 -0700537
538 /* marshall the parameters */
539 bp = call->request;
540 bp[0] = htonl(FSFETCHDATA64);
541 bp[1] = htonl(vnode->fid.vid);
542 bp[2] = htonl(vnode->fid.vnode);
543 bp[3] = htonl(vnode->fid.unique);
David Howells196ee9c2017-01-05 10:38:34 +0000544 bp[4] = htonl(upper_32_bits(req->pos));
545 bp[5] = htonl(lower_32_bits(req->pos));
David Howellsb9b1f8d2007-05-10 03:15:21 -0700546 bp[6] = 0;
David Howells196ee9c2017-01-05 10:38:34 +0000547 bp[7] = htonl(lower_32_bits(req->len));
David Howellsb9b1f8d2007-05-10 03:15:21 -0700548
David Howells196ee9c2017-01-05 10:38:34 +0000549 atomic_inc(&req->usage);
David Howellsd2ddc772017-11-02 15:27:50 +0000550 call->cb_break = fc->cb_break;
551 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000552 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000553 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsb9b1f8d2007-05-10 03:15:21 -0700554}
555
David Howells08e0e7c2007-04-26 15:55:03 -0700556/*
557 * fetch data from a file
558 */
David Howellsd2ddc772017-11-02 15:27:50 +0000559int afs_fs_fetch_data(struct afs_fs_cursor *fc, struct afs_read *req)
David Howells08e0e7c2007-04-26 15:55:03 -0700560{
David Howellsd2ddc772017-11-02 15:27:50 +0000561 struct afs_vnode *vnode = fc->vnode;
David Howells08e0e7c2007-04-26 15:55:03 -0700562 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000563 struct afs_net *net = afs_v2net(vnode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700564 __be32 *bp;
565
David Howells196ee9c2017-01-05 10:38:34 +0000566 if (upper_32_bits(req->pos) ||
567 upper_32_bits(req->len) ||
568 upper_32_bits(req->pos + req->len))
David Howellsd2ddc772017-11-02 15:27:50 +0000569 return afs_fs_fetch_data64(fc, req);
David Howellsb9b1f8d2007-05-10 03:15:21 -0700570
David Howells08e0e7c2007-04-26 15:55:03 -0700571 _enter("");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700572
David Howellsf044c882017-11-02 15:27:45 +0000573 call = afs_alloc_flat_call(net, &afs_RXFSFetchData, 24, (21 + 3 + 6) * 4);
David Howells08e0e7c2007-04-26 15:55:03 -0700574 if (!call)
575 return -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576
David Howellsd2ddc772017-11-02 15:27:50 +0000577 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000578 call->reply[0] = vnode;
579 call->reply[1] = NULL; /* volsync */
580 call->reply[2] = req;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700581
582 /* marshall the parameters */
David Howells08e0e7c2007-04-26 15:55:03 -0700583 bp = call->request;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700584 bp[0] = htonl(FSFETCHDATA);
David Howells08e0e7c2007-04-26 15:55:03 -0700585 bp[1] = htonl(vnode->fid.vid);
586 bp[2] = htonl(vnode->fid.vnode);
587 bp[3] = htonl(vnode->fid.unique);
David Howells196ee9c2017-01-05 10:38:34 +0000588 bp[4] = htonl(lower_32_bits(req->pos));
589 bp[5] = htonl(lower_32_bits(req->len));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700590
David Howells196ee9c2017-01-05 10:38:34 +0000591 atomic_inc(&req->usage);
David Howellsd2ddc772017-11-02 15:27:50 +0000592 call->cb_break = fc->cb_break;
593 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000594 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000595 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsec268152007-04-26 15:49:28 -0700596}
David Howells260a9802007-04-26 15:59:35 -0700597
598/*
599 * deliver reply data to an FS.CreateFile or an FS.MakeDir
600 */
David Howellsd0016482016-08-30 20:42:14 +0100601static int afs_deliver_fs_create_vnode(struct afs_call *call)
David Howells260a9802007-04-26 15:59:35 -0700602{
David Howells97e30432017-11-02 15:27:48 +0000603 struct afs_vnode *vnode = call->reply[0];
David Howells260a9802007-04-26 15:59:35 -0700604 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100605 int ret;
David Howells260a9802007-04-26 15:59:35 -0700606
David Howellsd0016482016-08-30 20:42:14 +0100607 _enter("{%u}", call->unmarshall);
David Howells260a9802007-04-26 15:59:35 -0700608
David Howellsd0016482016-08-30 20:42:14 +0100609 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100610 if (ret < 0)
611 return ret;
David Howells260a9802007-04-26 15:59:35 -0700612
613 /* unmarshall the reply once we've received all of it */
614 bp = call->buffer;
David Howells97e30432017-11-02 15:27:48 +0000615 xdr_decode_AFSFid(&bp, call->reply[1]);
616 xdr_decode_AFSFetchStatus(&bp, call->reply[2], NULL, NULL);
David Howells31143d52007-05-09 02:33:46 -0700617 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
David Howells97e30432017-11-02 15:27:48 +0000618 xdr_decode_AFSCallBack_raw(&bp, call->reply[3]);
619 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells260a9802007-04-26 15:59:35 -0700620
621 _leave(" = 0 [done]");
622 return 0;
623}
624
625/*
626 * FS.CreateFile and FS.MakeDir operation type
627 */
David Howells025db802017-11-02 15:27:51 +0000628static const struct afs_call_type afs_RXFSCreateFile = {
629 .name = "FS.CreateFile",
630 .op = afs_FS_CreateFile,
631 .deliver = afs_deliver_fs_create_vnode,
632 .destructor = afs_flat_call_destructor,
633};
634
635static const struct afs_call_type afs_RXFSMakeDir = {
636 .name = "FS.MakeDir",
637 .op = afs_FS_MakeDir,
David Howells260a9802007-04-26 15:59:35 -0700638 .deliver = afs_deliver_fs_create_vnode,
David Howells260a9802007-04-26 15:59:35 -0700639 .destructor = afs_flat_call_destructor,
640};
641
642/*
643 * create a file or make a directory
644 */
David Howells8b2a4642017-11-02 15:27:50 +0000645int afs_fs_create(struct afs_fs_cursor *fc,
David Howells260a9802007-04-26 15:59:35 -0700646 const char *name,
647 umode_t mode,
648 struct afs_fid *newfid,
649 struct afs_file_status *newstatus,
David Howellsd2ddc772017-11-02 15:27:50 +0000650 struct afs_callback *newcb)
David Howells260a9802007-04-26 15:59:35 -0700651{
David Howellsd2ddc772017-11-02 15:27:50 +0000652 struct afs_vnode *vnode = fc->vnode;
David Howells260a9802007-04-26 15:59:35 -0700653 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000654 struct afs_net *net = afs_v2net(vnode);
David Howells260a9802007-04-26 15:59:35 -0700655 size_t namesz, reqsz, padsz;
656 __be32 *bp;
657
658 _enter("");
659
660 namesz = strlen(name);
661 padsz = (4 - (namesz & 3)) & 3;
662 reqsz = (5 * 4) + namesz + padsz + (6 * 4);
663
David Howells025db802017-11-02 15:27:51 +0000664 call = afs_alloc_flat_call(
665 net, S_ISDIR(mode) ? &afs_RXFSMakeDir : &afs_RXFSCreateFile,
666 reqsz, (3 + 21 + 21 + 3 + 6) * 4);
David Howells260a9802007-04-26 15:59:35 -0700667 if (!call)
668 return -ENOMEM;
669
David Howellsd2ddc772017-11-02 15:27:50 +0000670 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000671 call->reply[0] = vnode;
672 call->reply[1] = newfid;
673 call->reply[2] = newstatus;
674 call->reply[3] = newcb;
David Howells260a9802007-04-26 15:59:35 -0700675
676 /* marshall the parameters */
677 bp = call->request;
678 *bp++ = htonl(S_ISDIR(mode) ? FSMAKEDIR : FSCREATEFILE);
679 *bp++ = htonl(vnode->fid.vid);
680 *bp++ = htonl(vnode->fid.vnode);
681 *bp++ = htonl(vnode->fid.unique);
682 *bp++ = htonl(namesz);
683 memcpy(bp, name, namesz);
684 bp = (void *) bp + namesz;
685 if (padsz > 0) {
686 memset(bp, 0, padsz);
687 bp = (void *) bp + padsz;
688 }
Marc Dionneab94f5d2017-03-16 16:27:47 +0000689 *bp++ = htonl(AFS_SET_MODE | AFS_SET_MTIME);
690 *bp++ = htonl(vnode->vfs_inode.i_mtime.tv_sec); /* mtime */
David Howells260a9802007-04-26 15:59:35 -0700691 *bp++ = 0; /* owner */
692 *bp++ = 0; /* group */
693 *bp++ = htonl(mode & S_IALLUGO); /* unix mode */
694 *bp++ = 0; /* segment size */
695
David Howellsd2ddc772017-11-02 15:27:50 +0000696 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000697 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000698 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells260a9802007-04-26 15:59:35 -0700699}
700
701/*
702 * deliver reply data to an FS.RemoveFile or FS.RemoveDir
703 */
David Howellsd0016482016-08-30 20:42:14 +0100704static int afs_deliver_fs_remove(struct afs_call *call)
David Howells260a9802007-04-26 15:59:35 -0700705{
David Howells97e30432017-11-02 15:27:48 +0000706 struct afs_vnode *vnode = call->reply[0];
David Howells260a9802007-04-26 15:59:35 -0700707 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100708 int ret;
David Howells260a9802007-04-26 15:59:35 -0700709
David Howellsd0016482016-08-30 20:42:14 +0100710 _enter("{%u}", call->unmarshall);
David Howells260a9802007-04-26 15:59:35 -0700711
David Howellsd0016482016-08-30 20:42:14 +0100712 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100713 if (ret < 0)
714 return ret;
David Howells260a9802007-04-26 15:59:35 -0700715
716 /* unmarshall the reply once we've received all of it */
717 bp = call->buffer;
David Howells31143d52007-05-09 02:33:46 -0700718 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
David Howells97e30432017-11-02 15:27:48 +0000719 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells260a9802007-04-26 15:59:35 -0700720
721 _leave(" = 0 [done]");
722 return 0;
723}
724
725/*
726 * FS.RemoveDir/FS.RemoveFile operation type
727 */
David Howells025db802017-11-02 15:27:51 +0000728static const struct afs_call_type afs_RXFSRemoveFile = {
729 .name = "FS.RemoveFile",
730 .op = afs_FS_RemoveFile,
731 .deliver = afs_deliver_fs_remove,
732 .destructor = afs_flat_call_destructor,
733};
734
735static const struct afs_call_type afs_RXFSRemoveDir = {
736 .name = "FS.RemoveDir",
737 .op = afs_FS_RemoveDir,
David Howells260a9802007-04-26 15:59:35 -0700738 .deliver = afs_deliver_fs_remove,
David Howells260a9802007-04-26 15:59:35 -0700739 .destructor = afs_flat_call_destructor,
740};
741
742/*
743 * remove a file or directory
744 */
David Howellsd2ddc772017-11-02 15:27:50 +0000745int afs_fs_remove(struct afs_fs_cursor *fc, const char *name, bool isdir)
David Howells260a9802007-04-26 15:59:35 -0700746{
David Howellsd2ddc772017-11-02 15:27:50 +0000747 struct afs_vnode *vnode = fc->vnode;
David Howells260a9802007-04-26 15:59:35 -0700748 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000749 struct afs_net *net = afs_v2net(vnode);
David Howells260a9802007-04-26 15:59:35 -0700750 size_t namesz, reqsz, padsz;
751 __be32 *bp;
752
753 _enter("");
754
755 namesz = strlen(name);
756 padsz = (4 - (namesz & 3)) & 3;
757 reqsz = (5 * 4) + namesz + padsz;
758
David Howells025db802017-11-02 15:27:51 +0000759 call = afs_alloc_flat_call(
760 net, isdir ? &afs_RXFSRemoveDir : &afs_RXFSRemoveFile,
761 reqsz, (21 + 6) * 4);
David Howells260a9802007-04-26 15:59:35 -0700762 if (!call)
763 return -ENOMEM;
764
David Howellsd2ddc772017-11-02 15:27:50 +0000765 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000766 call->reply[0] = vnode;
David Howells260a9802007-04-26 15:59:35 -0700767
768 /* marshall the parameters */
769 bp = call->request;
770 *bp++ = htonl(isdir ? FSREMOVEDIR : FSREMOVEFILE);
771 *bp++ = htonl(vnode->fid.vid);
772 *bp++ = htonl(vnode->fid.vnode);
773 *bp++ = htonl(vnode->fid.unique);
774 *bp++ = htonl(namesz);
775 memcpy(bp, name, namesz);
776 bp = (void *) bp + namesz;
777 if (padsz > 0) {
778 memset(bp, 0, padsz);
779 bp = (void *) bp + padsz;
780 }
781
David Howellsd2ddc772017-11-02 15:27:50 +0000782 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000783 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000784 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells260a9802007-04-26 15:59:35 -0700785}
786
787/*
788 * deliver reply data to an FS.Link
789 */
David Howellsd0016482016-08-30 20:42:14 +0100790static int afs_deliver_fs_link(struct afs_call *call)
David Howells260a9802007-04-26 15:59:35 -0700791{
David Howells97e30432017-11-02 15:27:48 +0000792 struct afs_vnode *dvnode = call->reply[0], *vnode = call->reply[1];
David Howells260a9802007-04-26 15:59:35 -0700793 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100794 int ret;
David Howells260a9802007-04-26 15:59:35 -0700795
David Howellsd0016482016-08-30 20:42:14 +0100796 _enter("{%u}", call->unmarshall);
David Howells260a9802007-04-26 15:59:35 -0700797
David Howellsd0016482016-08-30 20:42:14 +0100798 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100799 if (ret < 0)
800 return ret;
David Howells260a9802007-04-26 15:59:35 -0700801
802 /* unmarshall the reply once we've received all of it */
803 bp = call->buffer;
David Howells31143d52007-05-09 02:33:46 -0700804 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
805 xdr_decode_AFSFetchStatus(&bp, &dvnode->status, dvnode, NULL);
David Howells97e30432017-11-02 15:27:48 +0000806 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells260a9802007-04-26 15:59:35 -0700807
808 _leave(" = 0 [done]");
809 return 0;
810}
811
812/*
813 * FS.Link operation type
814 */
815static const struct afs_call_type afs_RXFSLink = {
816 .name = "FS.Link",
David Howells025db802017-11-02 15:27:51 +0000817 .op = afs_FS_Link,
David Howells260a9802007-04-26 15:59:35 -0700818 .deliver = afs_deliver_fs_link,
David Howells260a9802007-04-26 15:59:35 -0700819 .destructor = afs_flat_call_destructor,
820};
821
822/*
823 * make a hard link
824 */
David Howellsd2ddc772017-11-02 15:27:50 +0000825int afs_fs_link(struct afs_fs_cursor *fc, struct afs_vnode *vnode,
826 const char *name)
David Howells260a9802007-04-26 15:59:35 -0700827{
David Howellsd2ddc772017-11-02 15:27:50 +0000828 struct afs_vnode *dvnode = fc->vnode;
David Howells260a9802007-04-26 15:59:35 -0700829 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000830 struct afs_net *net = afs_v2net(vnode);
David Howells260a9802007-04-26 15:59:35 -0700831 size_t namesz, reqsz, padsz;
832 __be32 *bp;
833
834 _enter("");
835
836 namesz = strlen(name);
837 padsz = (4 - (namesz & 3)) & 3;
838 reqsz = (5 * 4) + namesz + padsz + (3 * 4);
839
David Howellsf044c882017-11-02 15:27:45 +0000840 call = afs_alloc_flat_call(net, &afs_RXFSLink, reqsz, (21 + 21 + 6) * 4);
David Howells260a9802007-04-26 15:59:35 -0700841 if (!call)
842 return -ENOMEM;
843
David Howellsd2ddc772017-11-02 15:27:50 +0000844 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000845 call->reply[0] = dvnode;
846 call->reply[1] = vnode;
David Howells260a9802007-04-26 15:59:35 -0700847
848 /* marshall the parameters */
849 bp = call->request;
850 *bp++ = htonl(FSLINK);
851 *bp++ = htonl(dvnode->fid.vid);
852 *bp++ = htonl(dvnode->fid.vnode);
853 *bp++ = htonl(dvnode->fid.unique);
854 *bp++ = htonl(namesz);
855 memcpy(bp, name, namesz);
856 bp = (void *) bp + namesz;
857 if (padsz > 0) {
858 memset(bp, 0, padsz);
859 bp = (void *) bp + padsz;
860 }
861 *bp++ = htonl(vnode->fid.vid);
862 *bp++ = htonl(vnode->fid.vnode);
863 *bp++ = htonl(vnode->fid.unique);
864
David Howellsd2ddc772017-11-02 15:27:50 +0000865 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000866 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000867 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells260a9802007-04-26 15:59:35 -0700868}
869
870/*
871 * deliver reply data to an FS.Symlink
872 */
David Howellsd0016482016-08-30 20:42:14 +0100873static int afs_deliver_fs_symlink(struct afs_call *call)
David Howells260a9802007-04-26 15:59:35 -0700874{
David Howells97e30432017-11-02 15:27:48 +0000875 struct afs_vnode *vnode = call->reply[0];
David Howells260a9802007-04-26 15:59:35 -0700876 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100877 int ret;
David Howells260a9802007-04-26 15:59:35 -0700878
David Howellsd0016482016-08-30 20:42:14 +0100879 _enter("{%u}", call->unmarshall);
David Howells260a9802007-04-26 15:59:35 -0700880
David Howellsd0016482016-08-30 20:42:14 +0100881 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100882 if (ret < 0)
883 return ret;
David Howells260a9802007-04-26 15:59:35 -0700884
885 /* unmarshall the reply once we've received all of it */
886 bp = call->buffer;
David Howells97e30432017-11-02 15:27:48 +0000887 xdr_decode_AFSFid(&bp, call->reply[1]);
888 xdr_decode_AFSFetchStatus(&bp, call->reply[2], NULL, NULL);
David Howells31143d52007-05-09 02:33:46 -0700889 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, NULL);
David Howells97e30432017-11-02 15:27:48 +0000890 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells260a9802007-04-26 15:59:35 -0700891
892 _leave(" = 0 [done]");
893 return 0;
894}
895
896/*
897 * FS.Symlink operation type
898 */
899static const struct afs_call_type afs_RXFSSymlink = {
900 .name = "FS.Symlink",
David Howells025db802017-11-02 15:27:51 +0000901 .op = afs_FS_Symlink,
David Howells260a9802007-04-26 15:59:35 -0700902 .deliver = afs_deliver_fs_symlink,
David Howells260a9802007-04-26 15:59:35 -0700903 .destructor = afs_flat_call_destructor,
904};
905
906/*
907 * create a symbolic link
908 */
David Howells8b2a4642017-11-02 15:27:50 +0000909int afs_fs_symlink(struct afs_fs_cursor *fc,
David Howells260a9802007-04-26 15:59:35 -0700910 const char *name,
911 const char *contents,
912 struct afs_fid *newfid,
David Howellsd2ddc772017-11-02 15:27:50 +0000913 struct afs_file_status *newstatus)
David Howells260a9802007-04-26 15:59:35 -0700914{
David Howellsd2ddc772017-11-02 15:27:50 +0000915 struct afs_vnode *vnode = fc->vnode;
David Howells260a9802007-04-26 15:59:35 -0700916 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +0000917 struct afs_net *net = afs_v2net(vnode);
David Howells260a9802007-04-26 15:59:35 -0700918 size_t namesz, reqsz, padsz, c_namesz, c_padsz;
919 __be32 *bp;
920
921 _enter("");
922
923 namesz = strlen(name);
924 padsz = (4 - (namesz & 3)) & 3;
925
926 c_namesz = strlen(contents);
927 c_padsz = (4 - (c_namesz & 3)) & 3;
928
929 reqsz = (6 * 4) + namesz + padsz + c_namesz + c_padsz + (6 * 4);
930
David Howellsf044c882017-11-02 15:27:45 +0000931 call = afs_alloc_flat_call(net, &afs_RXFSSymlink, reqsz,
David Howells260a9802007-04-26 15:59:35 -0700932 (3 + 21 + 21 + 6) * 4);
933 if (!call)
934 return -ENOMEM;
935
David Howellsd2ddc772017-11-02 15:27:50 +0000936 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +0000937 call->reply[0] = vnode;
938 call->reply[1] = newfid;
939 call->reply[2] = newstatus;
David Howells260a9802007-04-26 15:59:35 -0700940
941 /* marshall the parameters */
942 bp = call->request;
943 *bp++ = htonl(FSSYMLINK);
944 *bp++ = htonl(vnode->fid.vid);
945 *bp++ = htonl(vnode->fid.vnode);
946 *bp++ = htonl(vnode->fid.unique);
947 *bp++ = htonl(namesz);
948 memcpy(bp, name, namesz);
949 bp = (void *) bp + namesz;
950 if (padsz > 0) {
951 memset(bp, 0, padsz);
952 bp = (void *) bp + padsz;
953 }
954 *bp++ = htonl(c_namesz);
955 memcpy(bp, contents, c_namesz);
956 bp = (void *) bp + c_namesz;
957 if (c_padsz > 0) {
958 memset(bp, 0, c_padsz);
959 bp = (void *) bp + c_padsz;
960 }
Marc Dionneab94f5d2017-03-16 16:27:47 +0000961 *bp++ = htonl(AFS_SET_MODE | AFS_SET_MTIME);
962 *bp++ = htonl(vnode->vfs_inode.i_mtime.tv_sec); /* mtime */
David Howells260a9802007-04-26 15:59:35 -0700963 *bp++ = 0; /* owner */
964 *bp++ = 0; /* group */
965 *bp++ = htonl(S_IRWXUGO); /* unix mode */
966 *bp++ = 0; /* segment size */
967
David Howellsd2ddc772017-11-02 15:27:50 +0000968 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +0000969 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +0000970 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells260a9802007-04-26 15:59:35 -0700971}
972
973/*
974 * deliver reply data to an FS.Rename
975 */
David Howellsd0016482016-08-30 20:42:14 +0100976static int afs_deliver_fs_rename(struct afs_call *call)
David Howells260a9802007-04-26 15:59:35 -0700977{
David Howells97e30432017-11-02 15:27:48 +0000978 struct afs_vnode *orig_dvnode = call->reply[0], *new_dvnode = call->reply[1];
David Howells260a9802007-04-26 15:59:35 -0700979 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +0100980 int ret;
David Howells260a9802007-04-26 15:59:35 -0700981
David Howellsd0016482016-08-30 20:42:14 +0100982 _enter("{%u}", call->unmarshall);
David Howells260a9802007-04-26 15:59:35 -0700983
David Howellsd0016482016-08-30 20:42:14 +0100984 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +0100985 if (ret < 0)
986 return ret;
David Howells260a9802007-04-26 15:59:35 -0700987
988 /* unmarshall the reply once we've received all of it */
989 bp = call->buffer;
David Howells31143d52007-05-09 02:33:46 -0700990 xdr_decode_AFSFetchStatus(&bp, &orig_dvnode->status, orig_dvnode, NULL);
David Howells260a9802007-04-26 15:59:35 -0700991 if (new_dvnode != orig_dvnode)
David Howells31143d52007-05-09 02:33:46 -0700992 xdr_decode_AFSFetchStatus(&bp, &new_dvnode->status, new_dvnode,
993 NULL);
David Howells97e30432017-11-02 15:27:48 +0000994 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells260a9802007-04-26 15:59:35 -0700995
996 _leave(" = 0 [done]");
997 return 0;
998}
999
1000/*
1001 * FS.Rename operation type
1002 */
1003static const struct afs_call_type afs_RXFSRename = {
1004 .name = "FS.Rename",
David Howells025db802017-11-02 15:27:51 +00001005 .op = afs_FS_Rename,
David Howells260a9802007-04-26 15:59:35 -07001006 .deliver = afs_deliver_fs_rename,
David Howells260a9802007-04-26 15:59:35 -07001007 .destructor = afs_flat_call_destructor,
1008};
1009
1010/*
1011 * create a symbolic link
1012 */
David Howells8b2a4642017-11-02 15:27:50 +00001013int afs_fs_rename(struct afs_fs_cursor *fc,
David Howells260a9802007-04-26 15:59:35 -07001014 const char *orig_name,
1015 struct afs_vnode *new_dvnode,
David Howellsd2ddc772017-11-02 15:27:50 +00001016 const char *new_name)
David Howells260a9802007-04-26 15:59:35 -07001017{
David Howellsd2ddc772017-11-02 15:27:50 +00001018 struct afs_vnode *orig_dvnode = fc->vnode;
David Howells260a9802007-04-26 15:59:35 -07001019 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001020 struct afs_net *net = afs_v2net(orig_dvnode);
David Howells260a9802007-04-26 15:59:35 -07001021 size_t reqsz, o_namesz, o_padsz, n_namesz, n_padsz;
1022 __be32 *bp;
1023
1024 _enter("");
1025
1026 o_namesz = strlen(orig_name);
1027 o_padsz = (4 - (o_namesz & 3)) & 3;
1028
1029 n_namesz = strlen(new_name);
1030 n_padsz = (4 - (n_namesz & 3)) & 3;
1031
1032 reqsz = (4 * 4) +
1033 4 + o_namesz + o_padsz +
1034 (3 * 4) +
1035 4 + n_namesz + n_padsz;
1036
David Howellsf044c882017-11-02 15:27:45 +00001037 call = afs_alloc_flat_call(net, &afs_RXFSRename, reqsz, (21 + 21 + 6) * 4);
David Howells260a9802007-04-26 15:59:35 -07001038 if (!call)
1039 return -ENOMEM;
1040
David Howellsd2ddc772017-11-02 15:27:50 +00001041 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001042 call->reply[0] = orig_dvnode;
1043 call->reply[1] = new_dvnode;
David Howells260a9802007-04-26 15:59:35 -07001044
1045 /* marshall the parameters */
1046 bp = call->request;
1047 *bp++ = htonl(FSRENAME);
1048 *bp++ = htonl(orig_dvnode->fid.vid);
1049 *bp++ = htonl(orig_dvnode->fid.vnode);
1050 *bp++ = htonl(orig_dvnode->fid.unique);
1051 *bp++ = htonl(o_namesz);
1052 memcpy(bp, orig_name, o_namesz);
1053 bp = (void *) bp + o_namesz;
1054 if (o_padsz > 0) {
1055 memset(bp, 0, o_padsz);
1056 bp = (void *) bp + o_padsz;
1057 }
1058
1059 *bp++ = htonl(new_dvnode->fid.vid);
1060 *bp++ = htonl(new_dvnode->fid.vnode);
1061 *bp++ = htonl(new_dvnode->fid.unique);
1062 *bp++ = htonl(n_namesz);
1063 memcpy(bp, new_name, n_namesz);
1064 bp = (void *) bp + n_namesz;
1065 if (n_padsz > 0) {
1066 memset(bp, 0, n_padsz);
1067 bp = (void *) bp + n_padsz;
1068 }
1069
David Howellsd2ddc772017-11-02 15:27:50 +00001070 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001071 trace_afs_make_fs_call(call, &orig_dvnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001072 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells260a9802007-04-26 15:59:35 -07001073}
David Howells31143d52007-05-09 02:33:46 -07001074
1075/*
1076 * deliver reply data to an FS.StoreData
1077 */
David Howellsd0016482016-08-30 20:42:14 +01001078static int afs_deliver_fs_store_data(struct afs_call *call)
David Howells31143d52007-05-09 02:33:46 -07001079{
David Howells97e30432017-11-02 15:27:48 +00001080 struct afs_vnode *vnode = call->reply[0];
David Howells31143d52007-05-09 02:33:46 -07001081 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +01001082 int ret;
David Howells31143d52007-05-09 02:33:46 -07001083
David Howellsd0016482016-08-30 20:42:14 +01001084 _enter("");
David Howells31143d52007-05-09 02:33:46 -07001085
David Howellsd0016482016-08-30 20:42:14 +01001086 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +01001087 if (ret < 0)
1088 return ret;
David Howells31143d52007-05-09 02:33:46 -07001089
1090 /* unmarshall the reply once we've received all of it */
1091 bp = call->buffer;
1092 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode,
1093 &call->store_version);
David Howells97e30432017-11-02 15:27:48 +00001094 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells31143d52007-05-09 02:33:46 -07001095
1096 afs_pages_written_back(vnode, call);
1097
1098 _leave(" = 0 [done]");
1099 return 0;
1100}
1101
1102/*
1103 * FS.StoreData operation type
1104 */
1105static const struct afs_call_type afs_RXFSStoreData = {
1106 .name = "FS.StoreData",
David Howells025db802017-11-02 15:27:51 +00001107 .op = afs_FS_StoreData,
David Howells31143d52007-05-09 02:33:46 -07001108 .deliver = afs_deliver_fs_store_data,
David Howells31143d52007-05-09 02:33:46 -07001109 .destructor = afs_flat_call_destructor,
1110};
1111
David Howellsb9b1f8d2007-05-10 03:15:21 -07001112static const struct afs_call_type afs_RXFSStoreData64 = {
1113 .name = "FS.StoreData64",
David Howells025db802017-11-02 15:27:51 +00001114 .op = afs_FS_StoreData64,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001115 .deliver = afs_deliver_fs_store_data,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001116 .destructor = afs_flat_call_destructor,
1117};
1118
1119/*
1120 * store a set of pages to a very large file
1121 */
David Howells8b2a4642017-11-02 15:27:50 +00001122static int afs_fs_store_data64(struct afs_fs_cursor *fc,
David Howells4343d002017-11-02 15:27:52 +00001123 struct address_space *mapping,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001124 pgoff_t first, pgoff_t last,
1125 unsigned offset, unsigned to,
David Howellsd2ddc772017-11-02 15:27:50 +00001126 loff_t size, loff_t pos, loff_t i_size)
David Howellsb9b1f8d2007-05-10 03:15:21 -07001127{
David Howells4343d002017-11-02 15:27:52 +00001128 struct afs_vnode *vnode = fc->vnode;
David Howellsb9b1f8d2007-05-10 03:15:21 -07001129 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001130 struct afs_net *net = afs_v2net(vnode);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001131 __be32 *bp;
1132
1133 _enter(",%x,{%x:%u},,",
David Howells4343d002017-11-02 15:27:52 +00001134 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001135
David Howellsf044c882017-11-02 15:27:45 +00001136 call = afs_alloc_flat_call(net, &afs_RXFSStoreData64,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001137 (4 + 6 + 3 * 2) * 4,
1138 (21 + 6) * 4);
1139 if (!call)
1140 return -ENOMEM;
1141
David Howells4343d002017-11-02 15:27:52 +00001142 call->key = fc->key;
1143 call->mapping = mapping;
David Howells97e30432017-11-02 15:27:48 +00001144 call->reply[0] = vnode;
David Howellsb9b1f8d2007-05-10 03:15:21 -07001145 call->first = first;
1146 call->last = last;
1147 call->first_offset = offset;
1148 call->last_to = to;
1149 call->send_pages = true;
1150 call->store_version = vnode->status.data_version + 1;
1151
1152 /* marshall the parameters */
1153 bp = call->request;
1154 *bp++ = htonl(FSSTOREDATA64);
1155 *bp++ = htonl(vnode->fid.vid);
1156 *bp++ = htonl(vnode->fid.vnode);
1157 *bp++ = htonl(vnode->fid.unique);
1158
Marc Dionneab94f5d2017-03-16 16:27:47 +00001159 *bp++ = htonl(AFS_SET_MTIME); /* mask */
1160 *bp++ = htonl(vnode->vfs_inode.i_mtime.tv_sec); /* mtime */
David Howellsb9b1f8d2007-05-10 03:15:21 -07001161 *bp++ = 0; /* owner */
1162 *bp++ = 0; /* group */
1163 *bp++ = 0; /* unix mode */
1164 *bp++ = 0; /* segment size */
1165
1166 *bp++ = htonl(pos >> 32);
1167 *bp++ = htonl((u32) pos);
1168 *bp++ = htonl(size >> 32);
1169 *bp++ = htonl((u32) size);
1170 *bp++ = htonl(i_size >> 32);
1171 *bp++ = htonl((u32) i_size);
1172
David Howells025db802017-11-02 15:27:51 +00001173 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001174 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001175}
1176
David Howells31143d52007-05-09 02:33:46 -07001177/*
1178 * store a set of pages
1179 */
David Howells4343d002017-11-02 15:27:52 +00001180int afs_fs_store_data(struct afs_fs_cursor *fc, struct address_space *mapping,
David Howells31143d52007-05-09 02:33:46 -07001181 pgoff_t first, pgoff_t last,
David Howellsd2ddc772017-11-02 15:27:50 +00001182 unsigned offset, unsigned to)
David Howells31143d52007-05-09 02:33:46 -07001183{
David Howells4343d002017-11-02 15:27:52 +00001184 struct afs_vnode *vnode = fc->vnode;
David Howells31143d52007-05-09 02:33:46 -07001185 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001186 struct afs_net *net = afs_v2net(vnode);
David Howells31143d52007-05-09 02:33:46 -07001187 loff_t size, pos, i_size;
1188 __be32 *bp;
1189
1190 _enter(",%x,{%x:%u},,",
David Howells4343d002017-11-02 15:27:52 +00001191 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
David Howells31143d52007-05-09 02:33:46 -07001192
David Howells146a1192017-03-16 16:27:47 +00001193 size = (loff_t)to - (loff_t)offset;
David Howells31143d52007-05-09 02:33:46 -07001194 if (first != last)
1195 size += (loff_t)(last - first) << PAGE_SHIFT;
1196 pos = (loff_t)first << PAGE_SHIFT;
1197 pos += offset;
1198
1199 i_size = i_size_read(&vnode->vfs_inode);
1200 if (pos + size > i_size)
1201 i_size = size + pos;
1202
1203 _debug("size %llx, at %llx, i_size %llx",
1204 (unsigned long long) size, (unsigned long long) pos,
1205 (unsigned long long) i_size);
1206
David Howellsb9b1f8d2007-05-10 03:15:21 -07001207 if (pos >> 32 || i_size >> 32 || size >> 32 || (pos + size) >> 32)
David Howells4343d002017-11-02 15:27:52 +00001208 return afs_fs_store_data64(fc, mapping, first, last, offset, to,
David Howellsd2ddc772017-11-02 15:27:50 +00001209 size, pos, i_size);
David Howells31143d52007-05-09 02:33:46 -07001210
David Howellsf044c882017-11-02 15:27:45 +00001211 call = afs_alloc_flat_call(net, &afs_RXFSStoreData,
David Howells31143d52007-05-09 02:33:46 -07001212 (4 + 6 + 3) * 4,
1213 (21 + 6) * 4);
1214 if (!call)
1215 return -ENOMEM;
1216
David Howells4343d002017-11-02 15:27:52 +00001217 call->key = fc->key;
1218 call->mapping = mapping;
David Howells97e30432017-11-02 15:27:48 +00001219 call->reply[0] = vnode;
David Howells31143d52007-05-09 02:33:46 -07001220 call->first = first;
1221 call->last = last;
1222 call->first_offset = offset;
1223 call->last_to = to;
1224 call->send_pages = true;
1225 call->store_version = vnode->status.data_version + 1;
1226
1227 /* marshall the parameters */
1228 bp = call->request;
1229 *bp++ = htonl(FSSTOREDATA);
1230 *bp++ = htonl(vnode->fid.vid);
1231 *bp++ = htonl(vnode->fid.vnode);
1232 *bp++ = htonl(vnode->fid.unique);
1233
Marc Dionneab94f5d2017-03-16 16:27:47 +00001234 *bp++ = htonl(AFS_SET_MTIME); /* mask */
1235 *bp++ = htonl(vnode->vfs_inode.i_mtime.tv_sec); /* mtime */
David Howells31143d52007-05-09 02:33:46 -07001236 *bp++ = 0; /* owner */
1237 *bp++ = 0; /* group */
1238 *bp++ = 0; /* unix mode */
1239 *bp++ = 0; /* segment size */
1240
1241 *bp++ = htonl(pos);
1242 *bp++ = htonl(size);
1243 *bp++ = htonl(i_size);
1244
David Howellsd2ddc772017-11-02 15:27:50 +00001245 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001246 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001247 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells31143d52007-05-09 02:33:46 -07001248}
1249
1250/*
1251 * deliver reply data to an FS.StoreStatus
1252 */
David Howellsd0016482016-08-30 20:42:14 +01001253static int afs_deliver_fs_store_status(struct afs_call *call)
David Howells31143d52007-05-09 02:33:46 -07001254{
1255 afs_dataversion_t *store_version;
David Howells97e30432017-11-02 15:27:48 +00001256 struct afs_vnode *vnode = call->reply[0];
David Howells31143d52007-05-09 02:33:46 -07001257 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +01001258 int ret;
David Howells31143d52007-05-09 02:33:46 -07001259
David Howellsd0016482016-08-30 20:42:14 +01001260 _enter("");
David Howells31143d52007-05-09 02:33:46 -07001261
David Howellsd0016482016-08-30 20:42:14 +01001262 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +01001263 if (ret < 0)
1264 return ret;
David Howells31143d52007-05-09 02:33:46 -07001265
1266 /* unmarshall the reply once we've received all of it */
1267 store_version = NULL;
1268 if (call->operation_ID == FSSTOREDATA)
1269 store_version = &call->store_version;
1270
1271 bp = call->buffer;
1272 xdr_decode_AFSFetchStatus(&bp, &vnode->status, vnode, store_version);
David Howells97e30432017-11-02 15:27:48 +00001273 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howells31143d52007-05-09 02:33:46 -07001274
1275 _leave(" = 0 [done]");
1276 return 0;
1277}
1278
1279/*
1280 * FS.StoreStatus operation type
1281 */
1282static const struct afs_call_type afs_RXFSStoreStatus = {
1283 .name = "FS.StoreStatus",
David Howells025db802017-11-02 15:27:51 +00001284 .op = afs_FS_StoreStatus,
David Howells31143d52007-05-09 02:33:46 -07001285 .deliver = afs_deliver_fs_store_status,
David Howells31143d52007-05-09 02:33:46 -07001286 .destructor = afs_flat_call_destructor,
1287};
1288
1289static const struct afs_call_type afs_RXFSStoreData_as_Status = {
1290 .name = "FS.StoreData",
David Howells025db802017-11-02 15:27:51 +00001291 .op = afs_FS_StoreData,
David Howells31143d52007-05-09 02:33:46 -07001292 .deliver = afs_deliver_fs_store_status,
David Howells31143d52007-05-09 02:33:46 -07001293 .destructor = afs_flat_call_destructor,
1294};
1295
David Howellsb9b1f8d2007-05-10 03:15:21 -07001296static const struct afs_call_type afs_RXFSStoreData64_as_Status = {
1297 .name = "FS.StoreData64",
David Howells025db802017-11-02 15:27:51 +00001298 .op = afs_FS_StoreData64,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001299 .deliver = afs_deliver_fs_store_status,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001300 .destructor = afs_flat_call_destructor,
1301};
1302
1303/*
1304 * set the attributes on a very large file, using FS.StoreData rather than
1305 * FS.StoreStatus so as to alter the file size also
1306 */
David Howellsd2ddc772017-11-02 15:27:50 +00001307static int afs_fs_setattr_size64(struct afs_fs_cursor *fc, struct iattr *attr)
David Howellsb9b1f8d2007-05-10 03:15:21 -07001308{
David Howellsd2ddc772017-11-02 15:27:50 +00001309 struct afs_vnode *vnode = fc->vnode;
David Howellsb9b1f8d2007-05-10 03:15:21 -07001310 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001311 struct afs_net *net = afs_v2net(vnode);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001312 __be32 *bp;
1313
1314 _enter(",%x,{%x:%u},,",
David Howellsd2ddc772017-11-02 15:27:50 +00001315 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001316
1317 ASSERT(attr->ia_valid & ATTR_SIZE);
1318
David Howellsf044c882017-11-02 15:27:45 +00001319 call = afs_alloc_flat_call(net, &afs_RXFSStoreData64_as_Status,
David Howellsb9b1f8d2007-05-10 03:15:21 -07001320 (4 + 6 + 3 * 2) * 4,
1321 (21 + 6) * 4);
1322 if (!call)
1323 return -ENOMEM;
1324
David Howellsd2ddc772017-11-02 15:27:50 +00001325 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001326 call->reply[0] = vnode;
David Howellsb9b1f8d2007-05-10 03:15:21 -07001327 call->store_version = vnode->status.data_version + 1;
David Howellsb9b1f8d2007-05-10 03:15:21 -07001328
1329 /* marshall the parameters */
1330 bp = call->request;
1331 *bp++ = htonl(FSSTOREDATA64);
1332 *bp++ = htonl(vnode->fid.vid);
1333 *bp++ = htonl(vnode->fid.vnode);
1334 *bp++ = htonl(vnode->fid.unique);
1335
1336 xdr_encode_AFS_StoreStatus(&bp, attr);
1337
1338 *bp++ = 0; /* position of start of write */
1339 *bp++ = 0;
1340 *bp++ = 0; /* size of write */
1341 *bp++ = 0;
1342 *bp++ = htonl(attr->ia_size >> 32); /* new file length */
1343 *bp++ = htonl((u32) attr->ia_size);
1344
David Howellsd2ddc772017-11-02 15:27:50 +00001345 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001346 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001347 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001348}
1349
David Howells31143d52007-05-09 02:33:46 -07001350/*
1351 * set the attributes on a file, using FS.StoreData rather than FS.StoreStatus
1352 * so as to alter the file size also
1353 */
David Howellsd2ddc772017-11-02 15:27:50 +00001354static int afs_fs_setattr_size(struct afs_fs_cursor *fc, struct iattr *attr)
David Howells31143d52007-05-09 02:33:46 -07001355{
David Howellsd2ddc772017-11-02 15:27:50 +00001356 struct afs_vnode *vnode = fc->vnode;
David Howells31143d52007-05-09 02:33:46 -07001357 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001358 struct afs_net *net = afs_v2net(vnode);
David Howells31143d52007-05-09 02:33:46 -07001359 __be32 *bp;
1360
1361 _enter(",%x,{%x:%u},,",
David Howellsd2ddc772017-11-02 15:27:50 +00001362 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
David Howells31143d52007-05-09 02:33:46 -07001363
1364 ASSERT(attr->ia_valid & ATTR_SIZE);
David Howellsb9b1f8d2007-05-10 03:15:21 -07001365 if (attr->ia_size >> 32)
David Howellsd2ddc772017-11-02 15:27:50 +00001366 return afs_fs_setattr_size64(fc, attr);
David Howells31143d52007-05-09 02:33:46 -07001367
David Howellsf044c882017-11-02 15:27:45 +00001368 call = afs_alloc_flat_call(net, &afs_RXFSStoreData_as_Status,
David Howells31143d52007-05-09 02:33:46 -07001369 (4 + 6 + 3) * 4,
1370 (21 + 6) * 4);
1371 if (!call)
1372 return -ENOMEM;
1373
David Howellsd2ddc772017-11-02 15:27:50 +00001374 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001375 call->reply[0] = vnode;
David Howells31143d52007-05-09 02:33:46 -07001376 call->store_version = vnode->status.data_version + 1;
David Howells31143d52007-05-09 02:33:46 -07001377
1378 /* marshall the parameters */
1379 bp = call->request;
1380 *bp++ = htonl(FSSTOREDATA);
1381 *bp++ = htonl(vnode->fid.vid);
1382 *bp++ = htonl(vnode->fid.vnode);
1383 *bp++ = htonl(vnode->fid.unique);
1384
1385 xdr_encode_AFS_StoreStatus(&bp, attr);
1386
1387 *bp++ = 0; /* position of start of write */
1388 *bp++ = 0; /* size of write */
1389 *bp++ = htonl(attr->ia_size); /* new file length */
1390
David Howellsd2ddc772017-11-02 15:27:50 +00001391 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001392 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001393 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells31143d52007-05-09 02:33:46 -07001394}
1395
1396/*
1397 * set the attributes on a file, using FS.StoreData if there's a change in file
1398 * size, and FS.StoreStatus otherwise
1399 */
David Howellsd2ddc772017-11-02 15:27:50 +00001400int afs_fs_setattr(struct afs_fs_cursor *fc, struct iattr *attr)
David Howells31143d52007-05-09 02:33:46 -07001401{
David Howellsd2ddc772017-11-02 15:27:50 +00001402 struct afs_vnode *vnode = fc->vnode;
David Howells31143d52007-05-09 02:33:46 -07001403 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001404 struct afs_net *net = afs_v2net(vnode);
David Howells31143d52007-05-09 02:33:46 -07001405 __be32 *bp;
1406
1407 if (attr->ia_valid & ATTR_SIZE)
David Howellsd2ddc772017-11-02 15:27:50 +00001408 return afs_fs_setattr_size(fc, attr);
David Howells31143d52007-05-09 02:33:46 -07001409
1410 _enter(",%x,{%x:%u},,",
David Howellsd2ddc772017-11-02 15:27:50 +00001411 key_serial(fc->key), vnode->fid.vid, vnode->fid.vnode);
David Howells31143d52007-05-09 02:33:46 -07001412
David Howellsf044c882017-11-02 15:27:45 +00001413 call = afs_alloc_flat_call(net, &afs_RXFSStoreStatus,
David Howells31143d52007-05-09 02:33:46 -07001414 (4 + 6) * 4,
1415 (21 + 6) * 4);
1416 if (!call)
1417 return -ENOMEM;
1418
David Howellsd2ddc772017-11-02 15:27:50 +00001419 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001420 call->reply[0] = vnode;
David Howells31143d52007-05-09 02:33:46 -07001421
1422 /* marshall the parameters */
1423 bp = call->request;
1424 *bp++ = htonl(FSSTORESTATUS);
1425 *bp++ = htonl(vnode->fid.vid);
1426 *bp++ = htonl(vnode->fid.vnode);
1427 *bp++ = htonl(vnode->fid.unique);
1428
1429 xdr_encode_AFS_StoreStatus(&bp, attr);
1430
David Howellsd2ddc772017-11-02 15:27:50 +00001431 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001432 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001433 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells31143d52007-05-09 02:33:46 -07001434}
David Howells45222b92007-05-10 22:22:20 -07001435
1436/*
1437 * deliver reply data to an FS.GetVolumeStatus
1438 */
David Howellsd0016482016-08-30 20:42:14 +01001439static int afs_deliver_fs_get_volume_status(struct afs_call *call)
David Howells45222b92007-05-10 22:22:20 -07001440{
1441 const __be32 *bp;
1442 char *p;
1443 int ret;
1444
David Howellsd0016482016-08-30 20:42:14 +01001445 _enter("{%u}", call->unmarshall);
David Howells45222b92007-05-10 22:22:20 -07001446
1447 switch (call->unmarshall) {
1448 case 0:
1449 call->offset = 0;
1450 call->unmarshall++;
1451
1452 /* extract the returned status record */
1453 case 1:
1454 _debug("extract status");
David Howellsd0016482016-08-30 20:42:14 +01001455 ret = afs_extract_data(call, call->buffer,
1456 12 * 4, true);
David Howells372ee162016-08-03 14:11:40 +01001457 if (ret < 0)
1458 return ret;
David Howells45222b92007-05-10 22:22:20 -07001459
1460 bp = call->buffer;
David Howells97e30432017-11-02 15:27:48 +00001461 xdr_decode_AFSFetchVolumeStatus(&bp, call->reply[1]);
David Howells45222b92007-05-10 22:22:20 -07001462 call->offset = 0;
1463 call->unmarshall++;
1464
1465 /* extract the volume name length */
1466 case 2:
David Howellsd0016482016-08-30 20:42:14 +01001467 ret = afs_extract_data(call, &call->tmp, 4, true);
David Howells372ee162016-08-03 14:11:40 +01001468 if (ret < 0)
1469 return ret;
David Howells45222b92007-05-10 22:22:20 -07001470
1471 call->count = ntohl(call->tmp);
1472 _debug("volname length: %u", call->count);
1473 if (call->count >= AFSNAMEMAX)
1474 return -EBADMSG;
1475 call->offset = 0;
1476 call->unmarshall++;
1477
1478 /* extract the volume name */
1479 case 3:
1480 _debug("extract volname");
1481 if (call->count > 0) {
David Howells97e30432017-11-02 15:27:48 +00001482 ret = afs_extract_data(call, call->reply[2],
David Howellsd0016482016-08-30 20:42:14 +01001483 call->count, true);
David Howells372ee162016-08-03 14:11:40 +01001484 if (ret < 0)
1485 return ret;
David Howells45222b92007-05-10 22:22:20 -07001486 }
1487
David Howells97e30432017-11-02 15:27:48 +00001488 p = call->reply[2];
David Howells45222b92007-05-10 22:22:20 -07001489 p[call->count] = 0;
1490 _debug("volname '%s'", p);
1491
1492 call->offset = 0;
1493 call->unmarshall++;
1494
1495 /* extract the volume name padding */
1496 if ((call->count & 3) == 0) {
1497 call->unmarshall++;
1498 goto no_volname_padding;
1499 }
1500 call->count = 4 - (call->count & 3);
1501
1502 case 4:
David Howellsd0016482016-08-30 20:42:14 +01001503 ret = afs_extract_data(call, call->buffer,
1504 call->count, true);
David Howells372ee162016-08-03 14:11:40 +01001505 if (ret < 0)
1506 return ret;
David Howells45222b92007-05-10 22:22:20 -07001507
1508 call->offset = 0;
1509 call->unmarshall++;
1510 no_volname_padding:
1511
1512 /* extract the offline message length */
1513 case 5:
David Howellsd0016482016-08-30 20:42:14 +01001514 ret = afs_extract_data(call, &call->tmp, 4, true);
David Howells372ee162016-08-03 14:11:40 +01001515 if (ret < 0)
1516 return ret;
David Howells45222b92007-05-10 22:22:20 -07001517
1518 call->count = ntohl(call->tmp);
1519 _debug("offline msg length: %u", call->count);
1520 if (call->count >= AFSNAMEMAX)
1521 return -EBADMSG;
1522 call->offset = 0;
1523 call->unmarshall++;
1524
1525 /* extract the offline message */
1526 case 6:
1527 _debug("extract offline");
1528 if (call->count > 0) {
David Howells97e30432017-11-02 15:27:48 +00001529 ret = afs_extract_data(call, call->reply[2],
David Howellsd0016482016-08-30 20:42:14 +01001530 call->count, true);
David Howells372ee162016-08-03 14:11:40 +01001531 if (ret < 0)
1532 return ret;
David Howells45222b92007-05-10 22:22:20 -07001533 }
1534
David Howells97e30432017-11-02 15:27:48 +00001535 p = call->reply[2];
David Howells45222b92007-05-10 22:22:20 -07001536 p[call->count] = 0;
1537 _debug("offline '%s'", p);
1538
1539 call->offset = 0;
1540 call->unmarshall++;
1541
1542 /* extract the offline message padding */
1543 if ((call->count & 3) == 0) {
1544 call->unmarshall++;
1545 goto no_offline_padding;
1546 }
1547 call->count = 4 - (call->count & 3);
1548
1549 case 7:
David Howellsd0016482016-08-30 20:42:14 +01001550 ret = afs_extract_data(call, call->buffer,
1551 call->count, true);
David Howells372ee162016-08-03 14:11:40 +01001552 if (ret < 0)
1553 return ret;
David Howells45222b92007-05-10 22:22:20 -07001554
1555 call->offset = 0;
1556 call->unmarshall++;
1557 no_offline_padding:
1558
1559 /* extract the message of the day length */
1560 case 8:
David Howellsd0016482016-08-30 20:42:14 +01001561 ret = afs_extract_data(call, &call->tmp, 4, true);
David Howells372ee162016-08-03 14:11:40 +01001562 if (ret < 0)
1563 return ret;
David Howells45222b92007-05-10 22:22:20 -07001564
1565 call->count = ntohl(call->tmp);
1566 _debug("motd length: %u", call->count);
1567 if (call->count >= AFSNAMEMAX)
1568 return -EBADMSG;
1569 call->offset = 0;
1570 call->unmarshall++;
1571
1572 /* extract the message of the day */
1573 case 9:
1574 _debug("extract motd");
1575 if (call->count > 0) {
David Howells97e30432017-11-02 15:27:48 +00001576 ret = afs_extract_data(call, call->reply[2],
David Howellsd0016482016-08-30 20:42:14 +01001577 call->count, true);
David Howells372ee162016-08-03 14:11:40 +01001578 if (ret < 0)
1579 return ret;
David Howells45222b92007-05-10 22:22:20 -07001580 }
1581
David Howells97e30432017-11-02 15:27:48 +00001582 p = call->reply[2];
David Howells45222b92007-05-10 22:22:20 -07001583 p[call->count] = 0;
1584 _debug("motd '%s'", p);
1585
1586 call->offset = 0;
1587 call->unmarshall++;
1588
1589 /* extract the message of the day padding */
David Howellsd0016482016-08-30 20:42:14 +01001590 call->count = (4 - (call->count & 3)) & 3;
David Howells45222b92007-05-10 22:22:20 -07001591
1592 case 10:
David Howellsd0016482016-08-30 20:42:14 +01001593 ret = afs_extract_data(call, call->buffer,
1594 call->count, false);
David Howells372ee162016-08-03 14:11:40 +01001595 if (ret < 0)
1596 return ret;
David Howells45222b92007-05-10 22:22:20 -07001597
1598 call->offset = 0;
1599 call->unmarshall++;
David Howells45222b92007-05-10 22:22:20 -07001600 case 11:
David Howells45222b92007-05-10 22:22:20 -07001601 break;
1602 }
1603
David Howells45222b92007-05-10 22:22:20 -07001604 _leave(" = 0 [done]");
1605 return 0;
1606}
1607
1608/*
1609 * destroy an FS.GetVolumeStatus call
1610 */
1611static void afs_get_volume_status_call_destructor(struct afs_call *call)
1612{
David Howells97e30432017-11-02 15:27:48 +00001613 kfree(call->reply[2]);
1614 call->reply[2] = NULL;
David Howells45222b92007-05-10 22:22:20 -07001615 afs_flat_call_destructor(call);
1616}
1617
1618/*
1619 * FS.GetVolumeStatus operation type
1620 */
1621static const struct afs_call_type afs_RXFSGetVolumeStatus = {
1622 .name = "FS.GetVolumeStatus",
David Howells025db802017-11-02 15:27:51 +00001623 .op = afs_FS_GetVolumeStatus,
David Howells45222b92007-05-10 22:22:20 -07001624 .deliver = afs_deliver_fs_get_volume_status,
David Howells45222b92007-05-10 22:22:20 -07001625 .destructor = afs_get_volume_status_call_destructor,
1626};
1627
1628/*
1629 * fetch the status of a volume
1630 */
David Howells8b2a4642017-11-02 15:27:50 +00001631int afs_fs_get_volume_status(struct afs_fs_cursor *fc,
David Howellsd2ddc772017-11-02 15:27:50 +00001632 struct afs_volume_status *vs)
David Howells45222b92007-05-10 22:22:20 -07001633{
David Howellsd2ddc772017-11-02 15:27:50 +00001634 struct afs_vnode *vnode = fc->vnode;
David Howells45222b92007-05-10 22:22:20 -07001635 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001636 struct afs_net *net = afs_v2net(vnode);
David Howells45222b92007-05-10 22:22:20 -07001637 __be32 *bp;
1638 void *tmpbuf;
1639
1640 _enter("");
1641
1642 tmpbuf = kmalloc(AFSOPAQUEMAX, GFP_KERNEL);
1643 if (!tmpbuf)
1644 return -ENOMEM;
1645
David Howellsf044c882017-11-02 15:27:45 +00001646 call = afs_alloc_flat_call(net, &afs_RXFSGetVolumeStatus, 2 * 4, 12 * 4);
David Howells45222b92007-05-10 22:22:20 -07001647 if (!call) {
1648 kfree(tmpbuf);
1649 return -ENOMEM;
1650 }
1651
David Howellsd2ddc772017-11-02 15:27:50 +00001652 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001653 call->reply[0] = vnode;
1654 call->reply[1] = vs;
1655 call->reply[2] = tmpbuf;
David Howells45222b92007-05-10 22:22:20 -07001656
1657 /* marshall the parameters */
1658 bp = call->request;
1659 bp[0] = htonl(FSGETVOLUMESTATUS);
1660 bp[1] = htonl(vnode->fid.vid);
1661
David Howellsd2ddc772017-11-02 15:27:50 +00001662 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001663 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001664 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howells45222b92007-05-10 22:22:20 -07001665}
David Howellse8d6c552007-07-15 23:40:12 -07001666
1667/*
1668 * deliver reply data to an FS.SetLock, FS.ExtendLock or FS.ReleaseLock
1669 */
David Howellsd0016482016-08-30 20:42:14 +01001670static int afs_deliver_fs_xxxx_lock(struct afs_call *call)
David Howellse8d6c552007-07-15 23:40:12 -07001671{
1672 const __be32 *bp;
David Howells372ee162016-08-03 14:11:40 +01001673 int ret;
David Howellse8d6c552007-07-15 23:40:12 -07001674
David Howellsd0016482016-08-30 20:42:14 +01001675 _enter("{%u}", call->unmarshall);
David Howellse8d6c552007-07-15 23:40:12 -07001676
David Howellsd0016482016-08-30 20:42:14 +01001677 ret = afs_transfer_reply(call);
David Howells372ee162016-08-03 14:11:40 +01001678 if (ret < 0)
1679 return ret;
David Howellse8d6c552007-07-15 23:40:12 -07001680
1681 /* unmarshall the reply once we've received all of it */
1682 bp = call->buffer;
David Howells97e30432017-11-02 15:27:48 +00001683 /* xdr_decode_AFSVolSync(&bp, call->reply[X]); */
David Howellse8d6c552007-07-15 23:40:12 -07001684
1685 _leave(" = 0 [done]");
1686 return 0;
1687}
1688
1689/*
1690 * FS.SetLock operation type
1691 */
1692static const struct afs_call_type afs_RXFSSetLock = {
1693 .name = "FS.SetLock",
David Howells025db802017-11-02 15:27:51 +00001694 .op = afs_FS_SetLock,
David Howellse8d6c552007-07-15 23:40:12 -07001695 .deliver = afs_deliver_fs_xxxx_lock,
David Howellse8d6c552007-07-15 23:40:12 -07001696 .destructor = afs_flat_call_destructor,
1697};
1698
1699/*
1700 * FS.ExtendLock operation type
1701 */
1702static const struct afs_call_type afs_RXFSExtendLock = {
1703 .name = "FS.ExtendLock",
David Howells025db802017-11-02 15:27:51 +00001704 .op = afs_FS_ExtendLock,
David Howellse8d6c552007-07-15 23:40:12 -07001705 .deliver = afs_deliver_fs_xxxx_lock,
David Howellse8d6c552007-07-15 23:40:12 -07001706 .destructor = afs_flat_call_destructor,
1707};
1708
1709/*
1710 * FS.ReleaseLock operation type
1711 */
1712static const struct afs_call_type afs_RXFSReleaseLock = {
1713 .name = "FS.ReleaseLock",
David Howells025db802017-11-02 15:27:51 +00001714 .op = afs_FS_ReleaseLock,
David Howellse8d6c552007-07-15 23:40:12 -07001715 .deliver = afs_deliver_fs_xxxx_lock,
David Howellse8d6c552007-07-15 23:40:12 -07001716 .destructor = afs_flat_call_destructor,
1717};
1718
1719/*
David Howellsd2ddc772017-11-02 15:27:50 +00001720 * Set a lock on a file
David Howellse8d6c552007-07-15 23:40:12 -07001721 */
David Howellsd2ddc772017-11-02 15:27:50 +00001722int afs_fs_set_lock(struct afs_fs_cursor *fc, afs_lock_type_t type)
David Howellse8d6c552007-07-15 23:40:12 -07001723{
David Howellsd2ddc772017-11-02 15:27:50 +00001724 struct afs_vnode *vnode = fc->vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001725 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001726 struct afs_net *net = afs_v2net(vnode);
David Howellse8d6c552007-07-15 23:40:12 -07001727 __be32 *bp;
1728
1729 _enter("");
1730
David Howellsf044c882017-11-02 15:27:45 +00001731 call = afs_alloc_flat_call(net, &afs_RXFSSetLock, 5 * 4, 6 * 4);
David Howellse8d6c552007-07-15 23:40:12 -07001732 if (!call)
1733 return -ENOMEM;
1734
David Howellsd2ddc772017-11-02 15:27:50 +00001735 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001736 call->reply[0] = vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001737
1738 /* marshall the parameters */
1739 bp = call->request;
1740 *bp++ = htonl(FSSETLOCK);
1741 *bp++ = htonl(vnode->fid.vid);
1742 *bp++ = htonl(vnode->fid.vnode);
1743 *bp++ = htonl(vnode->fid.unique);
1744 *bp++ = htonl(type);
1745
David Howellsd2ddc772017-11-02 15:27:50 +00001746 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001747 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001748 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellse8d6c552007-07-15 23:40:12 -07001749}
1750
1751/*
1752 * extend a lock on a file
1753 */
David Howellsd2ddc772017-11-02 15:27:50 +00001754int afs_fs_extend_lock(struct afs_fs_cursor *fc)
David Howellse8d6c552007-07-15 23:40:12 -07001755{
David Howellsd2ddc772017-11-02 15:27:50 +00001756 struct afs_vnode *vnode = fc->vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001757 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001758 struct afs_net *net = afs_v2net(vnode);
David Howellse8d6c552007-07-15 23:40:12 -07001759 __be32 *bp;
1760
1761 _enter("");
1762
David Howellsf044c882017-11-02 15:27:45 +00001763 call = afs_alloc_flat_call(net, &afs_RXFSExtendLock, 4 * 4, 6 * 4);
David Howellse8d6c552007-07-15 23:40:12 -07001764 if (!call)
1765 return -ENOMEM;
1766
David Howellsd2ddc772017-11-02 15:27:50 +00001767 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001768 call->reply[0] = vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001769
1770 /* marshall the parameters */
1771 bp = call->request;
1772 *bp++ = htonl(FSEXTENDLOCK);
1773 *bp++ = htonl(vnode->fid.vid);
1774 *bp++ = htonl(vnode->fid.vnode);
1775 *bp++ = htonl(vnode->fid.unique);
1776
David Howellsd2ddc772017-11-02 15:27:50 +00001777 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001778 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001779 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellse8d6c552007-07-15 23:40:12 -07001780}
1781
1782/*
1783 * release a lock on a file
1784 */
David Howellsd2ddc772017-11-02 15:27:50 +00001785int afs_fs_release_lock(struct afs_fs_cursor *fc)
David Howellse8d6c552007-07-15 23:40:12 -07001786{
David Howellsd2ddc772017-11-02 15:27:50 +00001787 struct afs_vnode *vnode = fc->vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001788 struct afs_call *call;
David Howellsf044c882017-11-02 15:27:45 +00001789 struct afs_net *net = afs_v2net(vnode);
David Howellse8d6c552007-07-15 23:40:12 -07001790 __be32 *bp;
1791
1792 _enter("");
1793
David Howellsf044c882017-11-02 15:27:45 +00001794 call = afs_alloc_flat_call(net, &afs_RXFSReleaseLock, 4 * 4, 6 * 4);
David Howellse8d6c552007-07-15 23:40:12 -07001795 if (!call)
1796 return -ENOMEM;
1797
David Howellsd2ddc772017-11-02 15:27:50 +00001798 call->key = fc->key;
David Howells97e30432017-11-02 15:27:48 +00001799 call->reply[0] = vnode;
David Howellse8d6c552007-07-15 23:40:12 -07001800
1801 /* marshall the parameters */
1802 bp = call->request;
1803 *bp++ = htonl(FSRELEASELOCK);
1804 *bp++ = htonl(vnode->fid.vid);
1805 *bp++ = htonl(vnode->fid.vnode);
1806 *bp++ = htonl(vnode->fid.unique);
1807
David Howellsd2ddc772017-11-02 15:27:50 +00001808 afs_use_fs_server(call, fc->cbi);
David Howells025db802017-11-02 15:27:51 +00001809 trace_afs_make_fs_call(call, &vnode->fid);
David Howellsd2ddc772017-11-02 15:27:50 +00001810 return afs_make_call(&fc->ac, call, GFP_NOFS, false);
David Howellsc435ee32017-11-02 15:27:49 +00001811}
1812
1813/*
1814 * Deliver reply data to an FS.GiveUpAllCallBacks operation.
1815 */
1816static int afs_deliver_fs_give_up_all_callbacks(struct afs_call *call)
1817{
1818 return afs_transfer_reply(call);
1819}
1820
1821/*
1822 * FS.GiveUpAllCallBacks operation type
1823 */
1824static const struct afs_call_type afs_RXFSGiveUpAllCallBacks = {
1825 .name = "FS.GiveUpAllCallBacks",
David Howells025db802017-11-02 15:27:51 +00001826 .op = afs_FS_GiveUpAllCallBacks,
David Howellsc435ee32017-11-02 15:27:49 +00001827 .deliver = afs_deliver_fs_give_up_all_callbacks,
1828 .destructor = afs_flat_call_destructor,
1829};
1830
1831/*
1832 * Flush all the callbacks we have on a server.
1833 */
David Howellsd2ddc772017-11-02 15:27:50 +00001834int afs_fs_give_up_all_callbacks(struct afs_net *net,
1835 struct afs_server *server,
David Howells8b2a4642017-11-02 15:27:50 +00001836 struct afs_addr_cursor *ac,
David Howellsd2ddc772017-11-02 15:27:50 +00001837 struct key *key)
David Howellsc435ee32017-11-02 15:27:49 +00001838{
1839 struct afs_call *call;
1840 __be32 *bp;
1841
1842 _enter("");
1843
David Howellsd2ddc772017-11-02 15:27:50 +00001844 call = afs_alloc_flat_call(net, &afs_RXFSGiveUpAllCallBacks, 1 * 4, 0);
David Howellsc435ee32017-11-02 15:27:49 +00001845 if (!call)
1846 return -ENOMEM;
1847
1848 call->key = key;
1849
1850 /* marshall the parameters */
1851 bp = call->request;
1852 *bp++ = htonl(FSGIVEUPALLCALLBACKS);
1853
1854 /* Can't take a ref on server */
David Howellsd2ddc772017-11-02 15:27:50 +00001855 return afs_make_call(ac, call, GFP_NOFS, false);
1856}
1857
1858/*
1859 * Deliver reply data to an FS.GetCapabilities operation.
1860 */
1861static int afs_deliver_fs_get_capabilities(struct afs_call *call)
1862{
1863 u32 count;
1864 int ret;
1865
1866 _enter("{%u,%zu/%u}", call->unmarshall, call->offset, call->count);
1867
1868again:
1869 switch (call->unmarshall) {
1870 case 0:
1871 call->offset = 0;
1872 call->unmarshall++;
1873
1874 /* Extract the capabilities word count */
1875 case 1:
1876 ret = afs_extract_data(call, &call->tmp,
1877 1 * sizeof(__be32),
1878 true);
1879 if (ret < 0)
1880 return ret;
1881
1882 count = ntohl(call->tmp);
1883
1884 call->count = count;
1885 call->count2 = count;
1886 call->offset = 0;
1887 call->unmarshall++;
1888
1889 /* Extract capabilities words */
1890 case 2:
1891 count = min(call->count, 16U);
1892 ret = afs_extract_data(call, call->buffer,
1893 count * sizeof(__be32),
1894 call->count > 16);
1895 if (ret < 0)
1896 return ret;
1897
1898 /* TODO: Examine capabilities */
1899
1900 call->count -= count;
1901 if (call->count > 0)
1902 goto again;
1903 call->offset = 0;
1904 call->unmarshall++;
1905 break;
1906 }
1907
1908 _leave(" = 0 [done]");
1909 return 0;
1910}
1911
1912/*
1913 * FS.GetCapabilities operation type
1914 */
1915static const struct afs_call_type afs_RXFSGetCapabilities = {
1916 .name = "FS.GetCapabilities",
David Howells025db802017-11-02 15:27:51 +00001917 .op = afs_FS_GetCapabilities,
David Howellsd2ddc772017-11-02 15:27:50 +00001918 .deliver = afs_deliver_fs_get_capabilities,
1919 .destructor = afs_flat_call_destructor,
1920};
1921
1922/*
1923 * Probe a fileserver for the capabilities that it supports. This can
1924 * return up to 196 words.
1925 */
1926int afs_fs_get_capabilities(struct afs_net *net,
1927 struct afs_server *server,
1928 struct afs_addr_cursor *ac,
1929 struct key *key)
1930{
1931 struct afs_call *call;
1932 __be32 *bp;
1933
1934 _enter("");
1935
1936 call = afs_alloc_flat_call(net, &afs_RXFSGetCapabilities, 1 * 4, 16 * 4);
1937 if (!call)
1938 return -ENOMEM;
1939
1940 call->key = key;
1941
1942 /* marshall the parameters */
1943 bp = call->request;
1944 *bp++ = htonl(FSGETCAPABILITIES);
1945
1946 /* Can't take a ref on server */
David Howells025db802017-11-02 15:27:51 +00001947 trace_afs_make_fs_call(call, NULL);
David Howellsd2ddc772017-11-02 15:27:50 +00001948 return afs_make_call(ac, call, GFP_NOFS, false);
David Howellse8d6c552007-07-15 23:40:12 -07001949}