powerpc fixes for 4.13 #8
Just one fix, to add a barrier in the switch_mm() code to make sure the mm cpumask update is ordered vs the MMU starting to load translations. As far as we know no one's actually hit the bug, but that's just luck. Thanks to: Benjamin Herrenschmidt, Nicholas Piggin. -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIcBAABAgAGBQJZoAZDAAoJEFHr6jzI4aWAi3AQAJq4boEBqdmL042oNK4PWW0M uGfehNmtzCw9Hp8bPfzOf8NypJ51Kw7eDQELaeSaazKW+gffUCBeEsKGS7kmHvc+ x1tHxkXxI7PXuNIRojJg9y7rlKXdRym5SecvPSo1cm/c46RRWOlNGZaIwiHyrXSh eBjyP5EHu1HXpRxkcUh+//PQp2b+7SmgUYzSf0hA9UCtzSZSJr19DuY8uhetI9Ws AfjkO1uvb2KETqBVegGBpAruZzQtxqdtffd2HToSaCHUnAKma2iqUZqkqBNjL6OQ gSXWpXVInng/7ktrrfEgSiwlHns7pgHkxYHS8thDZqQpIt3GNsUg2UwpHGf6oL7V L+GtRp36LM91Ueq6KdlU7bJkmoiJ798Hnp3FOjpkqo+j/MGuCQDDDK4Ge1popehJ a17K7lE/FKGqNaFINc1Q6hnXg4MPyawAOLDlV839Ap5+ISPS6WcHaa1AgKjdQNkH fIkZZsYT531FIf853AjUGFw8frSlVfrHmIx9/HJOhEa1KHQhBqGRV1sWYEjuN6IB av+tQDlleG5aT641qhHlA/hN5DGrGZXLp8e6cFRufF+CSsRayL27u0Qw9pP9VZ3S bgfdnmZZyP23+bzaq/m/bjhRiOf0snSQPxIKe56KmNCJ8buTrGWDw4IuiPKB7Y6V 06vBFn7ZUP5aeHIZkS62 =IClj -----END PGP SIGNATURE----- Merge tag 'powerpc-4.13-8' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux Pull powerpc fix from Michael Ellerman: "Just one fix, to add a barrier in the switch_mm() code to make sure the mm cpumask update is ordered vs the MMU starting to load translations. As far as we know no one's actually hit the bug, but that's just luck. Thanks to Benjamin Herrenschmidt, Nicholas Piggin" * tag 'powerpc-4.13-8' of git://git.kernel.org/pub/scm/linux/kernel/git/powerpc/linux: powerpc/mm: Ensure cpumask update is ordered
This commit is contained in:
commit
42e6d5e5ee
|
@ -90,6 +90,24 @@ static inline void switch_mm_irqs_off(struct mm_struct *prev,
|
||||||
/* Mark this context has been used on the new CPU */
|
/* Mark this context has been used on the new CPU */
|
||||||
if (!cpumask_test_cpu(smp_processor_id(), mm_cpumask(next))) {
|
if (!cpumask_test_cpu(smp_processor_id(), mm_cpumask(next))) {
|
||||||
cpumask_set_cpu(smp_processor_id(), mm_cpumask(next));
|
cpumask_set_cpu(smp_processor_id(), mm_cpumask(next));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This full barrier orders the store to the cpumask above vs
|
||||||
|
* a subsequent operation which allows this CPU to begin loading
|
||||||
|
* translations for next.
|
||||||
|
*
|
||||||
|
* When using the radix MMU that operation is the load of the
|
||||||
|
* MMU context id, which is then moved to SPRN_PID.
|
||||||
|
*
|
||||||
|
* For the hash MMU it is either the first load from slb_cache
|
||||||
|
* in switch_slb(), and/or the store of paca->mm_ctx_id in
|
||||||
|
* copy_mm_to_paca().
|
||||||
|
*
|
||||||
|
* On the read side the barrier is in pte_xchg(), which orders
|
||||||
|
* the store to the PTE vs the load of mm_cpumask.
|
||||||
|
*/
|
||||||
|
smp_mb();
|
||||||
|
|
||||||
new_on_cpu = true;
|
new_on_cpu = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -87,6 +87,7 @@ static inline bool pte_xchg(pte_t *ptep, pte_t old, pte_t new)
|
||||||
unsigned long *p = (unsigned long *)ptep;
|
unsigned long *p = (unsigned long *)ptep;
|
||||||
__be64 prev;
|
__be64 prev;
|
||||||
|
|
||||||
|
/* See comment in switch_mm_irqs_off() */
|
||||||
prev = (__force __be64)__cmpxchg_u64(p, (__force unsigned long)pte_raw(old),
|
prev = (__force __be64)__cmpxchg_u64(p, (__force unsigned long)pte_raw(old),
|
||||||
(__force unsigned long)pte_raw(new));
|
(__force unsigned long)pte_raw(new));
|
||||||
|
|
||||||
|
|
|
@ -62,6 +62,7 @@ static inline bool pte_xchg(pte_t *ptep, pte_t old, pte_t new)
|
||||||
{
|
{
|
||||||
unsigned long *p = (unsigned long *)ptep;
|
unsigned long *p = (unsigned long *)ptep;
|
||||||
|
|
||||||
|
/* See comment in switch_mm_irqs_off() */
|
||||||
return pte_val(old) == __cmpxchg_u64(p, pte_val(old), pte_val(new));
|
return pte_val(old) == __cmpxchg_u64(p, pte_val(old), pte_val(new));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in New Issue