On Tue, 28 Apr 2020 15:23:42 +0200 SeongJae Park <sjpark@xxxxxxxxxx> wrote: > On Tue, 28 Apr 2020 13:27:04 +0100 Jonathan Cameron <Jonathan.Cameron@xxxxxxxxxx> wrote: > > > On Mon, 27 Apr 2020 14:04:27 +0200 > > SeongJae Park <sjpark@xxxxxxxxxx> wrote: > > > > > From: SeongJae Park <sjpark@xxxxxxxxx> > > > > > > Introduction > > > ============ > > > > > > Memory management decisions can be improved if finer data access information is > > > available. However, because such finer information usually comes with higher > > > overhead, most systems including Linux forgives the potential benefit and rely > > > on only coarse information or some light-weight heuristics. The pseudo-LRU and > > > the aggressive THP promotions are such examples. > > > > > > A number of data access pattern awared memory management optimizations (refer > > > to 'Appendix A' for more details) consistently say the potential benefit is not > > > small. However, none of those has successfully merged to the mainline Linux > > > kernel mainly due to the absence of a scalable and efficient data access > > > monitoring mechanism. Refer to 'Appendix B' to see the limitations of existing > > > memory monitoring mechanisms. > > > > > > DAMON is a data access monitoring subsystem for the problem. It is 1) accurate > > > enough to be used for the DRAM level memory management (a straightforward > > > DAMON-based optimization achieved up to 2.55x speedup), 2) light-weight enough > > > to be applied online (compared to a straightforward access monitoring scheme, > > > DAMON is up to 94,242.42x lighter) and 3) keeps predefined upper-bound overhead > > > regardless of the size of target workloads (thus scalable). Refer to 'Appendix > > > C' if you interested in how it is possible, and 'Appendix F' to know how the > > > numbers collected. > > > > > > DAMON has mainly designed for the kernel's memory management mechanisms. > > > However, because it is implemented as a standalone kernel module and provides > > > several interfaces, it can be used by a wide range of users including kernel > > > space programs, user space programs, programmers, and administrators. DAMON > > > is now supporting the monitoring only, but it will also provide simple and > > > convenient data access pattern awared memory managements by itself. Refer to > > > 'Appendix D' for more detailed expected usages of DAMON. > > > > [...] > > > > > > Future Plans > > > ============ > > > > > > This patchset is only for the first stage of DAMON. As soon as this patchset > > > is merged, official patchsets for below future plans will be posted. > > > > [...] > > > > > > Support Various Address Spaces > > > ------------------------------ > > > > > > Currently, DAMON supports virtual memory address spaces using PTE Accessed bits > > > as its access checking primitive. However, the core design of DAMON is not > > > dependent to such implementation details. In a future, DAMON will decouple > > > those and support various address spaces including physical memory. It will > > > further allow users to configure and even implement the primitives by > > > themselves for their special usecase. Monitoring of page cache, NUMA nodes, > > > specific files, or block devices would be examples of such usecases. > > > > > > An RFC patchset for this plan is already available > > > (https://lore.kernel.org/linux-mm/20200409094232.29680-1-sjpark@xxxxxxxxxx/). > > > > [...] > > > > > > Patch History > > > ============= > > > > > > The most biggest change in this version is support of minimal region size, > > > which defaults to 'PAGE_SIZE'. This change will reduce unnecessary region > > > splits and thus improve the quality of the output. In a future, we will be > > > able to make this configurable for support of various access check primitives > > > such as PMUs. > > > > That is a good improvement. Might be interesting to consider taking > > hugepages into account as well. > > Thanks! Kudos to Stefan and you for giving me the comments for the change. > > As abovely mentioned in 'Future Plans' section, DAMON will be highly > configurable. You can see the plan in more detail via the RFC patchset[1]. > Thus, the minimal region size will also be able to configured as users want, > including the size of the hugepage. > > [1] https://lore.kernel.org/linux-mm/20200409094232.29680-1-sjpark@xxxxxxxxxx/ > > > > > One issue I've noted is that we have a degeneracy problem with the current > > region merging and splitting that perhaps could do with a small tweak. > > > > Currently we can end with a very small number of regions because there > > is no limit on how many regions can be merged in a give pass for merging. > > However, splitting only doubles the number of regions. > > > > I've been experimenting with a few loops of the splitting algorithm to ensure > > we don't end up stuck with limited regions. I think the problem we are working > > around can be roughly described as: > > > > 1) Program allocates a lot of memory - not really touching much of it. > > 2) Damon fuses the large memory allocations in to one region because the > > access counts are always near 0. > > 3) Program finishes setup. > > 4) Program accesses a few pages in the huge reason a lot, but not that much > > for most of the rest. Taking an extreme option, the page in the middle > > gets all the accesses and the other 1G on either side gets none. > > 5) As a split always breaks the page in two, the chances of significantly > > different values for the two resulting regions is low (as we only sample > > the hot page occasionally). > > > > If we just run the splits twice if the number of regions < max regions / 4 > > then over time we should eventually get a region with the single hot page in it. > > We will get there faster if we split more (keeping below max regions). > > > > As we always remain below max regions, we are still obeying the fixed > > maximum overhead and actually monitoring at closer to the desired granularity. > > Good point. However, as you also mentioned, DAMON will slowly, but eventually > adjust the regions appropriately. > > And yes, your suggested solution will work pretty well. Indeed, my one > previous colleague found this problem on a few of special workloads and tried > the solution you suggested. The improvement was clear. > > However, I didn't adopt the solution due to below reasons. > > First, IMHO, this is an accuracy improvement, rather than bug fix. But the > extent of the enhancement didn't seem very critical to me. Most of other > workloads didn't show such problem (and thus improvement). Even with the > workloads showing the problem, the problem was not seem so critical. > > Second, if the low accuracy is problem, users could get higher accuracy by > simply adjusting the sampling interval and/or aggregation interval to lower > value. This is the supposed way to trade the accuracy with the overhead. I disagree. There is very little chance of getting out of this situation with the current splitting. Changing sampling and aggregation intervals doesn't actually help. Let's draw out an example to discuss. Toy state - taking just one block of memory. 0 = not accessed page (very cold) X = accessed page (extremely hot) First few cycles - no accesses in X.Regions list average value estimated by damon. Region C is needed to set the max and will never be aggregated. aggregation cycle then state. 0.start 0.accessed 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 X X X 0.regions (percent)| A (0) | B (0) | C(1)| 0.merge | A | C | 0.split | A | B | C | After a few cycles, hot page 1.start 1.accessed 0 0 0 0 0 0 0 0 0 0 0 X 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1.regions (acc_cnt)| A (1/18) | B (0) | C(1)| 1.merge | A | C | 1.split | A | B | C | 2.start 2.accessed 0 0 0 0 0 0 0 0 0 0 0 X 0 0 0 0 0 0 0 0 0 0 0 0 0 0 2.regions (acc_cnt)| A (1/12) | B (0) | C(1)| 2.merge | A | C | 2.split | A | B | C | 3.start 3.accessed 0 0 0 0 0 0 0 0 0 0 0 X 0 0 0 0 0 0 0 0 0 0 0 0 0 0 3.regions (acc_cnt)| A (0) | B (1/21) | C(1)| 3.merge | A | C | 3.split | A | B | C | Now make that 1000 pages long with the hot page at page 500. So the average best case we will ever get is a 1/500 * number of sample period between aggregations. So what are the chances of failing to aggregate on the sample after we split at that optimal point? We need to successfully sample that one page enough that we get it 10% of the time. I 'think' this a case of where the 10% point is on the CDF of a binomial f(1/N, M) where N is number of bins and Mis number of samples. Using matlab online I think the best chance you ever get is when you take 10 samples and need just one of them to be in the region. p = 1 - binocdf(0,10,1/N) For N = 500, p = 0.0198 For N = 1000, p = 0.0099 Someone with better maths than me can check. Now this just got us to the point where we won't aggregate the region for one round of aggregation. We may split it again and if the resulting region is small enough might not merge it the next aggregation cycle. So I'd argue that allowing at least 2 repeats of splitting is well worth while. It is just a couple of additional lines of code. > > Finally, I would like to keep code as simple as it can. > > For same reasons, I would like to keep the code as currently is until real user > problem is reported. If you have different opinions, please feel free to yell > at me. :) > > > Thanks, > SeongJae Park > > > > > Jonathan > > > > > > > > Changes from v8 > > > (https://lore.kernel.org/linux-mm/20200406130938.14066-1-sjpark@xxxxxxxxxx/) > > > - Make regions always aligned by minimal region size that can be changed > > > (Stefan Nuernberger) > > > - Store binary format version in the recording file (Stefan Nuernberger) > > > - Use 'int' for pid instead of 'unsigned long' (Stefan Nuernberger) > > > - Fix a race condition in damon thread termination (Stefan Nuernberger) > > > - Optimize random value generation and recording (Stefan Nuernberger) > > > - Clean up commit messages and comments (Stefan Nuernberger) > > > - Clean up code (Stefan Nuernberger) > > > - Use explicit signalling and 'do_exit()' for damon thread termination > > > - Add more typos to spelling.txt > > > - Update the performance evaluation results > > > - Describe future plans in the cover letter > > > > > > Changes from v7 > > > (https://lore.kernel.org/linux-mm/20200318112722.30143-1-sjpark@xxxxxxxxxx/) > > > - Cleanup variable names (Jonathan Cameron) > > > - Split sampling address setup from access_check() (Jonathan Cameron) > > > - Make sampling address to always locate in the region (Jonathan Cameron) > > > - Make initial region's sampling addr to be old (Jonathan Cameron) > > > - Split kdamond on/off function to seperate functions (Jonathan Cameron) > > > - Fix wrong kernel doc comments (Jonathan Cameron) > > > - Reset 'last_accessed' to false in kdamond_check_access() if necessary > > > - Rebase on v5.6 > > > > > > Changes from v6 > > > (https://lore.kernel.org/linux-mm/20200224123047.32506-1-sjpark@xxxxxxxxxx/) > > > - Wordsmith cover letter (Shakeel Butt) > > > - Cleanup code and commit messages (Jonathan Cameron) > > > - Avoid kthread_run() under spinlock critical section (Jonathan Cameron) > > > - Use kthread_stop() (Jonathan Cameron) > > > - Change tracepoint to trace regions (Jonathan Cameron) > > > - Implement API from the beginning (Jonathan Cameron) > > > - Fix typos (Jonathan Cameron) > > > - Fix access checking to properly handle regions smaller than single page > > > (Jonathan Cameron) > > > - Add found typos to 'scripts/spelling.txt' > > > - Add recent evaluation results including DAMON-based Operation Schemes > > > > > > Changes from v5 > > > (https://lore.kernel.org/linux-mm/20200217103110.30817-1-sjpark@xxxxxxxxxx/) > > > - Fix minor bugs (sampling, record attributes, debugfs and user space tool) > > > - selftests: Add debugfs interface tests for the bugs > > > - Modify the user space tool to use its self default values for parameters > > > - Fix pmg huge page access check > > > > > > Changes from v4 > > > (https://lore.kernel.org/linux-mm/20200210144812.26845-1-sjpark@xxxxxxxxxx/) > > > - Add 'Reviewed-by' for the kunit tests patch (Brendan Higgins) > > > - Make the unit test to depedns on 'DAMON=y' (Randy Dunlap and kbuild bot) > > > Reported-by: kbuild test robot <lkp@xxxxxxxxx> > > > - Fix m68k module build issue > > > Reported-by: kbuild test robot <lkp@xxxxxxxxx> > > > - Add selftests > > > - Seperate patches for low level users from core logics for better reading > > > - Clean up debugfs interface > > > - Trivial nitpicks > > > > > > Changes from v3 > > > (https://lore.kernel.org/linux-mm/20200204062312.19913-1-sj38.park@xxxxxxxxx/) > > > - Fix i386 build issue > > > Reported-by: kbuild test robot <lkp@xxxxxxxxx> > > > - Increase the default size of the monitoring result buffer to 1 MiB > > > - Fix misc bugs in debugfs interface > > > > > > Changes from v2 > > > (https://lore.kernel.org/linux-mm/20200128085742.14566-1-sjpark@xxxxxxxxxx/) > > > - Move MAINTAINERS changes to last commit (Brendan Higgins) > > > - Add descriptions for kunittest: why not only entire mappings and what the 4 > > > input sets are trying to test (Brendan Higgins) > > > - Remove 'kdamond_need_stop()' test (Brendan Higgins) > > > - Discuss about the 'perf mem' and DAMON (Peter Zijlstra) > > > - Make CV clearly say what it actually does (Peter Zijlstra) > > > - Answer why new module (Qian Cai) > > > - Diable DAMON by default (Randy Dunlap) > > > - Change the interface: Seperate recording attributes > > > (attrs, record, rules) and allow multiple kdamond instances > > > - Implement kernel API interface > > > > > > Changes from v1 > > > (https://lore.kernel.org/linux-mm/20200120162757.32375-1-sjpark@xxxxxxxxxx/) > > > - Rebase on v5.5 > > > - Add a tracepoint for integration with other tracers (Kirill A. Shutemov) > > > - document: Add more description for the user space tool (Brendan Higgins) > > > - unittest: Improve readability (Brendan Higgins) > > > - unittest: Use consistent name and helpers function (Brendan Higgins) > > > - Update PG_Young to avoid reclaim logic interference (Yunjae Lee) > > > > > > Changes from RFC > > > (https://lore.kernel.org/linux-mm/20200110131522.29964-1-sjpark@xxxxxxxxxx/) > > > - Specify an ambiguous plan of access pattern based mm optimizations > > > - Support loadable module build > > > - Cleanup code > > > > > > SeongJae Park (15): > > > scripts/spelling: Add a few more typos > > > mm/page_ext: Export lookup_page_ext() to GPL modules > > > mm: Introduce Data Access MONitor (DAMON) > > > mm/damon: Implement region based sampling > > > mm/damon: Adaptively adjust regions > > > mm/damon: Apply dynamic memory mapping changes > > > mm/damon: Implement callbacks > > > mm/damon: Implement access pattern recording > > > mm/damon: Add debugfs interface > > > mm/damon: Add tracepoints > > > tools: Add a minimal user-space tool for DAMON > > > Documentation/admin-guide/mm: Add a document for DAMON > > > mm/damon: Add kunit tests > > > mm/damon: Add user space selftests > > > MAINTAINERS: Update for DAMON > > > > > > .../admin-guide/mm/data_access_monitor.rst | 428 +++++ > > > Documentation/admin-guide/mm/index.rst | 1 + > > > MAINTAINERS | 12 + > > > include/linux/damon.h | 78 + > > > include/trace/events/damon.h | 43 + > > > mm/Kconfig | 23 + > > > mm/Makefile | 1 + > > > mm/damon-test.h | 615 +++++++ > > > mm/damon.c | 1494 +++++++++++++++++ > > > mm/page_ext.c | 1 + > > > scripts/spelling.txt | 8 + > > > tools/damon/.gitignore | 1 + > > > tools/damon/_dist.py | 36 + > > > tools/damon/_recfile.py | 23 + > > > tools/damon/bin2txt.py | 67 + > > > tools/damon/damo | 37 + > > > tools/damon/heats.py | 362 ++++ > > > tools/damon/nr_regions.py | 91 + > > > tools/damon/record.py | 212 +++ > > > tools/damon/report.py | 45 + > > > tools/damon/wss.py | 97 ++ > > > tools/testing/selftests/damon/Makefile | 7 + > > > .../selftests/damon/_chk_dependency.sh | 28 + > > > tools/testing/selftests/damon/_chk_record.py | 108 ++ > > > .../testing/selftests/damon/debugfs_attrs.sh | 139 ++ > > > .../testing/selftests/damon/debugfs_record.sh | 50 + > > > 26 files changed, 4007 insertions(+) > > > create mode 100644 Documentation/admin-guide/mm/data_access_monitor.rst > > > create mode 100644 include/linux/damon.h > > > create mode 100644 include/trace/events/damon.h > > > create mode 100644 mm/damon-test.h > > > create mode 100644 mm/damon.c > > > create mode 100644 tools/damon/.gitignore > > > create mode 100644 tools/damon/_dist.py > > > create mode 100644 tools/damon/_recfile.py > > > create mode 100644 tools/damon/bin2txt.py > > > create mode 100755 tools/damon/damo > > > create mode 100644 tools/damon/heats.py > > > create mode 100644 tools/damon/nr_regions.py > > > create mode 100644 tools/damon/record.py > > > create mode 100644 tools/damon/report.py > > > create mode 100644 tools/damon/wss.py > > > create mode 100644 tools/testing/selftests/damon/Makefile > > > create mode 100644 tools/testing/selftests/damon/_chk_dependency.sh > > > create mode 100644 tools/testing/selftests/damon/_chk_record.py > > > create mode 100755 tools/testing/selftests/damon/debugfs_attrs.sh > > > create mode 100755 tools/testing/selftests/damon/debugfs_record.sh