Commit 96965941 authored by David Hildenbrand's avatar David Hildenbrand Committed by Christian Borntraeger

s390x/mm: cleanup gmap_pte_op_walk()

gmap_mprotect_notify() refuses shadow gmaps. Turns out that
a) gmap_protect_range()
b) gmap_read_table()
c) gmap_pte_op_walk()

Are never called for gmap shadows. And never should be. This dates back
to gmap shadow prototypes where we allowed to call mprotect_notify() on
the gmap shadow (to get notified about the prefix pages getting removed).
This is avoided by always getting notified about any change on the gmap
shadow.

The only real function for walking page tables on shadow gmaps is
gmap_table_walk().

So, essentially, these functions should never get called and
gmap_pte_op_walk() can be cleaned up. Add some checks to callers of
gmap_pte_op_walk().
Signed-off-by: default avatarDavid Hildenbrand <david@redhat.com>
Message-Id: <20171110151805.7541-1-david@redhat.com>
Reviewed-by: default avatarJanosch Frank <frankja@linux.vnet.ibm.com>
Acked-by: default avatarCornelia Huck <cohuck@redhat.com>
Signed-off-by: default avatarChristian Borntraeger <borntraeger@de.ibm.com>
parent 50c4c4e2
...@@ -815,27 +815,17 @@ static inline unsigned long *gmap_table_walk(struct gmap *gmap, ...@@ -815,27 +815,17 @@ static inline unsigned long *gmap_table_walk(struct gmap *gmap,
* @ptl: pointer to the spinlock pointer * @ptl: pointer to the spinlock pointer
* *
* Returns a pointer to the locked pte for a guest address, or NULL * Returns a pointer to the locked pte for a guest address, or NULL
*
* Note: Can also be called for shadow gmaps.
*/ */
static pte_t *gmap_pte_op_walk(struct gmap *gmap, unsigned long gaddr, static pte_t *gmap_pte_op_walk(struct gmap *gmap, unsigned long gaddr,
spinlock_t **ptl) spinlock_t **ptl)
{ {
unsigned long *table; unsigned long *table;
if (gmap_is_shadow(gmap)) BUG_ON(gmap_is_shadow(gmap));
spin_lock(&gmap->guest_table_lock);
/* Walk the gmap page table, lock and get pte pointer */ /* Walk the gmap page table, lock and get pte pointer */
table = gmap_table_walk(gmap, gaddr, 1); /* get segment pointer */ table = gmap_table_walk(gmap, gaddr, 1); /* get segment pointer */
if (!table || *table & _SEGMENT_ENTRY_INVALID) { if (!table || *table & _SEGMENT_ENTRY_INVALID)
if (gmap_is_shadow(gmap))
spin_unlock(&gmap->guest_table_lock);
return NULL; return NULL;
}
if (gmap_is_shadow(gmap)) {
*ptl = &gmap->guest_table_lock;
return pte_offset_map((pmd_t *) table, gaddr);
}
return pte_alloc_map_lock(gmap->mm, (pmd_t *) table, gaddr, ptl); return pte_alloc_map_lock(gmap->mm, (pmd_t *) table, gaddr, ptl);
} }
...@@ -889,8 +879,6 @@ static void gmap_pte_op_end(spinlock_t *ptl) ...@@ -889,8 +879,6 @@ static void gmap_pte_op_end(spinlock_t *ptl)
* -EFAULT if gaddr is invalid (or mapping for shadows is missing). * -EFAULT if gaddr is invalid (or mapping for shadows is missing).
* *
* Called with sg->mm->mmap_sem in read. * Called with sg->mm->mmap_sem in read.
*
* Note: Can also be called for shadow gmaps.
*/ */
static int gmap_protect_range(struct gmap *gmap, unsigned long gaddr, static int gmap_protect_range(struct gmap *gmap, unsigned long gaddr,
unsigned long len, int prot, unsigned long bits) unsigned long len, int prot, unsigned long bits)
...@@ -900,6 +888,7 @@ static int gmap_protect_range(struct gmap *gmap, unsigned long gaddr, ...@@ -900,6 +888,7 @@ static int gmap_protect_range(struct gmap *gmap, unsigned long gaddr,
pte_t *ptep; pte_t *ptep;
int rc; int rc;
BUG_ON(gmap_is_shadow(gmap));
while (len) { while (len) {
rc = -EAGAIN; rc = -EAGAIN;
ptep = gmap_pte_op_walk(gmap, gaddr, &ptl); ptep = gmap_pte_op_walk(gmap, gaddr, &ptl);
...@@ -960,7 +949,8 @@ EXPORT_SYMBOL_GPL(gmap_mprotect_notify); ...@@ -960,7 +949,8 @@ EXPORT_SYMBOL_GPL(gmap_mprotect_notify);
* @val: pointer to the unsigned long value to return * @val: pointer to the unsigned long value to return
* *
* Returns 0 if the value was read, -ENOMEM if out of memory and -EFAULT * Returns 0 if the value was read, -ENOMEM if out of memory and -EFAULT
* if reading using the virtual address failed. * if reading using the virtual address failed. -EINVAL if called on a gmap
* shadow.
* *
* Called with gmap->mm->mmap_sem in read. * Called with gmap->mm->mmap_sem in read.
*/ */
...@@ -971,6 +961,9 @@ int gmap_read_table(struct gmap *gmap, unsigned long gaddr, unsigned long *val) ...@@ -971,6 +961,9 @@ int gmap_read_table(struct gmap *gmap, unsigned long gaddr, unsigned long *val)
pte_t *ptep, pte; pte_t *ptep, pte;
int rc; int rc;
if (gmap_is_shadow(gmap))
return -EINVAL;
while (1) { while (1) {
rc = -EAGAIN; rc = -EAGAIN;
ptep = gmap_pte_op_walk(gmap, gaddr, &ptl); ptep = gmap_pte_op_walk(gmap, gaddr, &ptl);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment