aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/ksm.h5
-rw-r--r--include/linux/mm.h17
-rw-r--r--include/linux/rmap.h8
-rw-r--r--mm/migrate.c11
-rw-r--r--mm/rmap.c7
-rw-r--r--mm/swapfile.c2
6 files changed, 35 insertions, 15 deletions
diff --git a/include/linux/ksm.h b/include/linux/ksm.h
index a485c14ecd5d..1401a313fa77 100644
--- a/include/linux/ksm.h
+++ b/include/linux/ksm.h
@@ -38,7 +38,8 @@ static inline void ksm_exit(struct mm_struct *mm)
38 */ 38 */
39static inline int PageKsm(struct page *page) 39static inline int PageKsm(struct page *page)
40{ 40{
41 return ((unsigned long)page->mapping == PAGE_MAPPING_ANON); 41 return ((unsigned long)page->mapping & PAGE_MAPPING_FLAGS) ==
42 (PAGE_MAPPING_ANON | PAGE_MAPPING_KSM);
42} 43}
43 44
44/* 45/*
@@ -47,7 +48,7 @@ static inline int PageKsm(struct page *page)
47static inline void page_add_ksm_rmap(struct page *page) 48static inline void page_add_ksm_rmap(struct page *page)
48{ 49{
49 if (atomic_inc_and_test(&page->_mapcount)) { 50 if (atomic_inc_and_test(&page->_mapcount)) {
50 page->mapping = (void *) PAGE_MAPPING_ANON; 51 page->mapping = (void *) (PAGE_MAPPING_ANON | PAGE_MAPPING_KSM);
51 __inc_zone_page_state(page, NR_ANON_PAGES); 52 __inc_zone_page_state(page, NR_ANON_PAGES);
52 } 53 }
53} 54}
diff --git a/include/linux/mm.h b/include/linux/mm.h
index 24c395694f4d..1202cd3121e1 100644
--- a/include/linux/mm.h
+++ b/include/linux/mm.h
@@ -620,13 +620,22 @@ void page_address_init(void);
620/* 620/*
621 * On an anonymous page mapped into a user virtual memory area, 621 * On an anonymous page mapped into a user virtual memory area,
622 * page->mapping points to its anon_vma, not to a struct address_space; 622 * page->mapping points to its anon_vma, not to a struct address_space;
623 * with the PAGE_MAPPING_ANON bit set to distinguish it. 623 * with the PAGE_MAPPING_ANON bit set to distinguish it. See rmap.h.
624 *
625 * On an anonymous page in a VM_MERGEABLE area, if CONFIG_KSM is enabled,
626 * the PAGE_MAPPING_KSM bit may be set along with the PAGE_MAPPING_ANON bit;
627 * and then page->mapping points, not to an anon_vma, but to a private
628 * structure which KSM associates with that merged page. See ksm.h.
629 *
630 * PAGE_MAPPING_KSM without PAGE_MAPPING_ANON is currently never used.
624 * 631 *
625 * Please note that, confusingly, "page_mapping" refers to the inode 632 * Please note that, confusingly, "page_mapping" refers to the inode
626 * address_space which maps the page from disk; whereas "page_mapped" 633 * address_space which maps the page from disk; whereas "page_mapped"
627 * refers to user virtual address space into which the page is mapped. 634 * refers to user virtual address space into which the page is mapped.
628 */ 635 */
629#define PAGE_MAPPING_ANON 1 636#define PAGE_MAPPING_ANON 1
637#define PAGE_MAPPING_KSM 2
638#define PAGE_MAPPING_FLAGS (PAGE_MAPPING_ANON | PAGE_MAPPING_KSM)
630 639
631extern struct address_space swapper_space; 640extern struct address_space swapper_space;
632static inline struct address_space *page_mapping(struct page *page) 641static inline struct address_space *page_mapping(struct page *page)
@@ -644,6 +653,12 @@ static inline struct address_space *page_mapping(struct page *page)
644 return mapping; 653 return mapping;
645} 654}
646 655
656/* Neutral page->mapping pointer to address_space or anon_vma or other */
657static inline void *page_rmapping(struct page *page)
658{
659 return (void *)((unsigned long)page->mapping & ~PAGE_MAPPING_FLAGS);
660}
661
647static inline int PageAnon(struct page *page) 662static inline int PageAnon(struct page *page)
648{ 663{
649 return ((unsigned long)page->mapping & PAGE_MAPPING_ANON) != 0; 664 return ((unsigned long)page->mapping & PAGE_MAPPING_ANON) != 0;
diff --git a/include/linux/rmap.h b/include/linux/rmap.h
index cb0ba7032609..1f65af44c6d2 100644
--- a/include/linux/rmap.h
+++ b/include/linux/rmap.h
@@ -39,6 +39,14 @@ struct anon_vma {
39 39
40#ifdef CONFIG_MMU 40#ifdef CONFIG_MMU
41 41
42static inline struct anon_vma *page_anon_vma(struct page *page)
43{
44 if (((unsigned long)page->mapping & PAGE_MAPPING_FLAGS) !=
45 PAGE_MAPPING_ANON)
46 return NULL;
47 return page_rmapping(page);
48}
49
42static inline void anon_vma_lock(struct vm_area_struct *vma) 50static inline void anon_vma_lock(struct vm_area_struct *vma)
43{ 51{
44 struct anon_vma *anon_vma = vma->anon_vma; 52 struct anon_vma *anon_vma = vma->anon_vma;
diff --git a/mm/migrate.c b/mm/migrate.c
index 576c25eeb1ca..367272d04423 100644
--- a/mm/migrate.c
+++ b/mm/migrate.c
@@ -172,17 +172,14 @@ static void remove_anon_migration_ptes(struct page *old, struct page *new)
172{ 172{
173 struct anon_vma *anon_vma; 173 struct anon_vma *anon_vma;
174 struct vm_area_struct *vma; 174 struct vm_area_struct *vma;
175 unsigned long mapping;
176
177 mapping = (unsigned long)new->mapping;
178
179 if (!mapping || (mapping & PAGE_MAPPING_ANON) == 0)
180 return;
181 175
182 /* 176 /*
183 * We hold the mmap_sem lock. So no need to call page_lock_anon_vma. 177 * We hold the mmap_sem lock. So no need to call page_lock_anon_vma.
184 */ 178 */
185 anon_vma = (struct anon_vma *) (mapping - PAGE_MAPPING_ANON); 179 anon_vma = page_anon_vma(new);
180 if (!anon_vma)
181 return;
182
186 spin_lock(&anon_vma->lock); 183 spin_lock(&anon_vma->lock);
187 184
188 list_for_each_entry(vma, &anon_vma->head, anon_vma_node) 185 list_for_each_entry(vma, &anon_vma->head, anon_vma_node)
diff --git a/mm/rmap.c b/mm/rmap.c
index 1a0ee6e634c2..f06cee48eca7 100644
--- a/mm/rmap.c
+++ b/mm/rmap.c
@@ -203,7 +203,7 @@ struct anon_vma *page_lock_anon_vma(struct page *page)
203 203
204 rcu_read_lock(); 204 rcu_read_lock();
205 anon_mapping = (unsigned long) page->mapping; 205 anon_mapping = (unsigned long) page->mapping;
206 if (!(anon_mapping & PAGE_MAPPING_ANON)) 206 if ((anon_mapping & PAGE_MAPPING_FLAGS) != PAGE_MAPPING_ANON)
207 goto out; 207 goto out;
208 if (!page_mapped(page)) 208 if (!page_mapped(page))
209 goto out; 209 goto out;
@@ -248,8 +248,7 @@ vma_address(struct page *page, struct vm_area_struct *vma)
248unsigned long page_address_in_vma(struct page *page, struct vm_area_struct *vma) 248unsigned long page_address_in_vma(struct page *page, struct vm_area_struct *vma)
249{ 249{
250 if (PageAnon(page)) { 250 if (PageAnon(page)) {
251 if ((void *)vma->anon_vma != 251 if (vma->anon_vma != page_anon_vma(page))
252 (void *)page->mapping - PAGE_MAPPING_ANON)
253 return -EFAULT; 252 return -EFAULT;
254 } else if (page->mapping && !(vma->vm_flags & VM_NONLINEAR)) { 253 } else if (page->mapping && !(vma->vm_flags & VM_NONLINEAR)) {
255 if (!vma->vm_file || 254 if (!vma->vm_file ||
@@ -513,7 +512,7 @@ int page_referenced(struct page *page,
513 referenced++; 512 referenced++;
514 513
515 *vm_flags = 0; 514 *vm_flags = 0;
516 if (page_mapped(page) && page->mapping) { 515 if (page_mapped(page) && page_rmapping(page)) {
517 if (PageAnon(page)) 516 if (PageAnon(page))
518 referenced += page_referenced_anon(page, mem_cont, 517 referenced += page_referenced_anon(page, mem_cont,
519 vm_flags); 518 vm_flags);
diff --git a/mm/swapfile.c b/mm/swapfile.c
index d5eb2e85600b..e74112e8e5f4 100644
--- a/mm/swapfile.c
+++ b/mm/swapfile.c
@@ -938,7 +938,7 @@ static int unuse_vma(struct vm_area_struct *vma,
938 unsigned long addr, end, next; 938 unsigned long addr, end, next;
939 int ret; 939 int ret;
940 940
941 if (page->mapping) { 941 if (page_anon_vma(page)) {
942 addr = page_address_in_vma(page, vma); 942 addr = page_address_in_vma(page, vma);
943 if (addr == -EFAULT) 943 if (addr == -EFAULT)
944 return 0; 944 return 0;