Skip to content

Commit 0b48d42

Browse files
committed
Merge branch 'for-3.3' of git://linux-nfs.org/~bfields/linux
* 'for-3.3' of git://linux-nfs.org/~bfields/linux: (31 commits) nfsd4: nfsd4_create_clid_dir return value is unused NFSD: Change name of extended attribute containing junction svcrpc: don't revert to SVC_POOL_DEFAULT on nfsd shutdown svcrpc: fix double-free on shutdown of nfsd after changing pool mode nfsd4: be forgiving in the absence of the recovery directory nfsd4: fix spurious 4.1 post-reboot failures NFSD: forget_delegations should use list_for_each_entry_safe NFSD: Only reinitilize the recall_lru list under the recall lock nfsd4: initialize special stateid's at compile time NFSd: use network-namespace-aware cache registering routines SUNRPC: create svc_xprt in proper network namespace svcrpc: update outdated BKL comment nfsd41: allow non-reclaim open-by-fh's in 4.1 svcrpc: avoid memory-corruption on pool shutdown svcrpc: destroy server sockets all at once svcrpc: make svc_delete_xprt static nfsd: Fix oops when parsing a 0 length export nfsd4: Use kmemdup rather than duplicating its implementation nfsd4: add a separate (lockowner, inode) lookup nfsd4: fix CONFIG_NFSD_FAULT_INJECTION compile error ...
2 parents 8e63dd6 + 7a6ef8c commit 0b48d42

26 files changed

+624
-169
lines changed

CREDITS

+5
Original file line numberDiff line numberDiff line change
@@ -514,6 +514,11 @@ S: Bessemerstraat 21
514514
S: Amsterdam
515515
S: The Netherlands
516516

517+
N: NeilBrown
518+
519+
P: 4096R/566281B9 1BC6 29EB D390 D870 7B5F 497A 39EC 9EDD 5662 81B9
520+
D: NFSD Maintainer 2000-2007
521+
517522
N: Zach Brown
518523
519524
D: maestro pci sound

Documentation/filesystems/nfs/00-INDEX

+2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22
- this file (nfs-related documentation).
33
Exporting
44
- explanation of how to make filesystems exportable.
5+
fault_injection.txt
6+
- information for using fault injection on the server
57
knfsd-stats.txt
68
- statistics which the NFS server makes available to user space.
79
nfs.txt
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
2+
Fault Injection
3+
===============
4+
Fault injection is a method for forcing errors that may not normally occur, or
5+
may be difficult to reproduce. Forcing these errors in a controlled environment
6+
can help the developer find and fix bugs before their code is shipped in a
7+
production system. Injecting an error on the Linux NFS server will allow us to
8+
observe how the client reacts and if it manages to recover its state correctly.
9+
10+
NFSD_FAULT_INJECTION must be selected when configuring the kernel to use this
11+
feature.
12+
13+
14+
Using Fault Injection
15+
=====================
16+
On the client, mount the fault injection server through NFS v4.0+ and do some
17+
work over NFS (open files, take locks, ...).
18+
19+
On the server, mount the debugfs filesystem to <debug_dir> and ls
20+
<debug_dir>/nfsd. This will show a list of files that will be used for
21+
injecting faults on the NFS server. As root, write a number n to the file
22+
corresponding to the action you want the server to take. The server will then
23+
process the first n items it finds. So if you want to forget 5 locks, echo '5'
24+
to <debug_dir>/nfsd/forget_locks. A value of 0 will tell the server to forget
25+
all corresponding items. A log message will be created containing the number
26+
of items forgotten (check dmesg).
27+
28+
Go back to work on the client and check if the client recovered from the error
29+
correctly.
30+
31+
32+
Available Faults
33+
================
34+
forget_clients:
35+
The NFS server keeps a list of clients that have placed a mount call. If
36+
this list is cleared, the server will have no knowledge of who the client
37+
is, forcing the client to reauthenticate with the server.
38+
39+
forget_openowners:
40+
The NFS server keeps a list of what files are currently opened and who
41+
they were opened by. Clearing this list will force the client to reopen
42+
its files.
43+
44+
forget_locks:
45+
The NFS server keeps a list of what files are currently locked in the VFS.
46+
Clearing this list will force the client to reclaim its locks (files are
47+
unlocked through the VFS as they are cleared from this list).
48+
49+
forget_delegations:
50+
A delegation is used to assure the client that a file, or part of a file,
51+
has not changed since the delegation was awarded. Clearing this list will
52+
force the client to reaquire its delegation before accessing the file
53+
again.
54+
55+
recall_delegations:
56+
Delegations can be recalled by the server when another client attempts to
57+
access a file. This test will notify the client that its delegation has
58+
been revoked, forcing the client to reaquire the delegation before using
59+
the file again.
60+
61+
62+
tools/nfs/inject_faults.sh script
63+
=================================
64+
This script has been created to ease the fault injection process. This script
65+
will detect the mounted debugfs directory and write to the files located there
66+
based on the arguments passed by the user. For example, running
67+
`inject_faults.sh forget_locks 1` as root will instruct the server to forget
68+
one lock. Running `inject_faults forget_locks` will instruct the server to
69+
forgetall locks.

MAINTAINERS

-1
Original file line numberDiff line numberDiff line change
@@ -3775,7 +3775,6 @@ S: Odd Fixes
37753775

37763776
KERNEL NFSD, SUNRPC, AND LOCKD SERVERS
37773777
M: "J. Bruce Fields" <[email protected]>
3778-
M: Neil Brown <[email protected]>
37793778
37803779
W: http://nfs.sourceforge.net/
37813780
S: Supported

fs/nfsd/Kconfig

+10
Original file line numberDiff line numberDiff line change
@@ -80,3 +80,13 @@ config NFSD_V4
8080
available from http://linux-nfs.org/.
8181

8282
If unsure, say N.
83+
84+
config NFSD_FAULT_INJECTION
85+
bool "NFS server manual fault injection"
86+
depends on NFSD_V4 && DEBUG_KERNEL
87+
help
88+
This option enables support for manually injecting faults
89+
into the NFS server. This is intended to be used for
90+
testing error recovery on the NFS client.
91+
92+
If unsure, say N.

fs/nfsd/Makefile

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ obj-$(CONFIG_NFSD) += nfsd.o
66

77
nfsd-y := nfssvc.o nfsctl.o nfsproc.o nfsfh.o vfs.o \
88
export.o auth.o lockd.o nfscache.o nfsxdr.o stats.o
9+
nfsd-$(CONFIG_NFSD_FAULT_INJECTION) += fault_inject.o
910
nfsd-$(CONFIG_NFSD_V2_ACL) += nfs2acl.o
1011
nfsd-$(CONFIG_NFSD_V3) += nfs3proc.o nfs3xdr.o
1112
nfsd-$(CONFIG_NFSD_V3_ACL) += nfs3acl.o

fs/nfsd/export.c

+6-6
Original file line numberDiff line numberDiff line change
@@ -87,7 +87,7 @@ static int expkey_parse(struct cache_detail *cd, char *mesg, int mlen)
8787
struct svc_expkey key;
8888
struct svc_expkey *ek = NULL;
8989

90-
if (mesg[mlen-1] != '\n')
90+
if (mlen < 1 || mesg[mlen-1] != '\n')
9191
return -EINVAL;
9292
mesg[mlen-1] = 0;
9393

@@ -1226,12 +1226,12 @@ nfsd_export_init(void)
12261226
int rv;
12271227
dprintk("nfsd: initializing export module.\n");
12281228

1229-
rv = cache_register(&svc_export_cache);
1229+
rv = cache_register_net(&svc_export_cache, &init_net);
12301230
if (rv)
12311231
return rv;
1232-
rv = cache_register(&svc_expkey_cache);
1232+
rv = cache_register_net(&svc_expkey_cache, &init_net);
12331233
if (rv)
1234-
cache_unregister(&svc_export_cache);
1234+
cache_unregister_net(&svc_export_cache, &init_net);
12351235
return rv;
12361236

12371237
}
@@ -1255,8 +1255,8 @@ nfsd_export_shutdown(void)
12551255

12561256
dprintk("nfsd: shutting down export module.\n");
12571257

1258-
cache_unregister(&svc_expkey_cache);
1259-
cache_unregister(&svc_export_cache);
1258+
cache_unregister_net(&svc_expkey_cache, &init_net);
1259+
cache_unregister_net(&svc_export_cache, &init_net);
12601260
svcauth_unix_purge();
12611261

12621262
dprintk("nfsd: export shutdown complete.\n");

fs/nfsd/fault_inject.c

+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
/*
2+
* Copyright (c) 2011 Bryan Schumaker <[email protected]>
3+
*
4+
* Uses debugfs to create fault injection points for client testing
5+
*/
6+
7+
#include <linux/types.h>
8+
#include <linux/fs.h>
9+
#include <linux/debugfs.h>
10+
#include <linux/module.h>
11+
12+
#include "state.h"
13+
#include "fault_inject.h"
14+
15+
struct nfsd_fault_inject_op {
16+
char *file;
17+
void (*func)(u64);
18+
};
19+
20+
static struct nfsd_fault_inject_op inject_ops[] = {
21+
{
22+
.file = "forget_clients",
23+
.func = nfsd_forget_clients,
24+
},
25+
{
26+
.file = "forget_locks",
27+
.func = nfsd_forget_locks,
28+
},
29+
{
30+
.file = "forget_openowners",
31+
.func = nfsd_forget_openowners,
32+
},
33+
{
34+
.file = "forget_delegations",
35+
.func = nfsd_forget_delegations,
36+
},
37+
{
38+
.file = "recall_delegations",
39+
.func = nfsd_recall_delegations,
40+
},
41+
};
42+
43+
static long int NUM_INJECT_OPS = sizeof(inject_ops) / sizeof(struct nfsd_fault_inject_op);
44+
static struct dentry *debug_dir;
45+
46+
static int nfsd_inject_set(void *op_ptr, u64 val)
47+
{
48+
struct nfsd_fault_inject_op *op = op_ptr;
49+
50+
if (val == 0)
51+
printk(KERN_INFO "NFSD Fault Injection: %s (all)", op->file);
52+
else
53+
printk(KERN_INFO "NFSD Fault Injection: %s (n = %llu)", op->file, val);
54+
55+
op->func(val);
56+
return 0;
57+
}
58+
59+
static int nfsd_inject_get(void *data, u64 *val)
60+
{
61+
return 0;
62+
}
63+
64+
DEFINE_SIMPLE_ATTRIBUTE(fops_nfsd, nfsd_inject_get, nfsd_inject_set, "%llu\n");
65+
66+
void nfsd_fault_inject_cleanup(void)
67+
{
68+
debugfs_remove_recursive(debug_dir);
69+
}
70+
71+
int nfsd_fault_inject_init(void)
72+
{
73+
unsigned int i;
74+
struct nfsd_fault_inject_op *op;
75+
mode_t mode = S_IFREG | S_IRUSR | S_IWUSR;
76+
77+
debug_dir = debugfs_create_dir("nfsd", NULL);
78+
if (!debug_dir)
79+
goto fail;
80+
81+
for (i = 0; i < NUM_INJECT_OPS; i++) {
82+
op = &inject_ops[i];
83+
if (!debugfs_create_file(op->file, mode, debug_dir, op, &fops_nfsd))
84+
goto fail;
85+
}
86+
return 0;
87+
88+
fail:
89+
nfsd_fault_inject_cleanup();
90+
return -ENOMEM;
91+
}

fs/nfsd/fault_inject.h

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
/*
2+
* Copyright (c) 2011 Bryan Schumaker <[email protected]>
3+
*
4+
* Function definitions for fault injection
5+
*/
6+
7+
#ifndef LINUX_NFSD_FAULT_INJECT_H
8+
#define LINUX_NFSD_FAULT_INJECT_H
9+
10+
#ifdef CONFIG_NFSD_FAULT_INJECTION
11+
int nfsd_fault_inject_init(void);
12+
void nfsd_fault_inject_cleanup(void);
13+
void nfsd_forget_clients(u64);
14+
void nfsd_forget_locks(u64);
15+
void nfsd_forget_openowners(u64);
16+
void nfsd_forget_delegations(u64);
17+
void nfsd_recall_delegations(u64);
18+
#else /* CONFIG_NFSD_FAULT_INJECTION */
19+
static inline int nfsd_fault_inject_init(void) { return 0; }
20+
static inline void nfsd_fault_inject_cleanup(void) {}
21+
static inline void nfsd_forget_clients(u64 num) {}
22+
static inline void nfsd_forget_locks(u64 num) {}
23+
static inline void nfsd_forget_openowners(u64 num) {}
24+
static inline void nfsd_forget_delegations(u64 num) {}
25+
static inline void nfsd_recall_delegations(u64 num) {}
26+
#endif /* CONFIG_NFSD_FAULT_INJECTION */
27+
28+
#endif /* LINUX_NFSD_FAULT_INJECT_H */

fs/nfsd/nfs4idmap.c

+6-5
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
#include <linux/seq_file.h>
3737
#include <linux/sched.h>
3838
#include <linux/slab.h>
39+
#include <net/net_namespace.h>
3940
#include "idmap.h"
4041
#include "nfsd.h"
4142

@@ -466,20 +467,20 @@ nfsd_idmap_init(void)
466467
{
467468
int rv;
468469

469-
rv = cache_register(&idtoname_cache);
470+
rv = cache_register_net(&idtoname_cache, &init_net);
470471
if (rv)
471472
return rv;
472-
rv = cache_register(&nametoid_cache);
473+
rv = cache_register_net(&nametoid_cache, &init_net);
473474
if (rv)
474-
cache_unregister(&idtoname_cache);
475+
cache_unregister_net(&idtoname_cache, &init_net);
475476
return rv;
476477
}
477478

478479
void
479480
nfsd_idmap_shutdown(void)
480481
{
481-
cache_unregister(&idtoname_cache);
482-
cache_unregister(&nametoid_cache);
482+
cache_unregister_net(&idtoname_cache, &init_net);
483+
cache_unregister_net(&nametoid_cache, &init_net);
483484
}
484485

485486
static int

fs/nfsd/nfs4proc.c

+3-4
Original file line numberDiff line numberDiff line change
@@ -266,10 +266,6 @@ do_open_fhandle(struct svc_rqst *rqstp, struct svc_fh *current_fh, struct nfsd4_
266266
{
267267
__be32 status;
268268

269-
/* Only reclaims from previously confirmed clients are valid */
270-
if ((status = nfs4_check_open_reclaim(&open->op_clientid)))
271-
return status;
272-
273269
/* We don't know the target directory, and therefore can not
274270
* set the change info
275271
*/
@@ -373,6 +369,9 @@ nfsd4_open(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
373369
break;
374370
case NFS4_OPEN_CLAIM_PREVIOUS:
375371
open->op_openowner->oo_flags |= NFS4_OO_CONFIRMED;
372+
status = nfs4_check_open_reclaim(&open->op_clientid);
373+
if (status)
374+
goto out;
376375
case NFS4_OPEN_CLAIM_FH:
377376
case NFS4_OPEN_CLAIM_DELEG_CUR_FH:
378377
status = do_open_fhandle(rqstp, &cstate->current_fh,

fs/nfsd/nfs4recover.c

+14-8
Original file line numberDiff line numberDiff line change
@@ -117,8 +117,7 @@ nfs4_make_rec_clidname(char *dname, struct xdr_netobj *clname)
117117
return status;
118118
}
119119

120-
int
121-
nfsd4_create_clid_dir(struct nfs4_client *clp)
120+
void nfsd4_create_clid_dir(struct nfs4_client *clp)
122121
{
123122
const struct cred *original_cred;
124123
char *dname = clp->cl_recdir;
@@ -127,13 +126,14 @@ nfsd4_create_clid_dir(struct nfs4_client *clp)
127126

128127
dprintk("NFSD: nfsd4_create_clid_dir for \"%s\"\n", dname);
129128

130-
if (!rec_file || clp->cl_firststate)
131-
return 0;
132-
129+
if (clp->cl_firststate)
130+
return;
133131
clp->cl_firststate = 1;
132+
if (!rec_file)
133+
return;
134134
status = nfs4_save_creds(&original_cred);
135135
if (status < 0)
136-
return status;
136+
return;
137137

138138
dir = rec_file->f_path.dentry;
139139
/* lock the parent */
@@ -144,8 +144,15 @@ nfsd4_create_clid_dir(struct nfs4_client *clp)
144144
status = PTR_ERR(dentry);
145145
goto out_unlock;
146146
}
147-
status = -EEXIST;
148147
if (dentry->d_inode)
148+
/*
149+
* In the 4.1 case, where we're called from
150+
* reclaim_complete(), records from the previous reboot
151+
* may still be left, so this is OK.
152+
*
153+
* In the 4.0 case, we should never get here; but we may
154+
* as well be forgiving and just succeed silently.
155+
*/
149156
goto out_put;
150157
status = mnt_want_write_file(rec_file);
151158
if (status)
@@ -164,7 +171,6 @@ nfsd4_create_clid_dir(struct nfs4_client *clp)
164171
" and is writeable", status,
165172
user_recovery_dirname);
166173
nfs4_reset_creds(original_cred);
167-
return status;
168174
}
169175

170176
typedef int (recdir_func)(struct dentry *, struct dentry *);

0 commit comments

Comments
 (0)