#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/malloc.h>
#include <sys/msan.h>
#include <sys/proc.h>
#include <sys/smp.h>
#include <sys/dtrace_impl.h>
#include <sys/dtrace_bsd.h>
#include <cddl/dev/dtrace/dtrace_cddl.h>
#include <machine/clock.h>
#include <machine/cpufunc.h>
#include <machine/frame.h>
#include <machine/md_var.h>
#include <machine/psl.h>
#include <machine/trap.h>
#include <vm/pmap.h>
extern void dtrace_getnanotime(struct timespec *tsp);
extern int (*dtrace_invop_jump_addr)(struct trapframe *);
int dtrace_invop(uintptr_t, struct trapframe *, void **);
int dtrace_invop_start(struct trapframe *frame);
void dtrace_invop_init(void);
void dtrace_invop_uninit(void);
typedef struct dtrace_invop_hdlr {
int (*dtih_func)(uintptr_t, struct trapframe *, uintptr_t);
struct dtrace_invop_hdlr *dtih_next;
} dtrace_invop_hdlr_t;
dtrace_invop_hdlr_t *dtrace_invop_hdlr;
int
dtrace_invop(uintptr_t addr, struct trapframe *frame, void **scratch)
{
struct thread *td;
dtrace_invop_hdlr_t *hdlr;
int rval;
kmsan_mark(frame, sizeof(*frame), KMSAN_STATE_INITED);
td = curthread;
td->t_dtrace_trapframe = frame;
rval = 0;
for (hdlr = dtrace_invop_hdlr; hdlr != NULL; hdlr = hdlr->dtih_next) {
rval = hdlr->dtih_func(addr, frame, (uintptr_t)scratch);
if (rval != 0)
break;
}
td->t_dtrace_trapframe = NULL;
return (rval);
}
void
dtrace_invop_add(int (*func)(uintptr_t, struct trapframe *, uintptr_t))
{
dtrace_invop_hdlr_t *hdlr;
hdlr = kmem_alloc(sizeof (dtrace_invop_hdlr_t), KM_SLEEP);
hdlr->dtih_func = func;
hdlr->dtih_next = dtrace_invop_hdlr;
dtrace_invop_hdlr = hdlr;
}
void
dtrace_invop_remove(int (*func)(uintptr_t, struct trapframe *, uintptr_t))
{
dtrace_invop_hdlr_t *hdlr = dtrace_invop_hdlr, *prev = NULL;
for (;;) {
if (hdlr == NULL)
panic("attempt to remove non-existent invop handler");
if (hdlr->dtih_func == func)
break;
prev = hdlr;
hdlr = hdlr->dtih_next;
}
if (prev == NULL) {
ASSERT(dtrace_invop_hdlr == hdlr);
dtrace_invop_hdlr = hdlr->dtih_next;
} else {
ASSERT(dtrace_invop_hdlr != hdlr);
prev->dtih_next = hdlr->dtih_next;
}
kmem_free(hdlr, 0);
}
void
dtrace_invop_init(void)
{
dtrace_invop_jump_addr = dtrace_invop_start;
}
void
dtrace_invop_uninit(void)
{
dtrace_invop_jump_addr = NULL;
}
void
dtrace_toxic_ranges(void (*func)(uintptr_t base, uintptr_t limit))
{
(*func)(0, la57 ? (uintptr_t)addr_P5Tmap : (uintptr_t)addr_P4Tmap);
}
#ifdef notyet
void
dtrace_safe_synchronous_signal(void)
{
kthread_t *t = curthread;
struct regs *rp = lwptoregs(ttolwp(t));
size_t isz = t->t_dtrace_npc - t->t_dtrace_pc;
ASSERT(t->t_dtrace_on);
if (rp->r_pc < t->t_dtrace_scrpc ||
rp->r_pc > t->t_dtrace_astpc + isz) {
t->t_dtrace_ft = 0;
} else if (rp->r_pc == t->t_dtrace_scrpc ||
rp->r_pc == t->t_dtrace_astpc) {
rp->r_pc = t->t_dtrace_pc;
t->t_dtrace_ft = 0;
}
}
int
dtrace_safe_defer_signal(void)
{
kthread_t *t = curthread;
struct regs *rp = lwptoregs(ttolwp(t));
size_t isz = t->t_dtrace_npc - t->t_dtrace_pc;
ASSERT(t->t_dtrace_on);
if (rp->r_pc < t->t_dtrace_scrpc ||
rp->r_pc > t->t_dtrace_astpc + isz) {
t->t_dtrace_ft = 0;
return (0);
}
if (rp->r_pc >= t->t_dtrace_scrpc + isz &&
rp->r_pc < t->t_dtrace_astpc) {
#ifdef __amd64
if (t->t_dtrace_reg != 0 &&
rp->r_pc == t->t_dtrace_scrpc + isz) {
switch (t->t_dtrace_reg) {
case REG_RAX:
rp->r_rax = t->t_dtrace_regv;
break;
case REG_RCX:
rp->r_rcx = t->t_dtrace_regv;
break;
case REG_R8:
rp->r_r8 = t->t_dtrace_regv;
break;
case REG_R9:
rp->r_r9 = t->t_dtrace_regv;
break;
}
}
#endif
rp->r_pc = t->t_dtrace_npc;
t->t_dtrace_ft = 0;
return (0);
}
if (!t->t_dtrace_step) {
ASSERT(rp->r_pc < t->t_dtrace_astpc);
rp->r_pc += t->t_dtrace_astpc - t->t_dtrace_scrpc;
t->t_dtrace_step = 1;
}
t->t_dtrace_ast = 1;
return (1);
}
#endif
static int64_t tgt_cpu_tsc;
static int64_t hst_cpu_tsc;
static int64_t tsc_skew[MAXCPU];
static uint64_t nsec_scale;
#define SCALE_SHIFT 28
static void
dtrace_gethrtime_init_cpu(void *arg)
{
uintptr_t cpu = (uintptr_t) arg;
if (cpu == curcpu)
tgt_cpu_tsc = rdtsc();
else
hst_cpu_tsc = rdtsc();
}
static void
dtrace_gethrtime_init(void *arg)
{
struct pcpu *pc;
uint64_t tsc_f;
cpuset_t map;
int i;
tsc_f = atomic_load_acq_64(&tsc_freq);
KASSERT(tsc_f > (NANOSEC >> (32 - SCALE_SHIFT)),
("TSC frequency is too low"));
nsec_scale = ((uint64_t)NANOSEC << SCALE_SHIFT) / tsc_f;
if (vm_guest != VM_GUEST_NO)
return;
sched_pin();
tsc_skew[curcpu] = 0;
CPU_FOREACH(i) {
if (i == curcpu)
continue;
pc = pcpu_find(i);
CPU_SETOF(PCPU_GET(cpuid), &map);
CPU_SET(pc->pc_cpuid, &map);
smp_rendezvous_cpus(map, NULL,
dtrace_gethrtime_init_cpu,
smp_no_rendezvous_barrier, (void *)(uintptr_t) i);
tsc_skew[i] = tgt_cpu_tsc - hst_cpu_tsc;
}
sched_unpin();
}
SYSINIT(dtrace_gethrtime_init, SI_SUB_DTRACE, SI_ORDER_ANY,
dtrace_gethrtime_init, NULL);
uint64_t
dtrace_gethrtime(void)
{
uint64_t tsc;
uint32_t lo, hi;
register_t rflags;
rflags = intr_disable();
tsc = rdtsc() - tsc_skew[curcpu];
intr_restore(rflags);
lo = tsc;
hi = tsc >> 32;
return (((lo * nsec_scale) >> SCALE_SHIFT) +
((hi * nsec_scale) << (32 - SCALE_SHIFT)));
}
uint64_t
dtrace_gethrestime(void)
{
struct timespec current_time;
dtrace_getnanotime(¤t_time);
return (current_time.tv_sec * 1000000000ULL + current_time.tv_nsec);
}
int
dtrace_trap(struct trapframe *frame, u_int type)
{
uint16_t nofault;
sched_pin();
nofault = cpu_core[curcpu].cpuc_dtrace_flags & CPU_DTRACE_NOFAULT;
sched_unpin();
if (nofault) {
KASSERT((read_rflags() & PSL_I) == 0, ("interrupts enabled"));
switch (type) {
case T_PROTFLT:
cpu_core[curcpu].cpuc_dtrace_flags |= CPU_DTRACE_ILLOP;
frame->tf_rip += dtrace_instr_size((uint8_t *) frame->tf_rip);
return (1);
case T_PAGEFLT:
cpu_core[curcpu].cpuc_dtrace_flags |= CPU_DTRACE_BADADDR;
cpu_core[curcpu].cpuc_dtrace_illval = frame->tf_addr;
frame->tf_rip += dtrace_instr_size((uint8_t *) frame->tf_rip);
return (1);
default:
break;
}
}
return (0);
}