Re: [PATCH 7/9] riscv: bpf: Avoid breaking W^X
From: Jisheng Zhang
Date: Sun Jun 13 2021 - 13:12:48 EST
Hi,
On Fri, 11 Jun 2021 18:41:16 +0200
Andreas Schwab <schwab@xxxxxxxxxxxxxx> wrote:
> On Jun 12 2021, Jisheng Zhang wrote:
>
> > I reproduced an kernel panic with the defconfig on qemu, but I'm not sure whether
> > this is the issue you saw, I will check.
> >
> > 0.161959] futex hash table entries: 512 (order: 3, 32768 bytes, linear)
> > [ 0.167028] pinctrl core: initialized pinctrl subsystem
> > [ 0.190727] Unable to handle kernel paging request at virtual address ffffffff81651bd8
> > [ 0.191361] Oops [#1]
> > [ 0.191509] Modules linked in:
> > [ 0.191814] CPU: 0 PID: 1 Comm: swapper/0 Not tainted 5.13.0-rc5-default+ #3
> > [ 0.192179] Hardware name: riscv-virtio,qemu (DT)
> > [ 0.192492] epc : __memset+0xc4/0xfc
> > [ 0.192712] ra : skb_flow_dissector_init+0x22/0x86
>
> Yes, that's the same.
>
> Andreas.
>
I think I found the root cause: commit 2bfc6cd81bd ("move kernel mapping
outside of linear mapping") moves BPF JIT region after the kernel:
#define BPF_JIT_REGION_START PFN_ALIGN((unsigned long)&_end)
The &_end is unlikely aligned with PMD SIZE, so the front bpf jit region
sits with kernel .data section in one PMD. But kenrel is mapped in PMD SIZE,
so when bpf_jit_binary_lock_ro() is called to make the first bpf jit prog
ROX, we will make part of kernel .data section RO too, so when we write, for example
memset the .data section, MMU will trigger store page fault.
To fix the issue, we need to make the bpf jit region PMD size aligned by either
patch BPF_JIT_REGION_START to align on PMD size rather than PAGE SIZE, or
something as below patch to move the BPF region before modules region:
diff --git a/arch/riscv/include/asm/pgtable.h b/arch/riscv/include/asm/pgtable.h
index 9469f464e71a..997b894edbc2 100644
--- a/arch/riscv/include/asm/pgtable.h
+++ b/arch/riscv/include/asm/pgtable.h
@@ -31,8 +31,8 @@
#define BPF_JIT_REGION_SIZE (SZ_128M)
#ifdef CONFIG_64BIT
/* KASLR should leave at least 128MB for BPF after the kernel */
-#define BPF_JIT_REGION_START PFN_ALIGN((unsigned long)&_end)
-#define BPF_JIT_REGION_END (BPF_JIT_REGION_START + BPF_JIT_REGION_SIZE)
+#define BPF_JIT_REGION_START (BPF_JIT_REGION_END - BPF_JIT_REGION_SIZE)
+#define BPF_JIT_REGION_END (MODULES_VADDR)
#else
#define BPF_JIT_REGION_START (PAGE_OFFSET - BPF_JIT_REGION_SIZE)
#define BPF_JIT_REGION_END (VMALLOC_END)
@@ -40,8 +40,8 @@
/* Modules always live before the kernel */
#ifdef CONFIG_64BIT
-#define MODULES_VADDR (PFN_ALIGN((unsigned long)&_end) - SZ_2G)
#define MODULES_END (PFN_ALIGN((unsigned long)&_start))
+#define MODULES_VADDR (MODULES_END - SZ_128M)
#endif
can you please try it? Per my test, the issue is fixed.
Thanks