summaryrefslogtreecommitdiffstats
path: root/sys/arm
diff options
context:
space:
mode:
authorraj <raj@FreeBSD.org>2008-10-13 18:42:25 +0000
committerraj <raj@FreeBSD.org>2008-10-13 18:42:25 +0000
commite725b1fa27cadbe3a6f57fc73839bbfea449a623 (patch)
treee81229bde6de4f202b8a72e7a215d453d54fae94 /sys/arm
parentb9c565987ac8d8b7f1fedf4ed5b320ae753180b2 (diff)
downloadFreeBSD-src-e725b1fa27cadbe3a6f57fc73839bbfea449a623.zip
FreeBSD-src-e725b1fa27cadbe3a6f57fc73839bbfea449a623.tar.gz
Do not use cached page for temporary mapping in pmap_zero_page_generic()
The physical page which we clear is accessed via additional temp kernel mapping for the period of zeroing operation. However in systems with virtual d-cache (most ARMs) when write-allocate feature is enabled, we can have modified but unflushed content pertaining to this physical page still in the d-cache due to its primary (pre-existing) mapping. In such scenario that cached content upon flush is likely to overwrite [portions of] the physical page we want to zero here.. This is a general problem with multiple virtual mappings covering the same physical page with write-allocate and virtual d-cache: there is inherent potential for corruptions of this kind, which are not easily resolved; it is best policy that such multiple mappings be not allowed. Obtained from: Marvell, Semihalf
Diffstat (limited to 'sys/arm')
-rw-r--r--sys/arm/arm/pmap.c18
1 files changed, 8 insertions, 10 deletions
diff --git a/sys/arm/arm/pmap.c b/sys/arm/arm/pmap.c
index 3b5ae96..e1a8dc0 100644
--- a/sys/arm/arm/pmap.c
+++ b/sys/arm/arm/pmap.c
@@ -3850,21 +3850,19 @@ pmap_zero_page_generic(vm_paddr_t phys, int off, int size)
mtx_lock(&cmtx);
/*
- * Hook in the page, zero it, and purge the cache for that
- * zeroed page. Invalidate the TLB as needed.
+ * Hook in the page, zero it, invalidate the TLB as needed.
+ *
+ * Note the temporary zero-page mapping must be a non-cached page in
+ * ordert to work without corruption when write-allocate is enabled.
*/
- *cdst_pte = L2_S_PROTO | phys |
- L2_S_PROT(PTE_KERNEL, VM_PROT_WRITE) | pte_l2_s_cache_mode;
- PTE_SYNC(cdst_pte);
+ *cdst_pte = L2_S_PROTO | phys | L2_S_PROT(PTE_KERNEL, VM_PROT_WRITE);
cpu_tlb_flushD_SE(cdstp);
cpu_cpwait();
- if (off || size != PAGE_SIZE) {
+ if (off || size != PAGE_SIZE)
bzero((void *)(cdstp + off), size);
- cpu_dcache_wbinv_range(cdstp + off, size);
- } else {
+ else
bzero_page(cdstp);
- cpu_dcache_wbinv_range(cdstp, PAGE_SIZE);
- }
+
mtx_unlock(&cmtx);
#endif
}
OpenPOWER on IntegriCloud