callback_proc.c 9.9 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
19 20 21 22

__be32 nfs4_callback_getattr(struct cb_getattrargs *args,
			     struct cb_getattrres *res,
			     struct cb_process_state *cps)
L
Linus Torvalds 已提交
23 24 25 26
{
	struct nfs_delegation *delegation;
	struct nfs_inode *nfsi;
	struct inode *inode;
27

28 29 30 31
	res->status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
	if (!cps->clp) /* Always set for v4.0. Set in cb_sequence for v4.1 */
		goto out;

L
Linus Torvalds 已提交
32 33
	res->bitmap[0] = res->bitmap[1] = 0;
	res->status = htonl(NFS4ERR_BADHANDLE);
34 35

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

38
	inode = nfs_delegation_find_inode(cps->clp, &args->fh);
L
Linus Torvalds 已提交
39
	if (inode == NULL)
40
		goto out;
L
Linus Torvalds 已提交
41
	nfsi = NFS_I(inode);
42 43
	rcu_read_lock();
	delegation = rcu_dereference(nfsi->delegation);
L
Linus Torvalds 已提交
44 45 46
	if (delegation == NULL || (delegation->type & FMODE_WRITE) == 0)
		goto out_iput;
	res->size = i_size_read(inode);
47 48 49
	res->change_attr = delegation->change_attr;
	if (nfsi->npages != 0)
		res->change_attr++;
L
Linus Torvalds 已提交
50 51 52 53 54 55 56 57
	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:
58
	rcu_read_unlock();
L
Linus Torvalds 已提交
59 60
	iput(inode);
out:
61
	dprintk("%s: exit with status = %d\n", __func__, ntohl(res->status));
L
Linus Torvalds 已提交
62 63 64
	return res->status;
}

65 66
__be32 nfs4_callback_recall(struct cb_recallargs *args, void *dummy,
			    struct cb_process_state *cps)
L
Linus Torvalds 已提交
67 68
{
	struct inode *inode;
69
	__be32 res;
L
Linus Torvalds 已提交
70
	
71 72
	res = htonl(NFS4ERR_OP_NOT_IN_SESSION);
	if (!cps->clp) /* Always set for v4.0. Set in cb_sequence for v4.1 */
L
Linus Torvalds 已提交
73
		goto out;
74 75

	dprintk("NFS: RECALL callback request from %s\n",
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
		rpc_peeraddr2str(cps->clp->cl_rpcclient, RPC_DISPLAY_ADDR));

	res = htonl(NFS4ERR_BADHANDLE);
	inode = nfs_delegation_find_inode(cps->clp, &args->fh);
	if (inode == NULL)
		goto out;
	/* Set up a helper thread to actually return the delegation */
	switch (nfs_async_inode_return_delegation(inode, &args->stateid)) {
	case 0:
		res = 0;
		break;
	case -ENOENT:
		if (res != 0)
			res = htonl(NFS4ERR_BAD_STATEID);
		break;
	default:
		res = htonl(NFS4ERR_RESOURCE);
	}
	iput(inode);
L
Linus Torvalds 已提交
95
out:
96
	dprintk("%s: exit with status = %d\n", __func__, ntohl(res));
L
Linus Torvalds 已提交
97 98
	return res;
}
99

A
Alexandros Batsakis 已提交
100 101 102 103 104 105 106 107
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;
}

108 109
#if defined(CONFIG_NFS_V4_1)

F
Fred Isaman 已提交
110 111 112 113 114 115
__be32 nfs4_callback_layoutrecall(struct cb_layoutrecallargs *args,
				  void *dummy, struct cb_process_state *cps)
{
	return cpu_to_be32(NFS4ERR_NOTSUPP); /* STUB */
}

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 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
/*
 * 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;
}

242
__be32 nfs4_callback_sequence(struct cb_sequenceargs *args,
243 244
			      struct cb_sequenceres *res,
			      struct cb_process_state *cps)
245
{
246
	struct nfs_client *clp;
247 248
	int i;
	__be32 status;
249

250 251
	cps->clp = NULL;

252
	status = htonl(NFS4ERR_BADSESSION);
253 254 255 256 257 258
	/* Incoming session must match the callback session */
	if (memcmp(&args->csa_sessionid, cps->svc_sid, NFS4_MAX_SESSIONID_LEN))
		goto out;

	clp = nfs4_find_client_sessionid(args->csa_addr,
					 &args->csa_sessionid, 1);
259 260 261
	if (clp == NULL)
		goto out;

262 263 264 265 266 267
	/* state manager is resetting the session */
	if (test_bit(NFS4_SESSION_DRAINING, &clp->cl_session->session_state)) {
		status = NFS4ERR_DELAY;
		goto out;
	}

268
	status = validate_seqid(&clp->cl_session->bc_slot_table, args);
269
	if (status)
270
		goto out;
271

272 273 274 275 276 277 278
	/*
	 * 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);
279
		goto out;
280 281
	}

282 283 284 285 286 287
	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;
288
	nfs4_cb_take_slot(clp);
289
	cps->clp = clp; /* put in nfs4_callback_compound */
290

291
out:
292 293 294 295
	for (i = 0; i < args->csa_nrclists; i++)
		kfree(args->csa_rclists[i].rcl_refcalls);
	kfree(args->csa_rclists);

296 297 298 299
	if (status == htonl(NFS4ERR_RETRY_UNCACHED_REP)) {
		cps->drc_status = status;
		status = 0;
	} else
300
		res->csr_status = status;
301

302 303 304
	dprintk("%s: exit with status = %d res->csr_status %d\n", __func__,
		ntohl(status), ntohl(res->csr_status));
	return status;
305 306
}

307 308
__be32 nfs4_callback_recallany(struct cb_recallanyargs *args, void *dummy,
			       struct cb_process_state *cps)
309
{
310
	__be32 status;
311 312 313
	fmode_t flags = 0;

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
314
	if (!cps->clp) /* set in cb_sequence */
315 316 317
		goto out;

	dprintk("NFS: RECALL_ANY callback request from %s\n",
318
		rpc_peeraddr2str(cps->clp->cl_rpcclient, RPC_DISPLAY_ADDR));
319 320 321 322 323 324 325 326 327

	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)
328
		nfs_expire_all_delegation_types(cps->clp, flags);
329 330 331 332 333
	status = htonl(NFS4_OK);
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
A
Andy Adamson 已提交
334 335

/* Reduce the fore channel's max_slots to the target value */
336 337
__be32 nfs4_callback_recallslot(struct cb_recallslotargs *args, void *dummy,
				struct cb_process_state *cps)
A
Andy Adamson 已提交
338 339
{
	struct nfs4_slot_table *fc_tbl;
340
	__be32 status;
A
Andy Adamson 已提交
341 342

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
343
	if (!cps->clp) /* set in cb_sequence */
A
Andy Adamson 已提交
344 345 346
		goto out;

	dprintk("NFS: CB_RECALL_SLOT request from %s target max slots %d\n",
347
		rpc_peeraddr2str(cps->clp->cl_rpcclient, RPC_DISPLAY_ADDR),
A
Andy Adamson 已提交
348 349
		args->crsa_target_max_slots);

350
	fc_tbl = &cps->clp->cl_session->fc_slot_table;
A
Andy Adamson 已提交
351 352

	status = htonl(NFS4ERR_BAD_HIGH_SLOT);
353
	if (args->crsa_target_max_slots > fc_tbl->max_slots ||
A
Andy Adamson 已提交
354
	    args->crsa_target_max_slots < 1)
355
		goto out;
356 357 358

	status = htonl(NFS4_OK);
	if (args->crsa_target_max_slots == fc_tbl->max_slots)
359
		goto out;
A
Andy Adamson 已提交
360 361

	fc_tbl->target_max_slots = args->crsa_target_max_slots;
362
	nfs41_handle_recall_slot(cps->clp);
A
Andy Adamson 已提交
363 364 365 366
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
367
#endif /* CONFIG_NFS_V4_1 */