diff options
author | David Howells <dhowells@redhat.com> | 2012-12-05 08:34:45 -0500 |
---|---|---|
committer | David Howells <dhowells@redhat.com> | 2012-12-20 17:07:40 -0500 |
commit | b4cf1e08c8ac95eff65faa53904f7f13ac78194b (patch) | |
tree | 3a79581dc049450dd0a448fd618f6097a69a8696 | |
parent | de242c0b8b365a9e348bf53143e18e9d8c9cfae8 (diff) |
CacheFiles: Add missing retrieval completions
CacheFiles is missing some calls to fscache_retrieval_complete() in the error
handling/collision paths of its reader functions.
This can be seen by the following assertion tripping in fscache_put_operation()
whereby the operation being destroyed is still in the in-progress state and has
not been cancelled or completed:
FS-Cache: Assertion failed
3 == 5 is false
------------[ cut here ]------------
kernel BUG at fs/fscache/operation.c:408!
invalid opcode: 0000 [#1] SMP
CPU 2
Modules linked in: xfs ioatdma dca loop joydev evdev
psmouse dcdbas pcspkr serio_raw i5000_edac edac_core i5k_amb shpchp
pci_hotplug sg sr_mod]
Pid: 8062, comm: httpd Not tainted 3.1.0-rc8 #1 Dell Inc. PowerEdge 1950/0DT097
RIP: 0010:[<ffffffff81197b24>] [<ffffffff81197b24>] fscache_put_operation+0x304/0x330
RSP: 0018:ffff880062f739d8 EFLAGS: 00010296
RAX: 0000000000000025 RBX: ffff8800c5122e84 RCX: ffffffff81ddf040
RDX: 00000000ffffffff RSI: 0000000000000082 RDI: ffffffff81ddef30
RBP: ffff880062f739f8 R08: 0000000000000005 R09: 0000000000000000
R10: 0000000000000000 R11: 0000000000000003 R12: ffff8800c5122e40
R13: ffff880037a2cd20 R14: ffff880087c7a058 R15: ffff880087c7a000
FS: 00007f63dcf636e0(0000) GS:ffff88022fc80000(0000) knlGS:0000000000000000
CS: 0010 DS: 0000 ES: 0000 CR0: 0000000080050033
CR2: 00007f0c0a91f000 CR3: 0000000062ec2000 CR4: 00000000000006e0
DR0: 0000000000000000 DR1: 0000000000000000 DR2: 0000000000000000
DR3: 0000000000000000 DR6: 00000000ffff0ff0 DR7: 0000000000000400
Process httpd (pid: 8062, threadinfo ffff880062f72000, task ffff880087e58000)
Stack:
ffff880062f73bf8 0000000000000000 ffff880062f73bf8 ffff880037a2cd20
ffff880062f73a68 ffffffff8119aa7e ffff88006540e000 ffff880062f73ad4
ffff88008e9a4308 ffff880037a2cd20 ffff880062f73a48 ffff8800c5122e40
Call Trace:
[<ffffffff8119aa7e>] __fscache_read_or_alloc_pages+0x1fe/0x530
[<ffffffff81250780>] __nfs_readpages_from_fscache+0x70/0x1c0
[<ffffffff8123142a>] nfs_readpages+0xca/0x1e0
[<ffffffff815f3c06>] ? rpc_do_put_task+0x36/0x50
[<ffffffff8122755b>] ? alloc_nfs_open_context+0x4b/0x110
[<ffffffff815ecd1a>] ? rpc_call_sync+0x5a/0x70
[<ffffffff810e7e9a>] __do_page_cache_readahead+0x1ca/0x270
[<ffffffff810e7f61>] ra_submit+0x21/0x30
[<ffffffff810e818d>] ondemand_readahead+0x11d/0x250
[<ffffffff810e83b6>] page_cache_sync_readahead+0x36/0x60
[<ffffffff810dffa4>] generic_file_aio_read+0x454/0x770
[<ffffffff81224ce1>] nfs_file_read+0xe1/0x130
[<ffffffff81121bd9>] do_sync_read+0xd9/0x120
[<ffffffff8114088f>] ? mntput+0x1f/0x40
[<ffffffff811238cb>] ? fput+0x1cb/0x260
[<ffffffff81122938>] vfs_read+0xc8/0x180
[<ffffffff81122af5>] sys_read+0x55/0x90
Reported-by: Mark Moseley <moseleymark@gmail.com>
Signed-off-by: David Howells <dhowells@redhat.com>
-rw-r--r-- | fs/cachefiles/rdwr.c | 14 | ||||
-rw-r--r-- | fs/fscache/page.c | 2 |
2 files changed, 12 insertions, 4 deletions
diff --git a/fs/cachefiles/rdwr.c b/fs/cachefiles/rdwr.c index 2c994885520a..480992259707 100644 --- a/fs/cachefiles/rdwr.c +++ b/fs/cachefiles/rdwr.c | |||
@@ -361,8 +361,10 @@ out: | |||
361 | 361 | ||
362 | read_error: | 362 | read_error: |
363 | _debug("read error %d", ret); | 363 | _debug("read error %d", ret); |
364 | if (ret == -ENOMEM) | 364 | if (ret == -ENOMEM) { |
365 | fscache_retrieval_complete(op, 1); | ||
365 | goto out; | 366 | goto out; |
367 | } | ||
366 | io_error: | 368 | io_error: |
367 | cachefiles_io_error_obj(object, "Page read error on backing file"); | 369 | cachefiles_io_error_obj(object, "Page read error on backing file"); |
368 | fscache_retrieval_complete(op, 1); | 370 | fscache_retrieval_complete(op, 1); |
@@ -551,6 +553,7 @@ static int cachefiles_read_backing_file(struct cachefiles_object *object, | |||
551 | if (ret < 0) { | 553 | if (ret < 0) { |
552 | if (ret == -EEXIST) { | 554 | if (ret == -EEXIST) { |
553 | page_cache_release(netpage); | 555 | page_cache_release(netpage); |
556 | fscache_retrieval_complete(op, 1); | ||
554 | continue; | 557 | continue; |
555 | } | 558 | } |
556 | goto nomem; | 559 | goto nomem; |
@@ -627,6 +630,7 @@ static int cachefiles_read_backing_file(struct cachefiles_object *object, | |||
627 | if (ret < 0) { | 630 | if (ret < 0) { |
628 | if (ret == -EEXIST) { | 631 | if (ret == -EEXIST) { |
629 | page_cache_release(netpage); | 632 | page_cache_release(netpage); |
633 | fscache_retrieval_complete(op, 1); | ||
630 | continue; | 634 | continue; |
631 | } | 635 | } |
632 | goto nomem; | 636 | goto nomem; |
@@ -645,9 +649,9 @@ static int cachefiles_read_backing_file(struct cachefiles_object *object, | |||
645 | 649 | ||
646 | /* the netpage is unlocked and marked up to date here */ | 650 | /* the netpage is unlocked and marked up to date here */ |
647 | fscache_end_io(op, netpage, 0); | 651 | fscache_end_io(op, netpage, 0); |
648 | fscache_retrieval_complete(op, 1); | ||
649 | page_cache_release(netpage); | 652 | page_cache_release(netpage); |
650 | netpage = NULL; | 653 | netpage = NULL; |
654 | fscache_retrieval_complete(op, 1); | ||
651 | continue; | 655 | continue; |
652 | } | 656 | } |
653 | 657 | ||
@@ -682,15 +686,17 @@ out: | |||
682 | nomem: | 686 | nomem: |
683 | _debug("nomem"); | 687 | _debug("nomem"); |
684 | ret = -ENOMEM; | 688 | ret = -ENOMEM; |
685 | goto out; | 689 | goto record_page_complete; |
686 | 690 | ||
687 | read_error: | 691 | read_error: |
688 | _debug("read error %d", ret); | 692 | _debug("read error %d", ret); |
689 | if (ret == -ENOMEM) | 693 | if (ret == -ENOMEM) |
690 | goto out; | 694 | goto record_page_complete; |
691 | io_error: | 695 | io_error: |
692 | cachefiles_io_error_obj(object, "Page read error on backing file"); | 696 | cachefiles_io_error_obj(object, "Page read error on backing file"); |
693 | ret = -ENOBUFS; | 697 | ret = -ENOBUFS; |
698 | record_page_complete: | ||
699 | fscache_retrieval_complete(op, 1); | ||
694 | goto out; | 700 | goto out; |
695 | } | 701 | } |
696 | 702 | ||
diff --git a/fs/fscache/page.c b/fs/fscache/page.c index 7bf9d2557052..4dbbca162620 100644 --- a/fs/fscache/page.c +++ b/fs/fscache/page.c | |||
@@ -329,6 +329,8 @@ check_if_dead: | |||
329 | return -ENOBUFS; | 329 | return -ENOBUFS; |
330 | } | 330 | } |
331 | if (unlikely(fscache_object_is_dead(object))) { | 331 | if (unlikely(fscache_object_is_dead(object))) { |
332 | pr_err("%s() = -ENOBUFS [obj dead %d]", __func__, op->op.state); | ||
333 | fscache_cancel_op(&op->op); | ||
332 | fscache_stat(stat_object_dead); | 334 | fscache_stat(stat_object_dead); |
333 | return -ENOBUFS; | 335 | return -ENOBUFS; |
334 | } | 336 | } |