diff options
| author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-07-08 10:24:10 -0400 |
|---|---|---|
| committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-07-08 10:32:08 -0400 |
| commit | 4035c2487f179327fae87af3477659402b797584 (patch) | |
| tree | 098bb16500f88046493d2247359ca139b3f5611c | |
| parent | 6b16351acbd415e66ba16bf7d473ece1574cf0bc (diff) | |
NFS: Fix list manipulation snafus in fs/nfs/direct.c
Fix 2 bugs in nfs_direct_write_reschedule:
- The request needs to be removed from the 'reqs' list before it can
be added to 'failed'.
- Fix an infinite loop if the 'failed' list is non-empty.
Reported-by: Julia Lawall <julia.lawall@lip6.fr>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
| -rw-r--r-- | fs/nfs/direct.c | 6 |
1 files changed, 5 insertions, 1 deletions
diff --git a/fs/nfs/direct.c b/fs/nfs/direct.c index 9a4cbfc85d81..48253372ab1d 100644 --- a/fs/nfs/direct.c +++ b/fs/nfs/direct.c | |||
| @@ -484,6 +484,7 @@ static void nfs_direct_write_reschedule(struct nfs_direct_req *dreq) | |||
| 484 | 484 | ||
| 485 | list_for_each_entry_safe(req, tmp, &reqs, wb_list) { | 485 | list_for_each_entry_safe(req, tmp, &reqs, wb_list) { |
| 486 | if (!nfs_pageio_add_request(&desc, req)) { | 486 | if (!nfs_pageio_add_request(&desc, req)) { |
| 487 | nfs_list_remove_request(req); | ||
| 487 | nfs_list_add_request(req, &failed); | 488 | nfs_list_add_request(req, &failed); |
| 488 | spin_lock(cinfo.lock); | 489 | spin_lock(cinfo.lock); |
| 489 | dreq->flags = 0; | 490 | dreq->flags = 0; |
| @@ -494,8 +495,11 @@ static void nfs_direct_write_reschedule(struct nfs_direct_req *dreq) | |||
| 494 | } | 495 | } |
| 495 | nfs_pageio_complete(&desc); | 496 | nfs_pageio_complete(&desc); |
| 496 | 497 | ||
| 497 | while (!list_empty(&failed)) | 498 | while (!list_empty(&failed)) { |
| 499 | req = nfs_list_entry(failed.next); | ||
| 500 | nfs_list_remove_request(req); | ||
| 498 | nfs_unlock_and_release_request(req); | 501 | nfs_unlock_and_release_request(req); |
| 502 | } | ||
| 499 | 503 | ||
| 500 | if (put_dreq(dreq)) | 504 | if (put_dreq(dreq)) |
| 501 | nfs_direct_write_complete(dreq, dreq->inode); | 505 | nfs_direct_write_complete(dreq, dreq->inode); |
