fsclient.c 9.5 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 151 152 153 154

/*
 * FS.FetchStatus operation type
 */
static const struct afs_call_type afs_RXFSFetchStatus = {
	.deliver	= afs_deliver_fs_fetch_status,
	.abort_to_error	= afs_abort_to_error,
	.destructor	= afs_flat_call_destructor,
};
L
Linus Torvalds 已提交
155 156 157 158

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

167
	_enter("");
L
Linus Torvalds 已提交
168

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

173 174 175 176
	call->reply = vnode;
	call->reply2 = volsync;
	call->service_id = FS_SERVICE;
	call->port = htons(AFS_FS_PORT);
L
Linus Torvalds 已提交
177 178

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

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

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

199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
	_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 已提交
215

216 217 218 219 220 221 222 223 224 225 226 227 228
		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 已提交
229

230 231 232 233 234 235 236 237 238 239 240 241
		/* 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 已提交
242

243 244
		call->offset = 0;
		call->unmarshall++;
L
Linus Torvalds 已提交
245

246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
		/* 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 已提交
263

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

271 272 273 274 275
	if (!last)
		return 0;

	_leave(" = 0 [done]");
	return 0;
D
David Howells 已提交
276
}
L
Linus Torvalds 已提交
277 278

/*
279
 * FS.FetchData operation type
L
Linus Torvalds 已提交
280
 */
281 282 283 284 285 286 287 288 289 290 291 292 293 294 295
static const struct afs_call_type afs_RXFSFetchData = {
	.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,
		      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 已提交
296
{
297
	struct afs_call *call;
L
Linus Torvalds 已提交
298 299
	__be32 *bp;

300
	_enter("");
L
Linus Torvalds 已提交
301

302 303 304
	call = afs_alloc_flat_call(&afs_RXFSFetchData, 24, 120);
	if (!call)
		return -ENOMEM;
L
Linus Torvalds 已提交
305

306 307 308 309 310
	call->reply = vnode;
	call->reply2 = volsync;
	call->reply3 = buffer;
	call->service_id = FS_SERVICE;
	call->port = htons(AFS_FS_PORT);
L
Linus Torvalds 已提交
311 312

	/* marshall the parameters */
313 314 315 316 317 318 319
	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 已提交
320

321 322
	return afs_make_call(&server->addr, call, GFP_NOFS, wait_mode);
}
L
Linus Torvalds 已提交
323

324 325 326 327 328 329 330
/*
 * 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 已提交
331

332 333 334
	if (skb->len > 0)
		return -EBADMSG; /* shouldn't be any reply data */
	return 0;
D
David Howells 已提交
335
}
L
Linus Torvalds 已提交
336 337

/*
338 339 340 341 342 343 344 345 346 347 348
 * FS.GiveUpCallBacks operation type
 */
static const struct afs_call_type afs_RXFSGiveUpCallBacks = {
	.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 已提交
349
 */
350 351
int afs_fs_give_up_callbacks(struct afs_server *server,
			     const struct afs_wait_mode *wait_mode)
L
Linus Torvalds 已提交
352
{
353 354 355 356
	struct afs_call *call;
	size_t ncallbacks;
	__be32 *bp, *tp;
	int loop;
L
Linus Torvalds 已提交
357

358 359
	ncallbacks = CIRC_CNT(server->cb_break_head, server->cb_break_tail,
			      ARRAY_SIZE(server->cb_break));
L
Linus Torvalds 已提交
360

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

363 364 365 366
	if (ncallbacks == 0)
		return 0;
	if (ncallbacks > AFSCBMAX)
		ncallbacks = AFSCBMAX;
L
Linus Torvalds 已提交
367

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

370 371 372 373 374 375 376
	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 已提交
377 378

	/* marshall the parameters */
379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
	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 已提交
400 401
	}

402 403
	ASSERT(ncallbacks > 0);
	wake_up_nr(&server->cb_break_waitq, ncallbacks);
L
Linus Torvalds 已提交
404

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