2007-10-22 18:52:47 +08:00
|
|
|
/*
|
2012-07-20 17:15:04 +08:00
|
|
|
* Copyright IBM Corp. 2007, 2011
|
2007-10-22 18:52:47 +08:00
|
|
|
* Author(s): Martin Schwidefsky <schwidefsky@de.ibm.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/sched.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/errno.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/gfp.h>
|
2007-10-22 18:52:47 +08:00
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/swap.h>
|
|
|
|
#include <linux/smp.h>
|
|
|
|
#include <linux/highmem.h>
|
|
|
|
#include <linux/pagemap.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/quicklist.h>
|
2010-10-25 22:10:11 +08:00
|
|
|
#include <linux/rcupdate.h>
|
2011-07-24 16:48:20 +08:00
|
|
|
#include <linux/slab.h>
|
2013-04-17 23:36:29 +08:00
|
|
|
#include <linux/swapops.h>
|
2007-10-22 18:52:47 +08:00
|
|
|
|
|
|
|
#include <asm/pgtable.h>
|
|
|
|
#include <asm/pgalloc.h>
|
|
|
|
#include <asm/tlb.h>
|
|
|
|
#include <asm/tlbflush.h>
|
2008-02-10 01:24:37 +08:00
|
|
|
#include <asm/mmu_context.h>
|
2007-10-22 18:52:47 +08:00
|
|
|
|
|
|
|
#ifndef CONFIG_64BIT
|
|
|
|
#define ALLOC_ORDER 1
|
2011-06-06 20:14:41 +08:00
|
|
|
#define FRAG_MASK 0x0f
|
2007-10-22 18:52:47 +08:00
|
|
|
#else
|
|
|
|
#define ALLOC_ORDER 2
|
2011-06-06 20:14:41 +08:00
|
|
|
#define FRAG_MASK 0x03
|
2007-10-22 18:52:47 +08:00
|
|
|
#endif
|
|
|
|
|
2009-06-12 16:26:33 +08:00
|
|
|
|
2011-05-23 16:24:23 +08:00
|
|
|
unsigned long *crst_table_alloc(struct mm_struct *mm)
|
2007-10-22 18:52:47 +08:00
|
|
|
{
|
|
|
|
struct page *page = alloc_pages(GFP_KERNEL, ALLOC_ORDER);
|
|
|
|
|
|
|
|
if (!page)
|
|
|
|
return NULL;
|
|
|
|
return (unsigned long *) page_to_phys(page);
|
|
|
|
}
|
|
|
|
|
2010-10-25 22:10:11 +08:00
|
|
|
void crst_table_free(struct mm_struct *mm, unsigned long *table)
|
|
|
|
{
|
2011-05-23 16:24:23 +08:00
|
|
|
free_pages((unsigned long) table, ALLOC_ORDER);
|
2010-10-25 22:10:11 +08:00
|
|
|
}
|
|
|
|
|
2008-02-10 01:24:37 +08:00
|
|
|
#ifdef CONFIG_64BIT
|
2013-10-28 21:48:30 +08:00
|
|
|
static void __crst_table_upgrade(void *arg)
|
|
|
|
{
|
|
|
|
struct mm_struct *mm = arg;
|
|
|
|
|
|
|
|
if (current->active_mm == mm)
|
s390/uaccess: rework uaccess code - fix locking issues
The current uaccess code uses a page table walk in some circumstances,
e.g. in case of the in atomic futex operations or if running on old
hardware which doesn't support the mvcos instruction.
However it turned out that the page table walk code does not correctly
lock page tables when accessing page table entries.
In other words: a different cpu may invalidate a page table entry while
the current cpu inspects the pte. This may lead to random data corruption.
Adding correct locking however isn't trivial for all uaccess operations.
Especially copy_in_user() is problematic since that requires to hold at
least two locks, but must be protected against ABBA deadlock when a
different cpu also performs a copy_in_user() operation.
So the solution is a different approach where we change address spaces:
User space runs in primary address mode, or access register mode within
vdso code, like it currently already does.
The kernel usually also runs in home space mode, however when accessing
user space the kernel switches to primary or secondary address mode if
the mvcos instruction is not available or if a compare-and-swap (futex)
instruction on a user space address is performed.
KVM however is special, since that requires the kernel to run in home
address space while implicitly accessing user space with the sie
instruction.
So we end up with:
User space:
- runs in primary or access register mode
- cr1 contains the user asce
- cr7 contains the user asce
- cr13 contains the kernel asce
Kernel space:
- runs in home space mode
- cr1 contains the user or kernel asce
-> the kernel asce is loaded when a uaccess requires primary or
secondary address mode
- cr7 contains the user or kernel asce, (changed with set_fs())
- cr13 contains the kernel asce
In case of uaccess the kernel changes to:
- primary space mode in case of a uaccess (copy_to_user) and uses
e.g. the mvcp instruction to access user space. However the kernel
will stay in home space mode if the mvcos instruction is available
- secondary space mode in case of futex atomic operations, so that the
instructions come from primary address space and data from secondary
space
In case of kvm the kernel runs in home space mode, but cr1 gets switched
to contain the gmap asce before the sie instruction gets executed. When
the sie instruction is finished cr1 will be switched back to contain the
user asce.
A context switch between two processes will always load the kernel asce
for the next process in cr1. So the first exit to user space is a bit
more expensive (one extra load control register instruction) than before,
however keeps the code rather simple.
In sum this means there is no need to perform any error prone page table
walks anymore when accessing user space.
The patch seems to be rather large, however it mainly removes the
the page table walk code and restores the previously deleted "standard"
uaccess code, with a couple of changes.
The uaccess without mvcos mode can be enforced with the "uaccess_primary"
kernel parameter.
Reported-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2014-03-21 17:42:25 +08:00
|
|
|
update_user_asce(mm, 1);
|
2013-10-28 21:48:30 +08:00
|
|
|
__tlb_flush_local();
|
|
|
|
}
|
|
|
|
|
2008-02-10 01:24:37 +08:00
|
|
|
int crst_table_upgrade(struct mm_struct *mm, unsigned long limit)
|
|
|
|
{
|
|
|
|
unsigned long *table, *pgd;
|
|
|
|
unsigned long entry;
|
2013-10-28 21:48:30 +08:00
|
|
|
int flush;
|
2008-02-10 01:24:37 +08:00
|
|
|
|
|
|
|
BUG_ON(limit > (1UL << 53));
|
2013-10-28 21:48:30 +08:00
|
|
|
flush = 0;
|
2008-02-10 01:24:37 +08:00
|
|
|
repeat:
|
2011-05-23 16:24:23 +08:00
|
|
|
table = crst_table_alloc(mm);
|
2008-02-10 01:24:37 +08:00
|
|
|
if (!table)
|
|
|
|
return -ENOMEM;
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_lock_bh(&mm->page_table_lock);
|
2008-02-10 01:24:37 +08:00
|
|
|
if (mm->context.asce_limit < limit) {
|
|
|
|
pgd = (unsigned long *) mm->pgd;
|
|
|
|
if (mm->context.asce_limit <= (1UL << 31)) {
|
|
|
|
entry = _REGION3_ENTRY_EMPTY;
|
|
|
|
mm->context.asce_limit = 1UL << 42;
|
|
|
|
mm->context.asce_bits = _ASCE_TABLE_LENGTH |
|
|
|
|
_ASCE_USER_BITS |
|
|
|
|
_ASCE_TYPE_REGION3;
|
|
|
|
} else {
|
|
|
|
entry = _REGION2_ENTRY_EMPTY;
|
|
|
|
mm->context.asce_limit = 1UL << 53;
|
|
|
|
mm->context.asce_bits = _ASCE_TABLE_LENGTH |
|
|
|
|
_ASCE_USER_BITS |
|
|
|
|
_ASCE_TYPE_REGION2;
|
|
|
|
}
|
|
|
|
crst_table_init(table, entry);
|
|
|
|
pgd_populate(mm, (pgd_t *) table, (pud_t *) pgd);
|
|
|
|
mm->pgd = (pgd_t *) table;
|
2009-03-18 20:27:36 +08:00
|
|
|
mm->task_size = mm->context.asce_limit;
|
2008-02-10 01:24:37 +08:00
|
|
|
table = NULL;
|
2013-10-28 21:48:30 +08:00
|
|
|
flush = 1;
|
2008-02-10 01:24:37 +08:00
|
|
|
}
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_unlock_bh(&mm->page_table_lock);
|
2008-02-10 01:24:37 +08:00
|
|
|
if (table)
|
|
|
|
crst_table_free(mm, table);
|
|
|
|
if (mm->context.asce_limit < limit)
|
|
|
|
goto repeat;
|
2013-10-28 21:48:30 +08:00
|
|
|
if (flush)
|
|
|
|
on_each_cpu(__crst_table_upgrade, mm, 0);
|
2008-02-10 01:24:37 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void crst_table_downgrade(struct mm_struct *mm, unsigned long limit)
|
|
|
|
{
|
|
|
|
pgd_t *pgd;
|
|
|
|
|
2014-04-03 19:54:59 +08:00
|
|
|
if (current->active_mm == mm) {
|
s390/uaccess: rework uaccess code - fix locking issues
The current uaccess code uses a page table walk in some circumstances,
e.g. in case of the in atomic futex operations or if running on old
hardware which doesn't support the mvcos instruction.
However it turned out that the page table walk code does not correctly
lock page tables when accessing page table entries.
In other words: a different cpu may invalidate a page table entry while
the current cpu inspects the pte. This may lead to random data corruption.
Adding correct locking however isn't trivial for all uaccess operations.
Especially copy_in_user() is problematic since that requires to hold at
least two locks, but must be protected against ABBA deadlock when a
different cpu also performs a copy_in_user() operation.
So the solution is a different approach where we change address spaces:
User space runs in primary address mode, or access register mode within
vdso code, like it currently already does.
The kernel usually also runs in home space mode, however when accessing
user space the kernel switches to primary or secondary address mode if
the mvcos instruction is not available or if a compare-and-swap (futex)
instruction on a user space address is performed.
KVM however is special, since that requires the kernel to run in home
address space while implicitly accessing user space with the sie
instruction.
So we end up with:
User space:
- runs in primary or access register mode
- cr1 contains the user asce
- cr7 contains the user asce
- cr13 contains the kernel asce
Kernel space:
- runs in home space mode
- cr1 contains the user or kernel asce
-> the kernel asce is loaded when a uaccess requires primary or
secondary address mode
- cr7 contains the user or kernel asce, (changed with set_fs())
- cr13 contains the kernel asce
In case of uaccess the kernel changes to:
- primary space mode in case of a uaccess (copy_to_user) and uses
e.g. the mvcp instruction to access user space. However the kernel
will stay in home space mode if the mvcos instruction is available
- secondary space mode in case of futex atomic operations, so that the
instructions come from primary address space and data from secondary
space
In case of kvm the kernel runs in home space mode, but cr1 gets switched
to contain the gmap asce before the sie instruction gets executed. When
the sie instruction is finished cr1 will be switched back to contain the
user asce.
A context switch between two processes will always load the kernel asce
for the next process in cr1. So the first exit to user space is a bit
more expensive (one extra load control register instruction) than before,
however keeps the code rather simple.
In sum this means there is no need to perform any error prone page table
walks anymore when accessing user space.
The patch seems to be rather large, however it mainly removes the
the page table walk code and restores the previously deleted "standard"
uaccess code, with a couple of changes.
The uaccess without mvcos mode can be enforced with the "uaccess_primary"
kernel parameter.
Reported-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2014-03-21 17:42:25 +08:00
|
|
|
clear_user_asce(mm, 1);
|
2013-10-28 21:48:30 +08:00
|
|
|
__tlb_flush_mm(mm);
|
2014-04-03 19:54:59 +08:00
|
|
|
}
|
2008-02-10 01:24:37 +08:00
|
|
|
while (mm->context.asce_limit > limit) {
|
|
|
|
pgd = mm->pgd;
|
|
|
|
switch (pgd_val(*pgd) & _REGION_ENTRY_TYPE_MASK) {
|
|
|
|
case _REGION_ENTRY_TYPE_R2:
|
|
|
|
mm->context.asce_limit = 1UL << 42;
|
|
|
|
mm->context.asce_bits = _ASCE_TABLE_LENGTH |
|
|
|
|
_ASCE_USER_BITS |
|
|
|
|
_ASCE_TYPE_REGION3;
|
|
|
|
break;
|
|
|
|
case _REGION_ENTRY_TYPE_R3:
|
|
|
|
mm->context.asce_limit = 1UL << 31;
|
|
|
|
mm->context.asce_bits = _ASCE_TABLE_LENGTH |
|
|
|
|
_ASCE_USER_BITS |
|
|
|
|
_ASCE_TYPE_SEGMENT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
}
|
|
|
|
mm->pgd = (pgd_t *) (pgd_val(*pgd) & _REGION_ENTRY_ORIGIN);
|
2009-03-18 20:27:36 +08:00
|
|
|
mm->task_size = mm->context.asce_limit;
|
2008-02-10 01:24:37 +08:00
|
|
|
crst_table_free(mm, (unsigned long *) pgd);
|
|
|
|
}
|
2013-10-28 21:48:30 +08:00
|
|
|
if (current->active_mm == mm)
|
s390/uaccess: rework uaccess code - fix locking issues
The current uaccess code uses a page table walk in some circumstances,
e.g. in case of the in atomic futex operations or if running on old
hardware which doesn't support the mvcos instruction.
However it turned out that the page table walk code does not correctly
lock page tables when accessing page table entries.
In other words: a different cpu may invalidate a page table entry while
the current cpu inspects the pte. This may lead to random data corruption.
Adding correct locking however isn't trivial for all uaccess operations.
Especially copy_in_user() is problematic since that requires to hold at
least two locks, but must be protected against ABBA deadlock when a
different cpu also performs a copy_in_user() operation.
So the solution is a different approach where we change address spaces:
User space runs in primary address mode, or access register mode within
vdso code, like it currently already does.
The kernel usually also runs in home space mode, however when accessing
user space the kernel switches to primary or secondary address mode if
the mvcos instruction is not available or if a compare-and-swap (futex)
instruction on a user space address is performed.
KVM however is special, since that requires the kernel to run in home
address space while implicitly accessing user space with the sie
instruction.
So we end up with:
User space:
- runs in primary or access register mode
- cr1 contains the user asce
- cr7 contains the user asce
- cr13 contains the kernel asce
Kernel space:
- runs in home space mode
- cr1 contains the user or kernel asce
-> the kernel asce is loaded when a uaccess requires primary or
secondary address mode
- cr7 contains the user or kernel asce, (changed with set_fs())
- cr13 contains the kernel asce
In case of uaccess the kernel changes to:
- primary space mode in case of a uaccess (copy_to_user) and uses
e.g. the mvcp instruction to access user space. However the kernel
will stay in home space mode if the mvcos instruction is available
- secondary space mode in case of futex atomic operations, so that the
instructions come from primary address space and data from secondary
space
In case of kvm the kernel runs in home space mode, but cr1 gets switched
to contain the gmap asce before the sie instruction gets executed. When
the sie instruction is finished cr1 will be switched back to contain the
user asce.
A context switch between two processes will always load the kernel asce
for the next process in cr1. So the first exit to user space is a bit
more expensive (one extra load control register instruction) than before,
however keeps the code rather simple.
In sum this means there is no need to perform any error prone page table
walks anymore when accessing user space.
The patch seems to be rather large, however it mainly removes the
the page table walk code and restores the previously deleted "standard"
uaccess code, with a couple of changes.
The uaccess without mvcos mode can be enforced with the "uaccess_primary"
kernel parameter.
Reported-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2014-03-21 17:42:25 +08:00
|
|
|
update_user_asce(mm, 1);
|
2008-02-10 01:24:37 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
#ifdef CONFIG_PGSTE
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_alloc - allocate a guest address space
|
|
|
|
* @mm: pointer to the parent mm_struct
|
|
|
|
*
|
|
|
|
* Returns a guest address space structure.
|
|
|
|
*/
|
|
|
|
struct gmap *gmap_alloc(struct mm_struct *mm)
|
2011-06-06 20:14:41 +08:00
|
|
|
{
|
2011-07-24 16:48:20 +08:00
|
|
|
struct gmap *gmap;
|
|
|
|
struct page *page;
|
|
|
|
unsigned long *table;
|
2011-06-06 20:14:41 +08:00
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
gmap = kzalloc(sizeof(struct gmap), GFP_KERNEL);
|
|
|
|
if (!gmap)
|
|
|
|
goto out;
|
|
|
|
INIT_LIST_HEAD(&gmap->crst_list);
|
|
|
|
gmap->mm = mm;
|
|
|
|
page = alloc_pages(GFP_KERNEL, ALLOC_ORDER);
|
|
|
|
if (!page)
|
|
|
|
goto out_free;
|
|
|
|
list_add(&page->lru, &gmap->crst_list);
|
|
|
|
table = (unsigned long *) page_to_phys(page);
|
|
|
|
crst_table_init(table, _REGION1_ENTRY_EMPTY);
|
|
|
|
gmap->table = table;
|
2011-09-20 23:07:28 +08:00
|
|
|
gmap->asce = _ASCE_TYPE_REGION1 | _ASCE_TABLE_LENGTH |
|
|
|
|
_ASCE_USER_BITS | __pa(table);
|
2011-07-24 16:48:20 +08:00
|
|
|
list_add(&gmap->list, &mm->context.gmap_list);
|
|
|
|
return gmap;
|
|
|
|
|
|
|
|
out_free:
|
|
|
|
kfree(gmap);
|
|
|
|
out:
|
|
|
|
return NULL;
|
2011-06-06 20:14:41 +08:00
|
|
|
}
|
2011-07-24 16:48:20 +08:00
|
|
|
EXPORT_SYMBOL_GPL(gmap_alloc);
|
2011-06-06 20:14:41 +08:00
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
static int gmap_unlink_segment(struct gmap *gmap, unsigned long *table)
|
|
|
|
{
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct gmap_rmap *rmap;
|
|
|
|
struct page *page;
|
|
|
|
|
2013-07-24 02:57:57 +08:00
|
|
|
if (*table & _SEGMENT_ENTRY_INVALID)
|
2011-07-24 16:48:20 +08:00
|
|
|
return 0;
|
|
|
|
page = pfn_to_page(*table >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
list_for_each_entry(rmap, &mp->mapper, list) {
|
|
|
|
if (rmap->entry != table)
|
|
|
|
continue;
|
|
|
|
list_del(&rmap->list);
|
|
|
|
kfree(rmap);
|
|
|
|
break;
|
|
|
|
}
|
2013-07-24 02:57:57 +08:00
|
|
|
*table = mp->vmaddr | _SEGMENT_ENTRY_INVALID | _SEGMENT_ENTRY_PROTECT;
|
2011-07-24 16:48:20 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gmap_flush_tlb(struct gmap *gmap)
|
|
|
|
{
|
|
|
|
if (MACHINE_HAS_IDTE)
|
2014-04-03 19:55:01 +08:00
|
|
|
__tlb_flush_asce(gmap->mm, (unsigned long) gmap->table |
|
2011-07-24 16:48:20 +08:00
|
|
|
_ASCE_TYPE_REGION1);
|
|
|
|
else
|
|
|
|
__tlb_flush_global();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_free - free a guest address space
|
|
|
|
* @gmap: pointer to the guest address space structure
|
2007-10-22 18:52:47 +08:00
|
|
|
*/
|
2011-07-24 16:48:20 +08:00
|
|
|
void gmap_free(struct gmap *gmap)
|
|
|
|
{
|
|
|
|
struct page *page, *next;
|
|
|
|
unsigned long *table;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
/* Flush tlb. */
|
|
|
|
if (MACHINE_HAS_IDTE)
|
2014-04-03 19:55:01 +08:00
|
|
|
__tlb_flush_asce(gmap->mm, (unsigned long) gmap->table |
|
2011-07-24 16:48:20 +08:00
|
|
|
_ASCE_TYPE_REGION1);
|
|
|
|
else
|
|
|
|
__tlb_flush_global();
|
|
|
|
|
|
|
|
/* Free all segment & region tables. */
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_lock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
list_for_each_entry_safe(page, next, &gmap->crst_list, lru) {
|
|
|
|
table = (unsigned long *) page_to_phys(page);
|
|
|
|
if ((*table & _REGION_ENTRY_TYPE_MASK) == 0)
|
|
|
|
/* Remove gmap rmap structures for segment table. */
|
|
|
|
for (i = 0; i < PTRS_PER_PMD; i++, table++)
|
|
|
|
gmap_unlink_segment(gmap, table);
|
|
|
|
__free_pages(page, ALLOC_ORDER);
|
|
|
|
}
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_unlock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
list_del(&gmap->list);
|
|
|
|
kfree(gmap);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_free);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_enable - switch primary space to the guest address space
|
|
|
|
* @gmap: pointer to the guest address space structure
|
|
|
|
*/
|
|
|
|
void gmap_enable(struct gmap *gmap)
|
|
|
|
{
|
|
|
|
S390_lowcore.gmap = (unsigned long) gmap;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_enable);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_disable - switch back to the standard primary address space
|
|
|
|
* @gmap: pointer to the guest address space structure
|
|
|
|
*/
|
|
|
|
void gmap_disable(struct gmap *gmap)
|
|
|
|
{
|
|
|
|
S390_lowcore.gmap = 0UL;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_disable);
|
|
|
|
|
2011-10-30 22:17:00 +08:00
|
|
|
/*
|
|
|
|
* gmap_alloc_table is assumed to be called with mmap_sem held
|
|
|
|
*/
|
2011-07-24 16:48:20 +08:00
|
|
|
static int gmap_alloc_table(struct gmap *gmap,
|
2013-09-07 00:48:58 +08:00
|
|
|
unsigned long *table, unsigned long init)
|
|
|
|
__releases(&gmap->mm->page_table_lock)
|
|
|
|
__acquires(&gmap->mm->page_table_lock)
|
2011-07-24 16:48:20 +08:00
|
|
|
{
|
|
|
|
struct page *page;
|
|
|
|
unsigned long *new;
|
|
|
|
|
[S390] kvm: fix sleeping function ... at mm/page_alloc.c:2260
commit cc772456ac9b460693492b3a3d89e8c81eda5874
[S390] fix list corruption in gmap reverse mapping
added a potential dead lock:
BUG: sleeping function called from invalid context at mm/page_alloc.c:2260
in_atomic(): 1, irqs_disabled(): 0, pid: 1108, name: qemu-system-s39
3 locks held by qemu-system-s39/1108:
#0: (&kvm->slots_lock){+.+.+.}, at: [<000003e004866542>] kvm_set_memory_region+0x3a/0x6c [kvm]
#1: (&mm->mmap_sem){++++++}, at: [<0000000000123790>] gmap_map_segment+0x9c/0x298
#2: (&(&mm->page_table_lock)->rlock){+.+.+.}, at: [<00000000001237a8>] gmap_map_segment+0xb4/0x298
CPU: 0 Not tainted 3.1.3 #45
Process qemu-system-s39 (pid: 1108, task: 00000004f8b3cb30, ksp: 00000004fd5978d0)
00000004fd5979a0 00000004fd597920 0000000000000002 0000000000000000
00000004fd5979c0 00000004fd597938 00000004fd597938 0000000000617e96
0000000000000000 00000004f8b3cf58 0000000000000000 0000000000000000
000000000000000d 000000000000000c 00000004fd597988 0000000000000000
0000000000000000 0000000000100a18 00000004fd597920 00000004fd597960
Call Trace:
([<0000000000100926>] show_trace+0xee/0x144)
[<0000000000131f3a>] __might_sleep+0x12a/0x158
[<0000000000217fb4>] __alloc_pages_nodemask+0x224/0xadc
[<0000000000123086>] gmap_alloc_table+0x46/0x114
[<000000000012395c>] gmap_map_segment+0x268/0x298
[<000003e00486b014>] kvm_arch_commit_memory_region+0x44/0x6c [kvm]
[<000003e004866414>] __kvm_set_memory_region+0x3b0/0x4a4 [kvm]
[<000003e004866554>] kvm_set_memory_region+0x4c/0x6c [kvm]
[<000003e004867c7a>] kvm_vm_ioctl+0x14a/0x314 [kvm]
[<0000000000292100>] do_vfs_ioctl+0x94/0x588
[<0000000000292688>] SyS_ioctl+0x94/0xac
[<000000000061e124>] sysc_noemu+0x22/0x28
[<000003fffcd5e7ca>] 0x3fffcd5e7ca
3 locks held by qemu-system-s39/1108:
#0: (&kvm->slots_lock){+.+.+.}, at: [<000003e004866542>] kvm_set_memory_region+0x3a/0x6c [kvm]
#1: (&mm->mmap_sem){++++++}, at: [<0000000000123790>] gmap_map_segment+0x9c/0x298
#2: (&(&mm->page_table_lock)->rlock){+.+.+.}, at: [<00000000001237a8>] gmap_map_segment+0xb4/0x298
Fix this by freeing the lock on the alloc path. This is ok, since the
gmap table is never freed until we call gmap_free, so the table we are
walking cannot go.
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2011-12-27 18:25:47 +08:00
|
|
|
/* since we dont free the gmap table until gmap_free we can unlock */
|
|
|
|
spin_unlock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
page = alloc_pages(GFP_KERNEL, ALLOC_ORDER);
|
[S390] kvm: fix sleeping function ... at mm/page_alloc.c:2260
commit cc772456ac9b460693492b3a3d89e8c81eda5874
[S390] fix list corruption in gmap reverse mapping
added a potential dead lock:
BUG: sleeping function called from invalid context at mm/page_alloc.c:2260
in_atomic(): 1, irqs_disabled(): 0, pid: 1108, name: qemu-system-s39
3 locks held by qemu-system-s39/1108:
#0: (&kvm->slots_lock){+.+.+.}, at: [<000003e004866542>] kvm_set_memory_region+0x3a/0x6c [kvm]
#1: (&mm->mmap_sem){++++++}, at: [<0000000000123790>] gmap_map_segment+0x9c/0x298
#2: (&(&mm->page_table_lock)->rlock){+.+.+.}, at: [<00000000001237a8>] gmap_map_segment+0xb4/0x298
CPU: 0 Not tainted 3.1.3 #45
Process qemu-system-s39 (pid: 1108, task: 00000004f8b3cb30, ksp: 00000004fd5978d0)
00000004fd5979a0 00000004fd597920 0000000000000002 0000000000000000
00000004fd5979c0 00000004fd597938 00000004fd597938 0000000000617e96
0000000000000000 00000004f8b3cf58 0000000000000000 0000000000000000
000000000000000d 000000000000000c 00000004fd597988 0000000000000000
0000000000000000 0000000000100a18 00000004fd597920 00000004fd597960
Call Trace:
([<0000000000100926>] show_trace+0xee/0x144)
[<0000000000131f3a>] __might_sleep+0x12a/0x158
[<0000000000217fb4>] __alloc_pages_nodemask+0x224/0xadc
[<0000000000123086>] gmap_alloc_table+0x46/0x114
[<000000000012395c>] gmap_map_segment+0x268/0x298
[<000003e00486b014>] kvm_arch_commit_memory_region+0x44/0x6c [kvm]
[<000003e004866414>] __kvm_set_memory_region+0x3b0/0x4a4 [kvm]
[<000003e004866554>] kvm_set_memory_region+0x4c/0x6c [kvm]
[<000003e004867c7a>] kvm_vm_ioctl+0x14a/0x314 [kvm]
[<0000000000292100>] do_vfs_ioctl+0x94/0x588
[<0000000000292688>] SyS_ioctl+0x94/0xac
[<000000000061e124>] sysc_noemu+0x22/0x28
[<000003fffcd5e7ca>] 0x3fffcd5e7ca
3 locks held by qemu-system-s39/1108:
#0: (&kvm->slots_lock){+.+.+.}, at: [<000003e004866542>] kvm_set_memory_region+0x3a/0x6c [kvm]
#1: (&mm->mmap_sem){++++++}, at: [<0000000000123790>] gmap_map_segment+0x9c/0x298
#2: (&(&mm->page_table_lock)->rlock){+.+.+.}, at: [<00000000001237a8>] gmap_map_segment+0xb4/0x298
Fix this by freeing the lock on the alloc path. This is ok, since the
gmap table is never freed until we call gmap_free, so the table we are
walking cannot go.
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2011-12-27 18:25:47 +08:00
|
|
|
spin_lock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
if (!page)
|
|
|
|
return -ENOMEM;
|
|
|
|
new = (unsigned long *) page_to_phys(page);
|
|
|
|
crst_table_init(new, init);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (*table & _REGION_ENTRY_INVALID) {
|
2011-07-24 16:48:20 +08:00
|
|
|
list_add(&page->lru, &gmap->crst_list);
|
|
|
|
*table = (unsigned long) new | _REGION_ENTRY_LENGTH |
|
|
|
|
(*table & _REGION_ENTRY_TYPE_MASK);
|
|
|
|
} else
|
|
|
|
__free_pages(page, ALLOC_ORDER);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_unmap_segment - unmap segment from the guest address space
|
|
|
|
* @gmap: pointer to the guest address space structure
|
|
|
|
* @addr: address in the guest address space
|
|
|
|
* @len: length of the memory area to unmap
|
|
|
|
*
|
2013-12-13 19:53:42 +08:00
|
|
|
* Returns 0 if the unmap succeeded, -EINVAL if not.
|
2011-07-24 16:48:20 +08:00
|
|
|
*/
|
|
|
|
int gmap_unmap_segment(struct gmap *gmap, unsigned long to, unsigned long len)
|
|
|
|
{
|
|
|
|
unsigned long *table;
|
|
|
|
unsigned long off;
|
|
|
|
int flush;
|
|
|
|
|
|
|
|
if ((to | len) & (PMD_SIZE - 1))
|
|
|
|
return -EINVAL;
|
|
|
|
if (len == 0 || to + len < to)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
flush = 0;
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_lock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
for (off = 0; off < len; off += PMD_SIZE) {
|
|
|
|
/* Walk the guest addr space page table */
|
|
|
|
table = gmap->table + (((to + off) >> 53) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (*table & _REGION_ENTRY_INVALID)
|
2011-09-26 22:40:34 +08:00
|
|
|
goto out;
|
2011-07-24 16:48:20 +08:00
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 42) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (*table & _REGION_ENTRY_INVALID)
|
2011-09-26 22:40:34 +08:00
|
|
|
goto out;
|
2011-07-24 16:48:20 +08:00
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 31) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (*table & _REGION_ENTRY_INVALID)
|
2011-09-26 22:40:34 +08:00
|
|
|
goto out;
|
2011-07-24 16:48:20 +08:00
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 20) & 0x7ff);
|
|
|
|
|
|
|
|
/* Clear segment table entry in guest address space. */
|
|
|
|
flush |= gmap_unlink_segment(gmap, table);
|
2013-07-24 02:57:57 +08:00
|
|
|
*table = _SEGMENT_ENTRY_INVALID;
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
2011-09-26 22:40:34 +08:00
|
|
|
out:
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_unlock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
if (flush)
|
|
|
|
gmap_flush_tlb(gmap);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_unmap_segment);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_mmap_segment - map a segment to the guest address space
|
|
|
|
* @gmap: pointer to the guest address space structure
|
|
|
|
* @from: source address in the parent address space
|
|
|
|
* @to: target address in the guest address space
|
|
|
|
*
|
2013-12-13 19:53:42 +08:00
|
|
|
* Returns 0 if the mmap succeeded, -EINVAL or -ENOMEM if not.
|
2011-07-24 16:48:20 +08:00
|
|
|
*/
|
|
|
|
int gmap_map_segment(struct gmap *gmap, unsigned long from,
|
|
|
|
unsigned long to, unsigned long len)
|
|
|
|
{
|
|
|
|
unsigned long *table;
|
|
|
|
unsigned long off;
|
|
|
|
int flush;
|
|
|
|
|
|
|
|
if ((from | to | len) & (PMD_SIZE - 1))
|
|
|
|
return -EINVAL;
|
2013-07-26 21:04:03 +08:00
|
|
|
if (len == 0 || from + len > TASK_MAX_SIZE ||
|
2011-07-24 16:48:20 +08:00
|
|
|
from + len < from || to + len < to)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
flush = 0;
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_lock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
for (off = 0; off < len; off += PMD_SIZE) {
|
|
|
|
/* Walk the gmap address space page table */
|
|
|
|
table = gmap->table + (((to + off) >> 53) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if ((*table & _REGION_ENTRY_INVALID) &&
|
2011-07-24 16:48:20 +08:00
|
|
|
gmap_alloc_table(gmap, table, _REGION2_ENTRY_EMPTY))
|
|
|
|
goto out_unmap;
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 42) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if ((*table & _REGION_ENTRY_INVALID) &&
|
2011-07-24 16:48:20 +08:00
|
|
|
gmap_alloc_table(gmap, table, _REGION3_ENTRY_EMPTY))
|
|
|
|
goto out_unmap;
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 31) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if ((*table & _REGION_ENTRY_INVALID) &&
|
2011-07-24 16:48:20 +08:00
|
|
|
gmap_alloc_table(gmap, table, _SEGMENT_ENTRY_EMPTY))
|
|
|
|
goto out_unmap;
|
|
|
|
table = (unsigned long *) (*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + (((to + off) >> 20) & 0x7ff);
|
|
|
|
|
|
|
|
/* Store 'from' address in an invalid segment table entry. */
|
|
|
|
flush |= gmap_unlink_segment(gmap, table);
|
2013-07-24 02:57:57 +08:00
|
|
|
*table = (from + off) | (_SEGMENT_ENTRY_INVALID |
|
|
|
|
_SEGMENT_ENTRY_PROTECT);
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_unlock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
if (flush)
|
|
|
|
gmap_flush_tlb(gmap);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_unmap:
|
2011-10-30 22:17:01 +08:00
|
|
|
spin_unlock(&gmap->mm->page_table_lock);
|
2011-07-24 16:48:20 +08:00
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
gmap_unmap_segment(gmap, to, len);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_map_segment);
|
|
|
|
|
2012-09-10 22:14:33 +08:00
|
|
|
static unsigned long *gmap_table_walk(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long *table;
|
|
|
|
|
|
|
|
table = gmap->table + ((address >> 53) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID))
|
2012-09-10 22:14:33 +08:00
|
|
|
return ERR_PTR(-EFAULT);
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 42) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID))
|
2012-09-10 22:14:33 +08:00
|
|
|
return ERR_PTR(-EFAULT);
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 31) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID))
|
2012-09-10 22:14:33 +08:00
|
|
|
return ERR_PTR(-EFAULT);
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 20) & 0x7ff);
|
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* __gmap_translate - translate a guest address to a user space address
|
|
|
|
* @address: guest address
|
|
|
|
* @gmap: pointer to guest mapping meta data structure
|
|
|
|
*
|
|
|
|
* Returns user space address which corresponds to the guest address or
|
|
|
|
* -EFAULT if no such mapping exists.
|
|
|
|
* This function does not establish potentially missing page table entries.
|
|
|
|
* The mmap_sem of the mm that belongs to the address space must be held
|
|
|
|
* when this function gets called.
|
|
|
|
*/
|
|
|
|
unsigned long __gmap_translate(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long *segment_ptr, vmaddr, segment;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct page *page;
|
|
|
|
|
|
|
|
current->thread.gmap_addr = address;
|
|
|
|
segment_ptr = gmap_table_walk(address, gmap);
|
|
|
|
if (IS_ERR(segment_ptr))
|
|
|
|
return PTR_ERR(segment_ptr);
|
|
|
|
/* Convert the gmap address to an mm address. */
|
|
|
|
segment = *segment_ptr;
|
2013-07-24 02:57:57 +08:00
|
|
|
if (!(segment & _SEGMENT_ENTRY_INVALID)) {
|
2012-09-10 22:14:33 +08:00
|
|
|
page = pfn_to_page(segment >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
return mp->vmaddr | (address & ~PMD_MASK);
|
2013-07-24 02:57:57 +08:00
|
|
|
} else if (segment & _SEGMENT_ENTRY_PROTECT) {
|
2012-09-10 22:14:33 +08:00
|
|
|
vmaddr = segment & _SEGMENT_ENTRY_ORIGIN;
|
|
|
|
return vmaddr | (address & ~PMD_MASK);
|
|
|
|
}
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(__gmap_translate);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_translate - translate a guest address to a user space address
|
|
|
|
* @address: guest address
|
|
|
|
* @gmap: pointer to guest mapping meta data structure
|
|
|
|
*
|
|
|
|
* Returns user space address which corresponds to the guest address or
|
|
|
|
* -EFAULT if no such mapping exists.
|
|
|
|
* This function does not establish potentially missing page table entries.
|
|
|
|
*/
|
|
|
|
unsigned long gmap_translate(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long rc;
|
|
|
|
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
|
|
|
rc = __gmap_translate(address, gmap);
|
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_translate);
|
|
|
|
|
2013-04-17 16:53:39 +08:00
|
|
|
static int gmap_connect_pgtable(unsigned long address, unsigned long segment,
|
|
|
|
unsigned long *segment_ptr, struct gmap *gmap)
|
2011-07-24 16:48:20 +08:00
|
|
|
{
|
2013-04-16 19:37:46 +08:00
|
|
|
unsigned long vmaddr;
|
2012-09-10 22:14:33 +08:00
|
|
|
struct vm_area_struct *vma;
|
2011-07-24 16:48:20 +08:00
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct gmap_rmap *rmap;
|
2012-09-10 22:14:33 +08:00
|
|
|
struct mm_struct *mm;
|
2011-07-24 16:48:20 +08:00
|
|
|
struct page *page;
|
|
|
|
pgd_t *pgd;
|
|
|
|
pud_t *pud;
|
|
|
|
pmd_t *pmd;
|
|
|
|
|
2013-04-16 19:37:46 +08:00
|
|
|
mm = gmap->mm;
|
|
|
|
vmaddr = segment & _SEGMENT_ENTRY_ORIGIN;
|
|
|
|
vma = find_vma(mm, vmaddr);
|
|
|
|
if (!vma || vma->vm_start > vmaddr)
|
|
|
|
return -EFAULT;
|
|
|
|
/* Walk the parent mm page table */
|
|
|
|
pgd = pgd_offset(mm, vmaddr);
|
|
|
|
pud = pud_alloc(mm, pgd, vmaddr);
|
|
|
|
if (!pud)
|
|
|
|
return -ENOMEM;
|
|
|
|
pmd = pmd_alloc(mm, pud, vmaddr);
|
|
|
|
if (!pmd)
|
|
|
|
return -ENOMEM;
|
|
|
|
if (!pmd_present(*pmd) &&
|
|
|
|
__pte_alloc(mm, vma, pmd, vmaddr))
|
|
|
|
return -ENOMEM;
|
mm: revert "thp: make MADV_HUGEPAGE check for mm->def_flags"
The main motivation behind this patch is to provide a way to disable THP
for jobs where the code cannot be modified, and using a malloc hook with
madvise is not an option (i.e. statically allocated data). This patch
allows us to do just that, without affecting other jobs running on the
system.
We need to do this sort of thing for jobs where THP hurts performance,
due to the possibility of increased remote memory accesses that can be
created by situations such as the following:
When you touch 1 byte of an untouched, contiguous 2MB chunk, a THP will
be handed out, and the THP will be stuck on whatever node the chunk was
originally referenced from. If many remote nodes need to do work on
that same chunk, they'll be making remote accesses.
With THP disabled, 4K pages can be handed out to separate nodes as
they're needed, greatly reducing the amount of remote accesses to
memory.
This patch is based on some of my work combined with some
suggestions/patches given by Oleg Nesterov. The main goal here is to
add a prctl switch to allow us to disable to THP on a per mm_struct
basis.
Here's a bit of test data with the new patch in place...
First with the flag unset:
# perf stat -a ./prctl_wrapper_mmv3 0 ./thp_pthread -C 0 -m 0 -c 512 -b 256g
Setting thp_disabled for this task...
thp_disable: 0
Set thp_disabled state to 0
Process pid = 18027
PF/
MAX MIN TOTCPU/ TOT_PF/ TOT_PF/ WSEC/
TYPE: CPUS WALL WALL SYS USER TOTCPU CPU WALL_SEC SYS_SEC CPU NODES
512 1.120 0.060 0.000 0.110 0.110 0.000 28571428864 -9223372036854775808 55803572 23
Performance counter stats for './prctl_wrapper_mmv3_hack 0 ./thp_pthread -C 0 -m 0 -c 512 -b 256g':
273719072.841402 task-clock # 641.026 CPUs utilized [100.00%]
1,008,986 context-switches # 0.000 M/sec [100.00%]
7,717 CPU-migrations # 0.000 M/sec [100.00%]
1,698,932 page-faults # 0.000 M/sec
355,222,544,890,379 cycles # 1.298 GHz [100.00%]
536,445,412,234,588 stalled-cycles-frontend # 151.02% frontend cycles idle [100.00%]
409,110,531,310,223 stalled-cycles-backend # 115.17% backend cycles idle [100.00%]
148,286,797,266,411 instructions # 0.42 insns per cycle
# 3.62 stalled cycles per insn [100.00%]
27,061,793,159,503 branches # 98.867 M/sec [100.00%]
1,188,655,196 branch-misses # 0.00% of all branches
427.001706337 seconds time elapsed
Now with the flag set:
# perf stat -a ./prctl_wrapper_mmv3 1 ./thp_pthread -C 0 -m 0 -c 512 -b 256g
Setting thp_disabled for this task...
thp_disable: 1
Set thp_disabled state to 1
Process pid = 144957
PF/
MAX MIN TOTCPU/ TOT_PF/ TOT_PF/ WSEC/
TYPE: CPUS WALL WALL SYS USER TOTCPU CPU WALL_SEC SYS_SEC CPU NODES
512 0.620 0.260 0.250 0.320 0.570 0.001 51612901376 128000000000 100806448 23
Performance counter stats for './prctl_wrapper_mmv3_hack 1 ./thp_pthread -C 0 -m 0 -c 512 -b 256g':
138789390.540183 task-clock # 641.959 CPUs utilized [100.00%]
534,205 context-switches # 0.000 M/sec [100.00%]
4,595 CPU-migrations # 0.000 M/sec [100.00%]
63,133,119 page-faults # 0.000 M/sec
147,977,747,269,768 cycles # 1.066 GHz [100.00%]
200,524,196,493,108 stalled-cycles-frontend # 135.51% frontend cycles idle [100.00%]
105,175,163,716,388 stalled-cycles-backend # 71.07% backend cycles idle [100.00%]
180,916,213,503,160 instructions # 1.22 insns per cycle
# 1.11 stalled cycles per insn [100.00%]
26,999,511,005,868 branches # 194.536 M/sec [100.00%]
714,066,351 branch-misses # 0.00% of all branches
216.196778807 seconds time elapsed
As with previous versions of the patch, We're getting about a 2x
performance increase here. Here's a link to the test case I used, along
with the little wrapper to activate the flag:
http://oss.sgi.com/projects/memtests/thp_pthread_mmprctlv3.tar.gz
This patch (of 3):
Revert commit 8e72033f2a48 and add in code to fix up any issues caused
by the revert.
The revert is necessary because hugepage_madvise would return -EINVAL
when VM_NOHUGEPAGE is set, which will break subsequent chunks of this
patch set.
Here's a snip of an e-mail from Gerald detailing the original purpose of
this code, and providing justification for the revert:
"The intent of commit 8e72033f2a48 was to guard against any future
programming errors that may result in an madvice(MADV_HUGEPAGE) on
guest mappings, which would crash the kernel.
Martin suggested adding the bit to arch/s390/mm/pgtable.c, if
8e72033f2a48 was to be reverted, because that check will also prevent
a kernel crash in the case described above, it will now send a
SIGSEGV instead.
This would now also allow to do the madvise on other parts, if
needed, so it is a more flexible approach. One could also say that
it would have been better to do it this way right from the
beginning..."
Signed-off-by: Alex Thorlton <athorlton@sgi.com>
Suggested-by: Oleg Nesterov <oleg@redhat.com>
Tested-by: Christian Borntraeger <borntraeger@de.ibm.com>
Cc: Gerald Schaefer <gerald.schaefer@de.ibm.com>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: "Kirill A. Shutemov" <kirill.shutemov@linux.intel.com>
Cc: Mel Gorman <mgorman@suse.de>
Cc: Rik van Riel <riel@redhat.com>
Cc: Ingo Molnar <mingo@kernel.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: "Eric W. Biederman" <ebiederm@xmission.com>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: David Rientjes <rientjes@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2014-04-08 06:37:09 +08:00
|
|
|
/* large pmds cannot yet be handled */
|
|
|
|
if (pmd_large(*pmd))
|
|
|
|
return -EFAULT;
|
2013-04-16 19:37:46 +08:00
|
|
|
/* pmd now points to a valid segment table entry. */
|
|
|
|
rmap = kmalloc(sizeof(*rmap), GFP_KERNEL|__GFP_REPEAT);
|
|
|
|
if (!rmap)
|
|
|
|
return -ENOMEM;
|
|
|
|
/* Link gmap segment table entry location to page table. */
|
|
|
|
page = pmd_page(*pmd);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
2013-04-17 16:53:39 +08:00
|
|
|
rmap->gmap = gmap;
|
2013-04-16 19:37:46 +08:00
|
|
|
rmap->entry = segment_ptr;
|
2013-05-29 19:08:39 +08:00
|
|
|
rmap->vmaddr = address & PMD_MASK;
|
2013-04-16 19:37:46 +08:00
|
|
|
spin_lock(&mm->page_table_lock);
|
|
|
|
if (*segment_ptr == segment) {
|
|
|
|
list_add(&rmap->list, &mp->mapper);
|
|
|
|
/* Set gmap segment table entry to page table. */
|
|
|
|
*segment_ptr = pmd_val(*pmd) & PAGE_MASK;
|
|
|
|
rmap = NULL;
|
|
|
|
}
|
|
|
|
spin_unlock(&mm->page_table_lock);
|
|
|
|
kfree(rmap);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gmap_disconnect_pgtable(struct mm_struct *mm, unsigned long *table)
|
|
|
|
{
|
|
|
|
struct gmap_rmap *rmap, *next;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct page *page;
|
|
|
|
int flush;
|
|
|
|
|
|
|
|
flush = 0;
|
|
|
|
spin_lock(&mm->page_table_lock);
|
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
list_for_each_entry_safe(rmap, next, &mp->mapper, list) {
|
2013-07-24 02:57:57 +08:00
|
|
|
*rmap->entry = mp->vmaddr | (_SEGMENT_ENTRY_INVALID |
|
|
|
|
_SEGMENT_ENTRY_PROTECT);
|
2013-04-16 19:37:46 +08:00
|
|
|
list_del(&rmap->list);
|
|
|
|
kfree(rmap);
|
|
|
|
flush = 1;
|
|
|
|
}
|
|
|
|
spin_unlock(&mm->page_table_lock);
|
|
|
|
if (flush)
|
|
|
|
__tlb_flush_global();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* this function is assumed to be called with mmap_sem held
|
|
|
|
*/
|
|
|
|
unsigned long __gmap_fault(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long *segment_ptr, segment;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct page *page;
|
|
|
|
int rc;
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
current->thread.gmap_addr = address;
|
2012-09-10 22:14:33 +08:00
|
|
|
segment_ptr = gmap_table_walk(address, gmap);
|
|
|
|
if (IS_ERR(segment_ptr))
|
2011-07-24 16:48:20 +08:00
|
|
|
return -EFAULT;
|
|
|
|
/* Convert the gmap address to an mm address. */
|
2013-04-16 19:37:46 +08:00
|
|
|
while (1) {
|
|
|
|
segment = *segment_ptr;
|
2013-07-24 02:57:57 +08:00
|
|
|
if (!(segment & _SEGMENT_ENTRY_INVALID)) {
|
2013-04-16 19:37:46 +08:00
|
|
|
/* Page table is present */
|
|
|
|
page = pfn_to_page(segment >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
return mp->vmaddr | (address & ~PMD_MASK);
|
|
|
|
}
|
2013-07-24 02:57:57 +08:00
|
|
|
if (!(segment & _SEGMENT_ENTRY_PROTECT))
|
2013-04-16 19:37:46 +08:00
|
|
|
/* Nothing mapped in the gmap address space. */
|
|
|
|
break;
|
2013-04-17 16:53:39 +08:00
|
|
|
rc = gmap_connect_pgtable(address, segment, segment_ptr, gmap);
|
2013-04-16 19:37:46 +08:00
|
|
|
if (rc)
|
|
|
|
return rc;
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
|
|
|
return -EFAULT;
|
2011-10-30 22:17:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned long gmap_fault(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long rc;
|
|
|
|
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
|
|
|
rc = __gmap_fault(address, gmap);
|
|
|
|
up_read(&gmap->mm->mmap_sem);
|
2011-07-24 16:48:20 +08:00
|
|
|
|
2011-10-30 22:17:02 +08:00
|
|
|
return rc;
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_fault);
|
|
|
|
|
2013-04-17 23:36:29 +08:00
|
|
|
static void gmap_zap_swap_entry(swp_entry_t entry, struct mm_struct *mm)
|
|
|
|
{
|
|
|
|
if (!non_swap_entry(entry))
|
|
|
|
dec_mm_counter(mm, MM_SWAPENTS);
|
|
|
|
else if (is_migration_entry(entry)) {
|
|
|
|
struct page *page = migration_entry_to_page(entry);
|
|
|
|
|
|
|
|
if (PageAnon(page))
|
|
|
|
dec_mm_counter(mm, MM_ANONPAGES);
|
|
|
|
else
|
|
|
|
dec_mm_counter(mm, MM_FILEPAGES);
|
|
|
|
}
|
|
|
|
free_swap_and_cache(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The mm->mmap_sem lock must be held
|
|
|
|
*/
|
|
|
|
static void gmap_zap_unused(struct mm_struct *mm, unsigned long address)
|
|
|
|
{
|
|
|
|
unsigned long ptev, pgstev;
|
|
|
|
spinlock_t *ptl;
|
|
|
|
pgste_t pgste;
|
|
|
|
pte_t *ptep, pte;
|
|
|
|
|
|
|
|
ptep = get_locked_pte(mm, address, &ptl);
|
|
|
|
if (unlikely(!ptep))
|
|
|
|
return;
|
|
|
|
pte = *ptep;
|
|
|
|
if (!pte_swap(pte))
|
|
|
|
goto out_pte;
|
|
|
|
/* Zap unused and logically-zero pages */
|
|
|
|
pgste = pgste_get_lock(ptep);
|
|
|
|
pgstev = pgste_val(pgste);
|
|
|
|
ptev = pte_val(pte);
|
|
|
|
if (((pgstev & _PGSTE_GPS_USAGE_MASK) == _PGSTE_GPS_USAGE_UNUSED) ||
|
|
|
|
((pgstev & _PGSTE_GPS_ZERO) && (ptev & _PAGE_INVALID))) {
|
|
|
|
gmap_zap_swap_entry(pte_to_swp_entry(pte), mm);
|
|
|
|
pte_clear(mm, address, ptep);
|
|
|
|
}
|
|
|
|
pgste_set_unlock(ptep, pgste);
|
|
|
|
out_pte:
|
|
|
|
pte_unmap_unlock(*ptep, ptl);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* this function is assumed to be called with mmap_sem held
|
|
|
|
*/
|
|
|
|
void __gmap_zap(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
unsigned long *table, *segment_ptr;
|
|
|
|
unsigned long segment, pgstev, ptev;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct page *page;
|
|
|
|
|
|
|
|
segment_ptr = gmap_table_walk(address, gmap);
|
|
|
|
if (IS_ERR(segment_ptr))
|
|
|
|
return;
|
|
|
|
segment = *segment_ptr;
|
|
|
|
if (segment & _SEGMENT_ENTRY_INVALID)
|
|
|
|
return;
|
|
|
|
page = pfn_to_page(segment >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
address = mp->vmaddr | (address & ~PMD_MASK);
|
|
|
|
/* Page table is present */
|
|
|
|
table = (unsigned long *)(segment & _SEGMENT_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 12) & 0xff);
|
|
|
|
pgstev = table[PTRS_PER_PTE];
|
|
|
|
ptev = table[0];
|
|
|
|
/* quick check, checked again with locks held */
|
|
|
|
if (((pgstev & _PGSTE_GPS_USAGE_MASK) == _PGSTE_GPS_USAGE_UNUSED) ||
|
|
|
|
((pgstev & _PGSTE_GPS_ZERO) && (ptev & _PAGE_INVALID)))
|
|
|
|
gmap_zap_unused(gmap->mm, address);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(__gmap_zap);
|
|
|
|
|
2011-10-30 22:17:03 +08:00
|
|
|
void gmap_discard(unsigned long from, unsigned long to, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
|
|
|
|
unsigned long *table, address, size;
|
|
|
|
struct vm_area_struct *vma;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct page *page;
|
|
|
|
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
|
|
|
address = from;
|
|
|
|
while (address < to) {
|
|
|
|
/* Walk the gmap address space page table */
|
|
|
|
table = gmap->table + ((address >> 53) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID)) {
|
2011-10-30 22:17:03 +08:00
|
|
|
address = (address + PMD_SIZE) & PMD_MASK;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 42) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID)) {
|
2011-10-30 22:17:03 +08:00
|
|
|
address = (address + PMD_SIZE) & PMD_MASK;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 31) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _REGION_ENTRY_INVALID)) {
|
2011-10-30 22:17:03 +08:00
|
|
|
address = (address + PMD_SIZE) & PMD_MASK;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
table = (unsigned long *)(*table & _REGION_ENTRY_ORIGIN);
|
|
|
|
table = table + ((address >> 20) & 0x7ff);
|
2013-07-24 02:57:57 +08:00
|
|
|
if (unlikely(*table & _SEGMENT_ENTRY_INVALID)) {
|
2011-10-30 22:17:03 +08:00
|
|
|
address = (address + PMD_SIZE) & PMD_MASK;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
page = pfn_to_page(*table >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
vma = find_vma(gmap->mm, mp->vmaddr);
|
|
|
|
size = min(to - address, PMD_SIZE - (address & ~PMD_MASK));
|
|
|
|
zap_page_range(vma, mp->vmaddr | (address & ~PMD_MASK),
|
|
|
|
size, NULL);
|
|
|
|
address = (address + PMD_SIZE) & PMD_MASK;
|
|
|
|
}
|
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_discard);
|
|
|
|
|
2013-04-17 16:53:39 +08:00
|
|
|
static LIST_HEAD(gmap_notifier_list);
|
|
|
|
static DEFINE_SPINLOCK(gmap_notifier_lock);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_register_ipte_notifier - register a pte invalidation callback
|
|
|
|
* @nb: pointer to the gmap notifier block
|
|
|
|
*/
|
|
|
|
void gmap_register_ipte_notifier(struct gmap_notifier *nb)
|
|
|
|
{
|
|
|
|
spin_lock(&gmap_notifier_lock);
|
|
|
|
list_add(&nb->list, &gmap_notifier_list);
|
|
|
|
spin_unlock(&gmap_notifier_lock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_register_ipte_notifier);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_unregister_ipte_notifier - remove a pte invalidation callback
|
|
|
|
* @nb: pointer to the gmap notifier block
|
|
|
|
*/
|
|
|
|
void gmap_unregister_ipte_notifier(struct gmap_notifier *nb)
|
|
|
|
{
|
|
|
|
spin_lock(&gmap_notifier_lock);
|
|
|
|
list_del_init(&nb->list);
|
|
|
|
spin_unlock(&gmap_notifier_lock);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_unregister_ipte_notifier);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_ipte_notify - mark a range of ptes for invalidation notification
|
|
|
|
* @gmap: pointer to guest mapping meta data structure
|
2014-03-19 17:13:22 +08:00
|
|
|
* @start: virtual address in the guest address space
|
2013-04-17 16:53:39 +08:00
|
|
|
* @len: size of area
|
|
|
|
*
|
|
|
|
* Returns 0 if for each page in the given range a gmap mapping exists and
|
|
|
|
* the invalidation notification could be set. If the gmap mapping is missing
|
|
|
|
* for one or more pages -EFAULT is returned. If no memory could be allocated
|
|
|
|
* -ENOMEM is returned. This function establishes missing page table entries.
|
|
|
|
*/
|
|
|
|
int gmap_ipte_notify(struct gmap *gmap, unsigned long start, unsigned long len)
|
|
|
|
{
|
|
|
|
unsigned long addr;
|
|
|
|
spinlock_t *ptl;
|
|
|
|
pte_t *ptep, entry;
|
|
|
|
pgste_t pgste;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if ((start & ~PAGE_MASK) || (len & ~PAGE_MASK))
|
|
|
|
return -EINVAL;
|
|
|
|
down_read(&gmap->mm->mmap_sem);
|
|
|
|
while (len) {
|
|
|
|
/* Convert gmap address and connect the page tables */
|
|
|
|
addr = __gmap_fault(start, gmap);
|
|
|
|
if (IS_ERR_VALUE(addr)) {
|
|
|
|
rc = addr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Get the page mapped */
|
2013-05-08 21:25:38 +08:00
|
|
|
if (fixup_user_fault(current, gmap->mm, addr, FAULT_FLAG_WRITE)) {
|
2013-04-17 16:53:39 +08:00
|
|
|
rc = -EFAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Walk the process page table, lock and get pte pointer */
|
|
|
|
ptep = get_locked_pte(gmap->mm, addr, &ptl);
|
|
|
|
if (unlikely(!ptep))
|
|
|
|
continue;
|
|
|
|
/* Set notification bit in the pgste of the pte */
|
|
|
|
entry = *ptep;
|
2013-07-24 02:57:57 +08:00
|
|
|
if ((pte_val(entry) & (_PAGE_INVALID | _PAGE_PROTECT)) == 0) {
|
2013-04-17 16:53:39 +08:00
|
|
|
pgste = pgste_get_lock(ptep);
|
2013-05-17 20:41:33 +08:00
|
|
|
pgste_val(pgste) |= PGSTE_IN_BIT;
|
2013-04-17 16:53:39 +08:00
|
|
|
pgste_set_unlock(ptep, pgste);
|
|
|
|
start += PAGE_SIZE;
|
|
|
|
len -= PAGE_SIZE;
|
|
|
|
}
|
|
|
|
spin_unlock(ptl);
|
|
|
|
}
|
|
|
|
up_read(&gmap->mm->mmap_sem);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_ipte_notify);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* gmap_do_ipte_notify - call all invalidation callbacks for a specific pte.
|
|
|
|
* @mm: pointer to the process mm_struct
|
|
|
|
* @pte: pointer to the page table entry
|
|
|
|
*
|
|
|
|
* This function is assumed to be called with the page table lock held
|
|
|
|
* for the pte to notify.
|
|
|
|
*/
|
2014-03-19 17:18:49 +08:00
|
|
|
void gmap_do_ipte_notify(struct mm_struct *mm, pte_t *pte)
|
2013-04-17 16:53:39 +08:00
|
|
|
{
|
|
|
|
unsigned long segment_offset;
|
|
|
|
struct gmap_notifier *nb;
|
|
|
|
struct gmap_pgtable *mp;
|
|
|
|
struct gmap_rmap *rmap;
|
|
|
|
struct page *page;
|
|
|
|
|
|
|
|
segment_offset = ((unsigned long) pte) & (255 * sizeof(pte_t));
|
|
|
|
segment_offset = segment_offset * (4096 / sizeof(pte_t));
|
|
|
|
page = pfn_to_page(__pa(pte) >> PAGE_SHIFT);
|
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
spin_lock(&gmap_notifier_lock);
|
|
|
|
list_for_each_entry(rmap, &mp->mapper, list) {
|
|
|
|
list_for_each_entry(nb, &gmap_notifier_list, list)
|
|
|
|
nb->notifier_call(rmap->gmap,
|
|
|
|
rmap->vmaddr + segment_offset);
|
|
|
|
}
|
|
|
|
spin_unlock(&gmap_notifier_lock);
|
|
|
|
}
|
2013-10-18 18:03:41 +08:00
|
|
|
EXPORT_SYMBOL_GPL(gmap_do_ipte_notify);
|
2013-04-17 16:53:39 +08:00
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
static inline int page_table_with_pgste(struct page *page)
|
|
|
|
{
|
|
|
|
return atomic_read(&page->_mapcount) == 0;
|
|
|
|
}
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
static inline unsigned long *page_table_alloc_pgste(struct mm_struct *mm,
|
|
|
|
unsigned long vmaddr)
|
2011-06-06 20:14:41 +08:00
|
|
|
{
|
|
|
|
struct page *page;
|
|
|
|
unsigned long *table;
|
2011-07-24 16:48:20 +08:00
|
|
|
struct gmap_pgtable *mp;
|
2011-06-06 20:14:41 +08:00
|
|
|
|
|
|
|
page = alloc_page(GFP_KERNEL|__GFP_REPEAT);
|
|
|
|
if (!page)
|
|
|
|
return NULL;
|
2011-07-24 16:48:20 +08:00
|
|
|
mp = kmalloc(sizeof(*mp), GFP_KERNEL|__GFP_REPEAT);
|
|
|
|
if (!mp) {
|
|
|
|
__free_page(page);
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-11-15 06:31:39 +08:00
|
|
|
if (!pgtable_page_ctor(page)) {
|
|
|
|
kfree(mp);
|
|
|
|
__free_page(page);
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-07-24 16:48:20 +08:00
|
|
|
mp->vmaddr = vmaddr & PMD_MASK;
|
|
|
|
INIT_LIST_HEAD(&mp->mapper);
|
|
|
|
page->index = (unsigned long) mp;
|
2013-07-26 21:04:02 +08:00
|
|
|
atomic_set(&page->_mapcount, 0);
|
2011-06-06 20:14:41 +08:00
|
|
|
table = (unsigned long *) page_to_phys(page);
|
2013-07-24 02:57:57 +08:00
|
|
|
clear_table(table, _PAGE_INVALID, PAGE_SIZE/2);
|
2013-10-18 18:03:41 +08:00
|
|
|
clear_table(table + PTRS_PER_PTE, 0, PAGE_SIZE/2);
|
2011-06-06 20:14:41 +08:00
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void page_table_free_pgste(unsigned long *table)
|
|
|
|
{
|
|
|
|
struct page *page;
|
2011-07-24 16:48:20 +08:00
|
|
|
struct gmap_pgtable *mp;
|
2011-06-06 20:14:41 +08:00
|
|
|
|
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
2011-07-24 16:48:20 +08:00
|
|
|
mp = (struct gmap_pgtable *) page->index;
|
|
|
|
BUG_ON(!list_empty(&mp->mapper));
|
2012-02-17 17:29:21 +08:00
|
|
|
pgtable_page_dtor(page);
|
2011-06-06 20:14:41 +08:00
|
|
|
atomic_set(&page->_mapcount, -1);
|
2011-07-24 16:48:20 +08:00
|
|
|
kfree(mp);
|
2011-06-06 20:14:41 +08:00
|
|
|
__free_page(page);
|
|
|
|
}
|
|
|
|
|
2014-01-29 23:02:32 +08:00
|
|
|
static inline unsigned long page_table_reset_pte(struct mm_struct *mm, pmd_t *pmd,
|
|
|
|
unsigned long addr, unsigned long end, bool init_skey)
|
2013-05-21 23:29:52 +08:00
|
|
|
{
|
|
|
|
pte_t *start_pte, *pte;
|
|
|
|
spinlock_t *ptl;
|
|
|
|
pgste_t pgste;
|
|
|
|
|
|
|
|
start_pte = pte_offset_map_lock(mm, pmd, addr, &ptl);
|
|
|
|
pte = start_pte;
|
|
|
|
do {
|
|
|
|
pgste = pgste_get_lock(pte);
|
|
|
|
pgste_val(pgste) &= ~_PGSTE_GPS_USAGE_MASK;
|
2014-01-29 23:02:32 +08:00
|
|
|
if (init_skey) {
|
|
|
|
unsigned long address;
|
|
|
|
|
|
|
|
pgste_val(pgste) &= ~(PGSTE_ACC_BITS | PGSTE_FP_BIT |
|
|
|
|
PGSTE_GR_BIT | PGSTE_GC_BIT);
|
|
|
|
|
|
|
|
/* skip invalid and not writable pages */
|
|
|
|
if (pte_val(*pte) & _PAGE_INVALID ||
|
|
|
|
!(pte_val(*pte) & _PAGE_WRITE)) {
|
|
|
|
pgste_set_unlock(pte, pgste);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
address = pte_val(*pte) & PAGE_MASK;
|
|
|
|
page_set_storage_key(address, PAGE_DEFAULT_KEY, 1);
|
|
|
|
}
|
2013-05-21 23:29:52 +08:00
|
|
|
pgste_set_unlock(pte, pgste);
|
|
|
|
} while (pte++, addr += PAGE_SIZE, addr != end);
|
|
|
|
pte_unmap_unlock(start_pte, ptl);
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2014-01-29 23:02:32 +08:00
|
|
|
static inline unsigned long page_table_reset_pmd(struct mm_struct *mm, pud_t *pud,
|
|
|
|
unsigned long addr, unsigned long end, bool init_skey)
|
2013-05-21 23:29:52 +08:00
|
|
|
{
|
|
|
|
unsigned long next;
|
|
|
|
pmd_t *pmd;
|
|
|
|
|
|
|
|
pmd = pmd_offset(pud, addr);
|
|
|
|
do {
|
|
|
|
next = pmd_addr_end(addr, end);
|
|
|
|
if (pmd_none_or_clear_bad(pmd))
|
|
|
|
continue;
|
2014-01-29 23:02:32 +08:00
|
|
|
next = page_table_reset_pte(mm, pmd, addr, next, init_skey);
|
2013-05-21 23:29:52 +08:00
|
|
|
} while (pmd++, addr = next, addr != end);
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2014-01-29 23:02:32 +08:00
|
|
|
static inline unsigned long page_table_reset_pud(struct mm_struct *mm, pgd_t *pgd,
|
|
|
|
unsigned long addr, unsigned long end, bool init_skey)
|
2013-05-21 23:29:52 +08:00
|
|
|
{
|
|
|
|
unsigned long next;
|
|
|
|
pud_t *pud;
|
|
|
|
|
|
|
|
pud = pud_offset(pgd, addr);
|
|
|
|
do {
|
|
|
|
next = pud_addr_end(addr, end);
|
|
|
|
if (pud_none_or_clear_bad(pud))
|
|
|
|
continue;
|
2014-01-29 23:02:32 +08:00
|
|
|
next = page_table_reset_pmd(mm, pud, addr, next, init_skey);
|
2013-05-21 23:29:52 +08:00
|
|
|
} while (pud++, addr = next, addr != end);
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2014-01-29 23:02:32 +08:00
|
|
|
void page_table_reset_pgste(struct mm_struct *mm, unsigned long start,
|
|
|
|
unsigned long end, bool init_skey)
|
2013-05-21 23:29:52 +08:00
|
|
|
{
|
|
|
|
unsigned long addr, next;
|
|
|
|
pgd_t *pgd;
|
|
|
|
|
|
|
|
addr = start;
|
|
|
|
down_read(&mm->mmap_sem);
|
|
|
|
pgd = pgd_offset(mm, addr);
|
|
|
|
do {
|
|
|
|
next = pgd_addr_end(addr, end);
|
|
|
|
if (pgd_none_or_clear_bad(pgd))
|
|
|
|
continue;
|
2014-01-29 23:02:32 +08:00
|
|
|
next = page_table_reset_pud(mm, pgd, addr, next, init_skey);
|
2013-05-21 23:29:52 +08:00
|
|
|
} while (pgd++, addr = next, addr != end);
|
|
|
|
up_read(&mm->mmap_sem);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(page_table_reset_pgste);
|
|
|
|
|
2013-05-27 16:42:04 +08:00
|
|
|
int set_guest_storage_key(struct mm_struct *mm, unsigned long addr,
|
|
|
|
unsigned long key, bool nq)
|
|
|
|
{
|
|
|
|
spinlock_t *ptl;
|
|
|
|
pgste_t old, new;
|
|
|
|
pte_t *ptep;
|
|
|
|
|
|
|
|
down_read(&mm->mmap_sem);
|
|
|
|
ptep = get_locked_pte(current->mm, addr, &ptl);
|
|
|
|
if (unlikely(!ptep)) {
|
|
|
|
up_read(&mm->mmap_sem);
|
|
|
|
return -EFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
new = old = pgste_get_lock(ptep);
|
|
|
|
pgste_val(new) &= ~(PGSTE_GR_BIT | PGSTE_GC_BIT |
|
|
|
|
PGSTE_ACC_BITS | PGSTE_FP_BIT);
|
|
|
|
pgste_val(new) |= (key & (_PAGE_CHANGED | _PAGE_REFERENCED)) << 48;
|
|
|
|
pgste_val(new) |= (key & (_PAGE_ACC_BITS | _PAGE_FP_BIT)) << 56;
|
|
|
|
if (!(pte_val(*ptep) & _PAGE_INVALID)) {
|
2013-07-24 04:11:42 +08:00
|
|
|
unsigned long address, bits, skey;
|
2013-05-27 16:42:04 +08:00
|
|
|
|
|
|
|
address = pte_val(*ptep) & PAGE_MASK;
|
2013-07-24 04:11:42 +08:00
|
|
|
skey = (unsigned long) page_get_storage_key(address);
|
2013-05-27 16:42:04 +08:00
|
|
|
bits = skey & (_PAGE_CHANGED | _PAGE_REFERENCED);
|
2013-07-24 04:11:42 +08:00
|
|
|
skey = key & (_PAGE_ACC_BITS | _PAGE_FP_BIT);
|
2013-05-27 16:42:04 +08:00
|
|
|
/* Set storage key ACC and FP */
|
2013-07-24 04:11:42 +08:00
|
|
|
page_set_storage_key(address, skey, !nq);
|
2013-05-27 16:42:04 +08:00
|
|
|
/* Merge host changed & referenced into pgste */
|
|
|
|
pgste_val(new) |= bits << 52;
|
|
|
|
}
|
|
|
|
/* changing the guest storage key is considered a change of the page */
|
|
|
|
if ((pgste_val(new) ^ pgste_val(old)) &
|
|
|
|
(PGSTE_ACC_BITS | PGSTE_FP_BIT | PGSTE_GR_BIT | PGSTE_GC_BIT))
|
2013-10-18 18:03:41 +08:00
|
|
|
pgste_val(new) |= PGSTE_UC_BIT;
|
2013-05-27 16:42:04 +08:00
|
|
|
|
|
|
|
pgste_set_unlock(ptep, new);
|
|
|
|
pte_unmap_unlock(*ptep, ptl);
|
|
|
|
up_read(&mm->mmap_sem);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(set_guest_storage_key);
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
#else /* CONFIG_PGSTE */
|
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
static inline int page_table_with_pgste(struct page *page)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
static inline unsigned long *page_table_alloc_pgste(struct mm_struct *mm,
|
|
|
|
unsigned long vmaddr)
|
|
|
|
{
|
2011-08-03 22:44:18 +08:00
|
|
|
return NULL;
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
|
|
|
|
2014-01-29 23:02:32 +08:00
|
|
|
void page_table_reset_pgste(struct mm_struct *mm, unsigned long start,
|
|
|
|
unsigned long end, bool init_skey)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-07-24 16:48:20 +08:00
|
|
|
static inline void page_table_free_pgste(unsigned long *table)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-04-16 19:37:46 +08:00
|
|
|
static inline void gmap_disconnect_pgtable(struct mm_struct *mm,
|
|
|
|
unsigned long *table)
|
2011-07-24 16:48:20 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_PGSTE */
|
|
|
|
|
|
|
|
static inline unsigned int atomic_xor_bits(atomic_t *v, unsigned int bits)
|
|
|
|
{
|
|
|
|
unsigned int old, new;
|
|
|
|
|
|
|
|
do {
|
|
|
|
old = atomic_read(v);
|
|
|
|
new = old ^ bits;
|
|
|
|
} while (atomic_cmpxchg(v, old, new) != old);
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* page table entry allocation/free routines.
|
|
|
|
*/
|
|
|
|
unsigned long *page_table_alloc(struct mm_struct *mm, unsigned long vmaddr)
|
2007-10-22 18:52:47 +08:00
|
|
|
{
|
2012-09-14 17:09:52 +08:00
|
|
|
unsigned long *uninitialized_var(table);
|
|
|
|
struct page *uninitialized_var(page);
|
2011-06-06 20:14:41 +08:00
|
|
|
unsigned int mask, bit;
|
2007-10-22 18:52:47 +08:00
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
if (mm_has_pgste(mm))
|
2011-07-24 16:48:20 +08:00
|
|
|
return page_table_alloc_pgste(mm, vmaddr);
|
2011-06-06 20:14:41 +08:00
|
|
|
/* Allocate fragments of a 4K page as 1K/2K page table */
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_lock_bh(&mm->context.list_lock);
|
2011-06-06 20:14:41 +08:00
|
|
|
mask = FRAG_MASK;
|
2008-02-10 01:24:35 +08:00
|
|
|
if (!list_empty(&mm->context.pgtable_list)) {
|
|
|
|
page = list_first_entry(&mm->context.pgtable_list,
|
|
|
|
struct page, lru);
|
2011-06-06 20:14:41 +08:00
|
|
|
table = (unsigned long *) page_to_phys(page);
|
|
|
|
mask = atomic_read(&page->_mapcount);
|
|
|
|
mask = mask | (mask >> 4);
|
2008-02-10 01:24:35 +08:00
|
|
|
}
|
2011-06-06 20:14:41 +08:00
|
|
|
if ((mask & FRAG_MASK) == FRAG_MASK) {
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_unlock_bh(&mm->context.list_lock);
|
2008-02-10 01:24:35 +08:00
|
|
|
page = alloc_page(GFP_KERNEL|__GFP_REPEAT);
|
|
|
|
if (!page)
|
2007-10-22 18:52:47 +08:00
|
|
|
return NULL;
|
2013-11-15 06:31:39 +08:00
|
|
|
if (!pgtable_page_ctor(page)) {
|
|
|
|
__free_page(page);
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-06-06 20:14:41 +08:00
|
|
|
atomic_set(&page->_mapcount, 1);
|
2008-02-10 01:24:35 +08:00
|
|
|
table = (unsigned long *) page_to_phys(page);
|
2013-07-24 02:57:57 +08:00
|
|
|
clear_table(table, _PAGE_INVALID, PAGE_SIZE);
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_lock_bh(&mm->context.list_lock);
|
2008-02-10 01:24:35 +08:00
|
|
|
list_add(&page->lru, &mm->context.pgtable_list);
|
2011-06-06 20:14:41 +08:00
|
|
|
} else {
|
|
|
|
for (bit = 1; mask & bit; bit <<= 1)
|
|
|
|
table += PTRS_PER_PTE;
|
|
|
|
mask = atomic_xor_bits(&page->_mapcount, bit);
|
|
|
|
if ((mask & FRAG_MASK) == FRAG_MASK)
|
|
|
|
list_del(&page->lru);
|
2007-10-22 18:52:47 +08:00
|
|
|
}
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_unlock_bh(&mm->context.list_lock);
|
2007-10-22 18:52:47 +08:00
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
void page_table_free(struct mm_struct *mm, unsigned long *table)
|
2010-10-25 22:10:11 +08:00
|
|
|
{
|
|
|
|
struct page *page;
|
2011-06-06 20:14:41 +08:00
|
|
|
unsigned int bit, mask;
|
2010-10-25 22:10:11 +08:00
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
|
|
|
if (page_table_with_pgste(page)) {
|
2013-04-16 19:37:46 +08:00
|
|
|
gmap_disconnect_pgtable(mm, table);
|
2011-06-06 20:14:41 +08:00
|
|
|
return page_table_free_pgste(table);
|
2011-07-24 16:48:20 +08:00
|
|
|
}
|
2011-06-06 20:14:41 +08:00
|
|
|
/* Free 1K/2K page table fragment of a 4K page */
|
|
|
|
bit = 1 << ((__pa(table) & ~PAGE_MASK)/(PTRS_PER_PTE*sizeof(pte_t)));
|
|
|
|
spin_lock_bh(&mm->context.list_lock);
|
|
|
|
if ((atomic_read(&page->_mapcount) & FRAG_MASK) != FRAG_MASK)
|
|
|
|
list_del(&page->lru);
|
|
|
|
mask = atomic_xor_bits(&page->_mapcount, bit);
|
|
|
|
if (mask & FRAG_MASK)
|
|
|
|
list_add(&page->lru, &mm->context.pgtable_list);
|
|
|
|
spin_unlock_bh(&mm->context.list_lock);
|
|
|
|
if (mask == 0) {
|
2010-10-25 22:10:11 +08:00
|
|
|
pgtable_page_dtor(page);
|
2011-06-06 20:14:41 +08:00
|
|
|
atomic_set(&page->_mapcount, -1);
|
2010-10-25 22:10:11 +08:00
|
|
|
__free_page(page);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
static void __page_table_free_rcu(void *table, unsigned bit)
|
2007-10-22 18:52:47 +08:00
|
|
|
{
|
2008-02-10 01:24:35 +08:00
|
|
|
struct page *page;
|
2007-10-22 18:52:47 +08:00
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
if (bit == FRAG_MASK)
|
|
|
|
return page_table_free_pgste(table);
|
|
|
|
/* Free 1K/2K page table fragment of a 4K page */
|
2008-02-10 01:24:35 +08:00
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
2011-06-06 20:14:41 +08:00
|
|
|
if (atomic_xor_bits(&page->_mapcount, bit) == 0) {
|
2008-02-10 01:24:35 +08:00
|
|
|
pgtable_page_dtor(page);
|
2011-06-06 20:14:41 +08:00
|
|
|
atomic_set(&page->_mapcount, -1);
|
2008-02-10 01:24:35 +08:00
|
|
|
__free_page(page);
|
|
|
|
}
|
|
|
|
}
|
2007-10-22 18:52:47 +08:00
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
void page_table_free_rcu(struct mmu_gather *tlb, unsigned long *table)
|
2010-10-25 22:10:11 +08:00
|
|
|
{
|
2011-06-06 20:14:41 +08:00
|
|
|
struct mm_struct *mm;
|
2010-10-25 22:10:11 +08:00
|
|
|
struct page *page;
|
2011-06-06 20:14:41 +08:00
|
|
|
unsigned int bit, mask;
|
2010-10-25 22:10:11 +08:00
|
|
|
|
2011-06-06 20:14:41 +08:00
|
|
|
mm = tlb->mm;
|
2013-07-26 21:04:02 +08:00
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
|
|
|
if (page_table_with_pgste(page)) {
|
2013-04-16 19:37:46 +08:00
|
|
|
gmap_disconnect_pgtable(mm, table);
|
2011-06-06 20:14:41 +08:00
|
|
|
table = (unsigned long *) (__pa(table) | FRAG_MASK);
|
|
|
|
tlb_remove_table(tlb, table);
|
|
|
|
return;
|
2010-10-25 22:10:11 +08:00
|
|
|
}
|
2011-06-06 20:14:41 +08:00
|
|
|
bit = 1 << ((__pa(table) & ~PAGE_MASK) / (PTRS_PER_PTE*sizeof(pte_t)));
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_lock_bh(&mm->context.list_lock);
|
2011-06-06 20:14:41 +08:00
|
|
|
if ((atomic_read(&page->_mapcount) & FRAG_MASK) != FRAG_MASK)
|
|
|
|
list_del(&page->lru);
|
|
|
|
mask = atomic_xor_bits(&page->_mapcount, bit | (bit << 4));
|
|
|
|
if (mask & FRAG_MASK)
|
|
|
|
list_add_tail(&page->lru, &mm->context.pgtable_list);
|
2010-10-25 22:10:11 +08:00
|
|
|
spin_unlock_bh(&mm->context.list_lock);
|
2011-06-06 20:14:41 +08:00
|
|
|
table = (unsigned long *) (__pa(table) | (bit << 4));
|
|
|
|
tlb_remove_table(tlb, table);
|
|
|
|
}
|
|
|
|
|
2013-09-07 01:10:48 +08:00
|
|
|
static void __tlb_remove_table(void *_table)
|
2011-06-06 20:14:41 +08:00
|
|
|
{
|
2011-10-30 22:16:08 +08:00
|
|
|
const unsigned long mask = (FRAG_MASK << 4) | FRAG_MASK;
|
|
|
|
void *table = (void *)((unsigned long) _table & ~mask);
|
|
|
|
unsigned type = (unsigned long) _table & mask;
|
2011-06-06 20:14:41 +08:00
|
|
|
|
|
|
|
if (type)
|
|
|
|
__page_table_free_rcu(table, type);
|
|
|
|
else
|
|
|
|
free_pages((unsigned long) table, ALLOC_ORDER);
|
2010-10-25 22:10:11 +08:00
|
|
|
}
|
|
|
|
|
2012-04-11 20:28:07 +08:00
|
|
|
static void tlb_remove_table_smp_sync(void *arg)
|
|
|
|
{
|
|
|
|
/* Simply deliver the interrupt */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tlb_remove_table_one(void *table)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* This isn't an RCU grace period and hence the page-tables cannot be
|
|
|
|
* assumed to be actually RCU-freed.
|
|
|
|
*
|
|
|
|
* It is however sufficient for software page-table walkers that rely
|
|
|
|
* on IRQ disabling. See the comment near struct mmu_table_batch.
|
|
|
|
*/
|
|
|
|
smp_call_function(tlb_remove_table_smp_sync, NULL, 1);
|
|
|
|
__tlb_remove_table(table);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tlb_remove_table_rcu(struct rcu_head *head)
|
|
|
|
{
|
|
|
|
struct mmu_table_batch *batch;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
batch = container_of(head, struct mmu_table_batch, rcu);
|
|
|
|
|
|
|
|
for (i = 0; i < batch->nr; i++)
|
|
|
|
__tlb_remove_table(batch->tables[i]);
|
|
|
|
|
|
|
|
free_page((unsigned long)batch);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tlb_table_flush(struct mmu_gather *tlb)
|
|
|
|
{
|
|
|
|
struct mmu_table_batch **batch = &tlb->batch;
|
|
|
|
|
|
|
|
if (*batch) {
|
|
|
|
call_rcu_sched(&(*batch)->rcu, tlb_remove_table_rcu);
|
|
|
|
*batch = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void tlb_remove_table(struct mmu_gather *tlb, void *table)
|
|
|
|
{
|
|
|
|
struct mmu_table_batch **batch = &tlb->batch;
|
|
|
|
|
2013-08-16 19:31:40 +08:00
|
|
|
tlb->mm->context.flush_mm = 1;
|
2012-04-11 20:28:07 +08:00
|
|
|
if (*batch == NULL) {
|
|
|
|
*batch = (struct mmu_table_batch *)
|
|
|
|
__get_free_page(GFP_NOWAIT | __GFP_NOWARN);
|
|
|
|
if (*batch == NULL) {
|
2013-08-16 19:31:40 +08:00
|
|
|
__tlb_flush_mm_lazy(tlb->mm);
|
2012-04-11 20:28:07 +08:00
|
|
|
tlb_remove_table_one(table);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
(*batch)->nr = 0;
|
|
|
|
}
|
|
|
|
(*batch)->tables[(*batch)->nr++] = table;
|
|
|
|
if ((*batch)->nr == MAX_TABLE_BATCH)
|
2013-08-16 19:31:40 +08:00
|
|
|
tlb_flush_mmu(tlb);
|
2012-04-11 20:28:07 +08:00
|
|
|
}
|
2011-06-06 20:14:41 +08:00
|
|
|
|
2012-10-09 07:30:21 +08:00
|
|
|
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
2013-07-26 21:04:02 +08:00
|
|
|
static inline void thp_split_vma(struct vm_area_struct *vma)
|
2012-10-09 07:30:21 +08:00
|
|
|
{
|
|
|
|
unsigned long addr;
|
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
for (addr = vma->vm_start; addr < vma->vm_end; addr += PAGE_SIZE)
|
|
|
|
follow_page(vma, addr, FOLL_SPLIT);
|
2012-10-09 07:30:21 +08:00
|
|
|
}
|
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
static inline void thp_split_mm(struct mm_struct *mm)
|
2012-10-09 07:30:21 +08:00
|
|
|
{
|
2013-07-26 21:04:02 +08:00
|
|
|
struct vm_area_struct *vma;
|
2012-10-09 07:30:21 +08:00
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
for (vma = mm->mmap; vma != NULL; vma = vma->vm_next) {
|
2012-10-09 07:30:21 +08:00
|
|
|
thp_split_vma(vma);
|
|
|
|
vma->vm_flags &= ~VM_HUGEPAGE;
|
|
|
|
vma->vm_flags |= VM_NOHUGEPAGE;
|
|
|
|
}
|
2013-07-26 21:04:02 +08:00
|
|
|
mm->def_flags |= VM_NOHUGEPAGE;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static inline void thp_split_mm(struct mm_struct *mm)
|
|
|
|
{
|
2012-10-09 07:30:21 +08:00
|
|
|
}
|
|
|
|
#endif /* CONFIG_TRANSPARENT_HUGEPAGE */
|
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
static unsigned long page_table_realloc_pmd(struct mmu_gather *tlb,
|
|
|
|
struct mm_struct *mm, pud_t *pud,
|
|
|
|
unsigned long addr, unsigned long end)
|
|
|
|
{
|
|
|
|
unsigned long next, *table, *new;
|
|
|
|
struct page *page;
|
|
|
|
pmd_t *pmd;
|
|
|
|
|
|
|
|
pmd = pmd_offset(pud, addr);
|
|
|
|
do {
|
|
|
|
next = pmd_addr_end(addr, end);
|
|
|
|
again:
|
|
|
|
if (pmd_none_or_clear_bad(pmd))
|
|
|
|
continue;
|
|
|
|
table = (unsigned long *) pmd_deref(*pmd);
|
|
|
|
page = pfn_to_page(__pa(table) >> PAGE_SHIFT);
|
|
|
|
if (page_table_with_pgste(page))
|
|
|
|
continue;
|
|
|
|
/* Allocate new page table with pgstes */
|
|
|
|
new = page_table_alloc_pgste(mm, addr);
|
2013-10-31 17:01:16 +08:00
|
|
|
if (!new)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
spin_lock(&mm->page_table_lock);
|
|
|
|
if (likely((unsigned long *) pmd_deref(*pmd) == table)) {
|
|
|
|
/* Nuke pmd entry pointing to the "short" page table */
|
|
|
|
pmdp_flush_lazy(mm, addr, pmd);
|
|
|
|
pmd_clear(pmd);
|
|
|
|
/* Copy ptes from old table to new table */
|
|
|
|
memcpy(new, table, PAGE_SIZE/2);
|
|
|
|
clear_table(table, _PAGE_INVALID, PAGE_SIZE/2);
|
|
|
|
/* Establish new table */
|
|
|
|
pmd_populate(mm, pmd, (pte_t *) new);
|
|
|
|
/* Free old table with rcu, there might be a walker! */
|
|
|
|
page_table_free_rcu(tlb, table);
|
|
|
|
new = NULL;
|
|
|
|
}
|
|
|
|
spin_unlock(&mm->page_table_lock);
|
|
|
|
if (new) {
|
|
|
|
page_table_free_pgste(new);
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
} while (pmd++, addr = next, addr != end);
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long page_table_realloc_pud(struct mmu_gather *tlb,
|
|
|
|
struct mm_struct *mm, pgd_t *pgd,
|
|
|
|
unsigned long addr, unsigned long end)
|
|
|
|
{
|
|
|
|
unsigned long next;
|
|
|
|
pud_t *pud;
|
|
|
|
|
|
|
|
pud = pud_offset(pgd, addr);
|
|
|
|
do {
|
|
|
|
next = pud_addr_end(addr, end);
|
|
|
|
if (pud_none_or_clear_bad(pud))
|
|
|
|
continue;
|
|
|
|
next = page_table_realloc_pmd(tlb, mm, pud, addr, next);
|
2013-10-31 17:01:16 +08:00
|
|
|
if (unlikely(IS_ERR_VALUE(next)))
|
|
|
|
return next;
|
2013-07-26 21:04:02 +08:00
|
|
|
} while (pud++, addr = next, addr != end);
|
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2013-10-31 17:01:16 +08:00
|
|
|
static unsigned long page_table_realloc(struct mmu_gather *tlb, struct mm_struct *mm,
|
|
|
|
unsigned long addr, unsigned long end)
|
2013-07-26 21:04:02 +08:00
|
|
|
{
|
|
|
|
unsigned long next;
|
|
|
|
pgd_t *pgd;
|
|
|
|
|
|
|
|
pgd = pgd_offset(mm, addr);
|
|
|
|
do {
|
|
|
|
next = pgd_addr_end(addr, end);
|
|
|
|
if (pgd_none_or_clear_bad(pgd))
|
|
|
|
continue;
|
|
|
|
next = page_table_realloc_pud(tlb, mm, pgd, addr, next);
|
2013-10-31 17:01:16 +08:00
|
|
|
if (unlikely(IS_ERR_VALUE(next)))
|
|
|
|
return next;
|
2013-07-26 21:04:02 +08:00
|
|
|
} while (pgd++, addr = next, addr != end);
|
2013-10-31 17:01:16 +08:00
|
|
|
|
|
|
|
return 0;
|
2013-07-26 21:04:02 +08:00
|
|
|
}
|
|
|
|
|
2008-03-26 01:47:10 +08:00
|
|
|
/*
|
|
|
|
* switch on pgstes for its userspace process (for kvm)
|
|
|
|
*/
|
|
|
|
int s390_enable_sie(void)
|
|
|
|
{
|
|
|
|
struct task_struct *tsk = current;
|
2013-07-26 21:04:02 +08:00
|
|
|
struct mm_struct *mm = tsk->mm;
|
|
|
|
struct mmu_gather tlb;
|
2008-03-26 01:47:10 +08:00
|
|
|
|
2008-05-21 19:37:29 +08:00
|
|
|
/* Do we have pgstes? if yes, we are done */
|
2011-06-06 20:14:41 +08:00
|
|
|
if (mm_has_pgste(tsk->mm))
|
2008-05-21 19:37:29 +08:00
|
|
|
return 0;
|
2008-03-26 01:47:10 +08:00
|
|
|
|
2013-07-26 21:04:02 +08:00
|
|
|
down_write(&mm->mmap_sem);
|
2012-10-09 07:30:21 +08:00
|
|
|
/* split thp mappings and disable thp for future mappings */
|
|
|
|
thp_split_mm(mm);
|
2013-07-26 21:04:02 +08:00
|
|
|
/* Reallocate the page tables with pgstes */
|
2013-09-05 09:15:06 +08:00
|
|
|
tlb_gather_mmu(&tlb, mm, 0, TASK_SIZE);
|
2013-10-31 17:01:16 +08:00
|
|
|
if (!page_table_realloc(&tlb, mm, 0, TASK_SIZE))
|
|
|
|
mm->context.has_pgste = 1;
|
2013-09-05 09:15:06 +08:00
|
|
|
tlb_finish_mmu(&tlb, 0, TASK_SIZE);
|
2013-07-26 21:04:02 +08:00
|
|
|
up_write(&mm->mmap_sem);
|
|
|
|
return mm->context.has_pgste ? 0 : -ENOMEM;
|
2008-03-26 01:47:10 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(s390_enable_sie);
|
2009-06-16 16:30:26 +08:00
|
|
|
|
2014-01-15 01:10:17 +08:00
|
|
|
/*
|
|
|
|
* Enable storage key handling from now on and initialize the storage
|
|
|
|
* keys with the default key.
|
|
|
|
*/
|
|
|
|
void s390_enable_skey(void)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* To avoid races between multiple vcpus, ending in calling
|
|
|
|
* page_table_reset twice or more,
|
|
|
|
* the page_table_lock is taken for serialization.
|
|
|
|
*/
|
|
|
|
spin_lock(¤t->mm->page_table_lock);
|
|
|
|
if (mm_use_skey(current->mm)) {
|
|
|
|
spin_unlock(¤t->mm->page_table_lock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
current->mm->context.use_skey = 1;
|
|
|
|
spin_unlock(¤t->mm->page_table_lock);
|
|
|
|
page_table_reset_pgste(current->mm, 0, TASK_SIZE, true);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(s390_enable_skey);
|
|
|
|
|
2014-03-24 21:27:58 +08:00
|
|
|
/*
|
|
|
|
* Test and reset if a guest page is dirty
|
|
|
|
*/
|
|
|
|
bool gmap_test_and_clear_dirty(unsigned long address, struct gmap *gmap)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
spinlock_t *ptl;
|
|
|
|
bool dirty = false;
|
|
|
|
|
|
|
|
pte = get_locked_pte(gmap->mm, address, &ptl);
|
|
|
|
if (unlikely(!pte))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (ptep_test_and_clear_user_dirty(gmap->mm, address, pte))
|
|
|
|
dirty = true;
|
|
|
|
|
|
|
|
spin_unlock(ptl);
|
|
|
|
return dirty;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(gmap_test_and_clear_dirty);
|
|
|
|
|
2012-10-09 07:30:15 +08:00
|
|
|
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
2012-10-09 07:30:24 +08:00
|
|
|
int pmdp_clear_flush_young(struct vm_area_struct *vma, unsigned long address,
|
|
|
|
pmd_t *pmdp)
|
|
|
|
{
|
|
|
|
VM_BUG_ON(address & ~HPAGE_PMD_MASK);
|
|
|
|
/* No need to flush TLB
|
|
|
|
* On s390 reference bits are in storage key and never in TLB */
|
|
|
|
return pmdp_test_and_clear_young(vma, address, pmdp);
|
|
|
|
}
|
|
|
|
|
|
|
|
int pmdp_set_access_flags(struct vm_area_struct *vma,
|
|
|
|
unsigned long address, pmd_t *pmdp,
|
|
|
|
pmd_t entry, int dirty)
|
|
|
|
{
|
|
|
|
VM_BUG_ON(address & ~HPAGE_PMD_MASK);
|
|
|
|
|
|
|
|
if (pmd_same(*pmdp, entry))
|
|
|
|
return 0;
|
|
|
|
pmdp_invalidate(vma, address, pmdp);
|
|
|
|
set_pmd_at(vma->vm_mm, address, pmdp, entry);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-10-09 07:30:15 +08:00
|
|
|
static void pmdp_splitting_flush_sync(void *arg)
|
|
|
|
{
|
|
|
|
/* Simply deliver the interrupt */
|
|
|
|
}
|
|
|
|
|
|
|
|
void pmdp_splitting_flush(struct vm_area_struct *vma, unsigned long address,
|
|
|
|
pmd_t *pmdp)
|
|
|
|
{
|
|
|
|
VM_BUG_ON(address & ~HPAGE_PMD_MASK);
|
|
|
|
if (!test_and_set_bit(_SEGMENT_ENTRY_SPLIT_BIT,
|
|
|
|
(unsigned long *) pmdp)) {
|
|
|
|
/* need to serialize against gup-fast (IRQ disabled) */
|
|
|
|
smp_call_function(pmdp_splitting_flush_sync, NULL, 1);
|
|
|
|
}
|
|
|
|
}
|
2012-10-09 07:30:18 +08:00
|
|
|
|
2013-06-06 08:14:02 +08:00
|
|
|
void pgtable_trans_huge_deposit(struct mm_struct *mm, pmd_t *pmdp,
|
|
|
|
pgtable_t pgtable)
|
2012-10-09 07:30:18 +08:00
|
|
|
{
|
|
|
|
struct list_head *lh = (struct list_head *) pgtable;
|
|
|
|
|
2014-02-12 21:16:18 +08:00
|
|
|
assert_spin_locked(pmd_lockptr(mm, pmdp));
|
2012-10-09 07:30:18 +08:00
|
|
|
|
|
|
|
/* FIFO */
|
2013-11-15 06:30:59 +08:00
|
|
|
if (!pmd_huge_pte(mm, pmdp))
|
2012-10-09 07:30:18 +08:00
|
|
|
INIT_LIST_HEAD(lh);
|
|
|
|
else
|
2013-11-15 06:30:59 +08:00
|
|
|
list_add(lh, (struct list_head *) pmd_huge_pte(mm, pmdp));
|
|
|
|
pmd_huge_pte(mm, pmdp) = pgtable;
|
2012-10-09 07:30:18 +08:00
|
|
|
}
|
|
|
|
|
2013-06-06 08:14:02 +08:00
|
|
|
pgtable_t pgtable_trans_huge_withdraw(struct mm_struct *mm, pmd_t *pmdp)
|
2012-10-09 07:30:18 +08:00
|
|
|
{
|
|
|
|
struct list_head *lh;
|
|
|
|
pgtable_t pgtable;
|
|
|
|
pte_t *ptep;
|
|
|
|
|
2014-02-12 21:16:18 +08:00
|
|
|
assert_spin_locked(pmd_lockptr(mm, pmdp));
|
2012-10-09 07:30:18 +08:00
|
|
|
|
|
|
|
/* FIFO */
|
2013-11-15 06:30:59 +08:00
|
|
|
pgtable = pmd_huge_pte(mm, pmdp);
|
2012-10-09 07:30:18 +08:00
|
|
|
lh = (struct list_head *) pgtable;
|
|
|
|
if (list_empty(lh))
|
2013-11-15 06:30:59 +08:00
|
|
|
pmd_huge_pte(mm, pmdp) = NULL;
|
2012-10-09 07:30:18 +08:00
|
|
|
else {
|
2013-11-15 06:30:59 +08:00
|
|
|
pmd_huge_pte(mm, pmdp) = (pgtable_t) lh->next;
|
2012-10-09 07:30:18 +08:00
|
|
|
list_del(lh);
|
|
|
|
}
|
|
|
|
ptep = (pte_t *) pgtable;
|
2013-07-24 02:57:57 +08:00
|
|
|
pte_val(*ptep) = _PAGE_INVALID;
|
2012-10-09 07:30:18 +08:00
|
|
|
ptep++;
|
2013-07-24 02:57:57 +08:00
|
|
|
pte_val(*ptep) = _PAGE_INVALID;
|
2012-10-09 07:30:18 +08:00
|
|
|
return pgtable;
|
|
|
|
}
|
2012-10-09 07:30:15 +08:00
|
|
|
#endif /* CONFIG_TRANSPARENT_HUGEPAGE */
|