aboutsummaryrefslogtreecommitdiffstats
path: root/mm/rmap.c
diff options
context:
space:
mode:
authorMinchan Kim <minchan@kernel.org>2016-01-15 19:54:53 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2016-01-15 20:56:32 -0500
commit854e9ed09dedf0c19ac8640e91bcc74bc3f9e5c9 (patch)
tree4abd01fcd0378a3b3e39fb1c2de028b4076507ef /mm/rmap.c
parent17ec4cd985780a7e30aa45bb8f272237c12502a4 (diff)
mm: support madvise(MADV_FREE)
Linux doesn't have an ability to free pages lazy while other OS already have been supported that named by madvise(MADV_FREE). The gain is clear that kernel can discard freed pages rather than swapping out or OOM if memory pressure happens. Without memory pressure, freed pages would be reused by userspace without another additional overhead(ex, page fault + allocation + zeroing). Jason Evans said: : Facebook has been using MAP_UNINITIALIZED : (https://lkml.org/lkml/2012/1/18/308) in some of its applications for : several years, but there are operational costs to maintaining this : out-of-tree in our kernel and in jemalloc, and we are anxious to retire it : in favor of MADV_FREE. When we first enabled MAP_UNINITIALIZED it : increased throughput for much of our workload by ~5%, and although the : benefit has decreased using newer hardware and kernels, there is still : enough benefit that we cannot reasonably retire it without a replacement. : : Aside from Facebook operations, there are numerous broadly used : applications that would benefit from MADV_FREE. The ones that immediately : come to mind are redis, varnish, and MariaDB. I don't have much insight : into Android internals and development process, but I would hope to see : MADV_FREE support eventually end up there as well to benefit applications : linked with the integrated jemalloc. : : jemalloc will use MADV_FREE once it becomes available in the Linux kernel. : In fact, jemalloc already uses MADV_FREE or equivalent everywhere it's : available: *BSD, OS X, Windows, and Solaris -- every platform except Linux : (and AIX, but I'm not sure it even compiles on AIX). The lack of : MADV_FREE on Linux forced me down a long series of increasingly : sophisticated heuristics for madvise() volume reduction, and even so this : remains a common performance issue for people using jemalloc on Linux. : Please integrate MADV_FREE; many people will benefit substantially. How it works: When madvise syscall is called, VM clears dirty bit of ptes of the range. If memory pressure happens, VM checks dirty bit of page table and if it found still "clean", it means it's a "lazyfree pages" so VM could discard the page instead of swapping out. Once there was store operation for the page before VM peek a page to reclaim, dirty bit is set so VM can swap out the page instead of discarding. One thing we should notice is that basically, MADV_FREE relies on dirty bit in page table entry to decide whether VM allows to discard the page or not. IOW, if page table entry includes marked dirty bit, VM shouldn't discard the page. However, as a example, if swap-in by read fault happens, page table entry doesn't have dirty bit so MADV_FREE could discard the page wrongly. For avoiding the problem, MADV_FREE did more checks with PageDirty and PageSwapCache. It worked out because swapped-in page lives on swap cache and since it is evicted from the swap cache, the page has PG_dirty flag. So both page flags check effectively prevent wrong discarding by MADV_FREE. However, a problem in above logic is that swapped-in page has PG_dirty still after they are removed from swap cache so VM cannot consider the page as freeable any more even if madvise_free is called in future. Look at below example for detail. ptr = malloc(); memset(ptr); .. .. .. heavy memory pressure so all of pages are swapped out .. .. var = *ptr; -> a page swapped-in and could be removed from swapcache. Then, page table doesn't mark dirty bit and page descriptor includes PG_dirty .. .. madvise_free(ptr); -> It doesn't clear PG_dirty of the page. .. .. .. .. heavy memory pressure again. .. In this time, VM cannot discard the page because the page .. has *PG_dirty* To solve the problem, this patch clears PG_dirty if only the page is owned exclusively by current process when madvise is called because PG_dirty represents ptes's dirtiness in several processes so we could clear it only if we own it exclusively. Firstly, heavy users would be general allocators(ex, jemalloc, tcmalloc and hope glibc supports it) and jemalloc/tcmalloc already have supported the feature for other OS(ex, FreeBSD) barrios@blaptop:~/benchmark/ebizzy$ lscpu Architecture: x86_64 CPU op-mode(s): 32-bit, 64-bit Byte Order: Little Endian CPU(s): 12 On-line CPU(s) list: 0-11 Thread(s) per core: 1 Core(s) per socket: 1 Socket(s): 12 NUMA node(s): 1 Vendor ID: GenuineIntel CPU family: 6 Model: 2 Stepping: 3 CPU MHz: 3200.185 BogoMIPS: 6400.53 Virtualization: VT-x Hypervisor vendor: KVM Virtualization type: full L1d cache: 32K L1i cache: 32K L2 cache: 4096K NUMA node0 CPU(s): 0-11 ebizzy benchmark(./ebizzy -S 10 -n 512) Higher avg is better. vanilla-jemalloc MADV_free-jemalloc 1 thread records: 10 records: 10 avg: 2961.90 avg: 12069.70 std: 71.96(2.43%) std: 186.68(1.55%) max: 3070.00 max: 12385.00 min: 2796.00 min: 11746.00 2 thread records: 10 records: 10 avg: 5020.00 avg: 17827.00 std: 264.87(5.28%) std: 358.52(2.01%) max: 5244.00 max: 18760.00 min: 4251.00 min: 17382.00 4 thread records: 10 records: 10 avg: 8988.80 avg: 27930.80 std: 1175.33(13.08%) std: 3317.33(11.88%) max: 9508.00 max: 30879.00 min: 5477.00 min: 21024.00 8 thread records: 10 records: 10 avg: 13036.50 avg: 33739.40 std: 170.67(1.31%) std: 5146.22(15.25%) max: 13371.00 max: 40572.00 min: 12785.00 min: 24088.00 16 thread records: 10 records: 10 avg: 11092.40 avg: 31424.20 std: 710.60(6.41%) std: 3763.89(11.98%) max: 12446.00 max: 36635.00 min: 9949.00 min: 25669.00 32 thread records: 10 records: 10 avg: 11067.00 avg: 34495.80 std: 971.06(8.77%) std: 2721.36(7.89%) max: 12010.00 max: 38598.00 min: 9002.00 min: 30636.00 In summary, MADV_FREE is about much faster than MADV_DONTNEED. This patch (of 12): Add core MADV_FREE implementation. [akpm@linux-foundation.org: small cleanups] Signed-off-by: Minchan Kim <minchan@kernel.org> Acked-by: Michal Hocko <mhocko@suse.com> Acked-by: Hugh Dickins <hughd@google.com> Cc: Mika Penttil <mika.penttila@nextfour.com> Cc: Michael Kerrisk <mtk.manpages@gmail.com> Cc: Johannes Weiner <hannes@cmpxchg.org> Cc: Rik van Riel <riel@redhat.com> Cc: Mel Gorman <mgorman@suse.de> Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com> Cc: Jason Evans <je@fb.com> Cc: Daniel Micay <danielmicay@gmail.com> Cc: "Kirill A. Shutemov" <kirill@shutemov.name> Cc: Shaohua Li <shli@kernel.org> Cc: <yalin.wang2010@gmail.com> Cc: Andy Lutomirski <luto@amacapital.net> Cc: "James E.J. Bottomley" <jejb@parisc-linux.org> Cc: "Kirill A. Shutemov" <kirill@shutemov.name> Cc: "Shaohua Li" <shli@kernel.org> Cc: Andrea Arcangeli <aarcange@redhat.com> Cc: Arnd Bergmann <arnd@arndb.de> Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org> Cc: Catalin Marinas <catalin.marinas@arm.com> Cc: Chen Gang <gang.chen.5i5j@gmail.com> Cc: Chris Zankel <chris@zankel.net> Cc: Darrick J. Wong <darrick.wong@oracle.com> Cc: David S. Miller <davem@davemloft.net> Cc: Helge Deller <deller@gmx.de> Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru> Cc: Matt Turner <mattst88@gmail.com> Cc: Max Filippov <jcmvbkbc@gmail.com> Cc: Ralf Baechle <ralf@linux-mips.org> Cc: Richard Henderson <rth@twiddle.net> Cc: Roland Dreier <roland@kernel.org> Cc: Russell King <rmk@arm.linux.org.uk> Cc: Shaohua Li <shli@kernel.org> Cc: Will Deacon <will.deacon@arm.com> Cc: Wu Fengguang <fengguang.wu@intel.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/rmap.c')
-rw-r--r--mm/rmap.c36
1 files changed, 32 insertions, 4 deletions
diff --git a/mm/rmap.c b/mm/rmap.c
index cdc2a885a4cd..68af2e32f7ed 100644
--- a/mm/rmap.c
+++ b/mm/rmap.c
@@ -1411,6 +1411,11 @@ void page_remove_rmap(struct page *page, bool compound)
1411 */ 1411 */
1412} 1412}
1413 1413
1414struct rmap_private {
1415 enum ttu_flags flags;
1416 int lazyfreed;
1417};
1418
1414/* 1419/*
1415 * @arg: enum ttu_flags will be passed to this argument 1420 * @arg: enum ttu_flags will be passed to this argument
1416 */ 1421 */
@@ -1422,7 +1427,8 @@ static int try_to_unmap_one(struct page *page, struct vm_area_struct *vma,
1422 pte_t pteval; 1427 pte_t pteval;
1423 spinlock_t *ptl; 1428 spinlock_t *ptl;
1424 int ret = SWAP_AGAIN; 1429 int ret = SWAP_AGAIN;
1425 enum ttu_flags flags = (enum ttu_flags)arg; 1430 struct rmap_private *rp = arg;
1431 enum ttu_flags flags = rp->flags;
1426 1432
1427 /* munlock has nothing to gain from examining un-locked vmas */ 1433 /* munlock has nothing to gain from examining un-locked vmas */
1428 if ((flags & TTU_MUNLOCK) && !(vma->vm_flags & VM_LOCKED)) 1434 if ((flags & TTU_MUNLOCK) && !(vma->vm_flags & VM_LOCKED))
@@ -1514,6 +1520,14 @@ static int try_to_unmap_one(struct page *page, struct vm_area_struct *vma,
1514 * See handle_pte_fault() ... 1520 * See handle_pte_fault() ...
1515 */ 1521 */
1516 VM_BUG_ON_PAGE(!PageSwapCache(page), page); 1522 VM_BUG_ON_PAGE(!PageSwapCache(page), page);
1523
1524 if (!PageDirty(page) && (flags & TTU_LZFREE)) {
1525 /* It's a freeable page by MADV_FREE */
1526 dec_mm_counter(mm, MM_ANONPAGES);
1527 rp->lazyfreed++;
1528 goto discard;
1529 }
1530
1517 if (swap_duplicate(entry) < 0) { 1531 if (swap_duplicate(entry) < 0) {
1518 set_pte_at(mm, address, pte, pteval); 1532 set_pte_at(mm, address, pte, pteval);
1519 ret = SWAP_FAIL; 1533 ret = SWAP_FAIL;
@@ -1534,6 +1548,7 @@ static int try_to_unmap_one(struct page *page, struct vm_area_struct *vma,
1534 } else 1548 } else
1535 dec_mm_counter(mm, mm_counter_file(page)); 1549 dec_mm_counter(mm, mm_counter_file(page));
1536 1550
1551discard:
1537 page_remove_rmap(page, PageHuge(page)); 1552 page_remove_rmap(page, PageHuge(page));
1538 page_cache_release(page); 1553 page_cache_release(page);
1539 1554
@@ -1586,9 +1601,14 @@ static int page_not_mapped(struct page *page)
1586int try_to_unmap(struct page *page, enum ttu_flags flags) 1601int try_to_unmap(struct page *page, enum ttu_flags flags)
1587{ 1602{
1588 int ret; 1603 int ret;
1604 struct rmap_private rp = {
1605 .flags = flags,
1606 .lazyfreed = 0,
1607 };
1608
1589 struct rmap_walk_control rwc = { 1609 struct rmap_walk_control rwc = {
1590 .rmap_one = try_to_unmap_one, 1610 .rmap_one = try_to_unmap_one,
1591 .arg = (void *)flags, 1611 .arg = &rp,
1592 .done = page_not_mapped, 1612 .done = page_not_mapped,
1593 .anon_lock = page_lock_anon_vma_read, 1613 .anon_lock = page_lock_anon_vma_read,
1594 }; 1614 };
@@ -1608,8 +1628,11 @@ int try_to_unmap(struct page *page, enum ttu_flags flags)
1608 1628
1609 ret = rmap_walk(page, &rwc); 1629 ret = rmap_walk(page, &rwc);
1610 1630
1611 if (ret != SWAP_MLOCK && !page_mapped(page)) 1631 if (ret != SWAP_MLOCK && !page_mapped(page)) {
1612 ret = SWAP_SUCCESS; 1632 ret = SWAP_SUCCESS;
1633 if (rp.lazyfreed && !PageDirty(page))
1634 ret = SWAP_LZFREE;
1635 }
1613 return ret; 1636 return ret;
1614} 1637}
1615 1638
@@ -1631,9 +1654,14 @@ int try_to_unmap(struct page *page, enum ttu_flags flags)
1631int try_to_munlock(struct page *page) 1654int try_to_munlock(struct page *page)
1632{ 1655{
1633 int ret; 1656 int ret;
1657 struct rmap_private rp = {
1658 .flags = TTU_MUNLOCK,
1659 .lazyfreed = 0,
1660 };
1661
1634 struct rmap_walk_control rwc = { 1662 struct rmap_walk_control rwc = {
1635 .rmap_one = try_to_unmap_one, 1663 .rmap_one = try_to_unmap_one,
1636 .arg = (void *)TTU_MUNLOCK, 1664 .arg = &rp,
1637 .done = page_not_mapped, 1665 .done = page_not_mapped,
1638 .anon_lock = page_lock_anon_vma_read, 1666 .anon_lock = page_lock_anon_vma_read,
1639 1667