blob: e157cd70eb31090f71debf69a72b12fe45b7b19c [file] [log] [blame]
Brian Osmanb6705c22017-08-01 10:23:38 -04001/*
2 * Copyright 2017 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "SkCommandLineFlags.h"
9#include "SkImageInfo.h"
10#include "SkLeanWindows.h"
11#include "SkPoint.h"
12#include "SkRect.h"
13#include "SkTraceEvent.h"
14#include "Test.h"
15
16DEFINE_bool(slowTracingTest, false, "Artificially slow down tracing test to produce nicer JSON");
17
18namespace {
19
20/**
21 * Helper types for demonstrating usage of TRACE_EVENT_OBJECT_XXX macros.
22 */
23struct TracingShape {
24 TracingShape() {
25 TRACE_EVENT_OBJECT_CREATED_WITH_ID("skia.objects", this->typeName(), this);
26 }
27 virtual ~TracingShape() {
28 TRACE_EVENT_OBJECT_DELETED_WITH_ID("skia.objects", this->typeName(), this);
29 }
30 void traceSnapshot() {
31 // The state of an object can be specified at any point with the OBJECT_SNAPSHOT macro.
32 // This takes the "name" (actually the type name), the ID of the object (typically a
33 // pointer), and a single (unnnamed) argument, which is the "snapshot" of that object.
34 //
35 // Tracing viewer requires that all object macros use the same name and id for creation,
36 // deletion, and snapshots. However: It's convenient to put creation and deletion in the
37 // base-class constructor/destructor where the actual type name isn't known yet. That's
38 // what we're doing here. The JSON for snapshots can therefore include the actual type
39 // name, and a special tag that refers to the type name originally used at creation time.
40 // Skia's JSON tracer handles this automatically, so SNAPSHOT macros can simply use the
41 // derived type name, and the JSON will be formatted correctly to link the events.
42 TRACE_EVENT_OBJECT_SNAPSHOT_WITH_ID("skia.objects", this->typeName(), this,
43 TRACE_STR_COPY(this->toString().c_str()));
44 }
45
46 virtual const char* typeName() { return "TracingShape"; }
47 virtual SkString toString() { return SkString("Shape()"); }
48};
49
50struct TracingCircle : public TracingShape {
51 TracingCircle(SkPoint center, SkScalar radius) : fCenter(center), fRadius(radius) {}
52 const char* typeName() override { return "TracingCircle"; }
53 SkString toString() override {
54 return SkStringPrintf("Circle(%f, %f, %f)", fCenter.fX, fCenter.fY, fRadius);
55 }
56
57 SkPoint fCenter;
58 SkScalar fRadius;
59};
60
61struct TracingRect : public TracingShape {
62 TracingRect(SkRect rect) : fRect(rect) {}
63 const char* typeName() override { return "TracingRect"; }
64 SkString toString() override {
65 return SkStringPrintf("Rect(%f, %f, %f, %f)",
66 fRect.fLeft, fRect.fTop, fRect.fRight, fRect.fBottom);
67 }
68
69 SkRect fRect;
70};
71
72}
73
74static SkScalar gTracingTestWorkSink = 1.0f;
75
76static void do_work(int howMuchWork) {
77 // Do busy work so the trace marker durations are large enough to be readable in trace viewer
78 if (FLAGS_slowTracingTest) {
79 for (int i = 0; i < howMuchWork * 100; ++i) {
80 gTracingTestWorkSink += SkScalarSin(i);
81 }
82 }
83}
84
85static void test_trace_simple() {
86 // Simple event that lasts until the end of the current scope. TRACE_FUNC is an easy way
87 // to insert the current function name.
88 TRACE_EVENT0("skia", TRACE_FUNC);
89
90 {
91 // There are versions of the macro that take 1 or 2 named arguments. The arguments
92 // can be any simple type. Strings need to be static/literal - we just copy pointers.
93 // Argument names & values are shown when the event is selected in the viewer.
94 TRACE_EVENT1("skia", "Nested work",
95 "isBGRA", kN32_SkColorType == kBGRA_8888_SkColorType);
96 do_work(500);
97 }
98
99 {
100 // If you must copy a string as an argument value, use the TRACE_STR_COPY macro.
101 // This will instruct the tracing system (if one is active) to make a copy.
102 SkString message = SkStringPrintf("%s %s", "Hello", "World");
103 TRACE_EVENT1("skia", "Dynamic String", "message", TRACE_STR_COPY(message.c_str()));
104 do_work(500);
105 }
106}
107
108static void test_trace_counters() {
109 TRACE_EVENT0("skia", TRACE_FUNC);
110
111 {
112 TRACE_EVENT0("skia", "Single Counter");
113
114 // Counter macros allow recording a named value (which must be a 32-bit integer).
115 // The value will be graphed in the viewer.
116 for (int i = 0; i < 180; ++i) {
117 SkScalar rad = SkDegreesToRadians(SkIntToScalar(i));
118 TRACE_COUNTER1("skia", "sin", SkScalarSin(rad) * 1000.0f + 1000.0f);
119 do_work(10);
120 }
121 }
122
123 {
124 TRACE_EVENT0("skia", "Independent Counters");
125
126 // Recording multiple counters with separate COUNTER1 macros will make separate graphs.
127 for (int i = 0; i < 180; ++i) {
128 SkScalar rad = SkDegreesToRadians(SkIntToScalar(i));
129 SkScalar cos;
130 SkScalar sin = SkScalarSinCos(rad, &cos);
131 TRACE_COUNTER1("skia", "sin", sin * 1000.0f + 1000.0f);
132 TRACE_COUNTER1("skia", "cos", cos * 1000.0f + 1000.0f);
133 do_work(10);
134 }
135 }
136
137 {
138 TRACE_EVENT0("skia", "Stacked Counters");
139
140 // Two counters can be recorded together with COUNTER2. They will be graphed together,
141 // as a stacked bar graph. The combined graph needs a name, as does each data series.
142 for (int i = 0; i < 180; ++i) {
143 SkScalar rad = SkDegreesToRadians(SkIntToScalar(i));
144 SkScalar cos;
145 SkScalar sin = SkScalarSinCos(rad, &cos);
146 TRACE_COUNTER2("skia", "trig",
147 "sin", sin * 1000.0f + 1000.0f,
148 "cos", cos * 1000.0f + 1000.0f);
149 do_work(10);
150 }
151 }
152}
153
154static void test_trace_objects() {
155 TRACE_EVENT0("skia", TRACE_FUNC);
156
157 // Objects can be tracked through time with the TRACE_EVENT_OBJECT_ macros.
158 // The macros in use (and their idiosyncracies) are commented in the TracingShape class above.
159
160 TracingCircle* circle = new TracingCircle(SkPoint::Make(20, 20), 15);
161 circle->traceSnapshot();
162 do_work(100);
163
164 // Make another object. Objects with the same base type are shown in the same row in the viewer.
165 TracingRect* rect = new TracingRect(SkRect::MakeWH(100, 50));
166 rect->traceSnapshot();
167 do_work(100);
168
169 // We can create multiple snapshots of objects to reflect their state over time.
170 circle->fCenter.offset(10, 10);
171 circle->traceSnapshot();
172
173 {
174 // Other events (duration or instant) can refer directly to objects. For Skia's JSON
175 // tracer, having an argument whose name starts with '#' will trigger the creation of JSON
176 // that links the event to the object (with a direct link to the most recent snapshot).
177 TRACE_EVENT1("skia", "Processing Shape", "#shape", circle);
178 do_work(100);
179 }
180
181 delete circle;
182 delete rect;
183}
184
185DEF_TEST(Tracing, reporter) {
186 test_trace_simple();
187 test_trace_counters();
188 test_trace_objects();
189}