git: 0e69c959150c - main - dtrace: Fix up %rip for invop probes on x86
Date: Mon, 31 Oct 2022 23:12:17 UTC
The branch main has been updated by markj: URL: https://cgit.FreeBSD.org/src/commit/?id=0e69c959150c0ba38459e9121158016ee34b0d92 commit 0e69c959150c0ba38459e9121158016ee34b0d92 Author: Mark Johnston <markj@FreeBSD.org> AuthorDate: 2022-10-31 23:11:36 +0000 Commit: Mark Johnston <markj@FreeBSD.org> CommitDate: 2022-10-31 23:11:36 +0000 dtrace: Fix up %rip for invop probes on x86 When a breakpoint exception is raised, the saved value of %rip points to the instruction following the breakpoint. However, when fetching the value of %rip using regs[], it's more natural to provide the address of the breakpoint itself, so modify the kinst and fbt providers accordingly. Reported by: khng Reviewed by: christos, khng MFC after: 2 months Differential Revision: https://reviews.freebsd.org/D37218 --- sys/cddl/dev/fbt/x86/fbt_isa.c | 8 ++++++++ sys/cddl/dev/kinst/amd64/kinst_isa.c | 8 +++++++- 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/sys/cddl/dev/fbt/x86/fbt_isa.c b/sys/cddl/dev/fbt/x86/fbt_isa.c index 05ec87ab437f..b05ae4cb2c44 100644 --- a/sys/cddl/dev/fbt/x86/fbt_isa.c +++ b/sys/cddl/dev/fbt/x86/fbt_isa.c @@ -84,6 +84,12 @@ fbt_invop(uintptr_t addr, struct trapframe *frame, uintptr_t scratch __unused) if ((uintptr_t)fbt->fbtp_patchpoint != addr) continue; fbtrval = fbt->fbtp_rval; + + /* + * Report the address of the breakpoint for the benefit + * of consumers fetching register values with regs[]. + */ + frame->tf_rip--; for (; fbt != NULL; fbt = fbt->fbtp_tracenext) { ASSERT(fbt->fbtp_rval == fbtrval); if (fbt->fbtp_roffset == 0) { @@ -143,6 +149,8 @@ fbt_invop(uintptr_t addr, struct trapframe *frame, uintptr_t scratch __unused) cpu->cpu_dtrace_caller = 0; } } + /* Advance to the instruction following the breakpoint. */ + frame->tf_rip++; return (fbtrval); } diff --git a/sys/cddl/dev/kinst/amd64/kinst_isa.c b/sys/cddl/dev/kinst/amd64/kinst_isa.c index 6d8d5d521617..e47cfbbf4d4e 100644 --- a/sys/cddl/dev/kinst/amd64/kinst_isa.c +++ b/sys/cddl/dev/kinst/amd64/kinst_isa.c @@ -139,6 +139,12 @@ kinst_invop(uintptr_t addr, struct trapframe *frame, uintptr_t scratch) if (kp == NULL) return (0); + /* + * Report the address of the breakpoint for the benefit of consumers + * fetching register values with regs[]. + */ + frame->tf_rip--; + DTRACE_CPUFLAG_SET(CPU_DTRACE_NOFAULT); cpu->cpu_dtrace_caller = stack[0]; DTRACE_CPUFLAG_CLEAR(CPU_DTRACE_NOFAULT | CPU_DTRACE_BADADDR); @@ -162,7 +168,7 @@ kinst_invop(uintptr_t addr, struct trapframe *frame, uintptr_t scratch) if (kpmd->reg1 == -1 && kpmd->reg2 == -1) { /* rip-relative */ - rval = frame->tf_rip - 1 + kpmd->instlen; + rval = frame->tf_rip + kpmd->instlen; } else { /* indirect */ rval = kinst_regval(frame, kpmd->reg1) +