Re: [PATCH] perf_events: improve x86 event scheduling
From: Stephane Eranian
Date: Wed Jan 13 2010 - 11:52:42 EST
On Wed, Jan 13, 2010 at 5:29 PM, Peter Zijlstra <peterz@xxxxxxxxxxxxx> wrote:
> On Wed, 2010-01-13 at 10:54 +0100, Stephane Eranian wrote:
>
>> > One concern I do have is the loss of error checking on
>> > event_sched_in()'s event->pmu->enable(), that could be another
>> > 'hardware' PMU like breakpoints, in which case it could fail.
>> >
>> Well, x86_pmu_enable() does return an error code, so it is up
>> to the upper layer to handle the error gracefully and in particular
>> in perf_ctx_adjust_freq().
>
>> +static void event_sched_in(struct perf_event *event, int cpu)
>> +{
>> + Â Â Â event->state = PERF_EVENT_STATE_ACTIVE;
>> + Â Â Â event->oncpu = cpu;
>> + Â Â Â event->tstamp_running += event->ctx->time - event->tstamp_stopped;
>> + Â Â Â if (is_software_event(event))
>> + Â Â Â Â Â Â Â event->pmu->enable(event);
>> +}
>> +
>> +/*
>> + * Called to enable a whole group of events.
>> + * Returns 1 if the group was enabled, or -EAGAIN if it could not be.
>> + * Assumes the caller has disabled interrupts and has
>> + * frozen the PMU with hw_perf_save_disable.
>> + *
>> + * called with PMU disabled. If successful and return value 1,
>> + * then guaranteed to call perf_enable() and hw_perf_enable()
>> + */
>> +int hw_perf_group_sched_in(struct perf_event *leader,
>> + Â Â Â Â Â Â Âstruct perf_cpu_context *cpuctx,
>> + Â Â Â Â Â Â Âstruct perf_event_context *ctx, int cpu)
>> +{
>> + Â Â Â struct cpu_hw_events *cpuc = &per_cpu(cpu_hw_events, cpu);
>> + Â Â Â struct perf_event *sub;
>> + Â Â Â int assign[X86_PMC_IDX_MAX];
>> + Â Â Â int n, n0, ret;
>> +
>> + Â Â Â n0 = cpuc->n_events;
>> +
>> + Â Â Â n = collect_events(cpuc, leader, true);
>> + Â Â Â if (n < 0)
>> + Â Â Â Â Â Â Â return n;
>> +
>> + Â Â Â ret = x86_schedule_events(cpuc, n, assign);
>> + Â Â Â if (ret)
>> + Â Â Â Â Â Â Â return ret;
>> +
>> + Â Â Â /*
>> + Â Â Â Â* copy new assignment, now we know it is possible
>> + Â Â Â Â* will be used by hw_perf_enable()
>> + Â Â Â Â*/
>> + Â Â Â memcpy(cpuc->assign, assign, n*sizeof(int));
>> +
>> + Â Â Â cpuc->n_events = n;
>> + Â Â Â cpuc->n_added Â= n - n0;
>> +
>> + Â Â Â n = 1;
>> + Â Â Â event_sched_in(leader, cpu);
>> + Â Â Â list_for_each_entry(sub, &leader->sibling_list, group_entry) {
>> + Â Â Â Â Â Â Â if (sub->state != PERF_EVENT_STATE_OFF) {
>> + Â Â Â Â Â Â Â Â Â Â Â event_sched_in(sub, cpu);
>> + Â Â Â Â Â Â Â Â Â Â Â ++n;
>> + Â Â Â Â Â Â Â }
>> + Â Â Â }
>> + Â Â Â ctx->nr_active += n;
>> +
>> + Â Â Â /*
>> + Â Â Â Â* 1 means successful and events are active
>> + Â Â Â Â* This is not quite true because we defer
>> + Â Â Â Â* actual activation until hw_perf_enable() but
>> + Â Â Â Â* this way we* ensure caller won't try to enable
>> + Â Â Â Â* individual events
>> + Â Â Â Â*/
>> + Â Â Â return 1;
>> +}
>
> That most certainly looses error codes for all !is_x86_event() events in
> the group.
>
> So you either need to deal with that event_sched_in() failing, or
> guarantee that it always succeeds (forcing software events only for
> example).
>
True, but that one can be fixed because it is only called from
hw_perf_group_sched_in() which returns an error code.
The same code would have to be fixed on PPC as well.
>
--
Stephane Eranian | EMEA Software Engineering
Google France | 38 avenue de l'OpÃra | 75002 Paris
Tel : +33 (0) 1 42 68 53 00
This email may be confidential or privileged. If you received this
communication by mistake, please
don't forward it to anyone else, please erase all copies and
attachments, and please let me know that
it went to the wrong person. Thanks
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/