162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * KASAN for 64-bit Book3e powerpc
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright 2022, Christophe Leroy, CS GROUP France
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#define DISABLE_BRANCH_PROFILING
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/kasan.h>
1162306a36Sopenharmony_ci#include <linux/printk.h>
1262306a36Sopenharmony_ci#include <linux/memblock.h>
1362306a36Sopenharmony_ci#include <linux/set_memory.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci#include <asm/pgalloc.h>
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_cistatic inline bool kasan_pud_table(p4d_t p4d)
1862306a36Sopenharmony_ci{
1962306a36Sopenharmony_ci	return p4d_page(p4d) == virt_to_page(lm_alias(kasan_early_shadow_pud));
2062306a36Sopenharmony_ci}
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic inline bool kasan_pmd_table(pud_t pud)
2362306a36Sopenharmony_ci{
2462306a36Sopenharmony_ci	return pud_page(pud) == virt_to_page(lm_alias(kasan_early_shadow_pmd));
2562306a36Sopenharmony_ci}
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_cistatic inline bool kasan_pte_table(pmd_t pmd)
2862306a36Sopenharmony_ci{
2962306a36Sopenharmony_ci	return pmd_page(pmd) == virt_to_page(lm_alias(kasan_early_shadow_pte));
3062306a36Sopenharmony_ci}
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_cistatic int __init kasan_map_kernel_page(unsigned long ea, unsigned long pa, pgprot_t prot)
3362306a36Sopenharmony_ci{
3462306a36Sopenharmony_ci	pgd_t *pgdp;
3562306a36Sopenharmony_ci	p4d_t *p4dp;
3662306a36Sopenharmony_ci	pud_t *pudp;
3762306a36Sopenharmony_ci	pmd_t *pmdp;
3862306a36Sopenharmony_ci	pte_t *ptep;
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci	pgdp = pgd_offset_k(ea);
4162306a36Sopenharmony_ci	p4dp = p4d_offset(pgdp, ea);
4262306a36Sopenharmony_ci	if (kasan_pud_table(*p4dp)) {
4362306a36Sopenharmony_ci		pudp = memblock_alloc(PUD_TABLE_SIZE, PUD_TABLE_SIZE);
4462306a36Sopenharmony_ci		memcpy(pudp, kasan_early_shadow_pud, PUD_TABLE_SIZE);
4562306a36Sopenharmony_ci		p4d_populate(&init_mm, p4dp, pudp);
4662306a36Sopenharmony_ci	}
4762306a36Sopenharmony_ci	pudp = pud_offset(p4dp, ea);
4862306a36Sopenharmony_ci	if (kasan_pmd_table(*pudp)) {
4962306a36Sopenharmony_ci		pmdp = memblock_alloc(PMD_TABLE_SIZE, PMD_TABLE_SIZE);
5062306a36Sopenharmony_ci		memcpy(pmdp, kasan_early_shadow_pmd, PMD_TABLE_SIZE);
5162306a36Sopenharmony_ci		pud_populate(&init_mm, pudp, pmdp);
5262306a36Sopenharmony_ci	}
5362306a36Sopenharmony_ci	pmdp = pmd_offset(pudp, ea);
5462306a36Sopenharmony_ci	if (kasan_pte_table(*pmdp)) {
5562306a36Sopenharmony_ci		ptep = memblock_alloc(PTE_TABLE_SIZE, PTE_TABLE_SIZE);
5662306a36Sopenharmony_ci		memcpy(ptep, kasan_early_shadow_pte, PTE_TABLE_SIZE);
5762306a36Sopenharmony_ci		pmd_populate_kernel(&init_mm, pmdp, ptep);
5862306a36Sopenharmony_ci	}
5962306a36Sopenharmony_ci	ptep = pte_offset_kernel(pmdp, ea);
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_ci	__set_pte_at(&init_mm, ea, ptep, pfn_pte(pa >> PAGE_SHIFT, prot), 0);
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_ci	return 0;
6462306a36Sopenharmony_ci}
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_cistatic void __init kasan_init_phys_region(void *start, void *end)
6762306a36Sopenharmony_ci{
6862306a36Sopenharmony_ci	unsigned long k_start, k_end, k_cur;
6962306a36Sopenharmony_ci	void *va;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	if (start >= end)
7262306a36Sopenharmony_ci		return;
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	k_start = ALIGN_DOWN((unsigned long)kasan_mem_to_shadow(start), PAGE_SIZE);
7562306a36Sopenharmony_ci	k_end = ALIGN((unsigned long)kasan_mem_to_shadow(end), PAGE_SIZE);
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_ci	va = memblock_alloc(k_end - k_start, PAGE_SIZE);
7862306a36Sopenharmony_ci	for (k_cur = k_start; k_cur < k_end; k_cur += PAGE_SIZE, va += PAGE_SIZE)
7962306a36Sopenharmony_ci		kasan_map_kernel_page(k_cur, __pa(va), PAGE_KERNEL);
8062306a36Sopenharmony_ci}
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_civoid __init kasan_early_init(void)
8362306a36Sopenharmony_ci{
8462306a36Sopenharmony_ci	int i;
8562306a36Sopenharmony_ci	unsigned long addr;
8662306a36Sopenharmony_ci	pgd_t *pgd = pgd_offset_k(KASAN_SHADOW_START);
8762306a36Sopenharmony_ci	pte_t zero_pte = pfn_pte(virt_to_pfn(kasan_early_shadow_page), PAGE_KERNEL);
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_ci	BUILD_BUG_ON(!IS_ALIGNED(KASAN_SHADOW_START, PGDIR_SIZE));
9062306a36Sopenharmony_ci	BUILD_BUG_ON(!IS_ALIGNED(KASAN_SHADOW_END, PGDIR_SIZE));
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	for (i = 0; i < PTRS_PER_PTE; i++)
9362306a36Sopenharmony_ci		__set_pte_at(&init_mm, (unsigned long)kasan_early_shadow_page,
9462306a36Sopenharmony_ci			     &kasan_early_shadow_pte[i], zero_pte, 0);
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci	for (i = 0; i < PTRS_PER_PMD; i++)
9762306a36Sopenharmony_ci		pmd_populate_kernel(&init_mm, &kasan_early_shadow_pmd[i],
9862306a36Sopenharmony_ci				    kasan_early_shadow_pte);
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_ci	for (i = 0; i < PTRS_PER_PUD; i++)
10162306a36Sopenharmony_ci		pud_populate(&init_mm, &kasan_early_shadow_pud[i],
10262306a36Sopenharmony_ci			     kasan_early_shadow_pmd);
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_ci	for (addr = KASAN_SHADOW_START; addr != KASAN_SHADOW_END; addr += PGDIR_SIZE)
10562306a36Sopenharmony_ci		p4d_populate(&init_mm, p4d_offset(pgd++, addr), kasan_early_shadow_pud);
10662306a36Sopenharmony_ci}
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_civoid __init kasan_init(void)
10962306a36Sopenharmony_ci{
11062306a36Sopenharmony_ci	phys_addr_t start, end;
11162306a36Sopenharmony_ci	u64 i;
11262306a36Sopenharmony_ci	pte_t zero_pte = pfn_pte(virt_to_pfn(kasan_early_shadow_page), PAGE_KERNEL_RO);
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci	for_each_mem_range(i, &start, &end)
11562306a36Sopenharmony_ci		kasan_init_phys_region((void *)start, (void *)end);
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	if (IS_ENABLED(CONFIG_KASAN_VMALLOC))
11862306a36Sopenharmony_ci		kasan_remove_zero_shadow((void *)VMALLOC_START, VMALLOC_SIZE);
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	for (i = 0; i < PTRS_PER_PTE; i++)
12162306a36Sopenharmony_ci		__set_pte_at(&init_mm, (unsigned long)kasan_early_shadow_page,
12262306a36Sopenharmony_ci			     &kasan_early_shadow_pte[i], zero_pte, 0);
12362306a36Sopenharmony_ci
12462306a36Sopenharmony_ci	flush_tlb_kernel_range(KASAN_SHADOW_START, KASAN_SHADOW_END);
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_ci	memset(kasan_early_shadow_page, 0, PAGE_SIZE);
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	/* Enable error messages */
12962306a36Sopenharmony_ci	init_task.kasan_depth = 0;
13062306a36Sopenharmony_ci	pr_info("KASAN init done\n");
13162306a36Sopenharmony_ci}
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_civoid __init kasan_late_init(void) { }
134