On Mon, Dec 23, 2013 at 10:03:34AM +0100, Jan Kara wrote: > On Fri 20-12-13 18:42:43, Zheng Liu wrote: > > Hi all, > > > > I still have some ideas for improving the extents status tree shrinker. > > But before doing that, I think I should send out these patches for > > discussion so that I want to be sure that I am on the right way. I > > describe my work below, which contains four parts. The first part > > describes my work of improvment for the shrinker. In second part I > > describe the test case for measurement. In third part I measure the > > performance and compare my work with current implememtation on mainline > > kernel. The last part is TODO list. I describe some ideas that might > > improve the performance of extents status tree shrinker. > > > > Improvement > > =========== > > According to this bug report [1], the extents status tree shrinker might > > causes excessive time stall under heavy memory pressure. Due to lacking > > some details from this report, I am not sure whether the delalloc is > > enabled or not. But if I remember correctly Andreas has mentioned that > > he also notices that es shrinker takes too much time to try to reclaim > > some entries from extents status tree, and he observes that the app does > > a lot of buffered writes. So I suspect that the root cause is that > > currently es shrinker has to scan a lot of delayed entries that don't be > > reclaimed under some workloads. Hence it burns too much cpu time. > > > > 1. https://bugzilla.kernel.org/show_bug.cgi?id=62751 > > > > Why don't we discard delayed entry from extents status tree? Now we need > > to use delayed entry in extents status tree to reserve delayed space for > > delayed allocation, and support seek_data/hole. So we couldn't discard > > these delayed entries. > > > > In this patch series, it contains two patches. Patch 1 just improves the > > trace point in extents status tree. Later I will use them to calculate > > the latency of every function. This is a trival patch. So feel free to > > pick it up. > > > > Patch 2 tries to make es shrinker avoid scanning delayed entry. It adds > > a evictable list in order to track all reclaimable objects. Meanwhile > > a hlist_node structure is defined in 'extent_status' structure. The > > defect of using hlist_node is that it costs extra 1/3 memory space. On > > 64bits platform, extent_status structure occupies 48 bytes. After that > > it will take 64 bytes. Then the es shrinker can traverse on this list. > > It shouldn't encounter any delayed entry. So it reduces the time taking > > sbi->s_es_lru_lock and ei->i_es_lock due to it don't need to scan all > > delayed extries. > > > > Workload > > ======== > > For comparing the improvement, I use fio to generate a workload that > > could cause some fragmented files. For getting a extreme fragmented > > tree, I hack the code in ext4_es_can_be_merge() to always return 0. > > That means we don't try to merge any entry in extents status tree. > > Meanwhile vm.dirty_ratio, vm.dirty_background_ratio are adjusted to 80 > > and 60 on my machine. The purpose of doing this is to accumulate more > > delayed entries on extents status tree. The configure file of fio is > > like below: > > > > [global] > > ioengine=psync > > bs=4k > > directory=/mnt/sda1 > > thread > > group_reporting > > fallocate=0 > > direct=0 > > filesize=10g > > size=20g > > runtime=300 > > > > [io] > > rw=randwrite:32 > > rw_sequencer=sequential > > numjobs=25 > > nrfiles=10 > Well, my suspicion is the stall can be caused when we have lots of > relatively small files and we go through all the pain of scanning inode > list, then resorting it, and scanning again. Your workload doesn't > excercise this path at all if I understand right. You are right. My workload doesn't excercise your suspicion. But I am happy to create this workload and try again to look at the result of improvement. > > > sysctl settings: > > > > #!/bin/bash > > # > > # This script is used to adjust sysctl parameter to increase the memory > > # pressure and accumluate some delayed entries. > > > > sudo sysctl vm.dirty_ratio=80 > > sudo sysctl vm.dirty_background_ratio=60 > > > > Using this workload, the extents status tree will look like this: > > [0, 1)[1, 2)...[31, 32)[32, 51)[51, 52)[52, 53)...[101, 102)...[111, 112) > > [ D ][ D ]...[ D ][ H ][ D ][ D ]...[ W ]...[ U ] > > > > D: Delayed entry > > H: Hole entry > > W: Written entry > > U: Unwritten entry > > > > *NOTE* > > This workload couldn't reproduce the problem that is reported by [1]. > > > > I run this workload on my desktop, which has a Intel Core Duo CPU @ 3.0 > > GHz, 4G memeory and a HDD. I will run the same workload on a server to > > measure the result of improvement. > > > > Comparison > > =========== > > > > slabtop > > ------- > > slabtop is used to measure memory space which the slab of extent_status > > occupies. The following is the result of $(slabtop -o) > > > > vanilla: > > 718890 688677 95% 0.04K 7730 93 30920K ext4_extent_status > > patched: > > 718420 675933 94% 0.05K 10565 68 42260K ext4_extent_status > > > > vanilla patched +/- (%) > > ext4_extent_status 30920k 42260k 36.68 > > > > As I said above, adding hlist_node adds ~1/3 extra memory space. > > > > perf record -g > > -------------- > > The following is the result of $(perf diff), which compares the result > > of $(perf record -g). Here I cut all *_es_* functions and paste them > > here. > > > > # Event 'cycles' > > # > > # Baseline Delta Shared Object Symbol > > # ........ ....... .................. ............................................ > > # > > 1.82% +0.04% [ext4] [k] ext4_es_lookup_extent > > 0.55% +0.01% [ext4] [k] __es_tree_search > > 0.39% -0.05% [ext4] [k] __es_insert_extent > > 0.10% +0.02% [ext4] [k] ext4_es_insert_extent > > 0.08% +0.01% [ext4] [k] __es_remove_extent > > 0.07% +0.01% [ext4] [k] ext4_es_lru_add > > 0.01% -0.01% [ext4] [k] __es_try_to_reclaim_extents > > 0.00% [ext4] [k] ext4_es_free_extent > > 0.00% [ext4] [k] ext4_es_cache_extent > > > > perf lock record -g > > ------------------- > > Here is the result of $(perf lock report -k wait_max). > > > > Name acquired contended avg wait (ns) total wait (ns) max wait (ns) min wait (ns) > > vanilla: > > &(&sbi->s_es_lru... 152 1 87275 87275 87275 87275 > > patched: > > &(&sbi->s_es_lru... 148 0 0 0 0 0 > > > > 'sbi->s_es_lru_lock' protects the entire lru list of inodes which have > > reclaimable entries. When es shrinker tries to reclaim some entries, > > this lock will be taken during this process. From the result we can see > > that the improvement can reduce the waiting time. > > Well, this doesn't say too much. During the whole run there was exactly one > case where we contended for the s_es_lru_lock. It may be just luck you > didn't hit the contention in the 'patched' kernel. > > Also without having a reproducer it's a bit hard to judge whether we are > indeed improving a thing which needs improvement - i.e., whether the gain > in latency is worth the cost in memory usage. Yes, I am also afraid of this issue. But to be honest, it is diffculty for us to create a very similar workload before we can get more details. So maybe we can create some workloads to test different corner cases, and make sure that this improvement can bring benifits for all cases. Now we have two cases to measure two locks in es shrinker. 1. create some big file that have a lot of delayed extents 2. caeate a huge number of small files that have a few extents The first case can measure the overhead of s_es_lru_lock and i_es_lock. The second one can measure the overhead of s_es_lru_lock. More cases are welcome. Thanks, - Zheng > > > Honza > > > latency > > ------- > > Here I calculate the latency of the following operations from the > > result of trace points. (Unit: *us*) > > > > - es_find_delayed_extent_range - > > x vanilla > > + patched > > N Min Max Median Avg Stddev > > x 4425 0 54 1 1.0280226 1.0898311 > > + 7063 0 140 1 1.0127425 1.8662003 > > > > - es_insert_extent - > > x vanilla > > + patched > > N Min Max Median Avg Stddev > > x 26995 1 1270 2 4.1053899 20.26818 > > + 26559 1 440 2 3.0948454 7.6911348 > > > > - es_lookup_extent - > > x vanilla > > + patched > > N Min Max Median Avg Stddev > > x 32025 0 104 1 1.6082748 1.385739 > > + 31433 0 168 1 1.6219578 1.4846153 > > > > - es_shrink_scan - > > x vanilla > > + patched > > N Min Max Median Avg Stddev > > x 1219 24 1872 171 278.35603 275.8815 > > + 1405 17 339 86 95.057651 33.216144 > > > > The result of latency shows that the patch can reduce the latency when es > > shrinker tries to reclaim the entry. Meanwhile other operations don't be > > impacted too much. > > > > TODO list > > ========= > > > > Numa-aware shrinker > > ------------------- > > Now the shrinker has been numa-aware. But the es shrinker doesn't > > support it. So an improvement is to make es shrinker numa-aware. The > > key issue is that extent_status objects might not be on the same node > > with its inode. So that means we need to use lru_list on ext4_sb_info in > > order to make s_es_lru list numa-aware, and use lru_list on ext4_es_tree > > for making evictable list numa-aware. I am not sure it is worthwhile. > > > > rcu-skiplist > > ------------ > > Chris Mason has sent out a rcu-skiplist patch set. So an potential > > improvement here is to use rcu-skiplist in extents status tree to track > > all entries. Now we use rb-tree to do the same thing. But the defect > > of using rb-tree here is that we need to rotate the subtree when a node > > is inserted or removed. That means that rb-tree couldn't provide a very > > stable latency when it tries to balance the subtree. The skiplist does > > not need to do this. So I suspect that it would provide more smooth > > latency. > > > > Regards, > > - Zheng > > > > Zheng Liu (2): > > ext4: improve extents status tree trace point > > ext4: improve extents status tree shrinker to avoid scanning delayed entries > > > > fs/ext4/extents_status.c | 53 +++++++++++++++++++++++-------------------- > > fs/ext4/extents_status.h | 2 ++ > > include/trace/events/ext4.h | 46 +++++++++++++++++++++++++++++++++---- > > 3 files changed, 71 insertions(+), 30 deletions(-) > > > > -- > > 1.7.9.7 > > > > -- > > To unsubscribe from this list: send the line "unsubscribe linux-ext4" in > > the body of a message to majordomo@xxxxxxxxxxxxxxx > > More majordomo info at http://vger.kernel.org/majordomo-info.html > -- > Jan Kara <jack@xxxxxxx> > SUSE Labs, CR -- To unsubscribe from this list: send the line "unsubscribe linux-ext4" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html