2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-09-26 16:04:21 +10:00
/*
* MMU context allocation for 64 - bit kernels .
*
* Copyright ( C ) 2004 Anton Blanchard , IBM Corp . < anton @ samba . org >
*/
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/types.h>
# include <linux/mm.h>
2018-01-18 17:50:25 -08:00
# include <linux/pkeys.h>
2005-09-26 16:04:21 +10:00
# include <linux/spinlock.h>
# include <linux/idr.h>
2011-07-22 18:24:23 -04:00
# include <linux/export.h>
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 17:04:11 +09:00
# include <linux/gfp.h>
2011-05-02 20:43:04 +00:00
# include <linux/slab.h>
powerpc/64s: Trim offlined CPUs from mm_cpumasks
When offlining a CPU, powerpc/64s does not flush TLBs, rather it just
leaves the CPU set in mm_cpumasks, so it continues to receive TLBIEs
to manage its TLBs.
However the exit_flush_lazy_tlbs() function expects that after
returning, all CPUs (except self) have flushed TLBs for that mm, in
which case TLBIEL can be used for this flush. This breaks for offline
CPUs because they don't get the IPI to flush their TLB. This can lead
to stale translations.
Fix this by clearing the CPU from mm_cpumasks, then flushing all TLBs
before going offline.
These offlined CPU bits stuck in the cpumask also prevents the cpumask
from being trimmed back to local mode, which means continual broadcast
IPIs or TLBIEs are needed for TLB flushing. This patch prevents that
situation too.
A cast of many were involved in working this out, but in particular
Milton, Aneesh, Paul made key discoveries.
Fixes: 0cef77c7798a7 ("powerpc/64s/radix: flush remote CPUs out of single-threaded mm_cpumask")
Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Reviewed-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Milton Miller <miltonm@us.ibm.com>
Debugged-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Paul Mackerras <paulus@samba.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20201126102530.691335-5-npiggin@gmail.com
2020-11-26 20:25:30 +10:00
# include <linux/cpu.h>
2005-09-26 16:04:21 +10:00
# include <asm/mmu_context.h>
2013-04-28 09:37:33 +00:00
# include <asm/pgalloc.h>
2005-09-26 16:04:21 +10:00
2020-09-16 21:56:36 +10:00
# include "internal.h"
2010-02-07 12:30:12 +00:00
static DEFINE_IDA ( mmu_context_ida ) ;
2005-09-26 16:04:21 +10:00
2017-03-29 22:10:45 +11:00
static int alloc_context_id ( int min_id , int max_id )
2005-09-26 16:04:21 +10:00
{
2018-06-18 08:26:32 -04:00
return ida_alloc_range ( & mmu_context_ida , min_id , max_id , GFP_KERNEL ) ;
2009-11-02 12:02:30 +00:00
}
2017-03-29 22:00:46 +11:00
2017-03-22 09:07:00 +05:30
void hash__reserve_context_id ( int id )
{
2018-06-18 08:26:32 -04:00
int result = ida_alloc_range ( & mmu_context_ida , id , id , GFP_KERNEL ) ;
2017-03-22 09:07:00 +05:30
WARN ( result ! = id , " mmu: Failed to reserve context id %d (rc %d) \n " , id , result ) ;
}
2017-03-29 22:00:46 +11:00
int hash__alloc_context_id ( void )
{
2017-03-29 17:21:53 +11:00
unsigned long max ;
if ( mmu_has_feature ( MMU_FTR_68_BIT_VA ) )
max = MAX_USER_CONTEXT ;
else
max = MAX_USER_CONTEXT_65BIT_VA ;
return alloc_context_id ( MIN_USER_CONTEXT , max ) ;
2017-03-29 22:00:46 +11:00
}
EXPORT_SYMBOL_GPL ( hash__alloc_context_id ) ;
powerpc/mm/64s/hash: Reallocate context ids on fork
When using the Hash Page Table (HPT) MMU, userspace memory mappings
are managed at two levels. Firstly in the Linux page tables, much like
other architectures, and secondly in the SLB (Segment Lookaside
Buffer) and HPT. It's the SLB and HPT that are actually used by the
hardware to do translations.
As part of the series adding support for 4PB user virtual address
space using the hash MMU, we added support for allocating multiple
"context ids" per process, one for each 512TB chunk of address space.
These are tracked in an array called extended_id in the mm_context_t
of a process that has done a mapping above 512TB.
If such a process forks (ie. clone(2) without CLONE_VM set) it's mm is
copied, including the mm_context_t, and then init_new_context() is
called to reinitialise parts of the mm_context_t as appropriate to
separate the address spaces of the two processes.
The key step in ensuring the two processes have separate address
spaces is to allocate a new context id for the process, this is done
at the beginning of hash__init_new_context(). If we didn't allocate a
new context id then the two processes would share mappings as far as
the SLB and HPT are concerned, even though their Linux page tables
would be separate.
For mappings above 512TB, which use the extended_id array, we
neglected to allocate new context ids on fork, meaning the parent and
child use the same ids and therefore share those mappings even though
they're supposed to be separate. This can lead to the parent seeing
writes done by the child, which is essentially memory corruption.
There is an additional exposure which is that if the child process
exits, all its context ids are freed, including the context ids that
are still in use by the parent for mappings above 512TB. One or more
of those ids can then be reallocated to a third process, that process
can then read/write to the parent's mappings above 512TB. Additionally
if the freed id is used for the third process's primary context id,
then the parent is able to read/write to the third process's mappings
*below* 512TB.
All of these are fundamental failures to enforce separation between
processes. The only mitigating factor is that the bug only occurs if a
process creates mappings above 512TB, and most applications still do
not create such mappings.
Only machines using the hash page table MMU are affected, eg. PowerPC
970 (G5), PA6T, Power5/6/7/8/9. By default Power9 bare metal machines
(powernv) use the Radix MMU and are not affected, unless the machine
has been explicitly booted in HPT mode (using disable_radix on the
kernel command line). KVM guests on Power9 may be affected if the host
or guest is configured to use the HPT MMU. LPARs under PowerVM on
Power9 are affected as they always use the HPT MMU. Kernels built with
PAGE_SIZE=4K are not affected.
The fix is relatively simple, we need to reallocate context ids for
all extended mappings on fork.
Fixes: f384796c40dc ("powerpc/mm: Add support for handling > 512TB address in SLB miss")
Cc: stable@vger.kernel.org # v4.17+
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2019-06-12 23:35:07 +10:00
static int realloc_context_ids ( mm_context_t * ctx )
{
int i , id ;
/*
* id 0 ( aka . ctx - > id ) is special , we always allocate a new one , even if
* there wasn ' t one allocated previously ( which happens in the exec
* case where ctx is newly allocated ) .
*
* We have to be a bit careful here . We must keep the existing ids in
* the array , so that we can test if they ' re non - zero to decide if we
* need to allocate a new one . However in case of error we must free the
* ids we ' ve allocated but * not * any of the existing ones ( or risk a
* UAF ) . That ' s why we decrement i at the start of the error handling
* loop , to skip the id that we just tested but couldn ' t reallocate .
*/
for ( i = 0 ; i < ARRAY_SIZE ( ctx - > extended_id ) ; i + + ) {
if ( i = = 0 | | ctx - > extended_id [ i ] ) {
id = hash__alloc_context_id ( ) ;
if ( id < 0 )
goto error ;
ctx - > extended_id [ i ] = id ;
}
}
/* The caller expects us to return id */
return ctx - > id ;
error :
for ( i - - ; i > = 0 ; i - - ) {
if ( ctx - > extended_id [ i ] )
ida_free ( & mmu_context_ida , ctx - > extended_id [ i ] ) ;
}
return id ;
}
2017-03-29 22:36:56 +11:00
static int hash__init_new_context ( struct mm_struct * mm )
{
int index ;
2019-04-17 18:33:51 +05:30
mm - > context . hash_context = kmalloc ( sizeof ( struct hash_mm_context ) ,
GFP_KERNEL ) ;
2019-06-13 15:00:34 +10:00
if ( ! mm - > context . hash_context )
2019-04-17 18:33:50 +05:30
return - ENOMEM ;
2017-03-29 22:36:56 +11:00
/*
* The old code would re - promote on fork , we don ' t do that when using
* slices as it could cause problem promoting slices that have been
* forced down to 4 K .
*
* For book3s we have MMU_NO_CONTEXT set to be ~ 0. Hence check
* explicitly against context . id = = 0. This ensures that we properly
* initialize context slice details for newly allocated mm ' s ( which will
* have id = = 0 ) and don ' t alter context slice inherited via fork ( which
* will have id ! = 0 ) .
*
* We should not be calling init_new_context ( ) on init_mm . Hence a
* check against 0 is OK .
*/
2019-04-17 18:33:50 +05:30
if ( mm - > context . id = = 0 ) {
memset ( mm - > context . hash_context , 0 , sizeof ( struct hash_mm_context ) ) ;
2018-03-07 11:37:09 +10:00
slice_init_new_context_exec ( mm ) ;
2019-04-17 18:33:50 +05:30
} else {
/* This is fork. Copy hash_context details from current->mm */
memcpy ( mm - > context . hash_context , current - > mm - > context . hash_context , sizeof ( struct hash_mm_context ) ) ;
2019-04-17 18:33:51 +05:30
# ifdef CONFIG_PPC_SUBPAGE_PROT
/* inherit subpage prot detalis if we have one. */
if ( current - > mm - > context . hash_context - > spt ) {
mm - > context . hash_context - > spt = kmalloc ( sizeof ( struct subpage_prot_table ) ,
GFP_KERNEL ) ;
if ( ! mm - > context . hash_context - > spt ) {
kfree ( mm - > context . hash_context ) ;
return - ENOMEM ;
}
}
# endif
2019-06-13 15:00:34 +10:00
}
2019-04-17 18:33:50 +05:30
powerpc/mm/64s/hash: Reallocate context ids on fork
When using the Hash Page Table (HPT) MMU, userspace memory mappings
are managed at two levels. Firstly in the Linux page tables, much like
other architectures, and secondly in the SLB (Segment Lookaside
Buffer) and HPT. It's the SLB and HPT that are actually used by the
hardware to do translations.
As part of the series adding support for 4PB user virtual address
space using the hash MMU, we added support for allocating multiple
"context ids" per process, one for each 512TB chunk of address space.
These are tracked in an array called extended_id in the mm_context_t
of a process that has done a mapping above 512TB.
If such a process forks (ie. clone(2) without CLONE_VM set) it's mm is
copied, including the mm_context_t, and then init_new_context() is
called to reinitialise parts of the mm_context_t as appropriate to
separate the address spaces of the two processes.
The key step in ensuring the two processes have separate address
spaces is to allocate a new context id for the process, this is done
at the beginning of hash__init_new_context(). If we didn't allocate a
new context id then the two processes would share mappings as far as
the SLB and HPT are concerned, even though their Linux page tables
would be separate.
For mappings above 512TB, which use the extended_id array, we
neglected to allocate new context ids on fork, meaning the parent and
child use the same ids and therefore share those mappings even though
they're supposed to be separate. This can lead to the parent seeing
writes done by the child, which is essentially memory corruption.
There is an additional exposure which is that if the child process
exits, all its context ids are freed, including the context ids that
are still in use by the parent for mappings above 512TB. One or more
of those ids can then be reallocated to a third process, that process
can then read/write to the parent's mappings above 512TB. Additionally
if the freed id is used for the third process's primary context id,
then the parent is able to read/write to the third process's mappings
*below* 512TB.
All of these are fundamental failures to enforce separation between
processes. The only mitigating factor is that the bug only occurs if a
process creates mappings above 512TB, and most applications still do
not create such mappings.
Only machines using the hash page table MMU are affected, eg. PowerPC
970 (G5), PA6T, Power5/6/7/8/9. By default Power9 bare metal machines
(powernv) use the Radix MMU and are not affected, unless the machine
has been explicitly booted in HPT mode (using disable_radix on the
kernel command line). KVM guests on Power9 may be affected if the host
or guest is configured to use the HPT MMU. LPARs under PowerVM on
Power9 are affected as they always use the HPT MMU. Kernels built with
PAGE_SIZE=4K are not affected.
The fix is relatively simple, we need to reallocate context ids for
all extended mappings on fork.
Fixes: f384796c40dc ("powerpc/mm: Add support for handling > 512TB address in SLB miss")
Cc: stable@vger.kernel.org # v4.17+
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2019-06-12 23:35:07 +10:00
index = realloc_context_ids ( & mm - > context ) ;
2019-06-13 15:00:34 +10:00
if ( index < 0 ) {
# ifdef CONFIG_PPC_SUBPAGE_PROT
kfree ( mm - > context . hash_context - > spt ) ;
# endif
kfree ( mm - > context . hash_context ) ;
powerpc/mm/64s/hash: Reallocate context ids on fork
When using the Hash Page Table (HPT) MMU, userspace memory mappings
are managed at two levels. Firstly in the Linux page tables, much like
other architectures, and secondly in the SLB (Segment Lookaside
Buffer) and HPT. It's the SLB and HPT that are actually used by the
hardware to do translations.
As part of the series adding support for 4PB user virtual address
space using the hash MMU, we added support for allocating multiple
"context ids" per process, one for each 512TB chunk of address space.
These are tracked in an array called extended_id in the mm_context_t
of a process that has done a mapping above 512TB.
If such a process forks (ie. clone(2) without CLONE_VM set) it's mm is
copied, including the mm_context_t, and then init_new_context() is
called to reinitialise parts of the mm_context_t as appropriate to
separate the address spaces of the two processes.
The key step in ensuring the two processes have separate address
spaces is to allocate a new context id for the process, this is done
at the beginning of hash__init_new_context(). If we didn't allocate a
new context id then the two processes would share mappings as far as
the SLB and HPT are concerned, even though their Linux page tables
would be separate.
For mappings above 512TB, which use the extended_id array, we
neglected to allocate new context ids on fork, meaning the parent and
child use the same ids and therefore share those mappings even though
they're supposed to be separate. This can lead to the parent seeing
writes done by the child, which is essentially memory corruption.
There is an additional exposure which is that if the child process
exits, all its context ids are freed, including the context ids that
are still in use by the parent for mappings above 512TB. One or more
of those ids can then be reallocated to a third process, that process
can then read/write to the parent's mappings above 512TB. Additionally
if the freed id is used for the third process's primary context id,
then the parent is able to read/write to the third process's mappings
*below* 512TB.
All of these are fundamental failures to enforce separation between
processes. The only mitigating factor is that the bug only occurs if a
process creates mappings above 512TB, and most applications still do
not create such mappings.
Only machines using the hash page table MMU are affected, eg. PowerPC
970 (G5), PA6T, Power5/6/7/8/9. By default Power9 bare metal machines
(powernv) use the Radix MMU and are not affected, unless the machine
has been explicitly booted in HPT mode (using disable_radix on the
kernel command line). KVM guests on Power9 may be affected if the host
or guest is configured to use the HPT MMU. LPARs under PowerVM on
Power9 are affected as they always use the HPT MMU. Kernels built with
PAGE_SIZE=4K are not affected.
The fix is relatively simple, we need to reallocate context ids for
all extended mappings on fork.
Fixes: f384796c40dc ("powerpc/mm: Add support for handling > 512TB address in SLB miss")
Cc: stable@vger.kernel.org # v4.17+
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2019-06-12 23:35:07 +10:00
return index ;
2019-04-17 18:33:50 +05:30
}
2017-03-29 22:36:56 +11:00
2018-01-18 17:50:25 -08:00
pkey_mm_init ( mm ) ;
2017-03-29 22:36:56 +11:00
return index ;
}
2018-09-15 01:30:55 +10:00
void hash__setup_new_exec ( void )
{
slice_setup_new_exec ( ) ;
powerpc/64s/hash: Add a SLB preload cache
When switching processes, currently all user SLBEs are cleared, and a
few (exec_base, pc, and stack) are preloaded. In trivial testing with
small apps, this tends to miss the heap and low 256MB segments, and it
will also miss commonly accessed segments on large memory workloads.
Add a simple round-robin preload cache that just inserts the last SLB
miss into the head of the cache and preloads those at context switch
time. Every 256 context switches, the oldest entry is removed from the
cache to shrink the cache and require fewer slbmte if they are unused.
Much more could go into this, including into the SLB entry reclaim
side to track some LRU information etc, which would require a study of
large memory workloads. But this is a simple thing we can do now that
is an obvious win for common workloads.
With the full series, process switching speed on the context_switch
benchmark on POWER9/hash (with kernel speculation security masures
disabled) increases from 140K/s to 178K/s (27%).
POWER8 does not change much (within 1%), it's unclear why it does not
see a big gain like POWER9.
Booting to busybox init with 256MB segments has SLB misses go down
from 945 to 69, and with 1T segments 900 to 21. These could almost all
be eliminated by preloading a bit more carefully with ELF binary
loading.
Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2018-09-15 01:30:56 +10:00
slb_setup_new_exec ( ) ;
2018-09-15 01:30:55 +10:00
}
2017-03-29 22:36:56 +11:00
static int radix__init_new_context ( struct mm_struct * mm )
2016-04-29 23:26:02 +10:00
{
unsigned long rts_field ;
powerpc/mm/radix: Workaround prefetch issue with KVM
There's a somewhat architectural issue with Radix MMU and KVM.
When coming out of a guest with AIL (Alternate Interrupt Location, ie,
MMU enabled), we start executing hypervisor code with the PID register
still containing whatever the guest has been using.
The problem is that the CPU can (and will) then start prefetching or
speculatively load from whatever host context has that same PID (if
any), thus bringing translations for that context into the TLB, which
Linux doesn't know about.
This can cause stale translations and subsequent crashes.
Fixing this in a way that is neither racy nor a huge performance
impact is difficult. We could just make the host invalidations always
use broadcast forms but that would hurt single threaded programs for
example.
We chose to fix it instead by partitioning the PID space between guest
and host. This is possible because today Linux only use 19 out of the
20 bits of PID space, so existing guests will work if we make the host
use the top half of the 20 bits space.
We additionally add support for a property to indicate to Linux the
size of the PID register which will be useful if we eventually have
processors with a larger PID space available.
There is still an issue with malicious guests purposefully setting the
PID register to a value in the hosts PID range. Hopefully future HW
can prevent that, but in the meantime, we handle it with a pair of
kludges:
- On the way out of a guest, before we clear the current VCPU in the
PACA, we check the PID and if it's outside of the permitted range
we flush the TLB for that PID.
- When context switching, if the mm is "new" on that CPU (the
corresponding bit was set for the first time in the mm cpumask), we
check if any sibling thread is in KVM (has a non-NULL VCPU pointer
in the PACA). If that is the case, we also flush the PID for that
CPU (core).
This second part is needed to handle the case where a process is
migrated (or starts a new pthread) on a sibling thread of the CPU
coming out of KVM, as there's a window where stale translations can
exist before we detect it and flush them out.
A future optimization could be added by keeping track of whether the
PID has ever been used and avoid doing that for completely fresh PIDs.
We could similarily mark PIDs that have been the subject of a global
invalidation as "fresh". But for now this will do.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
[mpe: Rework the asm to build with CONFIG_PPC_RADIX_MMU=n, drop
unneeded include of kvm_book3s_asm.h]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2017-07-24 14:26:06 +10:00
int index , max_id ;
2017-03-29 22:36:56 +11:00
powerpc/mm/radix: Workaround prefetch issue with KVM
There's a somewhat architectural issue with Radix MMU and KVM.
When coming out of a guest with AIL (Alternate Interrupt Location, ie,
MMU enabled), we start executing hypervisor code with the PID register
still containing whatever the guest has been using.
The problem is that the CPU can (and will) then start prefetching or
speculatively load from whatever host context has that same PID (if
any), thus bringing translations for that context into the TLB, which
Linux doesn't know about.
This can cause stale translations and subsequent crashes.
Fixing this in a way that is neither racy nor a huge performance
impact is difficult. We could just make the host invalidations always
use broadcast forms but that would hurt single threaded programs for
example.
We chose to fix it instead by partitioning the PID space between guest
and host. This is possible because today Linux only use 19 out of the
20 bits of PID space, so existing guests will work if we make the host
use the top half of the 20 bits space.
We additionally add support for a property to indicate to Linux the
size of the PID register which will be useful if we eventually have
processors with a larger PID space available.
There is still an issue with malicious guests purposefully setting the
PID register to a value in the hosts PID range. Hopefully future HW
can prevent that, but in the meantime, we handle it with a pair of
kludges:
- On the way out of a guest, before we clear the current VCPU in the
PACA, we check the PID and if it's outside of the permitted range
we flush the TLB for that PID.
- When context switching, if the mm is "new" on that CPU (the
corresponding bit was set for the first time in the mm cpumask), we
check if any sibling thread is in KVM (has a non-NULL VCPU pointer
in the PACA). If that is the case, we also flush the PID for that
CPU (core).
This second part is needed to handle the case where a process is
migrated (or starts a new pthread) on a sibling thread of the CPU
coming out of KVM, as there's a window where stale translations can
exist before we detect it and flush them out.
A future optimization could be added by keeping track of whether the
PID has ever been used and avoid doing that for completely fresh PIDs.
We could similarily mark PIDs that have been the subject of a global
invalidation as "fresh". But for now this will do.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
[mpe: Rework the asm to build with CONFIG_PPC_RADIX_MMU=n, drop
unneeded include of kvm_book3s_asm.h]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2017-07-24 14:26:06 +10:00
max_id = ( 1 < < mmu_pid_bits ) - 1 ;
index = alloc_context_id ( mmu_base_pid , max_id ) ;
2017-03-29 22:36:56 +11:00
if ( index < 0 )
return index ;
2016-04-29 23:26:02 +10:00
/*
* set the process table entry ,
*/
2016-06-17 11:40:36 +05:30
rts_field = radix__get_tree_size ( ) ;
2016-04-29 23:26:02 +10:00
process_tb [ index ] . prtb0 = cpu_to_be64 ( rts_field | __pa ( mm - > pgd ) | RADIX_PGD_INDEX_SIZE ) ;
2017-03-29 22:36:56 +11:00
2017-07-07 16:12:16 -05:00
/*
* Order the above store with subsequent update of the PID
* register ( at which point HW can start loading / caching
* the entry ) and the corresponding load by the MMU from
* the L2 cache .
*/
asm volatile ( " ptesync;isync " : : : " memory " ) ;
2019-04-17 18:33:50 +05:30
mm - > context . hash_context = NULL ;
2017-04-03 19:51:44 +10:00
2017-03-29 22:36:56 +11:00
return index ;
2016-04-29 23:26:02 +10:00
}
2009-11-02 12:02:30 +00:00
int init_new_context ( struct task_struct * tsk , struct mm_struct * mm )
{
int index ;
2017-03-29 22:36:56 +11:00
if ( radix_enabled ( ) )
index = radix__init_new_context ( mm ) ;
else
index = hash__init_new_context ( mm ) ;
2009-11-02 12:02:30 +00:00
if ( index < 0 )
return index ;
2007-08-15 16:33:55 +10:00
mm - > context . id = index ;
2005-09-26 16:04:21 +10:00
2013-04-28 09:37:33 +00:00
mm - > context . pte_frag = NULL ;
2018-04-16 16:57:22 +05:30
mm - > context . pmd_frag = NULL ;
2015-06-05 16:35:24 +10:00
# ifdef CONFIG_SPAPR_TCE_IOMMU
2016-11-30 17:51:59 +11:00
mm_iommu_init ( mm ) ;
2013-04-28 09:37:33 +00:00
# endif
2017-07-24 14:28:02 +10:00
atomic_set ( & mm - > context . active_cpus , 0 ) ;
2018-03-23 09:29:05 +11:00
atomic_set ( & mm - > context . copros , 0 ) ;
2017-07-24 14:28:02 +10:00
2005-09-26 16:04:21 +10:00
return 0 ;
}
2009-11-02 12:02:30 +00:00
void __destroy_context ( int context_id )
2005-09-26 16:04:21 +10:00
{
2018-06-18 08:26:32 -04:00
ida_free ( & mmu_context_ida , context_id ) ;
2009-11-02 12:02:30 +00:00
}
EXPORT_SYMBOL_GPL ( __destroy_context ) ;
2005-09-26 16:04:21 +10:00
2018-03-26 15:34:48 +05:30
static void destroy_contexts ( mm_context_t * ctx )
{
int index , context_id ;
for ( index = 0 ; index < ARRAY_SIZE ( ctx - > extended_id ) ; index + + ) {
context_id = ctx - > extended_id [ index ] ;
if ( context_id )
2018-06-18 08:26:32 -04:00
ida_free ( & mmu_context_ida , context_id ) ;
2018-03-26 15:34:48 +05:30
}
2019-04-17 18:33:50 +05:30
kfree ( ctx - > hash_context ) ;
2018-03-26 15:34:48 +05:30
}
2018-04-16 16:57:22 +05:30
static void pmd_frag_destroy ( void * pmd_frag )
{
int count ;
struct page * page ;
page = virt_to_page ( pmd_frag ) ;
/* drop all the pending references */
count = ( ( unsigned long ) pmd_frag & ~ PAGE_MASK ) > > PMD_FRAG_SIZE_SHIFT ;
/* We allow PTE_FRAG_NR fragments from a PTE page */
2018-07-27 21:48:17 +10:00
if ( atomic_sub_and_test ( PMD_FRAG_NR - count , & page - > pt_frag_refcount ) ) {
2018-04-16 16:57:22 +05:30
pgtable_pmd_page_dtor ( page ) ;
2018-07-27 21:48:17 +10:00
__free_page ( page ) ;
2018-04-16 16:57:22 +05:30
}
}
2018-07-25 19:54:28 +10:00
static void destroy_pagetable_cache ( struct mm_struct * mm )
2018-04-16 16:57:22 +05:30
{
void * frag ;
frag = mm - > context . pte_frag ;
if ( frag )
pte_frag_destroy ( frag ) ;
frag = mm - > context . pmd_frag ;
if ( frag )
pmd_frag_destroy ( frag ) ;
return ;
}
2009-11-02 12:02:30 +00:00
void destroy_context ( struct mm_struct * mm )
{
2015-06-05 16:35:24 +10:00
# ifdef CONFIG_SPAPR_TCE_IOMMU
2016-11-30 17:52:05 +11:00
WARN_ON_ONCE ( ! list_empty ( & mm - > context . iommu_group_mem_list ) ) ;
2015-06-05 16:35:24 +10:00
# endif
2019-09-18 19:31:03 +05:30
/*
* For tasks which were successfully initialized we end up calling
* arch_exit_mmap ( ) which clears the process table entry . And
* arch_exit_mmap ( ) is called before the required fullmm TLB flush
* which does a RIC = 2 flush . Hence for an initialized task , we do clear
* any cached process table entries .
*
* The condition below handles the error case during task init . We have
* set the process table entry early and if we fail a task
* initialization , we need to ensure the process table entry is zeroed .
* We need not worry about process table entry caches because the task
* never ran with the PID value .
*/
2017-10-24 23:06:54 +10:00
if ( radix_enabled ( ) )
2019-09-18 19:31:03 +05:30
process_tb [ mm - > context . id ] . prtb0 = 0 ;
2017-10-24 23:06:54 +10:00
else
subpage_prot_free ( mm ) ;
2018-03-26 15:34:48 +05:30
destroy_contexts ( & mm - > context ) ;
2017-10-24 23:06:54 +10:00
mm - > context . id = MMU_NO_CONTEXT ;
}
void arch_exit_mmap ( struct mm_struct * mm )
{
2018-07-25 19:54:28 +10:00
destroy_pagetable_cache ( mm ) ;
2017-07-08 07:45:32 -05:00
if ( radix_enabled ( ) ) {
/*
* Radix doesn ' t have a valid bit in the process table
* entries . However we know that at least P9 implementation
* will avoid caching an entry with an invalid RTS field ,
* and 0 is invalid . So this will do .
2017-10-24 23:06:54 +10:00
*
* This runs before the " fullmm " tlb flush in exit_mmap ,
* which does a RIC = 2 tlbie to clear the process table
* entry . See the " fullmm " comments in tlb - radix . c .
*
* No barrier required here after the store because
* this process will do the invalidate , which starts with
* ptesync .
2017-07-08 07:45:32 -05:00
*/
process_tb [ mm - > context . id ] . prtb0 = 0 ;
2017-10-24 23:06:54 +10:00
}
2005-09-26 16:04:21 +10:00
}
2016-04-29 23:26:02 +10:00
# ifdef CONFIG_PPC_RADIX_MMU
void radix__switch_mmu_context ( struct mm_struct * prev , struct mm_struct * next )
{
2018-07-05 18:47:00 +10:00
mtspr ( SPRN_PID , next - > context . id ) ;
isync ( ) ;
2016-04-29 23:26:02 +10:00
}
# endif
powerpc/64s: Trim offlined CPUs from mm_cpumasks
When offlining a CPU, powerpc/64s does not flush TLBs, rather it just
leaves the CPU set in mm_cpumasks, so it continues to receive TLBIEs
to manage its TLBs.
However the exit_flush_lazy_tlbs() function expects that after
returning, all CPUs (except self) have flushed TLBs for that mm, in
which case TLBIEL can be used for this flush. This breaks for offline
CPUs because they don't get the IPI to flush their TLB. This can lead
to stale translations.
Fix this by clearing the CPU from mm_cpumasks, then flushing all TLBs
before going offline.
These offlined CPU bits stuck in the cpumask also prevents the cpumask
from being trimmed back to local mode, which means continual broadcast
IPIs or TLBIEs are needed for TLB flushing. This patch prevents that
situation too.
A cast of many were involved in working this out, but in particular
Milton, Aneesh, Paul made key discoveries.
Fixes: 0cef77c7798a7 ("powerpc/64s/radix: flush remote CPUs out of single-threaded mm_cpumask")
Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Reviewed-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Milton Miller <miltonm@us.ibm.com>
Debugged-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Paul Mackerras <paulus@samba.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20201126102530.691335-5-npiggin@gmail.com
2020-11-26 20:25:30 +10:00
/**
* cleanup_cpu_mmu_context - Clean up MMU details for this CPU ( newly offlined )
*
* This clears the CPU from mm_cpumask for all processes , and then flushes the
* local TLB to ensure TLB coherency in case the CPU is onlined again .
*
* KVM guest translations are not necessarily flushed here . If KVM started
* using mm_cpumask or the Linux APIs which do , this would have to be resolved .
*/
# ifdef CONFIG_HOTPLUG_CPU
void cleanup_cpu_mmu_context ( void )
{
int cpu = smp_processor_id ( ) ;
clear_tasks_mm_cpumask ( cpu ) ;
tlbiel_all ( ) ;
}
# endif