mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-16 18:26:42 +00:00
4bd1d80efb
Current implementation of update_mmu_cache function performs local TLB flush. It does not take into account ASID information. Besides, it does not take into account other harts currently running the same mm context or possible migration of the running context to other harts. Meanwhile TLB flush is not performed for every context switch if ASID support is enabled. Patch [1] proposed to add ASID support to update_mmu_cache to avoid flushing local TLB entirely. This patch takes into account other harts currently running the same mm context as well as possible migration of this context to other harts. For this purpose the approach from flush_icache_mm is reused. Remote harts currently running the same mm context are informed via SBI calls that they need to flush their local TLBs. All the other harts are marked as needing a deferred TLB flush when this mm context runs on them. [1] https://lore.kernel.org/linux-riscv/20220821013926.8968-1-tjytimi@163.com/ Signed-off-by: Sergey Matyukevich <sergey.matyukevich@syntacore.com> Fixes: 65d4b9c53017 ("RISC-V: Implement ASID allocator") Cc: stable@vger.kernel.org Link: https://lore.kernel.org/linux-riscv/20220829205219.283543-1-geomatsi@gmail.com/#t Signed-off-by: Palmer Dabbelt <palmer@rivosinc.com>
83 lines
2.0 KiB
C
83 lines
2.0 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
#include <linux/mm.h>
|
|
#include <linux/smp.h>
|
|
#include <linux/sched.h>
|
|
#include <asm/sbi.h>
|
|
#include <asm/mmu_context.h>
|
|
#include <asm/tlbflush.h>
|
|
|
|
void flush_tlb_all(void)
|
|
{
|
|
sbi_remote_sfence_vma(NULL, 0, -1);
|
|
}
|
|
|
|
static void __sbi_tlb_flush_range(struct mm_struct *mm, unsigned long start,
|
|
unsigned long size, unsigned long stride)
|
|
{
|
|
struct cpumask *pmask = &mm->context.tlb_stale_mask;
|
|
struct cpumask *cmask = mm_cpumask(mm);
|
|
unsigned int cpuid;
|
|
bool broadcast;
|
|
|
|
if (cpumask_empty(cmask))
|
|
return;
|
|
|
|
cpuid = get_cpu();
|
|
/* check if the tlbflush needs to be sent to other CPUs */
|
|
broadcast = cpumask_any_but(cmask, cpuid) < nr_cpu_ids;
|
|
if (static_branch_unlikely(&use_asid_allocator)) {
|
|
unsigned long asid = atomic_long_read(&mm->context.id);
|
|
|
|
/*
|
|
* TLB will be immediately flushed on harts concurrently
|
|
* executing this MM context. TLB flush on other harts
|
|
* is deferred until this MM context migrates there.
|
|
*/
|
|
cpumask_setall(pmask);
|
|
cpumask_clear_cpu(cpuid, pmask);
|
|
cpumask_andnot(pmask, pmask, cmask);
|
|
|
|
if (broadcast) {
|
|
sbi_remote_sfence_vma_asid(cmask, start, size, asid);
|
|
} else if (size <= stride) {
|
|
local_flush_tlb_page_asid(start, asid);
|
|
} else {
|
|
local_flush_tlb_all_asid(asid);
|
|
}
|
|
} else {
|
|
if (broadcast) {
|
|
sbi_remote_sfence_vma(cmask, start, size);
|
|
} else if (size <= stride) {
|
|
local_flush_tlb_page(start);
|
|
} else {
|
|
local_flush_tlb_all();
|
|
}
|
|
}
|
|
|
|
put_cpu();
|
|
}
|
|
|
|
void flush_tlb_mm(struct mm_struct *mm)
|
|
{
|
|
__sbi_tlb_flush_range(mm, 0, -1, PAGE_SIZE);
|
|
}
|
|
|
|
void flush_tlb_page(struct vm_area_struct *vma, unsigned long addr)
|
|
{
|
|
__sbi_tlb_flush_range(vma->vm_mm, addr, PAGE_SIZE, PAGE_SIZE);
|
|
}
|
|
|
|
void flush_tlb_range(struct vm_area_struct *vma, unsigned long start,
|
|
unsigned long end)
|
|
{
|
|
__sbi_tlb_flush_range(vma->vm_mm, start, end - start, PAGE_SIZE);
|
|
}
|
|
#ifdef CONFIG_TRANSPARENT_HUGEPAGE
|
|
void flush_pmd_tlb_range(struct vm_area_struct *vma, unsigned long start,
|
|
unsigned long end)
|
|
{
|
|
__sbi_tlb_flush_range(vma->vm_mm, start, end - start, PMD_SIZE);
|
|
}
|
|
#endif
|