diff options
author | Goldwyn Rodrigues <rgoldwyn@suse.de> | 2014-04-03 17:46:59 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2014-04-03 19:20:55 -0400 |
commit | 8ed6b23709b346f7bfc1edab47003a205a6a9f69 (patch) | |
tree | 61c8187bb57a42165ef838e21a268a346ab3af16 /fs/ocfs2/dcache.c | |
parent | 84d86f83f9d0e8431a3c9eae4c47e9d7ff49a411 (diff) |
ocfs2: revert iput deferring code in ocfs2_drop_dentry_lock
The following patches are reverted in this patch because these patches
caused performance regression in the remote unlink() calls.
ea455f8ab683 - ocfs2: Push out dropping of dentry lock to ocfs2_wq
f7b1aa69be13 - ocfs2: Fix deadlock on umount
5fd131893793 - ocfs2: Don't oops in ocfs2_kill_sb on a failed mount
Previous patches in this series removed the possible deadlocks from
downconvert thread so the above patches shouldn't be needed anymore.
The regression is caused because these patches delay the iput() in case
of dentry unlocks. This also delays the unlocking of the open lockres.
The open lockresource is required to test if the inode can be wiped from
disk or not. When the deleting node does not get the open lock, it
marks it as orphan (even though it is not in use by another
node/process) and causes a journal checkpoint. This delays operations
following the inode eviction. This also moves the inode to the orphaned
inode which further causes more I/O and a lot of unneccessary orphans.
The following script can be used to generate the load causing issues:
declare -a create
declare -a remove
declare -a iterations=(1 2 4 8 16 32 64 128 256 512 1024 2048 4096 8192 16384)
unique="`mktemp -u XXXXX`"
script="/tmp/idontknow-${unique}.sh"
cat <<EOF > "${script}"
for n in {1..8}; do mkdir -p test/dir\${n}
eval touch test/dir\${n}/foo{1.."\$1"}
done
EOF
chmod 700 "${script}"
function fcreate ()
{
exec 2>&1 /usr/bin/time --format=%E "${script}" "$1"
}
function fremove ()
{
exec 2>&1 /usr/bin/time --format=%E ssh node2 "cd `pwd`; rm -Rf test*"
}
function fcp ()
{
exec 2>&1 /usr/bin/time --format=%E ssh node3 "cd `pwd`; cp -R test test.new"
}
echo -------------------------------------------------
echo "| # files | create #s | copy #s | remove #s |"
echo -------------------------------------------------
for ((x=0; x < ${#iterations[*]} ; x++)) do
create[$x]="`fcreate ${iterations[$x]}`"
copy[$x]="`fcp ${iterations[$x]}`"
remove[$x]="`fremove`"
printf "| %8d | %9s | %9s | %9s |\n" ${iterations[$x]} ${create[$x]} ${copy[$x]} ${remove[$x]}
done
rm "${script}"
echo "------------------------"
Signed-off-by: Srinivas Eeda <srinivas.eeda@oracle.com>
Signed-off-by: Goldwyn Rodrigues <rgoldwyn@suse.com>
Signed-off-by: Jan Kara <jack@suse.cz>
Reviewed-by: Mark Fasheh <mfasheh@suse.de>
Cc: Joel Becker <jlbec@evilplan.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'fs/ocfs2/dcache.c')
-rw-r--r-- | fs/ocfs2/dcache.c | 61 |
1 files changed, 3 insertions, 58 deletions
diff --git a/fs/ocfs2/dcache.c b/fs/ocfs2/dcache.c index 0d3a97d2d5f6..e2e05a106beb 100644 --- a/fs/ocfs2/dcache.c +++ b/fs/ocfs2/dcache.c | |||
@@ -37,7 +37,6 @@ | |||
37 | #include "dlmglue.h" | 37 | #include "dlmglue.h" |
38 | #include "file.h" | 38 | #include "file.h" |
39 | #include "inode.h" | 39 | #include "inode.h" |
40 | #include "super.h" | ||
41 | #include "ocfs2_trace.h" | 40 | #include "ocfs2_trace.h" |
42 | 41 | ||
43 | void ocfs2_dentry_attach_gen(struct dentry *dentry) | 42 | void ocfs2_dentry_attach_gen(struct dentry *dentry) |
@@ -346,52 +345,6 @@ out_attach: | |||
346 | return ret; | 345 | return ret; |
347 | } | 346 | } |
348 | 347 | ||
349 | DEFINE_SPINLOCK(dentry_list_lock); | ||
350 | |||
351 | /* We limit the number of dentry locks to drop in one go. We have | ||
352 | * this limit so that we don't starve other users of ocfs2_wq. */ | ||
353 | #define DL_INODE_DROP_COUNT 64 | ||
354 | |||
355 | /* Drop inode references from dentry locks */ | ||
356 | static void __ocfs2_drop_dl_inodes(struct ocfs2_super *osb, int drop_count) | ||
357 | { | ||
358 | struct ocfs2_dentry_lock *dl; | ||
359 | |||
360 | spin_lock(&dentry_list_lock); | ||
361 | while (osb->dentry_lock_list && (drop_count < 0 || drop_count--)) { | ||
362 | dl = osb->dentry_lock_list; | ||
363 | osb->dentry_lock_list = dl->dl_next; | ||
364 | spin_unlock(&dentry_list_lock); | ||
365 | iput(dl->dl_inode); | ||
366 | kfree(dl); | ||
367 | spin_lock(&dentry_list_lock); | ||
368 | } | ||
369 | spin_unlock(&dentry_list_lock); | ||
370 | } | ||
371 | |||
372 | void ocfs2_drop_dl_inodes(struct work_struct *work) | ||
373 | { | ||
374 | struct ocfs2_super *osb = container_of(work, struct ocfs2_super, | ||
375 | dentry_lock_work); | ||
376 | |||
377 | __ocfs2_drop_dl_inodes(osb, DL_INODE_DROP_COUNT); | ||
378 | /* | ||
379 | * Don't queue dropping if umount is in progress. We flush the | ||
380 | * list in ocfs2_dismount_volume | ||
381 | */ | ||
382 | spin_lock(&dentry_list_lock); | ||
383 | if (osb->dentry_lock_list && | ||
384 | !ocfs2_test_osb_flag(osb, OCFS2_OSB_DROP_DENTRY_LOCK_IMMED)) | ||
385 | queue_work(ocfs2_wq, &osb->dentry_lock_work); | ||
386 | spin_unlock(&dentry_list_lock); | ||
387 | } | ||
388 | |||
389 | /* Flush the whole work queue */ | ||
390 | void ocfs2_drop_all_dl_inodes(struct ocfs2_super *osb) | ||
391 | { | ||
392 | __ocfs2_drop_dl_inodes(osb, -1); | ||
393 | } | ||
394 | |||
395 | /* | 348 | /* |
396 | * ocfs2_dentry_iput() and friends. | 349 | * ocfs2_dentry_iput() and friends. |
397 | * | 350 | * |
@@ -416,24 +369,16 @@ void ocfs2_drop_all_dl_inodes(struct ocfs2_super *osb) | |||
416 | static void ocfs2_drop_dentry_lock(struct ocfs2_super *osb, | 369 | static void ocfs2_drop_dentry_lock(struct ocfs2_super *osb, |
417 | struct ocfs2_dentry_lock *dl) | 370 | struct ocfs2_dentry_lock *dl) |
418 | { | 371 | { |
372 | iput(dl->dl_inode); | ||
419 | ocfs2_simple_drop_lockres(osb, &dl->dl_lockres); | 373 | ocfs2_simple_drop_lockres(osb, &dl->dl_lockres); |
420 | ocfs2_lock_res_free(&dl->dl_lockres); | 374 | ocfs2_lock_res_free(&dl->dl_lockres); |
421 | 375 | kfree(dl); | |
422 | /* We leave dropping of inode reference to ocfs2_wq as that can | ||
423 | * possibly lead to inode deletion which gets tricky */ | ||
424 | spin_lock(&dentry_list_lock); | ||
425 | if (!osb->dentry_lock_list && | ||
426 | !ocfs2_test_osb_flag(osb, OCFS2_OSB_DROP_DENTRY_LOCK_IMMED)) | ||
427 | queue_work(ocfs2_wq, &osb->dentry_lock_work); | ||
428 | dl->dl_next = osb->dentry_lock_list; | ||
429 | osb->dentry_lock_list = dl; | ||
430 | spin_unlock(&dentry_list_lock); | ||
431 | } | 376 | } |
432 | 377 | ||
433 | void ocfs2_dentry_lock_put(struct ocfs2_super *osb, | 378 | void ocfs2_dentry_lock_put(struct ocfs2_super *osb, |
434 | struct ocfs2_dentry_lock *dl) | 379 | struct ocfs2_dentry_lock *dl) |
435 | { | 380 | { |
436 | int unlock; | 381 | int unlock = 0; |
437 | 382 | ||
438 | BUG_ON(dl->dl_count == 0); | 383 | BUG_ON(dl->dl_count == 0); |
439 | 384 | ||