Module Name:    src
Committed By:   matt
Date:           Wed Jul 25 22:11:37 UTC 2012

Modified Files:
        src/sys/arch/powerpc/booke: e500_tlb.c

Log Message:
When dealing with kernel invalidations, make sure to use context-synchronizing
instructions.


To generate a diff of this commit:
cvs rdiff -u -r1.10 -r1.11 src/sys/arch/powerpc/booke/e500_tlb.c

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Modified files:

Index: src/sys/arch/powerpc/booke/e500_tlb.c
diff -u src/sys/arch/powerpc/booke/e500_tlb.c:1.10 src/sys/arch/powerpc/booke/e500_tlb.c:1.11
--- src/sys/arch/powerpc/booke/e500_tlb.c:1.10	Wed Jul 18 18:50:46 2012
+++ src/sys/arch/powerpc/booke/e500_tlb.c	Wed Jul 25 22:11:36 2012
@@ -1,4 +1,4 @@
-/*	$NetBSD: e500_tlb.c,v 1.10 2012/07/18 18:50:46 matt Exp $	*/
+/*	$NetBSD: e500_tlb.c,v 1.11 2012/07/25 22:11:36 matt Exp $	*/
 /*-
  * Copyright (c) 2010, 2011 The NetBSD Foundation, Inc.
  * All rights reserved.
@@ -38,7 +38,7 @@
 
 #include <sys/cdefs.h>
 
-__KERNEL_RCSID(0, "$NetBSD: e500_tlb.c,v 1.10 2012/07/18 18:50:46 matt Exp $");
+__KERNEL_RCSID(0, "$NetBSD: e500_tlb.c,v 1.11 2012/07/25 22:11:36 matt Exp $");
 
 #include <sys/param.h>
 
@@ -212,8 +212,7 @@ hwtlb_write(const struct e500_hwtlb hwtl
 #endif
 	__asm volatile("tlbwe");
 	if (needs_sync) {
-		__asm volatile("tlbsync");
-		__asm volatile("isync");
+		__asm volatile("tlbsync\n\tisync\n\tsync");
 	}
 
 	mtspr(SPR_MAS0, saved_mas0);
@@ -359,6 +358,7 @@ e500_tlb_invalidate_all(void)
 	 */
 #if 1
 	__asm volatile("tlbivax\t0, %0" :: "b"(4));	/* INV_ALL */
+	__asm volatile("tlbsync\n\tisync\n\tsync");
 #else
 	mtspr(SPR_MMUCSR0, MMUCSR0_TLB0_FL);
 	while (mfspr(SPR_MMUCSR0) != 0)
@@ -411,7 +411,7 @@ e500_tlb_invalidate_globals(void)
 			__asm volatile("tlbwe");
 		}
 	}
-	__asm volatile("isync");
+	__asm volatile("isync\n\tsync");
 	wrtee(msr);
 }
 
@@ -449,7 +449,7 @@ e500_tlb_invalidate_asids(tlb_asid_t asi
 			}
 		}
 	}
-	__asm volatile("isync");
+	__asm volatile("isync\n\tsync");
 	wrtee(msr);
 }
 
@@ -497,9 +497,26 @@ e500_tlb_invalidate_addr(vaddr_t va, tlb
 	/*
 	 * Bits 60 & 61 have meaning
 	 */
+	if (asid == KERNEL_PID) {
+		/*
+		 * For data accesses, the context-synchronizing instruction
+		 * before tlbwe or tlbivax ensures that all memory accesses
+		 * due to preceding instructions have completed to a point
+		 * at which they have reported all exceptions they will cause.
+		 */
+		__asm volatile("isync");
+	}
 	__asm volatile("tlbivax\t0, %0" :: "b"(va));
 	__asm volatile("tlbsync");
-	__asm volatile("tlbsync");
+	__asm volatile("tlbsync");	/* Why? */
+	if (asid == KERNEL_PID) {
+		/*
+		 * The context-synchronizing instruction after tlbwe or tlbivax
+		 * ensures that subsequent accesses (data and instruction) use
+		 * the updated value in any TLB entries affected.
+		 */
+		__asm volatile("isync\n\tsync");
+	}
 }
 
 static bool
@@ -527,8 +544,8 @@ e500_tlb_update_addr(vaddr_t va, tlb_asi
 	mtspr(SPR_MAS2, hwtlb.hwtlb_mas2);
 	mtspr(SPR_MAS3, hwtlb.hwtlb_mas3);
 	__asm volatile("tlbwe");
-	if (asid == 0)
-		__asm volatile("isync");
+	if (asid == KERNEL_PID)
+		__asm volatile("isync\n\tsync");
 	wrtee(msr);
 #if 0
 	if (asid)

Reply via email to