mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-23 16:53:58 -05:00
ARC: mm: vmalloc sync from kernel to user table to update PMD ...
... not PGD vmalloc() sets up the kernel page table (starting from @swapper_pg_dir). But when vmalloc area is accessed in context of a user task, say opening terminal in n_tty_open(), the user page tables need to be synced from kernel page tables so that TLB entry is created in "user context". The old code was doing this incorrectly, as it was updating the user pgd entry (first level itself) to point to kernel pud table (2nd level), effectively yanking away the entire user space translation with kernel one. The correct way to do this is to ONLY update a user space pgd/pud/pmd entry if it is not popluated already. This ensures that only the missing leaf pmd entry gets updated to point to relevant kernel pte table. From code change pov, we are chaging the pattern: p4d = p4d_offset(pgd, address); p4d_k = p4d_offset(pgd_k, address); if (!p4d_present(*p4d_k)) goto bad_area; set_p4d(p4d, *p4d_k); with p4d = p4d_offset(pgd, address); p4d_k = p4d_offset(pgd_k, address); if (p4d_none(*p4d_k)) goto bad_area; if (!p4d_present(*p4d)) set_p4d(p4d, *p4d_k); Signed-off-by: Vineet Gupta <vgupta@kernel.org>
This commit is contained in:
parent
8747ff704a
commit
56809a28d4
1 changed files with 12 additions and 12 deletions
|
@ -36,31 +36,31 @@ noinline static int handle_kernel_vaddr_fault(unsigned long address)
|
||||||
pgd = pgd_offset(current->active_mm, address);
|
pgd = pgd_offset(current->active_mm, address);
|
||||||
pgd_k = pgd_offset_k(address);
|
pgd_k = pgd_offset_k(address);
|
||||||
|
|
||||||
if (!pgd_present(*pgd_k))
|
if (pgd_none (*pgd_k))
|
||||||
goto bad_area;
|
goto bad_area;
|
||||||
|
if (!pgd_present(*pgd))
|
||||||
set_pgd(pgd, *pgd_k);
|
set_pgd(pgd, *pgd_k);
|
||||||
|
|
||||||
p4d = p4d_offset(pgd, address);
|
p4d = p4d_offset(pgd, address);
|
||||||
p4d_k = p4d_offset(pgd_k, address);
|
p4d_k = p4d_offset(pgd_k, address);
|
||||||
if (!p4d_present(*p4d_k))
|
if (p4d_none(*p4d_k))
|
||||||
goto bad_area;
|
goto bad_area;
|
||||||
|
if (!p4d_present(*p4d))
|
||||||
set_p4d(p4d, *p4d_k);
|
set_p4d(p4d, *p4d_k);
|
||||||
|
|
||||||
pud = pud_offset(p4d, address);
|
pud = pud_offset(p4d, address);
|
||||||
pud_k = pud_offset(p4d_k, address);
|
pud_k = pud_offset(p4d_k, address);
|
||||||
if (!pud_present(*pud_k))
|
if (pud_none(*pud_k))
|
||||||
goto bad_area;
|
goto bad_area;
|
||||||
|
if (!pud_present(*pud))
|
||||||
set_pud(pud, *pud_k);
|
set_pud(pud, *pud_k);
|
||||||
|
|
||||||
pmd = pmd_offset(pud, address);
|
pmd = pmd_offset(pud, address);
|
||||||
pmd_k = pmd_offset(pud_k, address);
|
pmd_k = pmd_offset(pud_k, address);
|
||||||
if (!pmd_present(*pmd_k))
|
if (pmd_none(*pmd_k))
|
||||||
goto bad_area;
|
goto bad_area;
|
||||||
|
if (!pmd_present(*pmd))
|
||||||
set_pmd(pmd, *pmd_k);
|
set_pmd(pmd, *pmd_k);
|
||||||
|
|
||||||
/* XXX: create the TLB entry here */
|
/* XXX: create the TLB entry here */
|
||||||
return 0;
|
return 0;
|
||||||
|
|
Loading…
Add table
Reference in a new issue