aboutsummaryrefslogtreecommitdiffstats
diff options
authorGreg Kroah-Hartman <gregkh@linuxfoundation.org>2025-05-07 15:57:26 +0200
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2025-05-07 15:57:26 +0200
commitbc19ceb00c67875cf7261c1c1f510740ffd16be5 (patch)
treede5c889fc5a91ff44aa6bd6a47afb18522767da3
parent4499baeb9fa326bdfa6aaafc011d96709e803683 (diff)
downloadstable-queue-bc19ceb00c67875cf7261c1c1f510740ffd16be5.tar.gz
6.6-stable patches
added patches: riscv-pass-patch_text-the-length-in-bytes.patch
-rw-r--r--queue-6.6/riscv-pass-patch_text-the-length-in-bytes.patch158
-rw-r--r--queue-6.6/series1
2 files changed, 159 insertions, 0 deletions
diff --git a/queue-6.6/riscv-pass-patch_text-the-length-in-bytes.patch b/queue-6.6/riscv-pass-patch_text-the-length-in-bytes.patch
new file mode 100644
index 00000000000..b7bd0d575ff
--- /dev/null
+++ b/queue-6.6/riscv-pass-patch_text-the-length-in-bytes.patch
@@ -0,0 +1,158 @@
+From 51781ce8f4486c3738a6c85175b599ad1be71f89 Mon Sep 17 00:00:00 2001
+From: Samuel Holland <samuel.holland@sifive.com>
+Date: Wed, 27 Mar 2024 09:04:44 -0700
+Subject: riscv: Pass patch_text() the length in bytes
+MIME-Version: 1.0
+Content-Type: text/plain; charset=UTF-8
+Content-Transfer-Encoding: 8bit
+
+From: Samuel Holland <samuel.holland@sifive.com>
+
+commit 51781ce8f4486c3738a6c85175b599ad1be71f89 upstream.
+
+patch_text_nosync() already handles an arbitrary length of code, so this
+removes a superfluous loop and reduces the number of icache flushes.
+
+Reviewed-by: Björn Töpel <bjorn@rivosinc.com>
+Signed-off-by: Samuel Holland <samuel.holland@sifive.com>
+Reviewed-by: Conor Dooley <conor.dooley@microchip.com>
+Link: https://lore.kernel.org/r/20240327160520.791322-6-samuel.holland@sifive.com
+Signed-off-by: Palmer Dabbelt <palmer@rivosinc.com>
+[apply to v6.6]
+Signed-off-by: Nam Cao <namcao@linutronix.de>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+---
+ arch/riscv/include/asm/patch.h | 2 +-
+ arch/riscv/kernel/patch.c | 14 +++++---------
+ arch/riscv/kernel/probes/kprobes.c | 18 ++++++++++--------
+ arch/riscv/net/bpf_jit_comp64.c | 7 ++++---
+ 4 files changed, 20 insertions(+), 21 deletions(-)
+
+--- a/arch/riscv/include/asm/patch.h
++++ b/arch/riscv/include/asm/patch.h
+@@ -9,7 +9,7 @@
+ int patch_insn_write(void *addr, const void *insn, size_t len);
+ int patch_text_nosync(void *addr, const void *insns, size_t len);
+ int patch_text_set_nosync(void *addr, u8 c, size_t len);
+-int patch_text(void *addr, u32 *insns, int ninsns);
++int patch_text(void *addr, u32 *insns, size_t len);
+
+ extern int riscv_patch_in_stop_machine;
+
+--- a/arch/riscv/kernel/patch.c
++++ b/arch/riscv/kernel/patch.c
+@@ -19,7 +19,7 @@
+ struct patch_insn {
+ void *addr;
+ u32 *insns;
+- int ninsns;
++ size_t len;
+ atomic_t cpu_count;
+ };
+
+@@ -234,14 +234,10 @@ NOKPROBE_SYMBOL(patch_text_nosync);
+ static int patch_text_cb(void *data)
+ {
+ struct patch_insn *patch = data;
+- unsigned long len;
+- int i, ret = 0;
++ int ret = 0;
+
+ if (atomic_inc_return(&patch->cpu_count) == num_online_cpus()) {
+- for (i = 0; ret == 0 && i < patch->ninsns; i++) {
+- len = GET_INSN_LENGTH(patch->insns[i]);
+- ret = patch_insn_write(patch->addr + i * len, &patch->insns[i], len);
+- }
++ ret = patch_insn_write(patch->addr, patch->insns, patch->len);
+ /*
+ * Make sure the patching store is effective *before* we
+ * increment the counter which releases all waiting CPUs
+@@ -262,13 +258,13 @@ static int patch_text_cb(void *data)
+ }
+ NOKPROBE_SYMBOL(patch_text_cb);
+
+-int patch_text(void *addr, u32 *insns, int ninsns)
++int patch_text(void *addr, u32 *insns, size_t len)
+ {
+ int ret;
+ struct patch_insn patch = {
+ .addr = addr,
+ .insns = insns,
+- .ninsns = ninsns,
++ .len = len,
+ .cpu_count = ATOMIC_INIT(0),
+ };
+
+--- a/arch/riscv/kernel/probes/kprobes.c
++++ b/arch/riscv/kernel/probes/kprobes.c
+@@ -23,13 +23,13 @@ post_kprobe_handler(struct kprobe *, str
+
+ static void __kprobes arch_prepare_ss_slot(struct kprobe *p)
+ {
++ size_t len = GET_INSN_LENGTH(p->opcode);
+ u32 insn = __BUG_INSN_32;
+- unsigned long offset = GET_INSN_LENGTH(p->opcode);
+
+- p->ainsn.api.restore = (unsigned long)p->addr + offset;
++ p->ainsn.api.restore = (unsigned long)p->addr + len;
+
+- patch_text_nosync(p->ainsn.api.insn, &p->opcode, 1);
+- patch_text_nosync((void *)p->ainsn.api.insn + offset, &insn, 1);
++ patch_text_nosync(p->ainsn.api.insn, &p->opcode, len);
++ patch_text_nosync((void *)p->ainsn.api.insn + len, &insn, GET_INSN_LENGTH(insn));
+ }
+
+ static void __kprobes arch_prepare_simulate(struct kprobe *p)
+@@ -116,16 +116,18 @@ void *alloc_insn_page(void)
+ /* install breakpoint in text */
+ void __kprobes arch_arm_kprobe(struct kprobe *p)
+ {
+- u32 insn = (p->opcode & __INSN_LENGTH_MASK) == __INSN_LENGTH_32 ?
+- __BUG_INSN_32 : __BUG_INSN_16;
++ size_t len = GET_INSN_LENGTH(p->opcode);
++ u32 insn = len == 4 ? __BUG_INSN_32 : __BUG_INSN_16;
+
+- patch_text(p->addr, &insn, 1);
++ patch_text(p->addr, &insn, len);
+ }
+
+ /* remove breakpoint from text */
+ void __kprobes arch_disarm_kprobe(struct kprobe *p)
+ {
+- patch_text(p->addr, &p->opcode, 1);
++ size_t len = GET_INSN_LENGTH(p->opcode);
++
++ patch_text(p->addr, &p->opcode, len);
+ }
+
+ void __kprobes arch_remove_kprobe(struct kprobe *p)
+--- a/arch/riscv/net/bpf_jit_comp64.c
++++ b/arch/riscv/net/bpf_jit_comp64.c
+@@ -14,6 +14,7 @@
+ #include "bpf_jit.h"
+
+ #define RV_FENTRY_NINSNS 2
++#define RV_FENTRY_NBYTES (RV_FENTRY_NINSNS * 4)
+
+ #define RV_REG_TCC RV_REG_A6
+ #define RV_REG_TCC_SAVED RV_REG_S6 /* Store A6 in S6 if program do calls */
+@@ -681,7 +682,7 @@ int bpf_arch_text_poke(void *ip, enum bp
+ if (ret)
+ return ret;
+
+- if (memcmp(ip, old_insns, RV_FENTRY_NINSNS * 4))
++ if (memcmp(ip, old_insns, RV_FENTRY_NBYTES))
+ return -EFAULT;
+
+ ret = gen_jump_or_nops(new_addr, ip, new_insns, is_call);
+@@ -690,8 +691,8 @@ int bpf_arch_text_poke(void *ip, enum bp
+
+ cpus_read_lock();
+ mutex_lock(&text_mutex);
+- if (memcmp(ip, new_insns, RV_FENTRY_NINSNS * 4))
+- ret = patch_text(ip, new_insns, RV_FENTRY_NINSNS);
++ if (memcmp(ip, new_insns, RV_FENTRY_NBYTES))
++ ret = patch_text(ip, new_insns, RV_FENTRY_NBYTES);
+ mutex_unlock(&text_mutex);
+ cpus_read_unlock();
+
diff --git a/queue-6.6/series b/queue-6.6/series
index 121d25fce4f..8e840f6b4f3 100644
--- a/queue-6.6/series
+++ b/queue-6.6/series
@@ -95,3 +95,4 @@ net-vertexcom-mse102x-add-range-check-for-cmd_rts.patch
net-vertexcom-mse102x-fix-rx-error-handling.patch
asoc-use-of_property_read_bool.patch
asoc-soc-core-stop-using-of_property_read_bool-for-non-boolean-properties.patch
+riscv-pass-patch_text-the-length-in-bytes.patch