diff options
author | NeilBrown <neilb@suse.de> | 2008-06-18 20:11:09 -0400 |
---|---|---|
committer | J. Bruce Fields <bfields@citi.umich.edu> | 2008-06-23 13:02:50 -0400 |
commit | 599eb3046a1380f31c65715f3940184c531c90cb (patch) | |
tree | 559719d700d80fc2acef1ff77506daa351cebba9 /fs/nfsd | |
parent | c7d106c90ec40a0e35a6960157b40f238627246e (diff) |
knfsd: nfsd: Handle ERESTARTSYS from syscalls.
OCFS2 can return -ERESTARTSYS from write requests (and possibly
elsewhere) if there is a signal pending.
If nfsd is shutdown (by sending a signal to each thread) while there
is still an IO load from the client, each thread could handle one last
request with a signal pending. This can result in -ERESTARTSYS
which is not understood by nfserrno() and so is reflected back to
the client as nfserr_io aka -EIO. This is wrong.
Instead, interpret ERESTARTSYS to mean "try again later" by returning
nfserr_jukebox. The client will resend and - if the server is
restarted - the write will (hopefully) be successful and everyone will
be happy.
The symptom that I narrowed down to this was:
copy a large file via NFS to an OCFS2 filesystem, and restart
the nfs server during the copy.
The 'cp' might get an -EIO, and the file will be corrupted -
presumably holes in the middle where writes appeared to fail.
Signed-off-by: Neil Brown <neilb@suse.de>
Signed-off-by: J. Bruce Fields <bfields@citi.umich.edu>
Diffstat (limited to 'fs/nfsd')
-rw-r--r-- | fs/nfsd/nfsproc.c | 1 |
1 files changed, 1 insertions, 0 deletions
diff --git a/fs/nfsd/nfsproc.c b/fs/nfsd/nfsproc.c index 6cfc96a12483..b5a20c486712 100644 --- a/fs/nfsd/nfsproc.c +++ b/fs/nfsd/nfsproc.c | |||
@@ -614,6 +614,7 @@ nfserrno (int errno) | |||
614 | #endif | 614 | #endif |
615 | { nfserr_stale, -ESTALE }, | 615 | { nfserr_stale, -ESTALE }, |
616 | { nfserr_jukebox, -ETIMEDOUT }, | 616 | { nfserr_jukebox, -ETIMEDOUT }, |
617 | { nfserr_jukebox, -ERESTARTSYS }, | ||
617 | { nfserr_dropit, -EAGAIN }, | 618 | { nfserr_dropit, -EAGAIN }, |
618 | { nfserr_dropit, -ENOMEM }, | 619 | { nfserr_dropit, -ENOMEM }, |
619 | { nfserr_badname, -ESRCH }, | 620 | { nfserr_badname, -ESRCH }, |