Commit 4373ea84 authored by Wendy Cheng's avatar Wendy Cheng Committed by J. Bruce Fields

lockd: unlock lockd locks associated with a given server ip

For high-availability NFS service, we generally need to be able to drop
file locks held on the exported filesystem before moving clients to a
new server.  Currently the only way to do that is by shutting down lockd
entirely, which is often undesireable (for example, if you want to
continue exporting other filesystems).

This patch allows the administrator to release all locks held by clients
accessing the client through a given server ip address, by echoing that
address to a new file, /proc/fs/nfsd/unlock_ip, as in:

shell> echo 10.1.1.2 > /proc/fs/nfsd/unlock_ip

The expected sequence of events can be:
1. Tear down the IP address
2. Unexport the path
3. Write IP to /proc/fs/nfsd/unlock_ip to unlock files
4. Signal peer to begin take-over.

For now we only support IPv4 addresses and NFSv2/v3 (NFSv4 locks are not
affected).

Also, if unmounting the filesystem is required, we assume at step 3 that
clients using the given server ip are the only clients holding locks on
the given filesystem; otherwise, an additional patch is required to
allow revoking all locks held by lockd on a given filesystem.
Signed-off-by: default avatarS. Wendy Cheng <wcheng@redhat.com>
Cc: Lon Hohberger  <lhh@redhat.com>
Cc: Christoph Hellwig <hch@lst.de>
Signed-off-by: default avatarJ. Bruce Fields <bfields@citi.umich.edu>

 fs/lockd/svcsubs.c          |   66 +++++++++++++++++++++++++++++++++++++++-----
 fs/nfsd/nfsctl.c            |   65 +++++++++++++++++++++++++++++++++++++++++++
 include/linux/lockd/lockd.h |    7 ++++
 3 files changed, 131 insertions(+), 7 deletions(-)
parent 9d91cdcc
...@@ -18,6 +18,8 @@ ...@@ -18,6 +18,8 @@
#include <linux/lockd/lockd.h> #include <linux/lockd/lockd.h>
#include <linux/lockd/share.h> #include <linux/lockd/share.h>
#include <linux/lockd/sm_inter.h> #include <linux/lockd/sm_inter.h>
#include <linux/module.h>
#include <linux/mount.h>
#define NLMDBG_FACILITY NLMDBG_SVCSUBS #define NLMDBG_FACILITY NLMDBG_SVCSUBS
...@@ -230,7 +232,7 @@ nlm_file_inuse(struct nlm_file *file) ...@@ -230,7 +232,7 @@ nlm_file_inuse(struct nlm_file *file)
* Loop over all files in the file table. * Loop over all files in the file table.
*/ */
static int static int
nlm_traverse_files(struct nlm_host *host, nlm_host_match_fn_t match) nlm_traverse_files(void *data, nlm_host_match_fn_t match)
{ {
struct hlist_node *pos, *next; struct hlist_node *pos, *next;
struct nlm_file *file; struct nlm_file *file;
...@@ -244,7 +246,7 @@ nlm_traverse_files(struct nlm_host *host, nlm_host_match_fn_t match) ...@@ -244,7 +246,7 @@ nlm_traverse_files(struct nlm_host *host, nlm_host_match_fn_t match)
/* Traverse locks, blocks and shares of this file /* Traverse locks, blocks and shares of this file
* and update file->f_locks count */ * and update file->f_locks count */
if (nlm_inspect_file(host, file, match)) if (nlm_inspect_file(data, file, match))
ret = 1; ret = 1;
mutex_lock(&nlm_file_mutex); mutex_lock(&nlm_file_mutex);
...@@ -303,21 +305,27 @@ nlm_release_file(struct nlm_file *file) ...@@ -303,21 +305,27 @@ nlm_release_file(struct nlm_file *file)
* Used by nlmsvc_invalidate_all * Used by nlmsvc_invalidate_all
*/ */
static int static int
nlmsvc_mark_host(struct nlm_host *host, struct nlm_host *dummy) nlmsvc_mark_host(void *data, struct nlm_host *dummy)
{ {
struct nlm_host *host = data;
host->h_inuse = 1; host->h_inuse = 1;
return 0; return 0;
} }
static int static int
nlmsvc_same_host(struct nlm_host *host, struct nlm_host *other) nlmsvc_same_host(void *data, struct nlm_host *other)
{ {
struct nlm_host *host = data;
return host == other; return host == other;
} }
static int static int
nlmsvc_is_client(struct nlm_host *host, struct nlm_host *dummy) nlmsvc_is_client(void *data, struct nlm_host *dummy)
{ {
struct nlm_host *host = data;
if (host->h_server) { if (host->h_server) {
/* we are destroying locks even though the client /* we are destroying locks even though the client
* hasn't asked us too, so don't unmonitor the * hasn't asked us too, so don't unmonitor the
...@@ -370,3 +378,21 @@ nlmsvc_invalidate_all(void) ...@@ -370,3 +378,21 @@ nlmsvc_invalidate_all(void)
*/ */
nlm_traverse_files(NULL, nlmsvc_is_client); nlm_traverse_files(NULL, nlmsvc_is_client);
} }
static int
nlmsvc_match_ip(void *datap, struct nlm_host *host)
{
__be32 *server_addr = datap;
return host->h_saddr.sin_addr.s_addr == *server_addr;
}
int
nlmsvc_unlock_all_by_ip(__be32 server_addr)
{
int ret;
ret = nlm_traverse_files(&server_addr, nlmsvc_match_ip);
return ret ? -EIO : 0;
}
EXPORT_SYMBOL_GPL(nlmsvc_unlock_all_by_ip);
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include <linux/seq_file.h> #include <linux/seq_file.h>
#include <linux/pagemap.h> #include <linux/pagemap.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/inet.h>
#include <linux/string.h> #include <linux/string.h>
#include <linux/smp_lock.h> #include <linux/smp_lock.h>
#include <linux/ctype.h> #include <linux/ctype.h>
...@@ -35,6 +36,7 @@ ...@@ -35,6 +36,7 @@
#include <linux/nfsd/cache.h> #include <linux/nfsd/cache.h>
#include <linux/nfsd/xdr.h> #include <linux/nfsd/xdr.h>
#include <linux/nfsd/syscall.h> #include <linux/nfsd/syscall.h>
#include <linux/lockd/lockd.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
#include <net/ipv6.h> #include <net/ipv6.h>
...@@ -53,6 +55,7 @@ enum { ...@@ -53,6 +55,7 @@ enum {
NFSD_Getfs, NFSD_Getfs,
NFSD_List, NFSD_List,
NFSD_Fh, NFSD_Fh,
NFSD_FO_UnlockIP,
NFSD_Threads, NFSD_Threads,
NFSD_Pool_Threads, NFSD_Pool_Threads,
NFSD_Versions, NFSD_Versions,
...@@ -89,6 +92,8 @@ static ssize_t write_leasetime(struct file *file, char *buf, size_t size); ...@@ -89,6 +92,8 @@ static ssize_t write_leasetime(struct file *file, char *buf, size_t size);
static ssize_t write_recoverydir(struct file *file, char *buf, size_t size); static ssize_t write_recoverydir(struct file *file, char *buf, size_t size);
#endif #endif
static ssize_t failover_unlock_ip(struct file *file, char *buf, size_t size);
static ssize_t (*write_op[])(struct file *, char *, size_t) = { static ssize_t (*write_op[])(struct file *, char *, size_t) = {
[NFSD_Svc] = write_svc, [NFSD_Svc] = write_svc,
[NFSD_Add] = write_add, [NFSD_Add] = write_add,
...@@ -98,6 +103,7 @@ static ssize_t (*write_op[])(struct file *, char *, size_t) = { ...@@ -98,6 +103,7 @@ static ssize_t (*write_op[])(struct file *, char *, size_t) = {
[NFSD_Getfd] = write_getfd, [NFSD_Getfd] = write_getfd,
[NFSD_Getfs] = write_getfs, [NFSD_Getfs] = write_getfs,
[NFSD_Fh] = write_filehandle, [NFSD_Fh] = write_filehandle,
[NFSD_FO_UnlockIP] = failover_unlock_ip,
[NFSD_Threads] = write_threads, [NFSD_Threads] = write_threads,
[NFSD_Pool_Threads] = write_pool_threads, [NFSD_Pool_Threads] = write_pool_threads,
[NFSD_Versions] = write_versions, [NFSD_Versions] = write_versions,
...@@ -298,6 +304,31 @@ static ssize_t write_getfd(struct file *file, char *buf, size_t size) ...@@ -298,6 +304,31 @@ static ssize_t write_getfd(struct file *file, char *buf, size_t size)
return err; return err;
} }
static ssize_t failover_unlock_ip(struct file *file, char *buf, size_t size)
{
__be32 server_ip;
char *fo_path, c;
int b1, b2, b3, b4;
/* sanity check */
if (size == 0)
return -EINVAL;
if (buf[size-1] != '\n')
return -EINVAL;
fo_path = buf;
if (qword_get(&buf, fo_path, size) < 0)
return -EINVAL;
/* get ipv4 address */
if (sscanf(fo_path, "%u.%u.%u.%u%c", &b1, &b2, &b3, &b4, &c) != 4)
return -EINVAL;
server_ip = htonl((((((b1<<8)|b2)<<8)|b3)<<8)|b4);
return nlmsvc_unlock_all_by_ip(server_ip);
}
static ssize_t write_filehandle(struct file *file, char *buf, size_t size) static ssize_t write_filehandle(struct file *file, char *buf, size_t size)
{ {
/* request is: /* request is:
...@@ -700,6 +731,8 @@ static int nfsd_fill_super(struct super_block * sb, void * data, int silent) ...@@ -700,6 +731,8 @@ static int nfsd_fill_super(struct super_block * sb, void * data, int silent)
[NFSD_Getfd] = {".getfd", &transaction_ops, S_IWUSR|S_IRUSR}, [NFSD_Getfd] = {".getfd", &transaction_ops, S_IWUSR|S_IRUSR},
[NFSD_Getfs] = {".getfs", &transaction_ops, S_IWUSR|S_IRUSR}, [NFSD_Getfs] = {".getfs", &transaction_ops, S_IWUSR|S_IRUSR},
[NFSD_List] = {"exports", &exports_operations, S_IRUGO}, [NFSD_List] = {"exports", &exports_operations, S_IRUGO},
[NFSD_FO_UnlockIP] = {"unlock_ip",
&transaction_ops, S_IWUSR|S_IRUSR},
[NFSD_Fh] = {"filehandle", &transaction_ops, S_IWUSR|S_IRUSR}, [NFSD_Fh] = {"filehandle", &transaction_ops, S_IWUSR|S_IRUSR},
[NFSD_Threads] = {"threads", &transaction_ops, S_IWUSR|S_IRUSR}, [NFSD_Threads] = {"threads", &transaction_ops, S_IWUSR|S_IRUSR},
[NFSD_Pool_Threads] = {"pool_threads", &transaction_ops, S_IWUSR|S_IRUSR}, [NFSD_Pool_Threads] = {"pool_threads", &transaction_ops, S_IWUSR|S_IRUSR},
......
...@@ -194,7 +194,7 @@ void nsm_release(struct nsm_handle *); ...@@ -194,7 +194,7 @@ void nsm_release(struct nsm_handle *);
* This is used in garbage collection and resource reclaim * This is used in garbage collection and resource reclaim
* A return value != 0 means destroy the lock/block/share * A return value != 0 means destroy the lock/block/share
*/ */
typedef int (*nlm_host_match_fn_t)(struct nlm_host *cur, struct nlm_host *ref); typedef int (*nlm_host_match_fn_t)(void *cur, struct nlm_host *ref);
/* /*
* Server-side lock handling * Server-side lock handling
...@@ -220,6 +220,11 @@ void nlmsvc_mark_resources(void); ...@@ -220,6 +220,11 @@ void nlmsvc_mark_resources(void);
void nlmsvc_free_host_resources(struct nlm_host *); void nlmsvc_free_host_resources(struct nlm_host *);
void nlmsvc_invalidate_all(void); void nlmsvc_invalidate_all(void);
/*
* Cluster failover support
*/
int nlmsvc_unlock_all_by_ip(__be32 server_addr);
static inline struct inode *nlmsvc_file_inode(struct nlm_file *file) static inline struct inode *nlmsvc_file_inode(struct nlm_file *file)
{ {
return file->f_file->f_path.dentry->d_inode; return file->f_file->f_path.dentry->d_inode;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment