callback_proc.c 9.7 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 257 258 259 260 261
	/* state manager is resetting the session */
	if (test_bit(NFS4_SESSION_DRAINING, &clp->cl_session->session_state)) {
		status = NFS4ERR_DELAY;
		goto out;
	}

262
	status = validate_seqid(&clp->cl_session->bc_slot_table, args);
263
	if (status)
264
		goto out;
265

266 267 268 269 270 271 272
	/*
	 * 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);
273
		goto out;
274 275
	}

276 277 278 279 280 281
	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;
282
	nfs4_cb_take_slot(clp);
283
	cps->clp = clp; /* put in nfs4_callback_compound */
284

285
out:
286 287 288 289
	for (i = 0; i < args->csa_nrclists; i++)
		kfree(args->csa_rclists[i].rcl_refcalls);
	kfree(args->csa_rclists);

290 291 292 293
	if (status == htonl(NFS4ERR_RETRY_UNCACHED_REP)) {
		cps->drc_status = status;
		status = 0;
	} else
294
		res->csr_status = status;
295

296 297 298
	dprintk("%s: exit with status = %d res->csr_status %d\n", __func__,
		ntohl(status), ntohl(res->csr_status));
	return status;
299 300
}

301 302
__be32 nfs4_callback_recallany(struct cb_recallanyargs *args, void *dummy,
			       struct cb_process_state *cps)
303
{
304
	__be32 status;
305 306 307
	fmode_t flags = 0;

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
308
	if (!cps->clp) /* set in cb_sequence */
309 310 311
		goto out;

	dprintk("NFS: RECALL_ANY callback request from %s\n",
312
		rpc_peeraddr2str(cps->clp->cl_rpcclient, RPC_DISPLAY_ADDR));
313 314 315 316 317 318 319 320 321

	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)
322
		nfs_expire_all_delegation_types(cps->clp, flags);
323 324 325 326 327
	status = htonl(NFS4_OK);
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
A
Andy Adamson 已提交
328 329

/* Reduce the fore channel's max_slots to the target value */
330 331
__be32 nfs4_callback_recallslot(struct cb_recallslotargs *args, void *dummy,
				struct cb_process_state *cps)
A
Andy Adamson 已提交
332 333
{
	struct nfs4_slot_table *fc_tbl;
334
	__be32 status;
A
Andy Adamson 已提交
335 336

	status = htonl(NFS4ERR_OP_NOT_IN_SESSION);
337
	if (!cps->clp) /* set in cb_sequence */
A
Andy Adamson 已提交
338 339 340
		goto out;

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

344
	fc_tbl = &cps->clp->cl_session->fc_slot_table;
A
Andy Adamson 已提交
345 346

	status = htonl(NFS4ERR_BAD_HIGH_SLOT);
347
	if (args->crsa_target_max_slots > fc_tbl->max_slots ||
A
Andy Adamson 已提交
348
	    args->crsa_target_max_slots < 1)
349
		goto out;
350 351 352

	status = htonl(NFS4_OK);
	if (args->crsa_target_max_slots == fc_tbl->max_slots)
353
		goto out;
A
Andy Adamson 已提交
354 355

	fc_tbl->target_max_slots = args->crsa_target_max_slots;
356
	nfs41_handle_recall_slot(cps->clp);
A
Andy Adamson 已提交
357 358 359 360
out:
	dprintk("%s: exit with status = %d\n", __func__, ntohl(status));
	return status;
}
361
#endif /* CONFIG_NFS_V4_1 */