]> www.infradead.org Git - nvme.git/commitdiff
riscv: misaligned: Add handling for ZCB instructions
authorNylon Chen <nylon.chen@sifive.com>
Fri, 11 Apr 2025 07:38:49 +0000 (15:38 +0800)
committerAlexandre Ghiti <alexghiti@rivosinc.com>
Mon, 5 May 2025 13:09:38 +0000 (13:09 +0000)
Add support for the Zcb extension's compressed half-word instructions
(C.LHU, C.LH, and C.SH) in the RISC-V misaligned access trap handler.

Signed-off-by: Zong Li <zong.li@sifive.com>
Signed-off-by: Nylon Chen <nylon.chen@sifive.com>
Fixes: 956d705dd279 ("riscv: Unaligned load/store handling for M_MODE")
Reviewed-by: Alexandre Ghiti <alexghiti@rivosinc.com>
Link: https://lore.kernel.org/r/20250411073850.3699180-2-nylon.chen@sifive.com
Signed-off-by: Alexandre Ghiti <alexghiti@rivosinc.com>
arch/riscv/kernel/traps_misaligned.c

index 4354c87c0376fd635017735645573fbb7f565fcf..dde5d11dc1b50dd2a308a6745e9f70996f4285d5 100644 (file)
 #define INSN_MATCH_C_FSWSP             0xe002
 #define INSN_MASK_C_FSWSP              0xe003
 
+#define INSN_MATCH_C_LHU               0x8400
+#define INSN_MASK_C_LHU                        0xfc43
+#define INSN_MATCH_C_LH                        0x8440
+#define INSN_MASK_C_LH                 0xfc43
+#define INSN_MATCH_C_SH                        0x8c00
+#define INSN_MASK_C_SH                 0xfc43
+
 #define INSN_LEN(insn)                 ((((insn) & 0x3) < 0x3) ? 2 : 4)
 
 #if defined(CONFIG_64BIT)
@@ -431,6 +438,13 @@ static int handle_scalar_misaligned_load(struct pt_regs *regs)
                fp = 1;
                len = 4;
 #endif
+       } else if ((insn & INSN_MASK_C_LHU) == INSN_MATCH_C_LHU) {
+               len = 2;
+               insn = RVC_RS2S(insn) << SH_RD;
+       } else if ((insn & INSN_MASK_C_LH) == INSN_MATCH_C_LH) {
+               len = 2;
+               shift = 8 * (sizeof(ulong) - len);
+               insn = RVC_RS2S(insn) << SH_RD;
        } else {
                regs->epc = epc;
                return -1;
@@ -530,6 +544,9 @@ static int handle_scalar_misaligned_store(struct pt_regs *regs)
                len = 4;
                val.data_ulong = GET_F32_RS2C(insn, regs);
 #endif
+       } else if ((insn & INSN_MASK_C_SH) == INSN_MATCH_C_SH) {
+               len = 2;
+               val.data_ulong = GET_RS2S(insn, regs);
        } else {
                regs->epc = epc;
                return -1;