Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

mm: refactor of vma_merge()

Patch series "Refactor of vma_merge and new merge call", v4.

I am currently working on my master's thesis trying to increase number of
merges of VMAs currently failing because of page offset incompatibility
and difference in their anon_vmas. The following refactor and added merge
call included in this series is just two smaller upgrades I created along
the way.


This patch (of 2):

Refactor vma_merge() to make it shorter and more understandable. Main
change is the elimination of code duplicity in the case of merge next
check. This is done by first doing checks and caching the results before
executing the merge itself. The variable 'area' is divided into 'mid' and
'res' as previously it was used for two purposes, as the middle VMA
between prev and next and also as the result of the merge itself. Exit
paths are also unified.

Link: https://lkml.kernel.org/r/20220603145719.1012094-1-matenajakub@gmail.com
Link: https://lkml.kernel.org/r/20220603145719.1012094-2-matenajakub@gmail.com
Signed-off-by: Jakub Matěna <matenajakub@gmail.com>
Reviewed-by: Vlastimil Babka <vbabka@suse.cz>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Mel Gorman <mgorman@techsingularity.net>
Cc: Matthew Wilcox <willy@infradead.org>
Cc: Liam Howlett <liam.howlett@oracle.com>
Cc: Hugh Dickins <hughd@google.com>
Cc: "Kirill A . Shutemov" <kirill@shutemov.name>
Cc: Rik van Riel <riel@surriel.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Peter Zijlstra (Intel) <peterz@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>

authored by

Jakub Matěna and committed by
Andrew Morton
eef19944 b3541d91

+37 -50
+37 -50
mm/mmap.c
··· 1005 1005 struct anon_vma_name *anon_name) 1006 1006 { 1007 1007 pgoff_t pglen = (end - addr) >> PAGE_SHIFT; 1008 - struct vm_area_struct *area, *next; 1009 - int err; 1008 + struct vm_area_struct *mid, *next, *res; 1009 + int err = -1; 1010 + bool merge_prev = false; 1011 + bool merge_next = false; 1010 1012 1011 1013 /* 1012 1014 * We later require that vma->vm_flags == vm_flags, ··· 1018 1016 return NULL; 1019 1017 1020 1018 next = find_vma(mm, prev ? prev->vm_end : 0); 1021 - area = next; 1022 - if (area && area->vm_end == end) /* cases 6, 7, 8 */ 1019 + mid = next; 1020 + if (next && next->vm_end == end) /* cases 6, 7, 8 */ 1023 1021 next = find_vma(mm, next->vm_end); 1024 1022 1025 1023 /* verify some invariant that must be enforced by the caller */ 1026 1024 VM_WARN_ON(prev && addr <= prev->vm_start); 1027 - VM_WARN_ON(area && end > area->vm_end); 1025 + VM_WARN_ON(mid && end > mid->vm_end); 1028 1026 VM_WARN_ON(addr >= end); 1029 1027 1030 - /* 1031 - * Can it merge with the predecessor? 1032 - */ 1028 + /* Can we merge the predecessor? */ 1033 1029 if (prev && prev->vm_end == addr && 1034 1030 mpol_equal(vma_policy(prev), policy) && 1035 1031 can_vma_merge_after(prev, vm_flags, 1036 1032 anon_vma, file, pgoff, 1037 1033 vm_userfaultfd_ctx, anon_name)) { 1038 - /* 1039 - * OK, it can. Can we now merge in the successor as well? 1040 - */ 1041 - if (next && end == next->vm_start && 1042 - mpol_equal(policy, vma_policy(next)) && 1043 - can_vma_merge_before(next, vm_flags, 1044 - anon_vma, file, 1045 - pgoff+pglen, 1046 - vm_userfaultfd_ctx, anon_name) && 1047 - is_mergeable_anon_vma(prev->anon_vma, 1048 - next->anon_vma, NULL)) { 1049 - /* cases 1, 6 */ 1050 - err = __vma_adjust(prev, prev->vm_start, 1051 - next->vm_end, prev->vm_pgoff, NULL, 1052 - prev); 1053 - } else /* cases 2, 5, 7 */ 1054 - err = __vma_adjust(prev, prev->vm_start, 1055 - end, prev->vm_pgoff, NULL, prev); 1056 - if (err) 1057 - return NULL; 1058 - khugepaged_enter_vma(prev, vm_flags); 1059 - return prev; 1034 + merge_prev = true; 1060 1035 } 1061 - 1062 - /* 1063 - * Can this new request be merged in front of next? 1064 - */ 1036 + /* Can we merge the successor? */ 1065 1037 if (next && end == next->vm_start && 1066 1038 mpol_equal(policy, vma_policy(next)) && 1067 1039 can_vma_merge_before(next, vm_flags, 1068 1040 anon_vma, file, pgoff+pglen, 1069 1041 vm_userfaultfd_ctx, anon_name)) { 1042 + merge_next = true; 1043 + } 1044 + /* Can we merge both the predecessor and the successor? */ 1045 + if (merge_prev && merge_next && 1046 + is_mergeable_anon_vma(prev->anon_vma, 1047 + next->anon_vma, NULL)) { /* cases 1, 6 */ 1048 + err = __vma_adjust(prev, prev->vm_start, 1049 + next->vm_end, prev->vm_pgoff, NULL, 1050 + prev); 1051 + res = prev; 1052 + } else if (merge_prev) { /* cases 2, 5, 7 */ 1053 + err = __vma_adjust(prev, prev->vm_start, 1054 + end, prev->vm_pgoff, NULL, prev); 1055 + res = prev; 1056 + } else if (merge_next) { 1070 1057 if (prev && addr < prev->vm_end) /* case 4 */ 1071 1058 err = __vma_adjust(prev, prev->vm_start, 1072 - addr, prev->vm_pgoff, NULL, next); 1073 - else { /* cases 3, 8 */ 1074 - err = __vma_adjust(area, addr, next->vm_end, 1075 - next->vm_pgoff - pglen, NULL, next); 1076 - /* 1077 - * In case 3 area is already equal to next and 1078 - * this is a noop, but in case 8 "area" has 1079 - * been removed and next was expanded over it. 1080 - */ 1081 - area = next; 1082 - } 1083 - if (err) 1084 - return NULL; 1085 - khugepaged_enter_vma(area, vm_flags); 1086 - return area; 1059 + addr, prev->vm_pgoff, NULL, next); 1060 + else /* cases 3, 8 */ 1061 + err = __vma_adjust(mid, addr, next->vm_end, 1062 + next->vm_pgoff - pglen, NULL, next); 1063 + res = next; 1087 1064 } 1088 1065 1089 - return NULL; 1066 + /* 1067 + * Cannot merge with predecessor or successor or error in __vma_adjust? 1068 + */ 1069 + if (err) 1070 + return NULL; 1071 + khugepaged_enter_vma(res, vm_flags); 1072 + return res; 1090 1073 } 1091 1074 1092 1075 /*