diff options
Diffstat (limited to 'mm/rmap.c')
| -rw-r--r-- | mm/rmap.c | 65 |
1 files changed, 65 insertions, 0 deletions
| @@ -434,6 +434,71 @@ int page_referenced(struct page *page, int is_locked) | |||
| 434 | return referenced; | 434 | return referenced; |
| 435 | } | 435 | } |
| 436 | 436 | ||
| 437 | static int page_mkclean_one(struct page *page, struct vm_area_struct *vma) | ||
| 438 | { | ||
| 439 | struct mm_struct *mm = vma->vm_mm; | ||
| 440 | unsigned long address; | ||
| 441 | pte_t *pte, entry; | ||
| 442 | spinlock_t *ptl; | ||
| 443 | int ret = 0; | ||
| 444 | |||
| 445 | address = vma_address(page, vma); | ||
| 446 | if (address == -EFAULT) | ||
| 447 | goto out; | ||
| 448 | |||
| 449 | pte = page_check_address(page, mm, address, &ptl); | ||
| 450 | if (!pte) | ||
| 451 | goto out; | ||
| 452 | |||
| 453 | if (!pte_dirty(*pte) && !pte_write(*pte)) | ||
| 454 | goto unlock; | ||
| 455 | |||
| 456 | entry = ptep_get_and_clear(mm, address, pte); | ||
| 457 | entry = pte_mkclean(entry); | ||
| 458 | entry = pte_wrprotect(entry); | ||
| 459 | ptep_establish(vma, address, pte, entry); | ||
| 460 | lazy_mmu_prot_update(entry); | ||
| 461 | ret = 1; | ||
| 462 | |||
| 463 | unlock: | ||
| 464 | pte_unmap_unlock(pte, ptl); | ||
| 465 | out: | ||
| 466 | return ret; | ||
| 467 | } | ||
| 468 | |||
| 469 | static int page_mkclean_file(struct address_space *mapping, struct page *page) | ||
| 470 | { | ||
| 471 | pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT); | ||
| 472 | struct vm_area_struct *vma; | ||
| 473 | struct prio_tree_iter iter; | ||
| 474 | int ret = 0; | ||
| 475 | |||
| 476 | BUG_ON(PageAnon(page)); | ||
| 477 | |||
| 478 | spin_lock(&mapping->i_mmap_lock); | ||
| 479 | vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) { | ||
| 480 | if (vma->vm_flags & VM_SHARED) | ||
| 481 | ret += page_mkclean_one(page, vma); | ||
| 482 | } | ||
| 483 | spin_unlock(&mapping->i_mmap_lock); | ||
| 484 | return ret; | ||
| 485 | } | ||
| 486 | |||
| 487 | int page_mkclean(struct page *page) | ||
| 488 | { | ||
| 489 | int ret = 0; | ||
| 490 | |||
| 491 | BUG_ON(!PageLocked(page)); | ||
| 492 | |||
| 493 | if (page_mapped(page)) { | ||
| 494 | struct address_space *mapping = page_mapping(page); | ||
| 495 | if (mapping) | ||
| 496 | ret = page_mkclean_file(mapping, page); | ||
| 497 | } | ||
| 498 | |||
| 499 | return ret; | ||
| 500 | } | ||
| 501 | |||
| 437 | /** | 502 | /** |
| 438 | * page_set_anon_rmap - setup new anonymous rmap | 503 | * page_set_anon_rmap - setup new anonymous rmap |
| 439 | * @page: the page to add the mapping to | 504 | * @page: the page to add the mapping to |
