diff options
Diffstat (limited to 'mm')
-rw-r--r-- | mm/madvise.c | 9 | ||||
-rw-r--r-- | mm/memory.c | 82 |
2 files changed, 22 insertions, 69 deletions
diff --git a/mm/madvise.c b/mm/madvise.c index a271adc93289..917754d26c17 100644 --- a/mm/madvise.c +++ b/mm/madvise.c | |||
@@ -278,14 +278,7 @@ static long madvise_dontneed(struct vm_area_struct *vma, | |||
278 | if (vma->vm_flags & (VM_LOCKED|VM_HUGETLB|VM_PFNMAP)) | 278 | if (vma->vm_flags & (VM_LOCKED|VM_HUGETLB|VM_PFNMAP)) |
279 | return -EINVAL; | 279 | return -EINVAL; |
280 | 280 | ||
281 | if (unlikely(vma->vm_flags & VM_NONLINEAR)) { | 281 | zap_page_range(vma, start, end - start, NULL); |
282 | struct zap_details details = { | ||
283 | .nonlinear_vma = vma, | ||
284 | .last_index = ULONG_MAX, | ||
285 | }; | ||
286 | zap_page_range(vma, start, end - start, &details); | ||
287 | } else | ||
288 | zap_page_range(vma, start, end - start, NULL); | ||
289 | return 0; | 282 | return 0; |
290 | } | 283 | } |
291 | 284 | ||
diff --git a/mm/memory.c b/mm/memory.c index 2c3536cc6c63..9a3e73b69dad 100644 --- a/mm/memory.c +++ b/mm/memory.c | |||
@@ -1082,6 +1082,7 @@ static unsigned long zap_pte_range(struct mmu_gather *tlb, | |||
1082 | spinlock_t *ptl; | 1082 | spinlock_t *ptl; |
1083 | pte_t *start_pte; | 1083 | pte_t *start_pte; |
1084 | pte_t *pte; | 1084 | pte_t *pte; |
1085 | swp_entry_t entry; | ||
1085 | 1086 | ||
1086 | again: | 1087 | again: |
1087 | init_rss_vec(rss); | 1088 | init_rss_vec(rss); |
@@ -1107,28 +1108,12 @@ again: | |||
1107 | if (details->check_mapping && | 1108 | if (details->check_mapping && |
1108 | details->check_mapping != page->mapping) | 1109 | details->check_mapping != page->mapping) |
1109 | continue; | 1110 | continue; |
1110 | /* | ||
1111 | * Each page->index must be checked when | ||
1112 | * invalidating or truncating nonlinear. | ||
1113 | */ | ||
1114 | if (details->nonlinear_vma && | ||
1115 | (page->index < details->first_index || | ||
1116 | page->index > details->last_index)) | ||
1117 | continue; | ||
1118 | } | 1111 | } |
1119 | ptent = ptep_get_and_clear_full(mm, addr, pte, | 1112 | ptent = ptep_get_and_clear_full(mm, addr, pte, |
1120 | tlb->fullmm); | 1113 | tlb->fullmm); |
1121 | tlb_remove_tlb_entry(tlb, pte, addr); | 1114 | tlb_remove_tlb_entry(tlb, pte, addr); |
1122 | if (unlikely(!page)) | 1115 | if (unlikely(!page)) |
1123 | continue; | 1116 | continue; |
1124 | if (unlikely(details) && details->nonlinear_vma | ||
1125 | && linear_page_index(details->nonlinear_vma, | ||
1126 | addr) != page->index) { | ||
1127 | pte_t ptfile = pgoff_to_pte(page->index); | ||
1128 | if (pte_soft_dirty(ptent)) | ||
1129 | ptfile = pte_file_mksoft_dirty(ptfile); | ||
1130 | set_pte_at(mm, addr, pte, ptfile); | ||
1131 | } | ||
1132 | if (PageAnon(page)) | 1117 | if (PageAnon(page)) |
1133 | rss[MM_ANONPAGES]--; | 1118 | rss[MM_ANONPAGES]--; |
1134 | else { | 1119 | else { |
@@ -1151,33 +1136,25 @@ again: | |||
1151 | } | 1136 | } |
1152 | continue; | 1137 | continue; |
1153 | } | 1138 | } |
1154 | /* | 1139 | /* If details->check_mapping, we leave swap entries. */ |
1155 | * If details->check_mapping, we leave swap entries; | ||
1156 | * if details->nonlinear_vma, we leave file entries. | ||
1157 | */ | ||
1158 | if (unlikely(details)) | 1140 | if (unlikely(details)) |
1159 | continue; | 1141 | continue; |
1160 | if (pte_file(ptent)) { | ||
1161 | if (unlikely(!(vma->vm_flags & VM_NONLINEAR))) | ||
1162 | print_bad_pte(vma, addr, ptent, NULL); | ||
1163 | } else { | ||
1164 | swp_entry_t entry = pte_to_swp_entry(ptent); | ||
1165 | 1142 | ||
1166 | if (!non_swap_entry(entry)) | 1143 | entry = pte_to_swp_entry(ptent); |
1167 | rss[MM_SWAPENTS]--; | 1144 | if (!non_swap_entry(entry)) |
1168 | else if (is_migration_entry(entry)) { | 1145 | rss[MM_SWAPENTS]--; |
1169 | struct page *page; | 1146 | else if (is_migration_entry(entry)) { |
1147 | struct page *page; | ||
1170 | 1148 | ||
1171 | page = migration_entry_to_page(entry); | 1149 | page = migration_entry_to_page(entry); |
1172 | 1150 | ||
1173 | if (PageAnon(page)) | 1151 | if (PageAnon(page)) |
1174 | rss[MM_ANONPAGES]--; | 1152 | rss[MM_ANONPAGES]--; |
1175 | else | 1153 | else |
1176 | rss[MM_FILEPAGES]--; | 1154 | rss[MM_FILEPAGES]--; |
1177 | } | ||
1178 | if (unlikely(!free_swap_and_cache(entry))) | ||
1179 | print_bad_pte(vma, addr, ptent, NULL); | ||
1180 | } | 1155 | } |
1156 | if (unlikely(!free_swap_and_cache(entry))) | ||
1157 | print_bad_pte(vma, addr, ptent, NULL); | ||
1181 | pte_clear_not_present_full(mm, addr, pte, tlb->fullmm); | 1158 | pte_clear_not_present_full(mm, addr, pte, tlb->fullmm); |
1182 | } while (pte++, addr += PAGE_SIZE, addr != end); | 1159 | } while (pte++, addr += PAGE_SIZE, addr != end); |
1183 | 1160 | ||
@@ -1277,7 +1254,7 @@ static void unmap_page_range(struct mmu_gather *tlb, | |||
1277 | pgd_t *pgd; | 1254 | pgd_t *pgd; |
1278 | unsigned long next; | 1255 | unsigned long next; |
1279 | 1256 | ||
1280 | if (details && !details->check_mapping && !details->nonlinear_vma) | 1257 | if (details && !details->check_mapping) |
1281 | details = NULL; | 1258 | details = NULL; |
1282 | 1259 | ||
1283 | BUG_ON(addr >= end); | 1260 | BUG_ON(addr >= end); |
@@ -1371,7 +1348,7 @@ void unmap_vmas(struct mmu_gather *tlb, | |||
1371 | * @vma: vm_area_struct holding the applicable pages | 1348 | * @vma: vm_area_struct holding the applicable pages |
1372 | * @start: starting address of pages to zap | 1349 | * @start: starting address of pages to zap |
1373 | * @size: number of bytes to zap | 1350 | * @size: number of bytes to zap |
1374 | * @details: details of nonlinear truncation or shared cache invalidation | 1351 | * @details: details of shared cache invalidation |
1375 | * | 1352 | * |
1376 | * Caller must protect the VMA list | 1353 | * Caller must protect the VMA list |
1377 | */ | 1354 | */ |
@@ -1397,7 +1374,7 @@ void zap_page_range(struct vm_area_struct *vma, unsigned long start, | |||
1397 | * @vma: vm_area_struct holding the applicable pages | 1374 | * @vma: vm_area_struct holding the applicable pages |
1398 | * @address: starting address of pages to zap | 1375 | * @address: starting address of pages to zap |
1399 | * @size: number of bytes to zap | 1376 | * @size: number of bytes to zap |
1400 | * @details: details of nonlinear truncation or shared cache invalidation | 1377 | * @details: details of shared cache invalidation |
1401 | * | 1378 | * |
1402 | * The range must fit into one VMA. | 1379 | * The range must fit into one VMA. |
1403 | */ | 1380 | */ |
@@ -2331,25 +2308,11 @@ static inline void unmap_mapping_range_tree(struct rb_root *root, | |||
2331 | } | 2308 | } |
2332 | } | 2309 | } |
2333 | 2310 | ||
2334 | static inline void unmap_mapping_range_list(struct list_head *head, | ||
2335 | struct zap_details *details) | ||
2336 | { | ||
2337 | struct vm_area_struct *vma; | ||
2338 | |||
2339 | /* | ||
2340 | * In nonlinear VMAs there is no correspondence between virtual address | ||
2341 | * offset and file offset. So we must perform an exhaustive search | ||
2342 | * across *all* the pages in each nonlinear VMA, not just the pages | ||
2343 | * whose virtual address lies outside the file truncation point. | ||
2344 | */ | ||
2345 | list_for_each_entry(vma, head, shared.nonlinear) { | ||
2346 | details->nonlinear_vma = vma; | ||
2347 | unmap_mapping_range_vma(vma, vma->vm_start, vma->vm_end, details); | ||
2348 | } | ||
2349 | } | ||
2350 | |||
2351 | /** | 2311 | /** |
2352 | * unmap_mapping_range - unmap the portion of all mmaps in the specified address_space corresponding to the specified page range in the underlying file. | 2312 | * unmap_mapping_range - unmap the portion of all mmaps in the specified |
2313 | * address_space corresponding to the specified page range in the underlying | ||
2314 | * file. | ||
2315 | * | ||
2353 | * @mapping: the address space containing mmaps to be unmapped. | 2316 | * @mapping: the address space containing mmaps to be unmapped. |
2354 | * @holebegin: byte in first page to unmap, relative to the start of | 2317 | * @holebegin: byte in first page to unmap, relative to the start of |
2355 | * the underlying file. This will be rounded down to a PAGE_SIZE | 2318 | * the underlying file. This will be rounded down to a PAGE_SIZE |
@@ -2378,7 +2341,6 @@ void unmap_mapping_range(struct address_space *mapping, | |||
2378 | } | 2341 | } |
2379 | 2342 | ||
2380 | details.check_mapping = even_cows? NULL: mapping; | 2343 | details.check_mapping = even_cows? NULL: mapping; |
2381 | details.nonlinear_vma = NULL; | ||
2382 | details.first_index = hba; | 2344 | details.first_index = hba; |
2383 | details.last_index = hba + hlen - 1; | 2345 | details.last_index = hba + hlen - 1; |
2384 | if (details.last_index < details.first_index) | 2346 | if (details.last_index < details.first_index) |
@@ -2388,8 +2350,6 @@ void unmap_mapping_range(struct address_space *mapping, | |||
2388 | i_mmap_lock_write(mapping); | 2350 | i_mmap_lock_write(mapping); |
2389 | if (unlikely(!RB_EMPTY_ROOT(&mapping->i_mmap))) | 2351 | if (unlikely(!RB_EMPTY_ROOT(&mapping->i_mmap))) |
2390 | unmap_mapping_range_tree(&mapping->i_mmap, &details); | 2352 | unmap_mapping_range_tree(&mapping->i_mmap, &details); |
2391 | if (unlikely(!list_empty(&mapping->i_mmap_nonlinear))) | ||
2392 | unmap_mapping_range_list(&mapping->i_mmap_nonlinear, &details); | ||
2393 | i_mmap_unlock_write(mapping); | 2353 | i_mmap_unlock_write(mapping); |
2394 | } | 2354 | } |
2395 | EXPORT_SYMBOL(unmap_mapping_range); | 2355 | EXPORT_SYMBOL(unmap_mapping_range); |