aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKeir Fraser <keir.fraser@citrix.com>2008-08-13 12:13:59 +0100
committerKeir Fraser <keir.fraser@citrix.com>2008-08-13 12:13:59 +0100
commitf507885bc303208bd2a346033cd7b8ff4128af0d (patch)
treee4cd64705ba0bc0c6f665af19eb55bff356417bc
parentc954127bc64c613ba7a7feff17a1646677a6977d (diff)
downloadxen-f507885bc303208bd2a346033cd7b8ff4128af0d.tar.gz
xen-f507885bc303208bd2a346033cd7b8ff4128af0d.tar.bz2
xen-f507885bc303208bd2a346033cd7b8ff4128af0d.zip
x86: Fix shadow code's handling of p2m superpage changes
When a p2m superpage entry is shattered, it's important not to unshadow any parts of the 2MB region that are still there afterwards. Otherwise shattering a superpage that contains the guest's top-level pagetable will cause the guest to be killed. Signed-off-by: Tim Deegan <Tim.Deegan@citrix.com>
-rw-r--r--xen/arch/x86/mm/shadow/common.c38
1 files changed, 30 insertions, 8 deletions
diff --git a/xen/arch/x86/mm/shadow/common.c b/xen/arch/x86/mm/shadow/common.c
index 57baa27655..34265fafee 100644
--- a/xen/arch/x86/mm/shadow/common.c
+++ b/xen/arch/x86/mm/shadow/common.c
@@ -3357,23 +3357,45 @@ shadow_write_p2m_entry(struct vcpu *v, unsigned long gfn,
}
}
- /* If we're removing a superpage mapping from the p2m, remove all the
- * MFNs covered by it from the shadows too. */
+ /* If we're removing a superpage mapping from the p2m, we need to check
+ * all the pages covered by it. If they're still there in the new
+ * scheme, that's OK, but otherwise they must be unshadowed. */
if ( level == 2 && (l1e_get_flags(*p) & _PAGE_PRESENT) &&
(l1e_get_flags(*p) & _PAGE_PSE) )
{
unsigned int i;
- mfn_t mfn = _mfn(l1e_get_pfn(*p));
+ cpumask_t flushmask;
+ mfn_t omfn = _mfn(l1e_get_pfn(*p));
+ mfn_t nmfn = _mfn(l1e_get_pfn(new));
+ l1_pgentry_t *npte = NULL;
p2m_type_t p2mt = p2m_flags_to_type(l1e_get_flags(*p));
- if ( p2m_is_valid(p2mt) && mfn_valid(mfn) )
+ if ( p2m_is_valid(p2mt) && mfn_valid(omfn) )
{
+ cpus_clear(flushmask);
+
+ /* If we're replacing a superpage with a normal L1 page, map it */
+ if ( (l1e_get_flags(new) & _PAGE_PRESENT)
+ && !(l1e_get_flags(new) & _PAGE_PSE)
+ && mfn_valid(nmfn) )
+ npte = map_domain_page(mfn_x(nmfn));
+
for ( i = 0; i < L1_PAGETABLE_ENTRIES; i++ )
{
- sh_remove_all_shadows_and_parents(v, mfn);
- if ( sh_remove_all_mappings(v, mfn) )
- flush_tlb_mask(d->domain_dirty_cpumask);
- mfn = _mfn(mfn_x(mfn) + 1);
+ if ( !npte
+ || !p2m_is_ram(p2m_flags_to_type(l1e_get_flags(npte[i])))
+ || l1e_get_pfn(npte[i]) != mfn_x(omfn) )
+ {
+ /* This GFN->MFN mapping has gone away */
+ sh_remove_all_shadows_and_parents(v, omfn);
+ if ( sh_remove_all_mappings(v, omfn) )
+ cpus_or(flushmask, flushmask, d->domain_dirty_cpumask);
+ }
+ omfn = _mfn(mfn_x(omfn) + 1);
}
+ flush_tlb_mask(flushmask);
+
+ if ( npte )
+ unmap_domain_page(npte);
}
}