diff options
author | Yan, Zheng <zheng.z.yan@intel.com> | 2013-05-03 02:15:52 -0400 |
---|---|---|
committer | Theodore Ts'o <tytso@mit.edu> | 2013-05-03 02:15:52 -0400 |
commit | e30b5dca15dea86aa697f9d58ff646294fe80d3d (patch) | |
tree | 549d1ff3f8e5ecb9675a2937113aa99e64ef3ea1 /fs/ext4/file.c | |
parent | 0d606e2c9fccdd4e67febf1e2da500e1bfe9e045 (diff) |
ext4: fix fio regression
We (Linux Kernel Performance project) found a regression introduced
by commit:
f7fec032aa ext4: track all extent status in extent status tree
The commit causes about 20% performance decrease in fio random write
test. Profiler shows that rb_next() uses a lot of CPU time. The call
stack is:
rb_next
ext4_es_find_delayed_extent
ext4_map_blocks
_ext4_get_block
ext4_get_block_write
__blockdev_direct_IO
ext4_direct_IO
generic_file_direct_write
__generic_file_aio_write
ext4_file_write
aio_rw_vect_retry
aio_run_iocb
do_io_submit
sys_io_submit
system_call_fastpath
io_submit
td_io_getevents
io_u_queued_complete
thread_main
main
__libc_start_main
The cause is that ext4_es_find_delayed_extent() doesn't have an
upper bound, it keeps searching until a delayed extent is found.
When there are a lots of non-delayed entries in the extent state
tree, ext4_es_find_delayed_extent() may uses a lot of CPU time.
Reported-by: LKP project <lkp@linux.intel.com>
Signed-off-by: Yan, Zheng <zheng.z.yan@intel.com>
Signed-off-by: Zheng Liu <wenqing.lz@taobao.com>
Cc: "Theodore Ts'o" <tytso@mit.edu>
Diffstat (limited to 'fs/ext4/file.c')
-rw-r--r-- | fs/ext4/file.c | 4 |
1 files changed, 2 insertions, 2 deletions
diff --git a/fs/ext4/file.c b/fs/ext4/file.c index 64848b595b24..1bc4523541d6 100644 --- a/fs/ext4/file.c +++ b/fs/ext4/file.c | |||
@@ -464,7 +464,7 @@ static loff_t ext4_seek_data(struct file *file, loff_t offset, loff_t maxsize) | |||
464 | * If there is a delay extent at this offset, | 464 | * If there is a delay extent at this offset, |
465 | * it will be as a data. | 465 | * it will be as a data. |
466 | */ | 466 | */ |
467 | ext4_es_find_delayed_extent(inode, last, &es); | 467 | ext4_es_find_delayed_extent_range(inode, last, last, &es); |
468 | if (es.es_len != 0 && in_range(last, es.es_lblk, es.es_len)) { | 468 | if (es.es_len != 0 && in_range(last, es.es_lblk, es.es_len)) { |
469 | if (last != start) | 469 | if (last != start) |
470 | dataoff = last << blkbits; | 470 | dataoff = last << blkbits; |
@@ -547,7 +547,7 @@ static loff_t ext4_seek_hole(struct file *file, loff_t offset, loff_t maxsize) | |||
547 | * If there is a delay extent at this offset, | 547 | * If there is a delay extent at this offset, |
548 | * we will skip this extent. | 548 | * we will skip this extent. |
549 | */ | 549 | */ |
550 | ext4_es_find_delayed_extent(inode, last, &es); | 550 | ext4_es_find_delayed_extent_range(inode, last, last, &es); |
551 | if (es.es_len != 0 && in_range(last, es.es_lblk, es.es_len)) { | 551 | if (es.es_len != 0 && in_range(last, es.es_lblk, es.es_len)) { |
552 | last = es.es_lblk + es.es_len; | 552 | last = es.es_lblk + es.es_len; |
553 | holeoff = last << blkbits; | 553 | holeoff = last << blkbits; |