callback_proc.c 9.5 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)

A
Alexandros Batsakis 已提交
110 111 112 113 114
int nfs41_validate_delegation_stateid(struct nfs_delegation *delegation, const nfs4_stateid *stateid)
{
	if (delegation == NULL)
		return 0;

115
	if (stateid->stateid.seqid != 0)
A
Alexandros Batsakis 已提交
116
		return 0;
117 118 119
	if (memcmp(&delegation->stateid.stateid.other,
		   &stateid->stateid.other,
		   NFS4_STATEID_OTHER_SIZE))
A
Alexandros Batsakis 已提交
120 121 122 123 124
		return 0;

	return 1;
}

125 126 127 128 129
/*
 * 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.
 *
130 131
 * 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
132 133 134 135 136 137
 * 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.
 */
138
static __be32
139
validate_seqid(struct nfs4_slot_table *tbl, struct cb_sequenceargs * args)
140 141 142 143
{
	struct nfs4_slot *slot;

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

146
	if (args->csa_slotid > NFS41_BC_MAX_CALLBACKS)
147 148
		return htonl(NFS4ERR_BADSLOT);

149
	slot = tbl->slots + args->csa_slotid;
150 151 152
	dprintk("%s slot table seqid: %d\n", __func__, slot->seq_nr);

	/* Normal */
153
	if (likely(args->csa_sequenceid == slot->seq_nr + 1)) {
154 155 156 157 158
		slot->seq_nr++;
		return htonl(NFS4_OK);
	}

	/* Replay */
159
	if (args->csa_sequenceid == slot->seq_nr) {
160
		dprintk("%s seqid %d is a replay\n",
161
			__func__, args->csa_sequenceid);
162 163 164 165 166 167 168
		/* 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);
169 170 171
	}

	/* Wraparound */
172
	if (args->csa_sequenceid == 1 && (slot->seq_nr + 1) == 0) {
173 174 175 176 177 178 179 180
		slot->seq_nr = 1;
		return htonl(NFS4_OK);
	}

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

181 182 183 184 185 186 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
/*
 * 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;
}

236
__be32 nfs4_callback_sequence(struct cb_sequenceargs *args,
237 238
			      struct cb_sequenceres *res,
			      struct cb_process_state *cps)
239
{
240
	struct nfs_client *clp;
241 242
	int i;
	__be32 status;
243

244 245
	cps->clp = NULL;

246
	status = htonl(NFS4ERR_BADSESSION);
247 248 249 250 251 252
	/* 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);
253 254 255
	if (clp == NULL)
		goto out;

256
	status = validate_seqid(&clp->cl_session->bc_slot_table, args);
257
	if (status)
258
		goto out;
259

260 261 262 263 264 265 266
	/*
	 * 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);
267
		goto out;
268 269
	}

270 271 272 273 274 275
	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;
276
	cps->clp = clp; /* put in nfs4_callback_compound */
277

278
out:
279 280 281 282
	for (i = 0; i < args->csa_nrclists; i++)
		kfree(args->csa_rclists[i].rcl_refcalls);
	kfree(args->csa_rclists);

283 284 285 286
	if (status == htonl(NFS4ERR_RETRY_UNCACHED_REP)) {
		cps->drc_status = status;
		status = 0;
	} else
287
		res->csr_status = status;
288

289 290 291
	dprintk("%s: exit with status = %d res->csr_status %d\n", __func__,
		ntohl(status), ntohl(res->csr_status));
	return status;
292 293
}

294 295
__be32 nfs4_callback_recallany(struct cb_recallanyargs *args, void *dummy,
			       struct cb_process_state *cps)
296
{
297
	__be32 status;
298 299 300
	fmode_t flags = 0;

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
301
	if (!cps->clp) /* set in cb_sequence */
302 303 304
		goto out;

	dprintk("NFS: RECALL_ANY callback request from %s\n",
305
		rpc_peeraddr2str(cps->clp->cl_rpcclient, RPC_DISPLAY_ADDR));
306 307 308 309 310 311 312 313 314

	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)
315
		nfs_expire_all_delegation_types(cps->clp, flags);
316 317 318 319 320
	status = htonl(NFS4_OK);
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
A
Andy Adamson 已提交
321 322

/* Reduce the fore channel's max_slots to the target value */
323 324
__be32 nfs4_callback_recallslot(struct cb_recallslotargs *args, void *dummy,
				struct cb_process_state *cps)
A
Andy Adamson 已提交
325 326
{
	struct nfs4_slot_table *fc_tbl;
327
	__be32 status;
A
Andy Adamson 已提交
328 329

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
330
	if (!cps->clp) /* set in cb_sequence */
A
Andy Adamson 已提交
331 332 333
		goto out;

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

337
	fc_tbl = &cps->clp->cl_session->fc_slot_table;
A
Andy Adamson 已提交
338 339

	status = htonl(NFS4ERR_BAD_HIGH_SLOT);
340
	if (args->crsa_target_max_slots > fc_tbl->max_slots ||
A
Andy Adamson 已提交
341
	    args->crsa_target_max_slots < 1)
342
		goto out;
343 344 345

	status = htonl(NFS4_OK);
	if (args->crsa_target_max_slots == fc_tbl->max_slots)
346
		goto out;
A
Andy Adamson 已提交
347 348

	fc_tbl->target_max_slots = args->crsa_target_max_slots;
349
	nfs41_handle_recall_slot(cps->clp);
A
Andy Adamson 已提交
350 351 352 353
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
354
#endif /* CONFIG_NFS_V4_1 */