aboutsummaryrefslogtreecommitdiffstats
path: root/arch/s390/mm
diff options
context:
space:
mode:
authorMartin Schwidefsky <schwidefsky@de.ibm.com>2013-10-18 06:03:41 -0400
committerChristian Borntraeger <borntraeger@de.ibm.com>2014-04-22 03:36:26 -0400
commit0a61b222df75a6a69dc34816f7db2f61fee8c935 (patch)
tree7fb9f148bc1b96bc02beed069d4490673d7798ab /arch/s390/mm
parent693ffc0802db41911ada95a3e77546f0ed1e7d00 (diff)
KVM: s390/mm: use software dirty bit detection for user dirty tracking
Switch the user dirty bit detection used for migration from the hardware provided host change-bit in the pgste to a fault based detection method. This reduced the dependency of the host from the storage key to a point where it becomes possible to enable the RCP bypass for KVM guests. The fault based dirty detection will only indicate changes caused by accesses via the guest address space. The hardware based method can detect all changes, even those caused by I/O or accesses via the kernel page table. The KVM/qemu code needs to take this into account. Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com> Signed-off-by: Dominik Dingel <dingel@linux.vnet.ibm.com> Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Diffstat (limited to 'arch/s390/mm')
-rw-r--r--arch/s390/mm/pgtable.c6
1 files changed, 3 insertions, 3 deletions
diff --git a/arch/s390/mm/pgtable.c b/arch/s390/mm/pgtable.c
index 02a8607bbeb5..1ddf975352a0 100644
--- a/arch/s390/mm/pgtable.c
+++ b/arch/s390/mm/pgtable.c
@@ -832,6 +832,7 @@ void gmap_do_ipte_notify(struct mm_struct *mm, pte_t *pte)
832 } 832 }
833 spin_unlock(&gmap_notifier_lock); 833 spin_unlock(&gmap_notifier_lock);
834} 834}
835EXPORT_SYMBOL_GPL(gmap_do_ipte_notify);
835 836
836static inline int page_table_with_pgste(struct page *page) 837static inline int page_table_with_pgste(struct page *page)
837{ 838{
@@ -864,8 +865,7 @@ static inline unsigned long *page_table_alloc_pgste(struct mm_struct *mm,
864 atomic_set(&page->_mapcount, 0); 865 atomic_set(&page->_mapcount, 0);
865 table = (unsigned long *) page_to_phys(page); 866 table = (unsigned long *) page_to_phys(page);
866 clear_table(table, _PAGE_INVALID, PAGE_SIZE/2); 867 clear_table(table, _PAGE_INVALID, PAGE_SIZE/2);
867 clear_table(table + PTRS_PER_PTE, PGSTE_HR_BIT | PGSTE_HC_BIT, 868 clear_table(table + PTRS_PER_PTE, 0, PAGE_SIZE/2);
868 PAGE_SIZE/2);
869 return table; 869 return table;
870} 870}
871 871
@@ -1005,7 +1005,7 @@ int set_guest_storage_key(struct mm_struct *mm, unsigned long addr,
1005 /* changing the guest storage key is considered a change of the page */ 1005 /* changing the guest storage key is considered a change of the page */
1006 if ((pgste_val(new) ^ pgste_val(old)) & 1006 if ((pgste_val(new) ^ pgste_val(old)) &
1007 (PGSTE_ACC_BITS | PGSTE_FP_BIT | PGSTE_GR_BIT | PGSTE_GC_BIT)) 1007 (PGSTE_ACC_BITS | PGSTE_FP_BIT | PGSTE_GR_BIT | PGSTE_GC_BIT))
1008 pgste_val(new) |= PGSTE_HC_BIT; 1008 pgste_val(new) |= PGSTE_UC_BIT;
1009 1009
1010 pgste_set_unlock(ptep, new); 1010 pgste_set_unlock(ptep, new);
1011 pte_unmap_unlock(*ptep, ptl); 1011 pte_unmap_unlock(*ptep, ptl);