Page MenuHomeFreeBSD

kinst: use per-probe trampolines in riscv
ClosedPublic

Authored by christos on Jul 10 2023, 8:37 PM.
Tags
None
Referenced Files
F108336342: D40963.id124639.diff
Thu, Jan 23, 11:37 PM
Unknown Object (File)
Fri, Jan 17, 3:21 PM
Unknown Object (File)
Mon, Jan 13, 6:11 PM
Unknown Object (File)
Thu, Jan 9, 5:50 AM
Unknown Object (File)
Thu, Jan 9, 5:26 AM
Unknown Object (File)
Sun, Jan 5, 4:06 AM
Unknown Object (File)
Dec 11 2024, 9:58 AM
Unknown Object (File)
Dec 2 2024, 1:13 PM
Subscribers

Diff Detail

Repository
rG FreeBSD src repository
Lint
Lint Not Applicable
Unit
Tests Not Applicable

Event Timeline

sys/cddl/dev/kinst/riscv/kinst_isa.c
17

What is a "per-CPU probe"?

295

This should be done using the DPCPU macros, same as intr_probe.

298–299

What if the cached value of sstatus is 0? Then the code doesn't distinguish between two different cases.

It would be better to have an explicit state machine. Try defining a per-CPU struct:

struct kinst_cpu_state {
    enum {
        KINST_PROBE_ARMED,
        KINST_PROBE_FIRED,
    } state;
    struct kinst_probe *kp;
    register_t sstatus;
};

When a probe is enabled, it enters state ARMED. When it fires, kinst_invop() calls dtrace_probe() and sets the state to FIRED. If single-stepping is done using a trampoline, then the second call to kinst_invop() will start in state FIRED, and there it will go back to ARMED. If single-stepping is done using emulation, then the state goes immediately from FIRED back to ARMED.

In particular, kinst_invop() should be simpler. Now that each probe has its own trampoline, and trampolines are executed with interrupts disabled, kinst_invop() doesn't need to check SPIE, and I think t_kinst_curprobe isn't needed.

538

Where does this trampoline get freed?

sys/cddl/dev/kinst/riscv/kinst_isa.c
17

It is the per-CPU equivalent of t_kinst_curprobe. I noticed that the kernel was crashing if I tried to fetch
t_kinst_curprobe when the thread had been executing with interrupts disabled, so I followed the per-CPU trampoline
scheme, which presumably to fix this issue.

538

In kinst_unload() when it calls kinst_trampoline_deinit().

sys/cddl/dev/kinst/riscv/kinst_isa.c
298–299

Since amd64 is going to eventually also use the double-breakpoint mechanism, and arm64 will need this as well, do you think we should add the struct definition to kinst.h and replace register_t sstatus with a generic register_t field?

sys/cddl/dev/kinst/riscv/kinst_isa.c
17

But what exactly is the issue? If the trampoline is executed with interrupts disabled, there should be no way to re-enter kinst_invop() while executing a trampoline.

298–299

I don't have a strong opinion on whether the implementation should be per-platform or generic. It's not a lot of code either way. I just find the current implementation hard to follow and believe it should be simpler.

538

It should be freed when the probe is destroyed, if possible. Otherwise the memory will remain allocated even when the trampoline is unused.

sys/cddl/dev/kinst/riscv/kinst_isa.c
17

Yeah, that's a left over I forgot to get rid of.

538

Then the trampoline allocations and deallocations can go to kinst_enable() and kinst_disable() respectively.

christos marked 7 inline comments as done.

Improve state chaching. Depends on D40962.

This is looking good, just minor comments.

sys/cddl/dev/kinst/riscv/kinst_isa.c
295–297

We can and should declare kp const. In particular, multiple CPUs might be sharing a reference to a single probe; we should make sure that kinst_invop() doesn't mutate any state in the probe itself.

329

I don't see a point in updating the state only to update it again a few lines below. This can just go after the emulate check.

331–332

We don't need to handle this case anymore, now that trampoline allocation is handled at the time that probes are enabled.

538

You need to check kp_tramp == NULL and return ENOMEM if it happens.

560–561

I think you can DPCPU_DEFINE a struct, there's no need for dynamic allocation. Then use DPCPU_PTR() to get a pointer to that struct.

sys/cddl/dev/kinst/riscv/kinst_isa.c
331–332

Sorry, I don't understand - why do we need to populate the trampoline upon every call to kinst_invop()? It needs to be done only once, at the time the probe is created.

christos marked 5 inline comments as done.

Address comments.

I'm happy with this change with the one comment addressed.

sys/cddl/dev/kinst/riscv/kinst_isa.c
560–561

This comment still applies. My point is that you're introducing an unnecessary layer of indirection here. You should be able to write

DPCPU_DEFINE_STATIC(struct kinst_cpu_state, kinst_state);

then, any time you want to access the state, write ks = DPCPU_PTR(kinst_state);.

Implementing it this way requires less code and is more efficient.

This revision is now accepted and ready to land.Jul 19 2023, 1:51 PM
This revision was automatically updated to reflect the committed changes.
christos marked 2 inline comments as done.