callback_proc.c 10.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9
/*
 * linux/fs/nfs/callback_proc.c
 *
 * Copyright (C) 2004 Trond Myklebust
 *
 * NFSv4 callback procedures
 */
#include <linux/nfs4.h>
#include <linux/nfs_fs.h>
10
#include <linux/slab.h>
11
#include "nfs4_fs.h"
L
Linus Torvalds 已提交
12 13
#include "callback.h"
#include "delegation.h"
14
#include "internal.h"
L
Linus Torvalds 已提交
15

16
#ifdef NFS_DEBUG
L
Linus Torvalds 已提交
17
#define NFSDBG_FACILITY NFSDBG_CALLBACK
18
#endif
L
Linus Torvalds 已提交
19
 
20
__be32 nfs4_callback_getattr(struct cb_getattrargs *args, struct cb_getattrres *res)
L
Linus Torvalds 已提交
21
{
22
	struct nfs_client *clp;
L
Linus Torvalds 已提交
23 24 25
	struct nfs_delegation *delegation;
	struct nfs_inode *nfsi;
	struct inode *inode;
26

L
Linus Torvalds 已提交
27 28
	res->bitmap[0] = res->bitmap[1] = 0;
	res->status = htonl(NFS4ERR_BADHANDLE);
29
	clp = nfs_find_client(args->addr, 4);
L
Linus Torvalds 已提交
30 31
	if (clp == NULL)
		goto out;
32 33 34 35

	dprintk("NFS: GETATTR callback request from %s\n",
		rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR));

L
Linus Torvalds 已提交
36 37 38 39
	inode = nfs_delegation_find_inode(clp, &args->fh);
	if (inode == NULL)
		goto out_putclient;
	nfsi = NFS_I(inode);
40 41
	rcu_read_lock();
	delegation = rcu_dereference(nfsi->delegation);
L
Linus Torvalds 已提交
42 43 44
	if (delegation == NULL || (delegation->type & FMODE_WRITE) == 0)
		goto out_iput;
	res->size = i_size_read(inode);
45 46 47
	res->change_attr = delegation->change_attr;
	if (nfsi->npages != 0)
		res->change_attr++;
L
Linus Torvalds 已提交
48 49 50 51 52 53 54 55
	res->ctime = inode->i_ctime;
	res->mtime = inode->i_mtime;
	res->bitmap[0] = (FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE) &
		args->bitmap[0];
	res->bitmap[1] = (FATTR4_WORD1_TIME_METADATA|FATTR4_WORD1_TIME_MODIFY) &
		args->bitmap[1];
	res->status = 0;
out_iput:
56
	rcu_read_unlock();
L
Linus Torvalds 已提交
57 58
	iput(inode);
out_putclient:
59
	nfs_put_client(clp);
L
Linus Torvalds 已提交
60
out:
61
	dprintk("%s: exit with status = %d\n", __func__, ntohl(res->status));
L
Linus Torvalds 已提交
62 63 64
	return res->status;
}

65
__be32 nfs4_callback_recall(struct cb_recallargs *args, void *dummy)
L
Linus Torvalds 已提交
66
{
67
	struct nfs_client *clp;
L
Linus Torvalds 已提交
68
	struct inode *inode;
69
	__be32 res;
L
Linus Torvalds 已提交
70 71
	
	res = htonl(NFS4ERR_BADHANDLE);
72
	clp = nfs_find_client(args->addr, 4);
L
Linus Torvalds 已提交
73 74
	if (clp == NULL)
		goto out;
75 76 77 78

	dprintk("NFS: RECALL callback request from %s\n",
		rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR));

79 80 81 82 83 84
	do {
		struct nfs_client *prev = clp;

		inode = nfs_delegation_find_inode(clp, &args->fh);
		if (inode != NULL) {
			/* Set up a helper thread to actually return the delegation */
85
			switch (nfs_async_inode_return_delegation(inode, &args->stateid)) {
86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
				case 0:
					res = 0;
					break;
				case -ENOENT:
					if (res != 0)
						res = htonl(NFS4ERR_BAD_STATEID);
					break;
				default:
					res = htonl(NFS4ERR_RESOURCE);
			}
			iput(inode);
		}
		clp = nfs_find_client_next(prev);
		nfs_put_client(prev);
	} while (clp != NULL);
L
Linus Torvalds 已提交
101
out:
102
	dprintk("%s: exit with status = %d\n", __func__, ntohl(res));
L
Linus Torvalds 已提交
103 104
	return res;
}
105

A
Alexandros Batsakis 已提交
106 107 108 109 110 111 112 113
int nfs4_validate_delegation_stateid(struct nfs_delegation *delegation, const nfs4_stateid *stateid)
{
	if (delegation == NULL || memcmp(delegation->stateid.data, stateid->data,
					 sizeof(delegation->stateid.data)) != 0)
		return 0;
	return 1;
}

114 115
#if defined(CONFIG_NFS_V4_1)

A
Alexandros Batsakis 已提交
116 117 118 119 120
int nfs41_validate_delegation_stateid(struct nfs_delegation *delegation, const nfs4_stateid *stateid)
{
	if (delegation == NULL)
		return 0;

121
	if (stateid->stateid.seqid != 0)
A
Alexandros Batsakis 已提交
122
		return 0;
123 124 125
	if (memcmp(&delegation->stateid.stateid.other,
		   &stateid->stateid.other,
		   NFS4_STATEID_OTHER_SIZE))
A
Alexandros Batsakis 已提交
126 127 128 129 130
		return 0;

	return 1;
}

131 132 133 134 135
/*
 * Validate the sequenceID sent by the server.
 * Return success if the sequenceID is one more than what we last saw on
 * this slot, accounting for wraparound.  Increments the slot's sequence.
 *
136 137
 * We don't yet implement a duplicate request cache, instead we set the
 * back channel ca_maxresponsesize_cached to zero. This is OK for now
138 139 140 141 142 143
 * since we only currently implement idempotent callbacks anyway.
 *
 * We have a single slot backchannel at this time, so we don't bother
 * checking the used_slots bit array on the table.  The lower layer guarantees
 * a single outstanding callback request at a time.
 */
144
static __be32
145
validate_seqid(struct nfs4_slot_table *tbl, struct cb_sequenceargs * args)
146 147 148 149
{
	struct nfs4_slot *slot;

	dprintk("%s enter. slotid %d seqid %d\n",
150
		__func__, args->csa_slotid, args->csa_sequenceid);
151

152
	if (args->csa_slotid > NFS41_BC_MAX_CALLBACKS)
153 154
		return htonl(NFS4ERR_BADSLOT);

155
	slot = tbl->slots + args->csa_slotid;
156 157 158
	dprintk("%s slot table seqid: %d\n", __func__, slot->seq_nr);

	/* Normal */
159
	if (likely(args->csa_sequenceid == slot->seq_nr + 1)) {
160 161 162 163 164
		slot->seq_nr++;
		return htonl(NFS4_OK);
	}

	/* Replay */
165
	if (args->csa_sequenceid == slot->seq_nr) {
166
		dprintk("%s seqid %d is a replay\n",
167
			__func__, args->csa_sequenceid);
168 169 170 171 172 173 174
		/* Signal process_op to set this error on next op */
		if (args->csa_cachethis == 0)
			return htonl(NFS4ERR_RETRY_UNCACHED_REP);

		/* The ca_maxresponsesize_cached is 0 with no DRC */
		else if (args->csa_cachethis == 1)
			return htonl(NFS4ERR_REP_TOO_BIG_TO_CACHE);
175 176 177
	}

	/* Wraparound */
178
	if (args->csa_sequenceid == 1 && (slot->seq_nr + 1) == 0) {
179 180 181 182 183 184 185 186
		slot->seq_nr = 1;
		return htonl(NFS4_OK);
	}

	/* Misordered request */
	return htonl(NFS4ERR_SEQ_MISORDERED);
}

187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222
/*
 * Returns a pointer to a held 'struct nfs_client' that matches the server's
 * address, major version number, and session ID.  It is the caller's
 * responsibility to release the returned reference.
 *
 * Returns NULL if there are no connections with sessions, or if no session
 * matches the one of interest.
 */
 static struct nfs_client *find_client_with_session(
	const struct sockaddr *addr, u32 nfsversion,
	struct nfs4_sessionid *sessionid)
{
	struct nfs_client *clp;

	clp = nfs_find_client(addr, 4);
	if (clp == NULL)
		return NULL;

	do {
		struct nfs_client *prev = clp;

		if (clp->cl_session != NULL) {
			if (memcmp(clp->cl_session->sess_id.data,
					sessionid->data,
					NFS4_MAX_SESSIONID_LEN) == 0) {
				/* Returns a held reference to clp */
				return clp;
			}
		}
		clp = nfs_find_client_next(prev);
		nfs_put_client(prev);
	} while (clp != NULL);

	return NULL;
}

223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277
/*
 * For each referring call triple, check the session's slot table for
 * a match.  If the slot is in use and the sequence numbers match, the
 * client is still waiting for a response to the original request.
 */
static bool referring_call_exists(struct nfs_client *clp,
				  uint32_t nrclists,
				  struct referring_call_list *rclists)
{
	bool status = 0;
	int i, j;
	struct nfs4_session *session;
	struct nfs4_slot_table *tbl;
	struct referring_call_list *rclist;
	struct referring_call *ref;

	/*
	 * XXX When client trunking is implemented, this becomes
	 * a session lookup from within the loop
	 */
	session = clp->cl_session;
	tbl = &session->fc_slot_table;

	for (i = 0; i < nrclists; i++) {
		rclist = &rclists[i];
		if (memcmp(session->sess_id.data,
			   rclist->rcl_sessionid.data,
			   NFS4_MAX_SESSIONID_LEN) != 0)
			continue;

		for (j = 0; j < rclist->rcl_nrefcalls; j++) {
			ref = &rclist->rcl_refcalls[j];

			dprintk("%s: sessionid %x:%x:%x:%x sequenceid %u "
				"slotid %u\n", __func__,
				((u32 *)&rclist->rcl_sessionid.data)[0],
				((u32 *)&rclist->rcl_sessionid.data)[1],
				((u32 *)&rclist->rcl_sessionid.data)[2],
				((u32 *)&rclist->rcl_sessionid.data)[3],
				ref->rc_sequenceid, ref->rc_slotid);

			spin_lock(&tbl->slot_tbl_lock);
			status = (test_bit(ref->rc_slotid, tbl->used_slots) &&
				  tbl->slots[ref->rc_slotid].seq_nr ==
					ref->rc_sequenceid);
			spin_unlock(&tbl->slot_tbl_lock);
			if (status)
				goto out;
		}
	}

out:
	return status;
}

278
__be32 nfs4_callback_sequence(struct cb_sequenceargs *args,
279 280
				struct cb_sequenceres *res)
{
281
	struct nfs_client *clp;
282 283
	int i;
	__be32 status;
284

285 286 287 288 289
	status = htonl(NFS4ERR_BADSESSION);
	clp = find_client_with_session(args->csa_addr, 4, &args->csa_sessionid);
	if (clp == NULL)
		goto out;

290
	status = validate_seqid(&clp->cl_session->bc_slot_table, args);
291 292 293
	if (status)
		goto out_putclient;

294 295 296 297 298 299 300 301 302 303
	/*
	 * Check for pending referring calls.  If a match is found, a
	 * related callback was received before the response to the original
	 * call.
	 */
	if (referring_call_exists(clp, args->csa_nrclists, args->csa_rclists)) {
		status = htonl(NFS4ERR_DELAY);
		goto out_putclient;
	}

304 305 306 307 308 309 310
	memcpy(&res->csr_sessionid, &args->csa_sessionid,
	       sizeof(res->csr_sessionid));
	res->csr_sequenceid = args->csa_sequenceid;
	res->csr_slotid = args->csa_slotid;
	res->csr_highestslotid = NFS41_BC_MAX_CALLBACKS - 1;
	res->csr_target_highestslotid = NFS41_BC_MAX_CALLBACKS - 1;

311 312 313
out_putclient:
	nfs_put_client(clp);
out:
314 315 316 317
	for (i = 0; i < args->csa_nrclists; i++)
		kfree(args->csa_rclists[i].rcl_refcalls);
	kfree(args->csa_rclists);

318 319 320 321 322 323 324
	if (status == htonl(NFS4ERR_RETRY_UNCACHED_REP))
		res->csr_status = 0;
	else
		res->csr_status = status;
	dprintk("%s: exit with status = %d res->csr_status %d\n", __func__,
		ntohl(status), ntohl(res->csr_status));
	return status;
325 326
}

327
__be32 nfs4_callback_recallany(struct cb_recallanyargs *args, void *dummy)
328 329
{
	struct nfs_client *clp;
330
	__be32 status;
331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
	fmode_t flags = 0;

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
	clp = nfs_find_client(args->craa_addr, 4);
	if (clp == NULL)
		goto out;

	dprintk("NFS: RECALL_ANY callback request from %s\n",
		rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR));

	if (test_bit(RCA4_TYPE_MASK_RDATA_DLG, (const unsigned long *)
		     &args->craa_type_mask))
		flags = FMODE_READ;
	if (test_bit(RCA4_TYPE_MASK_WDATA_DLG, (const unsigned long *)
		     &args->craa_type_mask))
		flags |= FMODE_WRITE;

	if (flags)
		nfs_expire_all_delegation_types(clp, flags);
	status = htonl(NFS4_OK);
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
A
Andy Adamson 已提交
355 356

/* Reduce the fore channel's max_slots to the target value */
357
__be32 nfs4_callback_recallslot(struct cb_recallslotargs *args, void *dummy)
A
Andy Adamson 已提交
358 359 360
{
	struct nfs_client *clp;
	struct nfs4_slot_table *fc_tbl;
361
	__be32 status;
A
Andy Adamson 已提交
362 363 364 365 366 367 368 369 370 371 372 373 374

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
	clp = nfs_find_client(args->crsa_addr, 4);
	if (clp == NULL)
		goto out;

	dprintk("NFS: CB_RECALL_SLOT request from %s target max slots %d\n",
		rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR),
		args->crsa_target_max_slots);

	fc_tbl = &clp->cl_session->fc_slot_table;

	status = htonl(NFS4ERR_BAD_HIGH_SLOT);
375
	if (args->crsa_target_max_slots > fc_tbl->max_slots ||
A
Andy Adamson 已提交
376
	    args->crsa_target_max_slots < 1)
377 378 379 380 381
		goto out_putclient;

	status = htonl(NFS4_OK);
	if (args->crsa_target_max_slots == fc_tbl->max_slots)
		goto out_putclient;
A
Andy Adamson 已提交
382 383 384

	fc_tbl->target_max_slots = args->crsa_target_max_slots;
	nfs41_handle_recall_slot(clp);
385
out_putclient:
A
Andy Adamson 已提交
386 387 388 389 390
	nfs_put_client(clp);	/* balance nfs_find_client */
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
391
#endif /* CONFIG_NFS_V4_1 */