host.c 15.5 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * linux/fs/lockd/host.c
 *
 * Management for NLM peer hosts. The nlm_host struct is shared
 * between client and server implementation. The only reason to
 * do so is to reduce code bloat.
 *
 * Copyright (C) 1996, Olaf Kirch <okir@monad.swb.de>
 */

#include <linux/types.h>
#include <linux/slab.h>
#include <linux/in.h>
14
#include <linux/in6.h>
L
Linus Torvalds 已提交
15 16 17
#include <linux/sunrpc/clnt.h>
#include <linux/sunrpc/svc.h>
#include <linux/lockd/lockd.h>
I
Ingo Molnar 已提交
18
#include <linux/mutex.h>
L
Linus Torvalds 已提交
19

20
#include <net/ipv6.h>
L
Linus Torvalds 已提交
21 22 23 24

#define NLMDBG_FACILITY		NLMDBG_HOSTCACHE
#define NLM_HOST_NRHASH		32
#define NLM_HOST_REBIND		(60 * HZ)
25 26
#define NLM_HOST_EXPIRE		(300 * HZ)
#define NLM_HOST_COLLECT	(120 * HZ)
L
Linus Torvalds 已提交
27

28
static struct hlist_head	nlm_hosts[NLM_HOST_NRHASH];
29 30 31 32 33 34 35 36 37 38 39 40

#define for_each_host(host, pos, chain, table) \
	for ((chain) = (table); \
	     (chain) < (table) + NLM_HOST_NRHASH; ++(chain)) \
		hlist_for_each_entry((host), (pos), (chain), h_hash)

#define for_each_host_safe(host, pos, next, chain, table) \
	for ((chain) = (table); \
	     (chain) < (table) + NLM_HOST_NRHASH; ++(chain)) \
		hlist_for_each_entry_safe((host), (pos), (next), \
						(chain), h_hash)

L
Linus Torvalds 已提交
41 42
static unsigned long		next_gc;
static int			nrhosts;
I
Ingo Molnar 已提交
43
static DEFINE_MUTEX(nlm_host_mutex);
L
Linus Torvalds 已提交
44 45 46

static void			nlm_gc_hosts(void);

47 48
struct nlm_lookup_host_info {
	const int		server;		/* search for server|client */
49 50
	const struct sockaddr	*sap;		/* address to search for */
	const size_t		salen;		/* it's length */
51 52 53 54
	const unsigned short	protocol;	/* transport to search for*/
	const u32		version;	/* NLM version to search for */
	const char		*hostname;	/* remote's hostname */
	const size_t		hostname_len;	/* it's length */
55
	const struct sockaddr	*src_sap;	/* our address (optional) */
56
	const size_t		src_len;	/* it's length */
57
	const int		noresvport;	/* use non-priv port */
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 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
/*
 * Hash function must work well on big- and little-endian platforms
 */
static unsigned int __nlm_hash32(const __be32 n)
{
	unsigned int hash = (__force u32)n ^ ((__force u32)n >> 16);
	return hash ^ (hash >> 8);
}

static unsigned int __nlm_hash_addr4(const struct sockaddr *sap)
{
	const struct sockaddr_in *sin = (struct sockaddr_in *)sap;
	return __nlm_hash32(sin->sin_addr.s_addr);
}

static unsigned int __nlm_hash_addr6(const struct sockaddr *sap)
{
	const struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)sap;
	const struct in6_addr addr = sin6->sin6_addr;
	return __nlm_hash32(addr.s6_addr32[0]) ^
	       __nlm_hash32(addr.s6_addr32[1]) ^
	       __nlm_hash32(addr.s6_addr32[2]) ^
	       __nlm_hash32(addr.s6_addr32[3]);
}

static unsigned int nlm_hash_address(const struct sockaddr *sap)
{
	unsigned int hash;

	switch (sap->sa_family) {
	case AF_INET:
		hash = __nlm_hash_addr4(sap);
		break;
	case AF_INET6:
		hash = __nlm_hash_addr6(sap);
		break;
	default:
		hash = 0;
	}
	return hash & (NLM_HOST_NRHASH - 1);
}

C
Chuck Lever 已提交
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
/*
 * Allocate and initialize an nlm_host.  Common to both client and server.
 */
static struct nlm_host *nlm_alloc_host(struct nlm_lookup_host_info *ni,
				       struct nsm_handle *nsm)
{
	struct nlm_host *host = NULL;
	unsigned long now = jiffies;

	if (nsm != NULL)
		atomic_inc(&nsm->sm_count);
	else {
		host = NULL;
		nsm = nsm_get_handle(ni->sap, ni->salen,
					ni->hostname, ni->hostname_len);
		if (unlikely(nsm == NULL)) {
			dprintk("lockd: %s failed; no nsm handle\n",
				__func__);
			goto out;
		}
	}

	host = kmalloc(sizeof(*host), GFP_KERNEL);
	if (unlikely(host == NULL)) {
		dprintk("lockd: %s failed; no memory\n", __func__);
		nsm_release(nsm);
		goto out;
	}

	memcpy(nlm_addr(host), ni->sap, ni->salen);
	host->h_addrlen    = ni->salen;
	rpc_set_port(nlm_addr(host), 0);
	host->h_srcaddrlen = 0;

	host->h_rpcclnt    = NULL;
	host->h_name	   = nsm->sm_name;
	host->h_version    = ni->version;
	host->h_proto      = ni->protocol;
	host->h_reclaiming = 0;
	host->h_server     = ni->server;
	host->h_noresvport = ni->noresvport;
	host->h_inuse      = 0;
	init_waitqueue_head(&host->h_gracewait);
	init_rwsem(&host->h_rwsem);
	host->h_state      = 0;
	host->h_nsmstate   = 0;
	host->h_pidcount   = 0;
	atomic_set(&host->h_count, 1);
	mutex_init(&host->h_mutex);
	host->h_nextrebind = now + NLM_HOST_REBIND;
	host->h_expires    = now + NLM_HOST_EXPIRE;
	INIT_LIST_HEAD(&host->h_lockowners);
	spin_lock_init(&host->h_lock);
	INIT_LIST_HEAD(&host->h_granted);
	INIT_LIST_HEAD(&host->h_reclaim);
	host->h_nsmhandle  = nsm;
	host->h_addrbuf    = nsm->sm_addrbuf;

out:
	return host;
}

L
Linus Torvalds 已提交
164 165 166
/*
 * Common host lookup routine for server & client
 */
167
static struct nlm_host *nlm_lookup_host(struct nlm_lookup_host_info *ni)
L
Linus Torvalds 已提交
168
{
169 170 171
	struct hlist_head *chain;
	struct hlist_node *pos;
	struct nlm_host	*host;
172
	struct nsm_handle *nsm = NULL;
L
Linus Torvalds 已提交
173

I
Ingo Molnar 已提交
174
	mutex_lock(&nlm_host_mutex);
L
Linus Torvalds 已提交
175 176 177 178

	if (time_after_eq(jiffies, next_gc))
		nlm_gc_hosts();

179 180 181 182 183 184 185
	/* We may keep several nlm_host objects for a peer, because each
	 * nlm_host is identified by
	 * (address, protocol, version, server/client)
	 * We could probably simplify this a little by putting all those
	 * different NLM rpc_clients into one single nlm_host object.
	 * This would allow us to have one nlm_host per address.
	 */
186
	chain = &nlm_hosts[nlm_hash_address(ni->sap)];
187
	hlist_for_each_entry(host, pos, chain, h_hash) {
188
		if (!rpc_cmp_addr(nlm_addr(host), ni->sap))
189 190 191
			continue;

		/* See if we have an NSM handle for this client */
192 193
		if (!nsm)
			nsm = host->h_nsmhandle;
194

195
		if (host->h_proto != ni->protocol)
L
Linus Torvalds 已提交
196
			continue;
197
		if (host->h_version != ni->version)
L
Linus Torvalds 已提交
198
			continue;
199
		if (host->h_server != ni->server)
L
Linus Torvalds 已提交
200
			continue;
T
Trond Myklebust 已提交
201
		if (ni->server && ni->src_len != 0 &&
202
		    !rpc_cmp_addr(nlm_srcaddr(host), ni->src_sap))
203
			continue;
L
Linus Torvalds 已提交
204

205 206 207 208
		/* Move to head of hash chain. */
		hlist_del(&host->h_hash);
		hlist_add_head(&host->h_hash, chain);

209
		nlm_get_host(host);
210 211
		dprintk("lockd: nlm_lookup_host found host %s (%s)\n",
				host->h_name, host->h_addrbuf);
212
		goto out;
L
Linus Torvalds 已提交
213 214
	}

C
Chuck Lever 已提交
215 216
	host = nlm_alloc_host(ni, nsm);
	if (unlikely(host == NULL))
217
		goto out;
C
Chuck Lever 已提交
218

219
	memcpy(nlm_srcaddr(host), ni->src_sap, ni->src_len);
T
Trond Myklebust 已提交
220
	host->h_srcaddrlen = ni->src_len;
221
	hlist_add_head(&host->h_hash, chain);
L
Linus Torvalds 已提交
222

223
	nrhosts++;
224 225 226 227

	dprintk("lockd: nlm_lookup_host created host %s\n",
			host->h_name);

228
out:
I
Ingo Molnar 已提交
229
	mutex_unlock(&nlm_host_mutex);
L
Linus Torvalds 已提交
230 231 232
	return host;
}

233
/*
C
Chuck Lever 已提交
234 235 236
 * Destroy an nlm_host and free associated resources
 *
 * Caller must hold nlm_host_mutex.
237
 */
C
Chuck Lever 已提交
238
static void nlm_destroy_host_locked(struct nlm_host *host)
239 240 241
{
	struct rpc_clnt	*clnt;

C
Chuck Lever 已提交
242 243
	dprintk("lockd: destroy host %s\n", host->h_name);

244 245 246
	BUG_ON(!list_empty(&host->h_lockowners));
	BUG_ON(atomic_read(&host->h_count));

C
Chuck Lever 已提交
247 248
	hlist_del_init(&host->h_hash);

249
	nsm_unmonitor(host);
250
	nsm_release(host->h_nsmhandle);
251

252 253 254
	clnt = host->h_rpcclnt;
	if (clnt != NULL)
		rpc_shutdown_client(clnt);
255
	kfree(host);
C
Chuck Lever 已提交
256 257

	nrhosts--;
258 259
}

260 261 262 263 264 265 266
/**
 * nlmclnt_lookup_host - Find an NLM host handle matching a remote server
 * @sap: network address of server
 * @salen: length of server address
 * @protocol: transport protocol to use
 * @version: NLM protocol version
 * @hostname: '\0'-terminated hostname of server
267
 * @noresvport: 1 if non-privileged port should be used
268 269 270 271 272
 *
 * Returns an nlm_host structure that matches the passed-in
 * [server address, transport protocol, NLM version, server hostname].
 * If one doesn't already exist in the host cache, a new handle is
 * created and returned.
273
 */
274 275 276
struct nlm_host *nlmclnt_lookup_host(const struct sockaddr *sap,
				     const size_t salen,
				     const unsigned short protocol,
277 278 279
				     const u32 version,
				     const char *hostname,
				     int noresvport)
280
{
281 282
	struct nlm_lookup_host_info ni = {
		.server		= 0,
283 284 285
		.sap		= sap,
		.salen		= salen,
		.protocol	= protocol,
286 287
		.version	= version,
		.hostname	= hostname,
288
		.hostname_len	= strlen(hostname),
289
		.noresvport	= noresvport,
290
	};
291

292 293
	dprintk("lockd: %s(host='%s', vers=%u, proto=%s)\n", __func__,
			(hostname ? hostname : "<none>"), version,
294
			(protocol == IPPROTO_UDP ? "udp" : "tcp"));
295 296

	return nlm_lookup_host(&ni);
297 298
}

299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315
/**
 * nlmsvc_lookup_host - Find an NLM host handle matching a remote client
 * @rqstp: incoming NLM request
 * @hostname: name of client host
 * @hostname_len: length of client hostname
 *
 * Returns an nlm_host structure that matches the [client address,
 * transport protocol, NLM version, client hostname] of the passed-in
 * NLM request.  If one doesn't already exist in the host cache, a
 * new handle is created and returned.
 *
 * Before possibly creating a new nlm_host, construct a sockaddr
 * for a specific source address in case the local system has
 * multiple network addresses.  The family of the address in
 * rq_daddr is guaranteed to be the same as the family of the
 * address in rq_addr, so it's safe to use the same family for
 * the source address.
316
 */
317 318 319
struct nlm_host *nlmsvc_lookup_host(const struct svc_rqst *rqstp,
				    const char *hostname,
				    const size_t hostname_len)
320
{
321
	struct sockaddr_in sin = {
322
		.sin_family	= AF_INET,
323 324 325
	};
	struct sockaddr_in6 sin6 = {
		.sin6_family	= AF_INET6,
326
	};
327 328
	struct nlm_lookup_host_info ni = {
		.server		= 1,
329 330
		.sap		= svc_addr(rqstp),
		.salen		= rqstp->rq_addrlen,
331 332 333 334
		.protocol	= rqstp->rq_prot,
		.version	= rqstp->rq_vers,
		.hostname	= hostname,
		.hostname_len	= hostname_len,
335
		.src_len	= rqstp->rq_addrlen,
336 337 338 339 340
	};

	dprintk("lockd: %s(host='%*s', vers=%u, proto=%s)\n", __func__,
			(int)hostname_len, hostname, rqstp->rq_vers,
			(rqstp->rq_prot == IPPROTO_UDP ? "udp" : "tcp"));
341

342 343 344 345 346 347 348 349 350 351 352 353 354
	switch (ni.sap->sa_family) {
	case AF_INET:
		sin.sin_addr.s_addr = rqstp->rq_daddr.addr.s_addr;
		ni.src_sap = (struct sockaddr *)&sin;
		break;
	case AF_INET6:
		ipv6_addr_copy(&sin6.sin6_addr, &rqstp->rq_daddr.addr6);
		ni.src_sap = (struct sockaddr *)&sin6;
		break;
	default:
		return NULL;
	}

355
	return nlm_lookup_host(&ni);
356 357
}

L
Linus Torvalds 已提交
358 359 360 361 362 363 364 365
/*
 * Create the NLM RPC client for an NLM peer
 */
struct rpc_clnt *
nlm_bind_host(struct nlm_host *host)
{
	struct rpc_clnt	*clnt;

366 367
	dprintk("lockd: nlm_bind_host %s (%s)\n",
			host->h_name, host->h_addrbuf);
L
Linus Torvalds 已提交
368 369

	/* Lock host handle */
T
Trond Myklebust 已提交
370
	mutex_lock(&host->h_mutex);
L
Linus Torvalds 已提交
371 372 373 374 375

	/* If we've already created an RPC client, check whether
	 * RPC rebind is required
	 */
	if ((clnt = host->h_rpcclnt) != NULL) {
376
		if (time_after_eq(jiffies, host->h_nextrebind)) {
377
			rpc_force_rebind(clnt);
L
Linus Torvalds 已提交
378
			host->h_nextrebind = jiffies + NLM_HOST_REBIND;
379
			dprintk("lockd: next rebind in %lu jiffies\n",
L
Linus Torvalds 已提交
380 381 382
					host->h_nextrebind - jiffies);
		}
	} else {
383
		unsigned long increment = nlmsvc_timeout;
384 385 386 387 388 389 390
		struct rpc_timeout timeparms = {
			.to_initval	= increment,
			.to_increment	= increment,
			.to_maxval	= increment * 6UL,
			.to_retries	= 5U,
		};
		struct rpc_create_args args = {
391
			.net		= &init_net,
392
			.protocol	= host->h_proto,
393 394
			.address	= nlm_addr(host),
			.addrsize	= host->h_addrlen,
395 396 397 398 399
			.timeout	= &timeparms,
			.servername	= host->h_name,
			.program	= &nlm_program,
			.version	= host->h_version,
			.authflavor	= RPC_AUTH_UNIX,
400
			.flags		= (RPC_CLNT_CREATE_NOPING |
401 402 403
					   RPC_CLNT_CREATE_AUTOBIND),
		};

404 405 406 407 408 409 410
		/*
		 * lockd retries server side blocks automatically so we want
		 * those to be soft RPC calls. Client side calls need to be
		 * hard RPC tasks.
		 */
		if (!host->h_server)
			args.flags |= RPC_CLNT_CREATE_HARDRTRY;
411 412
		if (host->h_noresvport)
			args.flags |= RPC_CLNT_CREATE_NONPRIVPORT;
T
Trond Myklebust 已提交
413 414
		if (host->h_srcaddrlen)
			args.saddress = nlm_srcaddr(host);
415

416 417 418 419 420 421 422
		clnt = rpc_create(&args);
		if (!IS_ERR(clnt))
			host->h_rpcclnt = clnt;
		else {
			printk("lockd: couldn't create RPC handle for %s\n", host->h_name);
			clnt = NULL;
		}
L
Linus Torvalds 已提交
423 424
	}

T
Trond Myklebust 已提交
425
	mutex_unlock(&host->h_mutex);
L
Linus Torvalds 已提交
426 427 428 429 430 431 432 433 434 435 436
	return clnt;
}

/*
 * Force a portmap lookup of the remote lockd port
 */
void
nlm_rebind_host(struct nlm_host *host)
{
	dprintk("lockd: rebind host %s\n", host->h_name);
	if (host->h_rpcclnt && time_after_eq(jiffies, host->h_nextrebind)) {
437
		rpc_force_rebind(host->h_rpcclnt);
L
Linus Torvalds 已提交
438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462
		host->h_nextrebind = jiffies + NLM_HOST_REBIND;
	}
}

/*
 * Increment NLM host count
 */
struct nlm_host * nlm_get_host(struct nlm_host *host)
{
	if (host) {
		dprintk("lockd: get host %s\n", host->h_name);
		atomic_inc(&host->h_count);
		host->h_expires = jiffies + NLM_HOST_EXPIRE;
	}
	return host;
}

/*
 * Release NLM host after use
 */
void nlm_release_host(struct nlm_host *host)
{
	if (host != NULL) {
		dprintk("lockd: release host %s\n", host->h_name);
		BUG_ON(atomic_read(&host->h_count) < 0);
463 464 465 466 467
		if (atomic_dec_and_test(&host->h_count)) {
			BUG_ON(!list_empty(&host->h_lockowners));
			BUG_ON(!list_empty(&host->h_granted));
			BUG_ON(!list_empty(&host->h_reclaim));
		}
L
Linus Torvalds 已提交
468 469 470
	}
}

471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495
static struct nlm_host *next_host_state(struct hlist_head *cache,
					struct nsm_handle *nsm,
					const struct nlm_reboot *info)
{
	struct nlm_host *host = NULL;
	struct hlist_head *chain;
	struct hlist_node *pos;

	mutex_lock(&nlm_host_mutex);
	for_each_host(host, pos, chain, cache) {
		if (host->h_nsmhandle == nsm
		    && host->h_nsmstate != info->state) {
			host->h_nsmstate = info->state;
			host->h_state++;

			nlm_get_host(host);
			mutex_unlock(&nlm_host_mutex);
			goto out;
		}
	}
out:
	mutex_unlock(&nlm_host_mutex);
	return host;
}

496 497 498 499 500 501
/**
 * nlm_host_rebooted - Release all resources held by rebooted host
 * @info: pointer to decoded results of NLM_SM_NOTIFY call
 *
 * We were notified that the specified host has rebooted.  Release
 * all resources held by that peer.
502
 */
503
void nlm_host_rebooted(const struct nlm_reboot *info)
504
{
505
	struct nsm_handle *nsm;
506
	struct nlm_host	*host;
507

508 509
	nsm = nsm_reboot_lookup(info);
	if (unlikely(nsm == NULL))
510
		return;
511 512 513 514 515 516

	/* Mark all hosts tied to this NSM state as having rebooted.
	 * We run the loop repeatedly, because we drop the host table
	 * lock for this.
	 * To avoid processing a host several times, we match the nsmstate.
	 */
517 518 519 520 521 522 523 524
	while ((host = next_host_state(nlm_hosts, nsm, info)) != NULL) {
		if (host->h_server) {
			/* We're server for this guy, just ditch
			 * all the locks he held. */
			nlmsvc_free_host_resources(host);
		} else {
			/* He's the server, initiate lock recovery. */
			nlmclnt_recovery(host);
525
		}
526
		nlm_release_host(host);
527
	}
528
	nsm_release(nsm);
529 530
}

L
Linus Torvalds 已提交
531 532 533 534 535 536 537
/*
 * Shut down the hosts module.
 * Note that this routine is called only at server shutdown time.
 */
void
nlm_shutdown_hosts(void)
{
538 539
	struct hlist_head *chain;
	struct hlist_node *pos;
L
Linus Torvalds 已提交
540 541 542
	struct nlm_host	*host;

	dprintk("lockd: shutting down host module\n");
I
Ingo Molnar 已提交
543
	mutex_lock(&nlm_host_mutex);
L
Linus Torvalds 已提交
544 545 546

	/* First, make all hosts eligible for gc */
	dprintk("lockd: nuking all hosts...\n");
547 548 549 550 551
	for_each_host(host, pos, chain, nlm_hosts) {
		host->h_expires = jiffies - 1;
		if (host->h_rpcclnt) {
			rpc_shutdown_client(host->h_rpcclnt);
			host->h_rpcclnt = NULL;
552
		}
L
Linus Torvalds 已提交
553 554 555 556
	}

	/* Then, perform a garbage collection pass */
	nlm_gc_hosts();
I
Ingo Molnar 已提交
557
	mutex_unlock(&nlm_host_mutex);
L
Linus Torvalds 已提交
558 559 560 561 562

	/* complain if any hosts are left */
	if (nrhosts) {
		printk(KERN_WARNING "lockd: couldn't shutdown host module!\n");
		dprintk("lockd: %d hosts left:\n", nrhosts);
563 564 565 566
		for_each_host(host, pos, chain, nlm_hosts) {
			dprintk("       %s (cnt %d use %d exp %ld)\n",
				host->h_name, atomic_read(&host->h_count),
				host->h_inuse, host->h_expires);
L
Linus Torvalds 已提交
567 568 569 570 571 572 573 574 575 576 577 578
		}
	}
}

/*
 * Garbage collect any unused NLM hosts.
 * This GC combines reference counting for async operations with
 * mark & sweep for resources held by remote clients.
 */
static void
nlm_gc_hosts(void)
{
579 580 581
	struct hlist_head *chain;
	struct hlist_node *pos, *next;
	struct nlm_host	*host;
L
Linus Torvalds 已提交
582 583

	dprintk("lockd: host garbage collection\n");
584 585
	for_each_host(host, pos, chain, nlm_hosts)
		host->h_inuse = 0;
L
Linus Torvalds 已提交
586 587 588 589

	/* Mark all hosts that hold locks, blocks or shares */
	nlmsvc_mark_resources();

590 591 592 593 594 595 596 597
	for_each_host_safe(host, pos, next, chain, nlm_hosts) {
		if (atomic_read(&host->h_count) || host->h_inuse
		 || time_before(jiffies, host->h_expires)) {
			dprintk("nlm_gc_hosts skipping %s "
				"(cnt %d use %d exp %ld)\n",
				host->h_name, atomic_read(&host->h_count),
				host->h_inuse, host->h_expires);
			continue;
L
Linus Torvalds 已提交
598
		}
C
Chuck Lever 已提交
599
		nlm_destroy_host_locked(host);
L
Linus Torvalds 已提交
600 601 602 603
	}

	next_gc = jiffies + NLM_HOST_COLLECT;
}