diff options
Diffstat (limited to 'trace/control-target.c')
-rw-r--r-- | trace/control-target.c | 81 |
1 files changed, 73 insertions, 8 deletions
diff --git a/trace/control-target.c b/trace/control-target.c index 74c029accc..7ebf6e0bcb 100644 --- a/trace/control-target.c +++ b/trace/control-target.c @@ -9,10 +9,31 @@ #include "qemu/osdep.h" #include "cpu.h" +#include "trace.h" #include "trace/control.h" #include "translate-all.h" +void trace_event_set_state_dynamic_init(TraceEvent *ev, bool state) +{ + bool state_pre; + assert(trace_event_get_state_static(ev)); + /* + * We ignore the "vcpu" property here, since no vCPUs have been created + * yet. Then dstate can only be 1 or 0. + */ + state_pre = *ev->dstate; + if (state_pre != state) { + if (state) { + trace_events_enabled_count++; + *ev->dstate = 1; + } else { + trace_events_enabled_count--; + *ev->dstate = 0; + } + } +} + void trace_event_set_state_dynamic(TraceEvent *ev, bool state) { CPUState *vcpu; @@ -22,32 +43,76 @@ void trace_event_set_state_dynamic(TraceEvent *ev, bool state) trace_event_set_vcpu_state_dynamic(vcpu, ev, state); } } else { - TraceEventID id = trace_event_get_id(ev); - trace_events_enabled_count += state - trace_events_dstate[id]; - trace_events_dstate[id] = state; + /* Without the "vcpu" property, dstate can only be 1 or 0 */ + bool state_pre = *ev->dstate; + if (state_pre != state) { + if (state) { + trace_events_enabled_count++; + *ev->dstate = 1; + } else { + trace_events_enabled_count--; + *ev->dstate = 0; + } + } } } void trace_event_set_vcpu_state_dynamic(CPUState *vcpu, TraceEvent *ev, bool state) { - TraceEventID id; - TraceEventVCPUID vcpu_id; + uint32_t vcpu_id; bool state_pre; assert(trace_event_get_state_static(ev)); assert(trace_event_is_vcpu(ev)); - id = trace_event_get_id(ev); vcpu_id = trace_event_get_vcpu_id(ev); state_pre = test_bit(vcpu_id, vcpu->trace_dstate); if (state_pre != state) { if (state) { trace_events_enabled_count++; set_bit(vcpu_id, vcpu->trace_dstate); - trace_events_dstate[id]++; + (*ev->dstate)++; } else { trace_events_enabled_count--; clear_bit(vcpu_id, vcpu->trace_dstate); - trace_events_dstate[id]--; + (*ev->dstate)--; + } + } +} + +static bool adding_first_cpu(void) +{ + CPUState *cpu; + size_t count = 0; + CPU_FOREACH(cpu) { + count++; + if (count > 1) { + return false; + } + } + return true; +} + +void trace_init_vcpu(CPUState *vcpu) +{ + TraceEventIter iter; + TraceEvent *ev; + trace_event_iter_init(&iter, NULL); + while ((ev = trace_event_iter_next(&iter)) != NULL) { + if (trace_event_is_vcpu(ev) && + trace_event_get_state_static(ev) && + trace_event_get_state_dynamic(ev)) { + if (adding_first_cpu()) { + /* check preconditions */ + assert(*ev->dstate == 1); + /* disable early-init state ... */ + *ev->dstate = 0; + trace_events_enabled_count--; + /* ... and properly re-enable */ + trace_event_set_vcpu_state_dynamic(vcpu, ev, true); + } else { + trace_event_set_vcpu_state_dynamic(vcpu, ev, true); + } } } + trace_guest_cpu_enter(vcpu); } |