[igt-dev] [PATCH i-g-t 01/21] scripts/trace.pl: Fix after intel_engine_notify removal
Chris Wilson
chris at chris-wilson.co.uk
Fri May 10 12:33:04 UTC 2019
Quoting Tvrtko Ursulin (2019-05-08 13:10:38)
> From: Tvrtko Ursulin <tvrtko.ursulin at intel.com>
>
> After the removal of engine global seqnos and the corresponding
> intel_engine_notify tracepoints the script needs to be adjusted to cope
> with the new state of things.
>
> To keep working it switches over using the dma_fence:dma_fence_signaled:
> tracepoint and keeps one extra internal map to connect the ctx-seqno pairs
> with engines.
>
> It also needs to key the completion events on the full engine/ctx/seqno
> tokens, and adjust correspondingly the timeline sorting logic.
>
> v2:
> * Do not use late notifications (received after context complete) when
> splitting up coalesced requests. They are now much more likely and can
> not be used.
>
> Signed-off-by: Tvrtko Ursulin <tvrtko.ursulin at intel.com>
> ---
> scripts/trace.pl | 82 ++++++++++++++++++++++++------------------------
> 1 file changed, 41 insertions(+), 41 deletions(-)
>
> diff --git a/scripts/trace.pl b/scripts/trace.pl
> index 18f9f3b18396..95dc3a645e8e 100755
> --- a/scripts/trace.pl
> +++ b/scripts/trace.pl
> @@ -27,7 +27,8 @@ use warnings;
> use 5.010;
>
> my $gid = 0;
> -my (%db, %queue, %submit, %notify, %rings, %ctxdb, %ringmap, %reqwait, %ctxtimelines);
> +my (%db, %queue, %submit, %notify, %rings, %ctxdb, %ringmap, %reqwait,
> + %ctxtimelines, %ctxengines);
> my @freqs;
So what's ctxengines? Or rings for that matter?
I take it ctxengines is really the last engine which we saw this context
execute on?
>
> my $max_items = 3000;
> @@ -66,7 +67,7 @@ Notes:
> i915:i915_request_submit, \
> i915:i915_request_in, \
> i915:i915_request_out, \
> - i915:intel_engine_notify, \
> + dma_fence:dma_fence_signaled, \
> i915:i915_request_wait_begin, \
> i915:i915_request_wait_end \
> [command-to-be-profiled]
> @@ -161,7 +162,7 @@ sub arg_trace
> 'i915:i915_request_submit',
> 'i915:i915_request_in',
> 'i915:i915_request_out',
> - 'i915:intel_engine_notify',
> + 'dma_fence:dma_fence_signaled',
> 'i915:i915_request_wait_begin',
> 'i915:i915_request_wait_end' );
>
> @@ -312,13 +313,6 @@ sub db_key
> return $ring . '/' . $ctx . '/' . $seqno;
> }
>
> -sub global_key
> -{
> - my ($ring, $seqno) = @_;
> -
> - return $ring . '/' . $seqno;
> -}
> -
> sub sanitize_ctx
> {
> my ($ctx, $ring) = @_;
> @@ -419,6 +413,8 @@ while (<>) {
> $req{'ring'} = $ring;
> $req{'seqno'} = $seqno;
> $req{'ctx'} = $ctx;
> + die if exists $ctxengines{$ctx} and $ctxengines{$ctx} ne $ring;
> + $ctxengines{$ctx} = $ring;
> $ctxtimelines{$ctx . '/' . $ring} = 1;
> $req{'name'} = $ctx . '/' . $seqno;
> $req{'global'} = $tp{'global'};
> @@ -429,16 +425,29 @@ while (<>) {
> $ringmap{$rings{$ring}} = $ring;
> $db{$key} = \%req;
> } elsif ($tp_name eq 'i915:i915_request_out:') {
> - my $gkey = global_key($ring, $tp{'global'});
> + my $gkey;
> +
# Must be paired with a previous i915_request_in
> + die unless exists $ctxengines{$ctx};
I'd suggest next unless, because there's always a change the capture is
started part way though someone's workload.
> + $gkey = db_key($ctxengines{$ctx}, $ctx, $seqno);
> +
> + if ($tp{'completed?'}) {
> + die unless exists $db{$key};
> + die unless exists $db{$key}->{'start'};
> + die if exists $db{$key}->{'end'};
> +
> + $db{$key}->{'end'} = $time;
> + $db{$key}->{'notify'} = $notify{$gkey}
> + if exists $notify{$gkey};
Hmm. With preempt-to-busy, a request can complete when we are no longer
tracking it (it completes before we preempt it).
They will still get the schedule-out tracepoint, but marked as
incomplete, and there will be a signaled tp later before we try and
resubmit.
> + } else {
> + delete $db{$key};
> + }
> + } elsif ($tp_name eq 'dma_fence:dma_fence_signaled:') {
> + my $gkey;
>
> - die unless exists $db{$key};
> - die unless exists $db{$key}->{'start'};
> - die if exists $db{$key}->{'end'};
> + die unless exists $ctxengines{$tp{'context'}};
>
> - $db{$key}->{'end'} = $time;
> - $db{$key}->{'notify'} = $notify{$gkey} if exists $notify{$gkey};
> - } elsif ($tp_name eq 'i915:intel_engine_notify:') {
> - my $gkey = global_key($ring, $seqno);
> + $gkey = db_key($ctxengines{$tp{'context'}}, $tp{'context'}, $tp{'seqno'});
>
> $notify{$gkey} = $time unless exists $notify{$gkey};
> } elsif ($tp_name eq 'i915:intel_gpu_freq_change:') {
> @@ -452,7 +461,7 @@ while (<>) {
> # find the largest seqno to be used for timeline sorting purposes.
> my $max_seqno = 0;
> foreach my $key (keys %db) {
> - my $gkey = global_key($db{$key}->{'ring'}, $db{$key}->{'global'});
> + my $gkey = db_key($db{$key}->{'ring'}, $db{$key}->{'ctx'}, $db{$key}->{'seqno'});
>
> die unless exists $db{$key}->{'start'};
>
> @@ -478,14 +487,13 @@ my $key_count = scalar(keys %db);
>
> my %engine_timelines;
>
> -sub sortEngine {
> - my $as = $db{$a}->{'global'};
> - my $bs = $db{$b}->{'global'};
> +sub sortStart {
> + my $as = $db{$a}->{'start'};
> + my $bs = $db{$b}->{'start'};
> my $val;
>
> $val = $as <=> $bs;
> -
> - die if $val == 0;
> + $val = $a cmp $b if $val == 0;
>
> return $val;
> }
> @@ -497,9 +505,7 @@ sub get_engine_timeline {
> return $engine_timelines{$ring} if exists $engine_timelines{$ring};
>
> @timeline = grep { $db{$_}->{'ring'} eq $ring } keys %db;
> - # FIXME seqno restart
> - @timeline = sort sortEngine @timeline;
> -
> + @timeline = sort sortStart @timeline;
> $engine_timelines{$ring} = \@timeline;
>
> return \@timeline;
> @@ -561,20 +567,10 @@ foreach my $gid (sort keys %rings) {
> $db{$key}->{'no-notify'} = 1;
> }
> $db{$key}->{'end'} = $end;
> + $db{$key}->{'notify'} = $end if $db{$key}->{'notify'} > $end;
> }
> }
>
> -sub sortStart {
> - my $as = $db{$a}->{'start'};
> - my $bs = $db{$b}->{'start'};
> - my $val;
> -
> - $val = $as <=> $bs;
> - $val = $a cmp $b if $val == 0;
> -
> - return $val;
> -}
> -
> my $re_sort = 1;
> my @sorted_keys;
>
> @@ -670,9 +666,13 @@ if ($correct_durations) {
> next unless exists $db{$key}->{'no-end'};
> last if $pos == $#{$timeline};
>
> - # Shift following request to start after the current one
> + # Shift following request to start after the current
> + # one, but only if that wouldn't make it zero duration,
> + # which would indicate notify arrived after context
> + # complete.
> $next_key = ${$timeline}[$pos + 1];
> - if (exists $db{$key}->{'notify'}) {
> + if (exists $db{$key}->{'notify'} and
> + $db{$key}->{'notify'} < $db{$key}->{'end'}) {
> $db{$next_key}->{'engine-start'} = $db{$next_key}->{'start'};
> $db{$next_key}->{'start'} = $db{$key}->{'notify'};
> $re_sort = 1;
> @@ -750,9 +750,9 @@ foreach my $gid (sort keys %rings) {
> # Extract all GPU busy intervals and sort them.
> foreach my $key (@sorted_keys) {
> next unless $db{$key}->{'ring'} eq $ring;
> + die if $db{$key}->{'start'} > $db{$key}->{'end'};
Heh, we're out of luck if we want to trace across seqno wraparound.
It makes enough sense,
Reviewed-by: Chris Wilson <chris at chris-wilson.co.uk>
-Chris
More information about the igt-dev
mailing list