diff options
author | Frederic Weisbecker <fweisbec@gmail.com> | 2009-05-08 08:21:33 -0400 |
---|---|---|
committer | Frederic Weisbecker <fweisbec@gmail.com> | 2009-09-14 01:18:19 -0400 |
commit | 09eb47a7c52ad535aafca889e0b936c445c375ce (patch) | |
tree | 81ebee49cb84f29e0aa3d884f6ed9c23ab044218 | |
parent | b1c839bb2d8d6f1f6bf48f5c657752b4963f88f8 (diff) |
kill-the-bkl/reiserfs: reduce number of contentions in search_by_key()
search_by_key() is a central function in reiserfs which searches
the patch in the fs tree from the root to a node given its key.
It is the function that is most requesting the write lock
because it's a path very often used.
Also we forget to release the lock while reading the next tree node,
making us holding the lock in a wasteful way.
Then we release the lock while reading the current node and its childs,
all-in-one. It should be safe because we have a reference to these
blocks and even if we read a block that will be concurrently changed,
we have an fs_changed check later that will make us retry the path from
the root.
[ Impact: release the write lock while unused in a hot path ]
Cc: Jeff Mahoney <jeffm@suse.com>
Cc: Chris Mason <chris.mason@oracle.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Alexander Beregalov <a.beregalov@gmail.com>
Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
-rw-r--r-- | fs/reiserfs/stree.c | 12 |
1 files changed, 11 insertions, 1 deletions
diff --git a/fs/reiserfs/stree.c b/fs/reiserfs/stree.c index 6ddcecb4e8ab..960c9114f6d3 100644 --- a/fs/reiserfs/stree.c +++ b/fs/reiserfs/stree.c | |||
@@ -529,6 +529,14 @@ static void search_by_key_reada(struct super_block *s, | |||
529 | for (i = 0; i < num; i++) { | 529 | for (i = 0; i < num; i++) { |
530 | bh[i] = sb_getblk(s, b[i]); | 530 | bh[i] = sb_getblk(s, b[i]); |
531 | } | 531 | } |
532 | /* | ||
533 | * We are going to read some blocks on which we | ||
534 | * have a reference. It's safe, though we might be | ||
535 | * reading blocks concurrently changed if we release | ||
536 | * the lock. But it's still fine because we check later | ||
537 | * if the tree changed | ||
538 | */ | ||
539 | reiserfs_write_unlock(s); | ||
532 | for (j = 0; j < i; j++) { | 540 | for (j = 0; j < i; j++) { |
533 | /* | 541 | /* |
534 | * note, this needs attention if we are getting rid of the BKL | 542 | * note, this needs attention if we are getting rid of the BKL |
@@ -626,10 +634,12 @@ int search_by_key(struct super_block *sb, const struct cpu_key *key, /* Key to s | |||
626 | if ((bh = last_element->pe_buffer = | 634 | if ((bh = last_element->pe_buffer = |
627 | sb_getblk(sb, block_number))) { | 635 | sb_getblk(sb, block_number))) { |
628 | if (!buffer_uptodate(bh) && reada_count > 1) | 636 | if (!buffer_uptodate(bh) && reada_count > 1) |
637 | /* will unlock the write lock */ | ||
629 | search_by_key_reada(sb, reada_bh, | 638 | search_by_key_reada(sb, reada_bh, |
630 | reada_blocks, reada_count); | 639 | reada_blocks, reada_count); |
640 | else | ||
641 | reiserfs_write_unlock(sb); | ||
631 | ll_rw_block(READ, 1, &bh); | 642 | ll_rw_block(READ, 1, &bh); |
632 | reiserfs_write_unlock(sb); | ||
633 | wait_on_buffer(bh); | 643 | wait_on_buffer(bh); |
634 | reiserfs_write_lock(sb); | 644 | reiserfs_write_lock(sb); |
635 | if (!buffer_uptodate(bh)) | 645 | if (!buffer_uptodate(bh)) |