]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
powerpc/64s/kuap: Add missing isync to KUAP restore paths
authorNicholas Piggin <npiggin@gmail.com>
Wed, 29 Apr 2020 06:56:50 +0000 (16:56 +1000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 24 Jun 2020 15:49:00 +0000 (17:49 +0200)
[ Upstream commit cb2b53cbffe3c388cd676b63f34e54ceb2643ae2 ]

Writing the AMR register is documented to require context
synchronizing operations before and after, for it to take effect as
expected. The KUAP restore at interrupt exit time deliberately avoids
the isync after the AMR update because it only needs to take effect
after the context synchronizing RFID that soon follows. Add a comment
for this.

The missing isync before the update doesn't have an obvious
justification, and seems it could theoretically allow a rogue user
access to leak past the AMR update. Add isyncs for these.

Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20200429065654.1677541-3-npiggin@gmail.com
Signed-off-by: Sasha Levin <sashal@kernel.org>
arch/powerpc/include/asm/book3s/64/kup-radix.h

index 3bcef989a35dfc083140813f73bf0e4856f244ef..101d60f16d466382608bad941482d3544d13ccfb 100644 (file)
@@ -16,7 +16,9 @@
 #ifdef CONFIG_PPC_KUAP
        BEGIN_MMU_FTR_SECTION_NESTED(67)
        ld      \gpr, STACK_REGS_KUAP(r1)
+       isync
        mtspr   SPRN_AMR, \gpr
+       /* No isync required, see kuap_restore_amr() */
        END_MMU_FTR_SECTION_NESTED_IFSET(MMU_FTR_RADIX_KUAP, 67)
 #endif
 .endm
 
 static inline void kuap_restore_amr(struct pt_regs *regs)
 {
-       if (mmu_has_feature(MMU_FTR_RADIX_KUAP))
+       if (mmu_has_feature(MMU_FTR_RADIX_KUAP)) {
+               isync();
                mtspr(SPRN_AMR, regs->kuap);
+               /*
+                * No isync required here because we are about to RFI back to
+                * previous context before any user accesses would be made,
+                * which is a CSI.
+                */
+       }
 }
 
 static inline void kuap_check_amr(void)