powerpc/e6500: hw tablewalk: clear TID in kernel indirect entries
authorScott Wood <scottwood@freescale.com>
Tue, 20 May 2014 04:04:55 +0000 (23:04 -0500)
committerScott Wood <scottwood@freescale.com>
Fri, 20 Jun 2014 23:48:29 +0000 (18:48 -0500)
Previously TID was being cleared before the tlbsx, but not after.  This
can lead to a multiway hit between a TLB entry with TID=0 (previously
inserted when PID=0) and a TLB entry with TID!=0 that matches PID.
This can theoretically result in undefined behavior, though we probably
get lucky due to the details of the overlap.  It also results in the
inability to use multihit detection to detect other conflicting TLB
entries, as well as poorer TLB utilization due to duplicating kernel
TLB entries.

Rather than try to patch up MAS1 after tlbsx, the entire value is
saved/restored as with MAS2.

I observed a slight improvement in TLB miss performance with this patch
applied.

Signed-off-by: Scott Wood <scottwood@freescale.com>
Reported-by: Ed Swarthout <ed.swarthout@freescale.com>
arch/powerpc/mm/tlb_low_64e.S

index 356e8b4..3298d10 100644 (file)
@@ -322,19 +322,17 @@ tlb_miss_common_e6500:
        b       1b
        .previous
 
-       mfspr   r15,SPRN_MAS2
+       mfspr   r15,SPRN_MAS1
+       mfspr   r10,SPRN_MAS2
 
        tlbsx   0,r16
+       mtspr   SPRN_MAS2,r10
        mfspr   r10,SPRN_MAS1
+       mtspr   SPRN_MAS1,r15
+
        andis.  r10,r10,MAS1_VALID@h
        bne     tlb_miss_done_e6500
 
-       /* Undo MAS-damage from the tlbsx */
-       mfspr   r10,SPRN_MAS1
-       oris    r10,r10,MAS1_VALID@h
-       mtspr   SPRN_MAS1,r10
-       mtspr   SPRN_MAS2,r15
-
        /* Now, we need to walk the page tables. First check if we are in
         * range.
         */