fsclient.c 9.7 KB
Newer Older
1
/* AFS File Server client stubs
L
Linus Torvalds 已提交
2
 *
3
 * Copyright (C) 2002, 2007 Red Hat, Inc. All Rights Reserved.
L
Linus Torvalds 已提交
4 5 6 7 8 9 10 11 12 13
 * Written by David Howells (dhowells@redhat.com)
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version
 * 2 of the License, or (at your option) any later version.
 */

#include <linux/init.h>
#include <linux/sched.h>
14
#include <linux/circ_buf.h>
L
Linus Torvalds 已提交
15
#include "internal.h"
16
#include "afs_fs.h"
L
Linus Torvalds 已提交
17 18

/*
19
 * decode an AFSFetchStatus block
L
Linus Torvalds 已提交
20
 */
21 22
static void xdr_decode_AFSFetchStatus(const __be32 **_bp,
				      struct afs_vnode *vnode)
L
Linus Torvalds 已提交
23
{
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
	const __be32 *bp = *_bp;
	umode_t mode;
	u64 data_version;
	u32 changed = 0; /* becomes non-zero if ctime-type changes seen */

#define EXTRACT(DST)				\
	do {					\
		u32 x = ntohl(*bp++);		\
		changed |= DST - x;		\
		DST = x;			\
	} while (0)

	vnode->status.if_version = ntohl(*bp++);
	EXTRACT(vnode->status.type);
	vnode->status.nlink = ntohl(*bp++);
	EXTRACT(vnode->status.size);
	data_version = ntohl(*bp++);
	EXTRACT(vnode->status.author);
	EXTRACT(vnode->status.owner);
	EXTRACT(vnode->status.caller_access); /* call ticket dependent */
	EXTRACT(vnode->status.anon_access);
	EXTRACT(vnode->status.mode);
	vnode->status.parent.vid = vnode->fid.vid;
	EXTRACT(vnode->status.parent.vnode);
	EXTRACT(vnode->status.parent.unique);
	bp++; /* seg size */
	vnode->status.mtime_client = ntohl(*bp++);
	vnode->status.mtime_server = ntohl(*bp++);
	bp++; /* group */
	bp++; /* sync counter */
	data_version |= (u64) ntohl(*bp++) << 32;
	bp++; /* spare2 */
	bp++; /* spare3 */
	bp++; /* spare4 */
	*_bp = bp;

	if (changed) {
		_debug("vnode changed");
		set_bit(AFS_VNODE_CHANGED, &vnode->flags);
		vnode->vfs_inode.i_uid		= vnode->status.owner;
		vnode->vfs_inode.i_size		= vnode->status.size;
		vnode->vfs_inode.i_version	= vnode->fid.unique;

		vnode->status.mode &= S_IALLUGO;
		mode = vnode->vfs_inode.i_mode;
		mode &= ~S_IALLUGO;
		mode |= vnode->status.mode;
		vnode->vfs_inode.i_mode = mode;
	}

	_debug("vnode time %lx, %lx",
	       vnode->status.mtime_client, vnode->status.mtime_server);
	vnode->vfs_inode.i_ctime.tv_sec	= vnode->status.mtime_server;
	vnode->vfs_inode.i_mtime	= vnode->vfs_inode.i_ctime;
	vnode->vfs_inode.i_atime	= vnode->vfs_inode.i_ctime;

	if (vnode->status.data_version != data_version) {
		_debug("vnode modified %llx", data_version);
		vnode->status.data_version = data_version;
		set_bit(AFS_VNODE_MODIFIED, &vnode->flags);
		set_bit(AFS_VNODE_ZAP_DATA, &vnode->flags);
L
Linus Torvalds 已提交
85
	}
D
David Howells 已提交
86
}
L
Linus Torvalds 已提交
87 88

/*
89
 * decode an AFSCallBack block
L
Linus Torvalds 已提交
90
 */
91
static void xdr_decode_AFSCallBack(const __be32 **_bp, struct afs_vnode *vnode)
L
Linus Torvalds 已提交
92
{
93
	const __be32 *bp = *_bp;
L
Linus Torvalds 已提交
94

95 96 97 98 99
	vnode->cb_version	= ntohl(*bp++);
	vnode->cb_expiry	= ntohl(*bp++);
	vnode->cb_type		= ntohl(*bp++);
	vnode->cb_expires	= vnode->cb_expiry + get_seconds();
	*_bp = bp;
D
David Howells 已提交
100
}
L
Linus Torvalds 已提交
101 102

/*
103
 * decode an AFSVolSync block
L
Linus Torvalds 已提交
104
 */
105 106
static void xdr_decode_AFSVolSync(const __be32 **_bp,
				  struct afs_volsync *volsync)
L
Linus Torvalds 已提交
107
{
108
	const __be32 *bp = *_bp;
L
Linus Torvalds 已提交
109

110 111 112 113 114 115 116 117
	volsync->creation = ntohl(*bp++);
	bp++; /* spare2 */
	bp++; /* spare3 */
	bp++; /* spare4 */
	bp++; /* spare5 */
	bp++; /* spare6 */
	*_bp = bp;
}
L
Linus Torvalds 已提交
118

119 120 121 122 123 124 125
/*
 * deliver reply data to an FS.FetchStatus
 */
static int afs_deliver_fs_fetch_status(struct afs_call *call,
				       struct sk_buff *skb, bool last)
{
	const __be32 *bp;
L
Linus Torvalds 已提交
126

127
	_enter(",,%u", last);
L
Linus Torvalds 已提交
128

129 130 131
	afs_transfer_reply(call, skb);
	if (!last)
		return 0;
L
Linus Torvalds 已提交
132

133 134
	if (call->reply_size != call->reply_max)
		return -EBADMSG;
L
Linus Torvalds 已提交
135

136 137 138 139 140 141
	/* unmarshall the reply once we've received all of it */
	bp = call->buffer;
	xdr_decode_AFSFetchStatus(&bp, call->reply);
	xdr_decode_AFSCallBack(&bp, call->reply);
	if (call->reply2)
		xdr_decode_AFSVolSync(&bp, call->reply2);
L
Linus Torvalds 已提交
142

143 144
	_leave(" = 0 [done]");
	return 0;
D
David Howells 已提交
145
}
146 147 148 149 150

/*
 * FS.FetchStatus operation type
 */
static const struct afs_call_type afs_RXFSFetchStatus = {
D
David Howells 已提交
151
	.name		= "FS.FetchStatus",
152 153 154 155
	.deliver	= afs_deliver_fs_fetch_status,
	.abort_to_error	= afs_abort_to_error,
	.destructor	= afs_flat_call_destructor,
};
L
Linus Torvalds 已提交
156 157 158 159

/*
 * fetch the status information for a file
 */
160
int afs_fs_fetch_file_status(struct afs_server *server,
D
David Howells 已提交
161
			     struct key *key,
162 163 164
			     struct afs_vnode *vnode,
			     struct afs_volsync *volsync,
			     const struct afs_wait_mode *wait_mode)
L
Linus Torvalds 已提交
165
{
166
	struct afs_call *call;
L
Linus Torvalds 已提交
167 168
	__be32 *bp;

D
David Howells 已提交
169
	_enter(",%x,,,", key_serial(key));
L
Linus Torvalds 已提交
170

171 172 173
	call = afs_alloc_flat_call(&afs_RXFSFetchStatus, 16, 120);
	if (!call)
		return -ENOMEM;
L
Linus Torvalds 已提交
174

D
David Howells 已提交
175
	call->key = key;
176 177 178 179
	call->reply = vnode;
	call->reply2 = volsync;
	call->service_id = FS_SERVICE;
	call->port = htons(AFS_FS_PORT);
L
Linus Torvalds 已提交
180 181

	/* marshall the parameters */
182
	bp = call->request;
L
Linus Torvalds 已提交
183 184 185 186 187
	bp[0] = htonl(FSFETCHSTATUS);
	bp[1] = htonl(vnode->fid.vid);
	bp[2] = htonl(vnode->fid.vnode);
	bp[3] = htonl(vnode->fid.unique);

188
	return afs_make_call(&server->addr, call, GFP_NOFS, wait_mode);
D
David Howells 已提交
189
}
L
Linus Torvalds 已提交
190 191

/*
192
 * deliver reply data to an FS.FetchData
L
Linus Torvalds 已提交
193
 */
194 195
static int afs_deliver_fs_fetch_data(struct afs_call *call,
				     struct sk_buff *skb, bool last)
L
Linus Torvalds 已提交
196
{
197 198 199
	const __be32 *bp;
	struct page *page;
	void *buffer;
L
Linus Torvalds 已提交
200 201
	int ret;

202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217
	_enter("{%u},{%u},%d", call->unmarshall, skb->len, last);

	switch (call->unmarshall) {
	case 0:
		call->offset = 0;
		call->unmarshall++;

		/* extract the returned data length */
	case 1:
		_debug("extract data length");
		ret = afs_extract_data(call, skb, last, &call->tmp, 4);
		switch (ret) {
		case 0:		break;
		case -EAGAIN:	return 0;
		default:	return ret;
		}
L
Linus Torvalds 已提交
218

219 220 221 222 223 224 225 226 227 228 229 230 231
		call->count = ntohl(call->tmp);
		_debug("DATA length: %u", call->count);
		if (call->count > PAGE_SIZE)
			return -EBADMSG;
		call->offset = 0;
		call->unmarshall++;

		if (call->count < PAGE_SIZE) {
			buffer = kmap_atomic(call->reply3, KM_USER0);
			memset(buffer + PAGE_SIZE - call->count, 0,
			       call->count);
			kunmap_atomic(buffer, KM_USER0);
		}
L
Linus Torvalds 已提交
232

233 234 235 236 237 238 239 240 241 242 243 244
		/* extract the returned data */
	case 2:
		_debug("extract data");
		page = call->reply3;
		buffer = kmap_atomic(page, KM_USER0);
		ret = afs_extract_data(call, skb, last, buffer, call->count);
		kunmap_atomic(buffer, KM_USER0);
		switch (ret) {
		case 0:		break;
		case -EAGAIN:	return 0;
		default:	return ret;
		}
L
Linus Torvalds 已提交
245

246 247
		call->offset = 0;
		call->unmarshall++;
L
Linus Torvalds 已提交
248

249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265
		/* extract the metadata */
	case 3:
		ret = afs_extract_data(call, skb, last, call->buffer, 120);
		switch (ret) {
		case 0:		break;
		case -EAGAIN:	return 0;
		default:	return ret;
		}

		bp = call->buffer;
		xdr_decode_AFSFetchStatus(&bp, call->reply);
		xdr_decode_AFSCallBack(&bp, call->reply);
		if (call->reply2)
			xdr_decode_AFSVolSync(&bp, call->reply2);

		call->offset = 0;
		call->unmarshall++;
L
Linus Torvalds 已提交
266

267 268 269 270 271
	case 4:
		_debug("trailer");
		if (skb->len != 0)
			return -EBADMSG;
		break;
L
Linus Torvalds 已提交
272 273
	}

274 275 276 277 278
	if (!last)
		return 0;

	_leave(" = 0 [done]");
	return 0;
D
David Howells 已提交
279
}
L
Linus Torvalds 已提交
280 281

/*
282
 * FS.FetchData operation type
L
Linus Torvalds 已提交
283
 */
284
static const struct afs_call_type afs_RXFSFetchData = {
D
David Howells 已提交
285
	.name		= "FS.FetchData",
286 287 288 289 290 291 292 293 294
	.deliver	= afs_deliver_fs_fetch_data,
	.abort_to_error	= afs_abort_to_error,
	.destructor	= afs_flat_call_destructor,
};

/*
 * fetch data from a file
 */
int afs_fs_fetch_data(struct afs_server *server,
D
David Howells 已提交
295
		      struct key *key,
296 297 298 299 300
		      struct afs_vnode *vnode,
		      off_t offset, size_t length,
		      struct page *buffer,
		      struct afs_volsync *volsync,
		      const struct afs_wait_mode *wait_mode)
L
Linus Torvalds 已提交
301
{
302
	struct afs_call *call;
L
Linus Torvalds 已提交
303 304
	__be32 *bp;

305
	_enter("");
L
Linus Torvalds 已提交
306

307 308 309
	call = afs_alloc_flat_call(&afs_RXFSFetchData, 24, 120);
	if (!call)
		return -ENOMEM;
L
Linus Torvalds 已提交
310

D
David Howells 已提交
311
	call->key = key;
312 313 314 315 316
	call->reply = vnode;
	call->reply2 = volsync;
	call->reply3 = buffer;
	call->service_id = FS_SERVICE;
	call->port = htons(AFS_FS_PORT);
L
Linus Torvalds 已提交
317 318

	/* marshall the parameters */
319 320 321 322 323 324 325
	bp = call->request;
	bp[0] = htonl(FSFETCHDATA);
	bp[1] = htonl(vnode->fid.vid);
	bp[2] = htonl(vnode->fid.vnode);
	bp[3] = htonl(vnode->fid.unique);
	bp[4] = htonl(offset);
	bp[5] = htonl(length);
L
Linus Torvalds 已提交
326

327 328
	return afs_make_call(&server->addr, call, GFP_NOFS, wait_mode);
}
L
Linus Torvalds 已提交
329

330 331 332 333 334 335 336
/*
 * deliver reply data to an FS.GiveUpCallBacks
 */
static int afs_deliver_fs_give_up_callbacks(struct afs_call *call,
					    struct sk_buff *skb, bool last)
{
	_enter(",{%u},%d", skb->len, last);
L
Linus Torvalds 已提交
337

338 339 340
	if (skb->len > 0)
		return -EBADMSG; /* shouldn't be any reply data */
	return 0;
D
David Howells 已提交
341
}
L
Linus Torvalds 已提交
342 343

/*
344 345 346
 * FS.GiveUpCallBacks operation type
 */
static const struct afs_call_type afs_RXFSGiveUpCallBacks = {
D
David Howells 已提交
347
	.name		= "FS.GiveUpCallBacks",
348 349 350 351 352 353 354 355
	.deliver	= afs_deliver_fs_give_up_callbacks,
	.abort_to_error	= afs_abort_to_error,
	.destructor	= afs_flat_call_destructor,
};

/*
 * give up a set of callbacks
 * - the callbacks are held in the server->cb_break ring
L
Linus Torvalds 已提交
356
 */
357 358
int afs_fs_give_up_callbacks(struct afs_server *server,
			     const struct afs_wait_mode *wait_mode)
L
Linus Torvalds 已提交
359
{
360 361 362 363
	struct afs_call *call;
	size_t ncallbacks;
	__be32 *bp, *tp;
	int loop;
L
Linus Torvalds 已提交
364

365 366
	ncallbacks = CIRC_CNT(server->cb_break_head, server->cb_break_tail,
			      ARRAY_SIZE(server->cb_break));
L
Linus Torvalds 已提交
367

368
	_enter("{%zu},", ncallbacks);
L
Linus Torvalds 已提交
369

370 371 372 373
	if (ncallbacks == 0)
		return 0;
	if (ncallbacks > AFSCBMAX)
		ncallbacks = AFSCBMAX;
L
Linus Torvalds 已提交
374

375
	_debug("break %zu callbacks", ncallbacks);
L
Linus Torvalds 已提交
376

377 378 379 380 381 382 383
	call = afs_alloc_flat_call(&afs_RXFSGiveUpCallBacks,
				   12 + ncallbacks * 6 * 4, 0);
	if (!call)
		return -ENOMEM;

	call->service_id = FS_SERVICE;
	call->port = htons(AFS_FS_PORT);
L
Linus Torvalds 已提交
384 385

	/* marshall the parameters */
386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406
	bp = call->request;
	tp = bp + 2 + ncallbacks * 3;
	*bp++ = htonl(FSGIVEUPCALLBACKS);
	*bp++ = htonl(ncallbacks);
	*tp++ = htonl(ncallbacks);

	atomic_sub(ncallbacks, &server->cb_break_n);
	for (loop = ncallbacks; loop > 0; loop--) {
		struct afs_callback *cb =
			&server->cb_break[server->cb_break_tail];

		*bp++ = htonl(cb->fid.vid);
		*bp++ = htonl(cb->fid.vnode);
		*bp++ = htonl(cb->fid.unique);
		*tp++ = htonl(cb->version);
		*tp++ = htonl(cb->expiry);
		*tp++ = htonl(cb->type);
		smp_mb();
		server->cb_break_tail =
			(server->cb_break_tail + 1) &
			(ARRAY_SIZE(server->cb_break) - 1);
L
Linus Torvalds 已提交
407 408
	}

409 410
	ASSERT(ncallbacks > 0);
	wake_up_nr(&server->cb_break_waitq, ncallbacks);
L
Linus Torvalds 已提交
411

412
	return afs_make_call(&server->addr, call, GFP_NOFS, wait_mode);
D
David Howells 已提交
413
}