Skip to content

Commit 3668499

Browse files
scottmayhewJ. Bruce Fields
authored andcommitted
nfsd: Register callbacks on the inetaddr_chain and inet6addr_chain
Register callbacks on inetaddr_chain and inet6addr_chain to trigger cleanup of nfsd transport sockets when an ip address is deleted. Signed-off-by: Scott Mayhew <[email protected]> Signed-off-by: J. Bruce Fields <[email protected]>
1 parent c3d4879 commit 3668499

File tree

1 file changed

+68
-0
lines changed

1 file changed

+68
-0
lines changed

fs/nfsd/nfssvc.c

Lines changed: 68 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -14,9 +14,13 @@
1414

1515
#include <linux/sunrpc/stats.h>
1616
#include <linux/sunrpc/svcsock.h>
17+
#include <linux/sunrpc/svc_xprt.h>
1718
#include <linux/lockd/bind.h>
1819
#include <linux/nfsacl.h>
1920
#include <linux/seq_file.h>
21+
#include <linux/inetdevice.h>
22+
#include <net/addrconf.h>
23+
#include <net/ipv6.h>
2024
#include <net/net_namespace.h>
2125
#include "nfsd.h"
2226
#include "cache.h"
@@ -306,10 +310,70 @@ static void nfsd_shutdown_net(struct net *net)
306310
nfsd_shutdown_generic();
307311
}
308312

313+
static int nfsd_inetaddr_event(struct notifier_block *this, unsigned long event,
314+
void *ptr)
315+
{
316+
struct in_ifaddr *ifa = (struct in_ifaddr *)ptr;
317+
struct net_device *dev = ifa->ifa_dev->dev;
318+
struct net *net = dev_net(dev);
319+
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
320+
struct sockaddr_in sin;
321+
322+
if (event != NETDEV_DOWN)
323+
goto out;
324+
325+
if (nn->nfsd_serv) {
326+
dprintk("nfsd_inetaddr_event: removed %pI4\n", &ifa->ifa_local);
327+
sin.sin_family = AF_INET;
328+
sin.sin_addr.s_addr = ifa->ifa_local;
329+
svc_age_temp_xprts_now(nn->nfsd_serv, (struct sockaddr *)&sin);
330+
}
331+
332+
out:
333+
return NOTIFY_DONE;
334+
}
335+
336+
static struct notifier_block nfsd_inetaddr_notifier = {
337+
.notifier_call = nfsd_inetaddr_event,
338+
};
339+
340+
#if IS_ENABLED(CONFIG_IPV6)
341+
static int nfsd_inet6addr_event(struct notifier_block *this,
342+
unsigned long event, void *ptr)
343+
{
344+
struct inet6_ifaddr *ifa = (struct inet6_ifaddr *)ptr;
345+
struct net_device *dev = ifa->idev->dev;
346+
struct net *net = dev_net(dev);
347+
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
348+
struct sockaddr_in6 sin6;
349+
350+
if (event != NETDEV_DOWN)
351+
goto out;
352+
353+
if (nn->nfsd_serv) {
354+
dprintk("nfsd_inet6addr_event: removed %pI6\n", &ifa->addr);
355+
sin6.sin6_family = AF_INET6;
356+
sin6.sin6_addr = ifa->addr;
357+
svc_age_temp_xprts_now(nn->nfsd_serv, (struct sockaddr *)&sin6);
358+
}
359+
360+
out:
361+
return NOTIFY_DONE;
362+
}
363+
364+
static struct notifier_block nfsd_inet6addr_notifier = {
365+
.notifier_call = nfsd_inet6addr_event,
366+
};
367+
#endif
368+
309369
static void nfsd_last_thread(struct svc_serv *serv, struct net *net)
310370
{
311371
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
312372

373+
unregister_inetaddr_notifier(&nfsd_inetaddr_notifier);
374+
#if IS_ENABLED(CONFIG_IPV6)
375+
unregister_inet6addr_notifier(&nfsd_inet6addr_notifier);
376+
#endif
313377
/*
314378
* write_ports can create the server without actually starting
315379
* any threads--if we get shut down before any threads are
@@ -425,6 +489,10 @@ int nfsd_create_serv(struct net *net)
425489
}
426490

427491
set_max_drc();
492+
register_inetaddr_notifier(&nfsd_inetaddr_notifier);
493+
#if IS_ENABLED(CONFIG_IPV6)
494+
register_inet6addr_notifier(&nfsd_inet6addr_notifier);
495+
#endif
428496
do_gettimeofday(&nn->nfssvc_boot); /* record boot time */
429497
return 0;
430498
}

0 commit comments

Comments
 (0)