mirror of
https://github.com/torvalds/linux.git
synced 2026-05-24 23:22:31 +02:00
Merge patch series "riscv: kprobes: simulate some instructions"
Nam Cao <namcaov@gmail.com> says:
Simulate some currently rejected instructions. Still to be simulated are:
- c.jal
- c.ebreak
* b4-shazam-merge:
riscv: kprobes: simulate c.beqz and c.bnez
riscv: kprobes: simulate c.jr and c.jalr instructions
riscv: kprobes: simulate c.j instruction
Link: https://lore.kernel.org/r/cover.1690704360.git.namcaov@gmail.com
Signed-off-by: Palmer Dabbelt <palmer@rivosinc.com>
This commit is contained in:
commit
82dfb5fde6
|
|
@ -29,13 +29,14 @@ riscv_probe_decode_insn(probe_opcode_t *addr, struct arch_probe_insn *api)
|
|||
* TODO: the REJECTED ones below need to be implemented
|
||||
*/
|
||||
#ifdef CONFIG_RISCV_ISA_C
|
||||
RISCV_INSN_REJECTED(c_j, insn);
|
||||
RISCV_INSN_REJECTED(c_jr, insn);
|
||||
RISCV_INSN_REJECTED(c_jal, insn);
|
||||
RISCV_INSN_REJECTED(c_jalr, insn);
|
||||
RISCV_INSN_REJECTED(c_beqz, insn);
|
||||
RISCV_INSN_REJECTED(c_bnez, insn);
|
||||
RISCV_INSN_REJECTED(c_ebreak, insn);
|
||||
|
||||
RISCV_INSN_SET_SIMULATE(c_j, insn);
|
||||
RISCV_INSN_SET_SIMULATE(c_jr, insn);
|
||||
RISCV_INSN_SET_SIMULATE(c_jalr, insn);
|
||||
RISCV_INSN_SET_SIMULATE(c_beqz, insn);
|
||||
RISCV_INSN_SET_SIMULATE(c_bnez, insn);
|
||||
#endif
|
||||
|
||||
RISCV_INSN_SET_SIMULATE(jal, insn);
|
||||
|
|
|
|||
|
|
@ -188,3 +188,108 @@ bool __kprobes simulate_branch(u32 opcode, unsigned long addr, struct pt_regs *r
|
|||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool __kprobes simulate_c_j(u32 opcode, unsigned long addr, struct pt_regs *regs)
|
||||
{
|
||||
/*
|
||||
* 15 13 12 2 1 0
|
||||
* | funct3 | offset[11|4|9:8|10|6|7|3:1|5] | opcode |
|
||||
* 3 11 2
|
||||
*/
|
||||
|
||||
s32 offset;
|
||||
|
||||
offset = ((opcode >> 3) & 0x7) << 1;
|
||||
offset |= ((opcode >> 11) & 0x1) << 4;
|
||||
offset |= ((opcode >> 2) & 0x1) << 5;
|
||||
offset |= ((opcode >> 7) & 0x1) << 6;
|
||||
offset |= ((opcode >> 6) & 0x1) << 7;
|
||||
offset |= ((opcode >> 9) & 0x3) << 8;
|
||||
offset |= ((opcode >> 8) & 0x1) << 10;
|
||||
offset |= ((opcode >> 12) & 0x1) << 11;
|
||||
|
||||
instruction_pointer_set(regs, addr + sign_extend32(offset, 11));
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool __kprobes simulate_c_jr_jalr(u32 opcode, unsigned long addr, struct pt_regs *regs,
|
||||
bool is_jalr)
|
||||
{
|
||||
/*
|
||||
* 15 12 11 7 6 2 1 0
|
||||
* | funct4 | rs1 | rs2 | op |
|
||||
* 4 5 5 2
|
||||
*/
|
||||
|
||||
unsigned long jump_addr;
|
||||
|
||||
u32 rs1 = (opcode >> 7) & 0x1f;
|
||||
|
||||
if (rs1 == 0) /* C.JR is only valid when rs1 != x0 */
|
||||
return false;
|
||||
|
||||
if (!rv_insn_reg_get_val(regs, rs1, &jump_addr))
|
||||
return false;
|
||||
|
||||
if (is_jalr && !rv_insn_reg_set_val(regs, 1, addr + 2))
|
||||
return false;
|
||||
|
||||
instruction_pointer_set(regs, jump_addr);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool __kprobes simulate_c_jr(u32 opcode, unsigned long addr, struct pt_regs *regs)
|
||||
{
|
||||
return simulate_c_jr_jalr(opcode, addr, regs, false);
|
||||
}
|
||||
|
||||
bool __kprobes simulate_c_jalr(u32 opcode, unsigned long addr, struct pt_regs *regs)
|
||||
{
|
||||
return simulate_c_jr_jalr(opcode, addr, regs, true);
|
||||
}
|
||||
|
||||
static bool __kprobes simulate_c_bnez_beqz(u32 opcode, unsigned long addr, struct pt_regs *regs,
|
||||
bool is_bnez)
|
||||
{
|
||||
/*
|
||||
* 15 13 12 10 9 7 6 2 1 0
|
||||
* | funct3 | offset[8|4:3] | rs1' | offset[7:6|2:1|5] | op |
|
||||
* 3 3 3 5 2
|
||||
*/
|
||||
|
||||
s32 offset;
|
||||
u32 rs1;
|
||||
unsigned long rs1_val;
|
||||
|
||||
rs1 = 0x8 | ((opcode >> 7) & 0x7);
|
||||
|
||||
if (!rv_insn_reg_get_val(regs, rs1, &rs1_val))
|
||||
return false;
|
||||
|
||||
if ((rs1_val != 0 && is_bnez) || (rs1_val == 0 && !is_bnez)) {
|
||||
offset = ((opcode >> 3) & 0x3) << 1;
|
||||
offset |= ((opcode >> 10) & 0x3) << 3;
|
||||
offset |= ((opcode >> 2) & 0x1) << 5;
|
||||
offset |= ((opcode >> 5) & 0x3) << 6;
|
||||
offset |= ((opcode >> 12) & 0x1) << 8;
|
||||
offset = sign_extend32(offset, 8);
|
||||
} else {
|
||||
offset = 2;
|
||||
}
|
||||
|
||||
instruction_pointer_set(regs, addr + offset);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool __kprobes simulate_c_bnez(u32 opcode, unsigned long addr, struct pt_regs *regs)
|
||||
{
|
||||
return simulate_c_bnez_beqz(opcode, addr, regs, true);
|
||||
}
|
||||
|
||||
bool __kprobes simulate_c_beqz(u32 opcode, unsigned long addr, struct pt_regs *regs)
|
||||
{
|
||||
return simulate_c_bnez_beqz(opcode, addr, regs, false);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -24,5 +24,10 @@ bool simulate_auipc(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
|||
bool simulate_branch(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_jal(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_jalr(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_c_j(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_c_jr(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_c_jalr(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_c_bnez(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
bool simulate_c_beqz(u32 opcode, unsigned long addr, struct pt_regs *regs);
|
||||
|
||||
#endif /* _RISCV_KERNEL_PROBES_SIMULATE_INSN_H */
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user