1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 // Test of classes in the tracked_objects.h classes.
6 
7 #include "base/tracked_objects.h"
8 
9 #include <stddef.h>
10 #include <stdint.h>
11 
12 #include "base/memory/scoped_ptr.h"
13 #include "base/process/process_handle.h"
14 #include "base/time/time.h"
15 #include "base/tracking_info.h"
16 #include "testing/gtest/include/gtest/gtest.h"
17 
18 const int kLineNumber = 1776;
19 const char kFile[] = "FixedUnitTestFileName";
20 const char kWorkerThreadName[] = "WorkerThread-1";
21 const char kMainThreadName[] = "SomeMainThreadName";
22 const char kStillAlive[] = "Still_Alive";
23 
24 namespace tracked_objects {
25 
26 class TrackedObjectsTest : public testing::Test {
27  protected:
TrackedObjectsTest()28   TrackedObjectsTest() {
29     // On entry, leak any database structures in case they are still in use by
30     // prior threads.
31     ThreadData::ShutdownSingleThreadedCleanup(true);
32 
33     test_time_ = 0;
34     ThreadData::SetAlternateTimeSource(&TrackedObjectsTest::GetTestTime);
35     ThreadData::now_function_is_time_ = true;
36   }
37 
~TrackedObjectsTest()38   ~TrackedObjectsTest() override {
39     // We should not need to leak any structures we create, since we are
40     // single threaded, and carefully accounting for items.
41     ThreadData::ShutdownSingleThreadedCleanup(false);
42   }
43 
44   // Reset the profiler state.
Reset()45   void Reset() {
46     ThreadData::ShutdownSingleThreadedCleanup(false);
47     test_time_ = 0;
48   }
49 
50   // Simulate a birth on the thread named |thread_name|, at the given
51   // |location|.
TallyABirth(const Location & location,const std::string & thread_name)52   void TallyABirth(const Location& location, const std::string& thread_name) {
53     // If the |thread_name| is empty, we don't initialize system with a thread
54     // name, so we're viewed as a worker thread.
55     if (!thread_name.empty())
56       ThreadData::InitializeThreadContext(kMainThreadName);
57 
58     // Do not delete |birth|.  We don't own it.
59     Births* birth = ThreadData::TallyABirthIfActive(location);
60 
61     if (ThreadData::status() == ThreadData::DEACTIVATED)
62       EXPECT_EQ(reinterpret_cast<Births*>(NULL), birth);
63     else
64       EXPECT_NE(reinterpret_cast<Births*>(NULL), birth);
65   }
66 
67   // Helper function to verify the most common test expectations.
ExpectSimpleProcessData(const ProcessDataSnapshot & process_data,const std::string & function_name,const std::string & birth_thread,const std::string & death_thread,int count,int run_ms,int queue_ms)68   void ExpectSimpleProcessData(const ProcessDataSnapshot& process_data,
69                                const std::string& function_name,
70                                const std::string& birth_thread,
71                                const std::string& death_thread,
72                                int count,
73                                int run_ms,
74                                int queue_ms) {
75     ASSERT_EQ(1u, process_data.phased_snapshots.size());
76     auto it = process_data.phased_snapshots.find(0);
77     ASSERT_TRUE(it != process_data.phased_snapshots.end());
78     const ProcessDataPhaseSnapshot& process_data_phase = it->second;
79 
80     ASSERT_EQ(1u, process_data_phase.tasks.size());
81 
82     EXPECT_EQ(kFile, process_data_phase.tasks[0].birth.location.file_name);
83     EXPECT_EQ(function_name,
84               process_data_phase.tasks[0].birth.location.function_name);
85     EXPECT_EQ(kLineNumber,
86               process_data_phase.tasks[0].birth.location.line_number);
87 
88     EXPECT_EQ(birth_thread, process_data_phase.tasks[0].birth.thread_name);
89 
90     EXPECT_EQ(count, process_data_phase.tasks[0].death_data.count);
91     EXPECT_EQ(count * run_ms,
92               process_data_phase.tasks[0].death_data.run_duration_sum);
93     EXPECT_EQ(run_ms, process_data_phase.tasks[0].death_data.run_duration_max);
94     EXPECT_EQ(run_ms,
95               process_data_phase.tasks[0].death_data.run_duration_sample);
96     EXPECT_EQ(count * queue_ms,
97               process_data_phase.tasks[0].death_data.queue_duration_sum);
98     EXPECT_EQ(queue_ms,
99               process_data_phase.tasks[0].death_data.queue_duration_max);
100     EXPECT_EQ(queue_ms,
101               process_data_phase.tasks[0].death_data.queue_duration_sample);
102 
103     EXPECT_EQ(death_thread, process_data_phase.tasks[0].death_thread_name);
104 
105     EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
106   }
107 
108   // Sets time that will be returned by ThreadData::Now().
SetTestTime(unsigned int test_time)109   static void SetTestTime(unsigned int test_time) { test_time_ = test_time; }
110 
111  private:
112   // Returns test time in milliseconds.
GetTestTime()113   static unsigned int GetTestTime() { return test_time_; }
114 
115   // Test time in milliseconds.
116   static unsigned int test_time_;
117 };
118 
119 // static
120 unsigned int TrackedObjectsTest::test_time_;
121 
TEST_F(TrackedObjectsTest,TaskStopwatchNoStartStop)122 TEST_F(TrackedObjectsTest, TaskStopwatchNoStartStop) {
123   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
124 
125   // Check that creating and destroying a stopwatch without starting it doesn't
126   // crash.
127   TaskStopwatch stopwatch;
128 }
129 
TEST_F(TrackedObjectsTest,MinimalStartupShutdown)130 TEST_F(TrackedObjectsTest, MinimalStartupShutdown) {
131   // Minimal test doesn't even create any tasks.
132   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
133 
134   EXPECT_FALSE(ThreadData::first());  // No activity even on this thread.
135   ThreadData* data = ThreadData::Get();
136   EXPECT_TRUE(ThreadData::first());  // Now class was constructed.
137   ASSERT_TRUE(data);
138   EXPECT_FALSE(data->next());
139   EXPECT_EQ(data, ThreadData::Get());
140   ThreadData::BirthMap birth_map;
141   ThreadData::DeathsSnapshot deaths;
142   data->SnapshotMaps(0, &birth_map, &deaths);
143   EXPECT_EQ(0u, birth_map.size());
144   EXPECT_EQ(0u, deaths.size());
145 
146   // Clean up with no leaking.
147   Reset();
148 
149   // Do it again, just to be sure we reset state completely.
150   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
151   EXPECT_FALSE(ThreadData::first());  // No activity even on this thread.
152   data = ThreadData::Get();
153   EXPECT_TRUE(ThreadData::first());  // Now class was constructed.
154   ASSERT_TRUE(data);
155   EXPECT_FALSE(data->next());
156   EXPECT_EQ(data, ThreadData::Get());
157   birth_map.clear();
158   deaths.clear();
159   data->SnapshotMaps(0, &birth_map, &deaths);
160   EXPECT_EQ(0u, birth_map.size());
161   EXPECT_EQ(0u, deaths.size());
162 }
163 
TEST_F(TrackedObjectsTest,TinyStartupShutdown)164 TEST_F(TrackedObjectsTest, TinyStartupShutdown) {
165   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
166 
167   // Instigate tracking on a single tracked object, on our thread.
168   const char kFunction[] = "TinyStartupShutdown";
169   Location location(kFunction, kFile, kLineNumber, NULL);
170   ThreadData::TallyABirthIfActive(location);
171 
172   ThreadData* data = ThreadData::first();
173   ASSERT_TRUE(data);
174   EXPECT_FALSE(data->next());
175   EXPECT_EQ(data, ThreadData::Get());
176   ThreadData::BirthMap birth_map;
177   ThreadData::DeathsSnapshot deaths;
178   data->SnapshotMaps(0, &birth_map, &deaths);
179   EXPECT_EQ(1u, birth_map.size());                         // 1 birth location.
180   EXPECT_EQ(1, birth_map.begin()->second->birth_count());  // 1 birth.
181   EXPECT_EQ(0u, deaths.size());                            // No deaths.
182 
183 
184   // Now instigate another birth, while we are timing the run of the first
185   // execution.
186   // Create a child (using the same birth location).
187   // TrackingInfo will call TallyABirth() during construction.
188   const int32_t start_time = 1;
189   base::TimeTicks kBogusBirthTime = base::TimeTicks() +
190       base::TimeDelta::FromMilliseconds(start_time);
191   base::TrackingInfo pending_task(location, kBogusBirthTime);
192   SetTestTime(1);
193   TaskStopwatch stopwatch;
194   stopwatch.Start();
195   // Finally conclude the outer run.
196   const int32_t time_elapsed = 1000;
197   SetTestTime(start_time + time_elapsed);
198   stopwatch.Stop();
199 
200   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
201 
202   birth_map.clear();
203   deaths.clear();
204   data->SnapshotMaps(0, &birth_map, &deaths);
205   EXPECT_EQ(1u, birth_map.size());                         // 1 birth location.
206   EXPECT_EQ(2, birth_map.begin()->second->birth_count());  // 2 births.
207   EXPECT_EQ(1u, deaths.size());                            // 1 location.
208   EXPECT_EQ(1, deaths.begin()->second.death_data.count);   // 1 death.
209 
210   // The births were at the same location as the one known death.
211   EXPECT_EQ(birth_map.begin()->second, deaths.begin()->first);
212 
213   ProcessDataSnapshot process_data;
214   ThreadData::Snapshot(0, &process_data);
215 
216   ASSERT_EQ(1u, process_data.phased_snapshots.size());
217   auto it = process_data.phased_snapshots.find(0);
218   ASSERT_TRUE(it != process_data.phased_snapshots.end());
219   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
220   ASSERT_EQ(1u, process_data_phase.tasks.size());
221   EXPECT_EQ(kFile, process_data_phase.tasks[0].birth.location.file_name);
222   EXPECT_EQ(kFunction,
223             process_data_phase.tasks[0].birth.location.function_name);
224   EXPECT_EQ(kLineNumber,
225             process_data_phase.tasks[0].birth.location.line_number);
226   EXPECT_EQ(kWorkerThreadName, process_data_phase.tasks[0].birth.thread_name);
227   EXPECT_EQ(1, process_data_phase.tasks[0].death_data.count);
228   EXPECT_EQ(time_elapsed,
229             process_data_phase.tasks[0].death_data.run_duration_sum);
230   EXPECT_EQ(time_elapsed,
231             process_data_phase.tasks[0].death_data.run_duration_max);
232   EXPECT_EQ(time_elapsed,
233             process_data_phase.tasks[0].death_data.run_duration_sample);
234   EXPECT_EQ(0, process_data_phase.tasks[0].death_data.queue_duration_sum);
235   EXPECT_EQ(0, process_data_phase.tasks[0].death_data.queue_duration_max);
236   EXPECT_EQ(0, process_data_phase.tasks[0].death_data.queue_duration_sample);
237   EXPECT_EQ(kWorkerThreadName, process_data_phase.tasks[0].death_thread_name);
238 }
239 
TEST_F(TrackedObjectsTest,DeathDataTestRecordDeath)240 TEST_F(TrackedObjectsTest, DeathDataTestRecordDeath) {
241   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
242 
243   scoped_ptr<DeathData> data(new DeathData());
244   ASSERT_NE(data, nullptr);
245   EXPECT_EQ(data->run_duration_sum(), 0);
246   EXPECT_EQ(data->run_duration_max(), 0);
247   EXPECT_EQ(data->run_duration_sample(), 0);
248   EXPECT_EQ(data->queue_duration_sum(), 0);
249   EXPECT_EQ(data->queue_duration_max(), 0);
250   EXPECT_EQ(data->queue_duration_sample(), 0);
251   EXPECT_EQ(data->count(), 0);
252   EXPECT_EQ(nullptr, data->last_phase_snapshot());
253 
254   int32_t run_ms = 42;
255   int32_t queue_ms = 8;
256 
257   const int kUnrandomInt = 0;  // Fake random int that ensure we sample data.
258   data->RecordDeath(queue_ms, run_ms, kUnrandomInt);
259   EXPECT_EQ(data->run_duration_sum(), run_ms);
260   EXPECT_EQ(data->run_duration_max(), run_ms);
261   EXPECT_EQ(data->run_duration_sample(), run_ms);
262   EXPECT_EQ(data->queue_duration_sum(), queue_ms);
263   EXPECT_EQ(data->queue_duration_max(), queue_ms);
264   EXPECT_EQ(data->queue_duration_sample(), queue_ms);
265   EXPECT_EQ(data->count(), 1);
266   EXPECT_EQ(nullptr, data->last_phase_snapshot());
267 
268   data->RecordDeath(queue_ms, run_ms, kUnrandomInt);
269   EXPECT_EQ(data->run_duration_sum(), run_ms + run_ms);
270   EXPECT_EQ(data->run_duration_max(), run_ms);
271   EXPECT_EQ(data->run_duration_sample(), run_ms);
272   EXPECT_EQ(data->queue_duration_sum(), queue_ms + queue_ms);
273   EXPECT_EQ(data->queue_duration_max(), queue_ms);
274   EXPECT_EQ(data->queue_duration_sample(), queue_ms);
275   EXPECT_EQ(data->count(), 2);
276   EXPECT_EQ(nullptr, data->last_phase_snapshot());
277 }
278 
TEST_F(TrackedObjectsTest,DeathDataTest2Phases)279 TEST_F(TrackedObjectsTest, DeathDataTest2Phases) {
280   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
281 
282   scoped_ptr<DeathData> data(new DeathData());
283   ASSERT_NE(data, nullptr);
284 
285   int32_t run_ms = 42;
286   int32_t queue_ms = 8;
287 
288   const int kUnrandomInt = 0;  // Fake random int that ensure we sample data.
289   data->RecordDeath(queue_ms, run_ms, kUnrandomInt);
290   data->RecordDeath(queue_ms, run_ms, kUnrandomInt);
291 
292   data->OnProfilingPhaseCompleted(123);
293   EXPECT_EQ(data->run_duration_sum(), run_ms + run_ms);
294   EXPECT_EQ(data->run_duration_max(), 0);
295   EXPECT_EQ(data->run_duration_sample(), run_ms);
296   EXPECT_EQ(data->queue_duration_sum(), queue_ms + queue_ms);
297   EXPECT_EQ(data->queue_duration_max(), 0);
298   EXPECT_EQ(data->queue_duration_sample(), queue_ms);
299   EXPECT_EQ(data->count(), 2);
300   ASSERT_NE(nullptr, data->last_phase_snapshot());
301   EXPECT_EQ(123, data->last_phase_snapshot()->profiling_phase);
302   EXPECT_EQ(2, data->last_phase_snapshot()->death_data.count);
303   EXPECT_EQ(2 * run_ms,
304             data->last_phase_snapshot()->death_data.run_duration_sum);
305   EXPECT_EQ(run_ms, data->last_phase_snapshot()->death_data.run_duration_max);
306   EXPECT_EQ(run_ms,
307             data->last_phase_snapshot()->death_data.run_duration_sample);
308   EXPECT_EQ(2 * queue_ms,
309             data->last_phase_snapshot()->death_data.queue_duration_sum);
310   EXPECT_EQ(queue_ms,
311             data->last_phase_snapshot()->death_data.queue_duration_max);
312   EXPECT_EQ(queue_ms,
313             data->last_phase_snapshot()->death_data.queue_duration_sample);
314   EXPECT_EQ(nullptr, data->last_phase_snapshot()->prev);
315 
316   int32_t run_ms1 = 21;
317   int32_t queue_ms1 = 4;
318 
319   data->RecordDeath(queue_ms1, run_ms1, kUnrandomInt);
320   EXPECT_EQ(data->run_duration_sum(), run_ms + run_ms + run_ms1);
321   EXPECT_EQ(data->run_duration_max(), run_ms1);
322   EXPECT_EQ(data->run_duration_sample(), run_ms1);
323   EXPECT_EQ(data->queue_duration_sum(), queue_ms + queue_ms + queue_ms1);
324   EXPECT_EQ(data->queue_duration_max(), queue_ms1);
325   EXPECT_EQ(data->queue_duration_sample(), queue_ms1);
326   EXPECT_EQ(data->count(), 3);
327   ASSERT_NE(nullptr, data->last_phase_snapshot());
328   EXPECT_EQ(123, data->last_phase_snapshot()->profiling_phase);
329   EXPECT_EQ(2, data->last_phase_snapshot()->death_data.count);
330   EXPECT_EQ(2 * run_ms,
331             data->last_phase_snapshot()->death_data.run_duration_sum);
332   EXPECT_EQ(run_ms, data->last_phase_snapshot()->death_data.run_duration_max);
333   EXPECT_EQ(run_ms,
334             data->last_phase_snapshot()->death_data.run_duration_sample);
335   EXPECT_EQ(2 * queue_ms,
336             data->last_phase_snapshot()->death_data.queue_duration_sum);
337   EXPECT_EQ(queue_ms,
338             data->last_phase_snapshot()->death_data.queue_duration_max);
339   EXPECT_EQ(queue_ms,
340             data->last_phase_snapshot()->death_data.queue_duration_sample);
341   EXPECT_EQ(nullptr, data->last_phase_snapshot()->prev);
342 }
343 
TEST_F(TrackedObjectsTest,Delta)344 TEST_F(TrackedObjectsTest, Delta) {
345   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
346 
347   DeathDataSnapshot snapshot;
348   snapshot.count = 10;
349   snapshot.run_duration_sum = 100;
350   snapshot.run_duration_max = 50;
351   snapshot.run_duration_sample = 25;
352   snapshot.queue_duration_sum = 200;
353   snapshot.queue_duration_max = 101;
354   snapshot.queue_duration_sample = 26;
355 
356   DeathDataSnapshot older_snapshot;
357   older_snapshot.count = 2;
358   older_snapshot.run_duration_sum = 95;
359   older_snapshot.run_duration_max = 48;
360   older_snapshot.run_duration_sample = 22;
361   older_snapshot.queue_duration_sum = 190;
362   older_snapshot.queue_duration_max = 99;
363   older_snapshot.queue_duration_sample = 21;
364 
365   const DeathDataSnapshot& delta = snapshot.Delta(older_snapshot);
366   EXPECT_EQ(8, delta.count);
367   EXPECT_EQ(5, delta.run_duration_sum);
368   EXPECT_EQ(50, delta.run_duration_max);
369   EXPECT_EQ(25, delta.run_duration_sample);
370   EXPECT_EQ(10, delta.queue_duration_sum);
371   EXPECT_EQ(101, delta.queue_duration_max);
372   EXPECT_EQ(26, delta.queue_duration_sample);
373 }
374 
TEST_F(TrackedObjectsTest,DeactivatedBirthOnlyToSnapshotWorkerThread)375 TEST_F(TrackedObjectsTest, DeactivatedBirthOnlyToSnapshotWorkerThread) {
376   // Start in the deactivated state.
377   ThreadData::InitializeAndSetTrackingStatus(ThreadData::DEACTIVATED);
378 
379   const char kFunction[] = "DeactivatedBirthOnlyToSnapshotWorkerThread";
380   Location location(kFunction, kFile, kLineNumber, NULL);
381   TallyABirth(location, std::string());
382 
383   ProcessDataSnapshot process_data;
384   ThreadData::Snapshot(0, &process_data);
385 
386   ASSERT_EQ(1u, process_data.phased_snapshots.size());
387 
388   auto it = process_data.phased_snapshots.find(0);
389   ASSERT_TRUE(it != process_data.phased_snapshots.end());
390   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
391 
392   ASSERT_EQ(0u, process_data_phase.tasks.size());
393 
394   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
395 }
396 
TEST_F(TrackedObjectsTest,DeactivatedBirthOnlyToSnapshotMainThread)397 TEST_F(TrackedObjectsTest, DeactivatedBirthOnlyToSnapshotMainThread) {
398   // Start in the deactivated state.
399   ThreadData::InitializeAndSetTrackingStatus(ThreadData::DEACTIVATED);
400 
401   const char kFunction[] = "DeactivatedBirthOnlyToSnapshotMainThread";
402   Location location(kFunction, kFile, kLineNumber, NULL);
403   TallyABirth(location, kMainThreadName);
404 
405   ProcessDataSnapshot process_data;
406   ThreadData::Snapshot(0, &process_data);
407 
408   ASSERT_EQ(1u, process_data.phased_snapshots.size());
409 
410   auto it = process_data.phased_snapshots.find(0);
411   ASSERT_TRUE(it != process_data.phased_snapshots.end());
412   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
413 
414   ASSERT_EQ(0u, process_data_phase.tasks.size());
415 
416   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
417 }
418 
TEST_F(TrackedObjectsTest,BirthOnlyToSnapshotWorkerThread)419 TEST_F(TrackedObjectsTest, BirthOnlyToSnapshotWorkerThread) {
420   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
421 
422   const char kFunction[] = "BirthOnlyToSnapshotWorkerThread";
423   Location location(kFunction, kFile, kLineNumber, NULL);
424   TallyABirth(location, std::string());
425 
426   ProcessDataSnapshot process_data;
427   ThreadData::Snapshot(0, &process_data);
428   ExpectSimpleProcessData(process_data, kFunction, kWorkerThreadName,
429                           kStillAlive, 1, 0, 0);
430 }
431 
TEST_F(TrackedObjectsTest,BirthOnlyToSnapshotMainThread)432 TEST_F(TrackedObjectsTest, BirthOnlyToSnapshotMainThread) {
433   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
434 
435   const char kFunction[] = "BirthOnlyToSnapshotMainThread";
436   Location location(kFunction, kFile, kLineNumber, NULL);
437   TallyABirth(location, kMainThreadName);
438 
439   ProcessDataSnapshot process_data;
440   ThreadData::Snapshot(0, &process_data);
441   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName, kStillAlive,
442                           1, 0, 0);
443 }
444 
TEST_F(TrackedObjectsTest,LifeCycleToSnapshotMainThread)445 TEST_F(TrackedObjectsTest, LifeCycleToSnapshotMainThread) {
446   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
447 
448   const char kFunction[] = "LifeCycleToSnapshotMainThread";
449   Location location(kFunction, kFile, kLineNumber, NULL);
450   TallyABirth(location, kMainThreadName);
451 
452   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
453   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
454   // TrackingInfo will call TallyABirth() during construction.
455   base::TrackingInfo pending_task(location, kDelayedStartTime);
456   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
457 
458   const unsigned int kStartOfRun = 5;
459   const unsigned int kEndOfRun = 7;
460   SetTestTime(kStartOfRun);
461   TaskStopwatch stopwatch;
462   stopwatch.Start();
463   SetTestTime(kEndOfRun);
464   stopwatch.Stop();
465 
466   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
467 
468   ProcessDataSnapshot process_data;
469   ThreadData::Snapshot(0, &process_data);
470   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName,
471                           kMainThreadName, 1, 2, 4);
472 }
473 
TEST_F(TrackedObjectsTest,TwoPhases)474 TEST_F(TrackedObjectsTest, TwoPhases) {
475   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
476 
477   const char kFunction[] = "TwoPhases";
478   Location location(kFunction, kFile, kLineNumber, NULL);
479   TallyABirth(location, kMainThreadName);
480 
481   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
482   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
483   // TrackingInfo will call TallyABirth() during construction.
484   base::TrackingInfo pending_task(location, kDelayedStartTime);
485   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
486 
487   const unsigned int kStartOfRun = 5;
488   const unsigned int kEndOfRun = 7;
489   SetTestTime(kStartOfRun);
490   TaskStopwatch stopwatch;
491   stopwatch.Start();
492   SetTestTime(kEndOfRun);
493   stopwatch.Stop();
494 
495   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
496 
497   ThreadData::OnProfilingPhaseCompleted(0);
498 
499   TallyABirth(location, kMainThreadName);
500 
501   const TrackedTime kTimePosted1 = TrackedTime::FromMilliseconds(9);
502   const base::TimeTicks kDelayedStartTime1 = base::TimeTicks();
503   // TrackingInfo will call TallyABirth() during construction.
504   base::TrackingInfo pending_task1(location, kDelayedStartTime1);
505   pending_task1.time_posted = kTimePosted1;  // Overwrite implied Now().
506 
507   const unsigned int kStartOfRun1 = 11;
508   const unsigned int kEndOfRun1 = 21;
509   SetTestTime(kStartOfRun1);
510   TaskStopwatch stopwatch1;
511   stopwatch1.Start();
512   SetTestTime(kEndOfRun1);
513   stopwatch1.Stop();
514 
515   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task1, stopwatch1);
516 
517   ProcessDataSnapshot process_data;
518   ThreadData::Snapshot(1, &process_data);
519 
520   ASSERT_EQ(2u, process_data.phased_snapshots.size());
521 
522   auto it0 = process_data.phased_snapshots.find(0);
523   ASSERT_TRUE(it0 != process_data.phased_snapshots.end());
524   const ProcessDataPhaseSnapshot& process_data_phase0 = it0->second;
525 
526   ASSERT_EQ(1u, process_data_phase0.tasks.size());
527 
528   EXPECT_EQ(kFile, process_data_phase0.tasks[0].birth.location.file_name);
529   EXPECT_EQ(kFunction,
530             process_data_phase0.tasks[0].birth.location.function_name);
531   EXPECT_EQ(kLineNumber,
532             process_data_phase0.tasks[0].birth.location.line_number);
533 
534   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].birth.thread_name);
535 
536   EXPECT_EQ(1, process_data_phase0.tasks[0].death_data.count);
537   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_sum);
538   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_max);
539   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_sample);
540   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_sum);
541   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_max);
542   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_sample);
543 
544   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].death_thread_name);
545 
546   auto it1 = process_data.phased_snapshots.find(1);
547   ASSERT_TRUE(it1 != process_data.phased_snapshots.end());
548   const ProcessDataPhaseSnapshot& process_data_phase1 = it1->second;
549 
550   ASSERT_EQ(1u, process_data_phase1.tasks.size());
551 
552   EXPECT_EQ(kFile, process_data_phase1.tasks[0].birth.location.file_name);
553   EXPECT_EQ(kFunction,
554             process_data_phase1.tasks[0].birth.location.function_name);
555   EXPECT_EQ(kLineNumber,
556             process_data_phase1.tasks[0].birth.location.line_number);
557 
558   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].birth.thread_name);
559 
560   EXPECT_EQ(1, process_data_phase1.tasks[0].death_data.count);
561   EXPECT_EQ(10, process_data_phase1.tasks[0].death_data.run_duration_sum);
562   EXPECT_EQ(10, process_data_phase1.tasks[0].death_data.run_duration_max);
563   EXPECT_EQ(10, process_data_phase1.tasks[0].death_data.run_duration_sample);
564   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.queue_duration_sum);
565   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.queue_duration_max);
566   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.queue_duration_sample);
567 
568   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].death_thread_name);
569 
570   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
571 }
572 
TEST_F(TrackedObjectsTest,ThreePhases)573 TEST_F(TrackedObjectsTest, ThreePhases) {
574   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
575 
576   const char kFunction[] = "ThreePhases";
577   Location location(kFunction, kFile, kLineNumber, NULL);
578 
579   // Phase 0
580   {
581     TallyABirth(location, kMainThreadName);
582 
583     // TrackingInfo will call TallyABirth() during construction.
584     SetTestTime(10);
585     base::TrackingInfo pending_task(location, base::TimeTicks());
586 
587     SetTestTime(17);
588     TaskStopwatch stopwatch;
589     stopwatch.Start();
590     SetTestTime(23);
591     stopwatch.Stop();
592 
593     ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
594   }
595 
596   ThreadData::OnProfilingPhaseCompleted(0);
597 
598   // Phase 1
599   {
600     TallyABirth(location, kMainThreadName);
601 
602     SetTestTime(30);
603     base::TrackingInfo pending_task(location, base::TimeTicks());
604 
605     SetTestTime(35);
606     TaskStopwatch stopwatch;
607     stopwatch.Start();
608     SetTestTime(39);
609     stopwatch.Stop();
610 
611     ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
612   }
613 
614   ThreadData::OnProfilingPhaseCompleted(1);
615 
616   // Phase 2
617   {
618     TallyABirth(location, kMainThreadName);
619 
620     // TrackingInfo will call TallyABirth() during construction.
621     SetTestTime(40);
622     base::TrackingInfo pending_task(location, base::TimeTicks());
623 
624     SetTestTime(43);
625     TaskStopwatch stopwatch;
626     stopwatch.Start();
627     SetTestTime(45);
628     stopwatch.Stop();
629 
630     ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
631   }
632 
633   // Snapshot and check results.
634   ProcessDataSnapshot process_data;
635   ThreadData::Snapshot(2, &process_data);
636 
637   ASSERT_EQ(3u, process_data.phased_snapshots.size());
638 
639   auto it0 = process_data.phased_snapshots.find(0);
640   ASSERT_TRUE(it0 != process_data.phased_snapshots.end());
641   const ProcessDataPhaseSnapshot& process_data_phase0 = it0->second;
642 
643   ASSERT_EQ(1u, process_data_phase0.tasks.size());
644 
645   EXPECT_EQ(kFile, process_data_phase0.tasks[0].birth.location.file_name);
646   EXPECT_EQ(kFunction,
647             process_data_phase0.tasks[0].birth.location.function_name);
648   EXPECT_EQ(kLineNumber,
649             process_data_phase0.tasks[0].birth.location.line_number);
650 
651   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].birth.thread_name);
652 
653   EXPECT_EQ(1, process_data_phase0.tasks[0].death_data.count);
654   EXPECT_EQ(6, process_data_phase0.tasks[0].death_data.run_duration_sum);
655   EXPECT_EQ(6, process_data_phase0.tasks[0].death_data.run_duration_max);
656   EXPECT_EQ(6, process_data_phase0.tasks[0].death_data.run_duration_sample);
657   EXPECT_EQ(7, process_data_phase0.tasks[0].death_data.queue_duration_sum);
658   EXPECT_EQ(7, process_data_phase0.tasks[0].death_data.queue_duration_max);
659   EXPECT_EQ(7, process_data_phase0.tasks[0].death_data.queue_duration_sample);
660 
661   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].death_thread_name);
662 
663   auto it1 = process_data.phased_snapshots.find(1);
664   ASSERT_TRUE(it1 != process_data.phased_snapshots.end());
665   const ProcessDataPhaseSnapshot& process_data_phase1 = it1->second;
666 
667   ASSERT_EQ(1u, process_data_phase1.tasks.size());
668 
669   EXPECT_EQ(kFile, process_data_phase1.tasks[0].birth.location.file_name);
670   EXPECT_EQ(kFunction,
671             process_data_phase1.tasks[0].birth.location.function_name);
672   EXPECT_EQ(kLineNumber,
673             process_data_phase1.tasks[0].birth.location.line_number);
674 
675   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].birth.thread_name);
676 
677   EXPECT_EQ(1, process_data_phase1.tasks[0].death_data.count);
678   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.run_duration_sum);
679   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.run_duration_max);
680   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.run_duration_sample);
681   EXPECT_EQ(5, process_data_phase1.tasks[0].death_data.queue_duration_sum);
682   EXPECT_EQ(5, process_data_phase1.tasks[0].death_data.queue_duration_max);
683   EXPECT_EQ(5, process_data_phase1.tasks[0].death_data.queue_duration_sample);
684 
685   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].death_thread_name);
686 
687   auto it2 = process_data.phased_snapshots.find(2);
688   ASSERT_TRUE(it2 != process_data.phased_snapshots.end());
689   const ProcessDataPhaseSnapshot& process_data_phase2 = it2->second;
690 
691   ASSERT_EQ(1u, process_data_phase2.tasks.size());
692 
693   EXPECT_EQ(kFile, process_data_phase2.tasks[0].birth.location.file_name);
694   EXPECT_EQ(kFunction,
695             process_data_phase2.tasks[0].birth.location.function_name);
696   EXPECT_EQ(kLineNumber,
697             process_data_phase2.tasks[0].birth.location.line_number);
698 
699   EXPECT_EQ(kMainThreadName, process_data_phase2.tasks[0].birth.thread_name);
700 
701   EXPECT_EQ(1, process_data_phase2.tasks[0].death_data.count);
702   EXPECT_EQ(2, process_data_phase2.tasks[0].death_data.run_duration_sum);
703   EXPECT_EQ(2, process_data_phase2.tasks[0].death_data.run_duration_max);
704   EXPECT_EQ(2, process_data_phase2.tasks[0].death_data.run_duration_sample);
705   EXPECT_EQ(3, process_data_phase2.tasks[0].death_data.queue_duration_sum);
706   EXPECT_EQ(3, process_data_phase2.tasks[0].death_data.queue_duration_max);
707   EXPECT_EQ(3, process_data_phase2.tasks[0].death_data.queue_duration_sample);
708 
709   EXPECT_EQ(kMainThreadName, process_data_phase2.tasks[0].death_thread_name);
710 
711   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
712 }
713 
TEST_F(TrackedObjectsTest,TwoPhasesSecondEmpty)714 TEST_F(TrackedObjectsTest, TwoPhasesSecondEmpty) {
715   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
716 
717   const char kFunction[] = "TwoPhasesSecondEmpty";
718   Location location(kFunction, kFile, kLineNumber, NULL);
719   ThreadData::InitializeThreadContext(kMainThreadName);
720 
721   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
722   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
723   // TrackingInfo will call TallyABirth() during construction.
724   base::TrackingInfo pending_task(location, kDelayedStartTime);
725   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
726 
727   const unsigned int kStartOfRun = 5;
728   const unsigned int kEndOfRun = 7;
729   SetTestTime(kStartOfRun);
730   TaskStopwatch stopwatch;
731   stopwatch.Start();
732   SetTestTime(kEndOfRun);
733   stopwatch.Stop();
734 
735   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
736 
737   ThreadData::OnProfilingPhaseCompleted(0);
738 
739   ProcessDataSnapshot process_data;
740   ThreadData::Snapshot(1, &process_data);
741 
742   ASSERT_EQ(2u, process_data.phased_snapshots.size());
743 
744   auto it0 = process_data.phased_snapshots.find(0);
745   ASSERT_TRUE(it0 != process_data.phased_snapshots.end());
746   const ProcessDataPhaseSnapshot& process_data_phase0 = it0->second;
747 
748   ASSERT_EQ(1u, process_data_phase0.tasks.size());
749 
750   EXPECT_EQ(kFile, process_data_phase0.tasks[0].birth.location.file_name);
751   EXPECT_EQ(kFunction,
752             process_data_phase0.tasks[0].birth.location.function_name);
753   EXPECT_EQ(kLineNumber,
754             process_data_phase0.tasks[0].birth.location.line_number);
755 
756   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].birth.thread_name);
757 
758   EXPECT_EQ(1, process_data_phase0.tasks[0].death_data.count);
759   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_sum);
760   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_max);
761   EXPECT_EQ(2, process_data_phase0.tasks[0].death_data.run_duration_sample);
762   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_sum);
763   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_max);
764   EXPECT_EQ(4, process_data_phase0.tasks[0].death_data.queue_duration_sample);
765 
766   EXPECT_EQ(kMainThreadName, process_data_phase0.tasks[0].death_thread_name);
767 
768   auto it1 = process_data.phased_snapshots.find(1);
769   ASSERT_TRUE(it1 != process_data.phased_snapshots.end());
770   const ProcessDataPhaseSnapshot& process_data_phase1 = it1->second;
771 
772   ASSERT_EQ(0u, process_data_phase1.tasks.size());
773 
774   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
775 }
776 
TEST_F(TrackedObjectsTest,TwoPhasesFirstEmpty)777 TEST_F(TrackedObjectsTest, TwoPhasesFirstEmpty) {
778   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
779 
780   ThreadData::OnProfilingPhaseCompleted(0);
781 
782   const char kFunction[] = "TwoPhasesSecondEmpty";
783   Location location(kFunction, kFile, kLineNumber, NULL);
784   ThreadData::InitializeThreadContext(kMainThreadName);
785 
786   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
787   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
788   // TrackingInfo will call TallyABirth() during construction.
789   base::TrackingInfo pending_task(location, kDelayedStartTime);
790   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
791 
792   const unsigned int kStartOfRun = 5;
793   const unsigned int kEndOfRun = 7;
794   SetTestTime(kStartOfRun);
795   TaskStopwatch stopwatch;
796   stopwatch.Start();
797   SetTestTime(kEndOfRun);
798   stopwatch.Stop();
799 
800   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
801 
802   ProcessDataSnapshot process_data;
803   ThreadData::Snapshot(1, &process_data);
804 
805   ASSERT_EQ(1u, process_data.phased_snapshots.size());
806 
807   auto it1 = process_data.phased_snapshots.find(1);
808   ASSERT_TRUE(it1 != process_data.phased_snapshots.end());
809   const ProcessDataPhaseSnapshot& process_data_phase1 = it1->second;
810 
811   ASSERT_EQ(1u, process_data_phase1.tasks.size());
812 
813   EXPECT_EQ(kFile, process_data_phase1.tasks[0].birth.location.file_name);
814   EXPECT_EQ(kFunction,
815             process_data_phase1.tasks[0].birth.location.function_name);
816   EXPECT_EQ(kLineNumber,
817             process_data_phase1.tasks[0].birth.location.line_number);
818 
819   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].birth.thread_name);
820 
821   EXPECT_EQ(1, process_data_phase1.tasks[0].death_data.count);
822   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.run_duration_sum);
823   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.run_duration_max);
824   EXPECT_EQ(2, process_data_phase1.tasks[0].death_data.run_duration_sample);
825   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.queue_duration_sum);
826   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.queue_duration_max);
827   EXPECT_EQ(4, process_data_phase1.tasks[0].death_data.queue_duration_sample);
828 
829   EXPECT_EQ(kMainThreadName, process_data_phase1.tasks[0].death_thread_name);
830 
831   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
832 }
833 
834 // We will deactivate tracking after the birth, and before the death, and
835 // demonstrate that the lifecycle is completely tallied. This ensures that
836 // our tallied births are matched by tallied deaths (except for when the
837 // task is still running, or is queued).
TEST_F(TrackedObjectsTest,LifeCycleMidDeactivatedToSnapshotMainThread)838 TEST_F(TrackedObjectsTest, LifeCycleMidDeactivatedToSnapshotMainThread) {
839   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
840 
841   const char kFunction[] = "LifeCycleMidDeactivatedToSnapshotMainThread";
842   Location location(kFunction, kFile, kLineNumber, NULL);
843   TallyABirth(location, kMainThreadName);
844 
845   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
846   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
847   // TrackingInfo will call TallyABirth() during construction.
848   base::TrackingInfo pending_task(location, kDelayedStartTime);
849   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
850 
851   // Turn off tracking now that we have births.
852   ThreadData::InitializeAndSetTrackingStatus(ThreadData::DEACTIVATED);
853 
854   const unsigned int kStartOfRun = 5;
855   const unsigned int kEndOfRun = 7;
856   SetTestTime(kStartOfRun);
857   TaskStopwatch stopwatch;
858   stopwatch.Start();
859   SetTestTime(kEndOfRun);
860   stopwatch.Stop();
861 
862   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
863 
864   ProcessDataSnapshot process_data;
865   ThreadData::Snapshot(0, &process_data);
866   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName,
867                           kMainThreadName, 1, 2, 4);
868 }
869 
870 // We will deactivate tracking before starting a life cycle, and neither
871 // the birth nor the death will be recorded.
TEST_F(TrackedObjectsTest,LifeCyclePreDeactivatedToSnapshotMainThread)872 TEST_F(TrackedObjectsTest, LifeCyclePreDeactivatedToSnapshotMainThread) {
873   // Start in the deactivated state.
874   ThreadData::InitializeAndSetTrackingStatus(ThreadData::DEACTIVATED);
875 
876   const char kFunction[] = "LifeCyclePreDeactivatedToSnapshotMainThread";
877   Location location(kFunction, kFile, kLineNumber, NULL);
878   TallyABirth(location, kMainThreadName);
879 
880   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
881   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
882   // TrackingInfo will call TallyABirth() during construction.
883   base::TrackingInfo pending_task(location, kDelayedStartTime);
884   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
885 
886   const unsigned int kStartOfRun = 5;
887   const unsigned int kEndOfRun = 7;
888   SetTestTime(kStartOfRun);
889   TaskStopwatch stopwatch;
890   stopwatch.Start();
891   SetTestTime(kEndOfRun);
892   stopwatch.Stop();
893 
894   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
895 
896   ProcessDataSnapshot process_data;
897   ThreadData::Snapshot(0, &process_data);
898 
899   ASSERT_EQ(1u, process_data.phased_snapshots.size());
900 
901   auto it = process_data.phased_snapshots.find(0);
902   ASSERT_TRUE(it != process_data.phased_snapshots.end());
903   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
904 
905   ASSERT_EQ(0u, process_data_phase.tasks.size());
906 
907   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
908 }
909 
TEST_F(TrackedObjectsTest,TwoLives)910 TEST_F(TrackedObjectsTest, TwoLives) {
911   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
912 
913   const char kFunction[] = "TwoLives";
914   Location location(kFunction, kFile, kLineNumber, NULL);
915   TallyABirth(location, kMainThreadName);
916 
917   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
918   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
919   // TrackingInfo will call TallyABirth() during construction.
920   base::TrackingInfo pending_task(location, kDelayedStartTime);
921   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
922 
923   const unsigned int kStartOfRun = 5;
924   const unsigned int kEndOfRun = 7;
925   SetTestTime(kStartOfRun);
926   TaskStopwatch stopwatch;
927   stopwatch.Start();
928   SetTestTime(kEndOfRun);
929   stopwatch.Stop();
930 
931   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
932 
933   // TrackingInfo will call TallyABirth() during construction.
934   base::TrackingInfo pending_task2(location, kDelayedStartTime);
935   pending_task2.time_posted = kTimePosted;  // Overwrite implied Now().
936   SetTestTime(kStartOfRun);
937   TaskStopwatch stopwatch2;
938   stopwatch2.Start();
939   SetTestTime(kEndOfRun);
940   stopwatch2.Stop();
941 
942   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task2, stopwatch2);
943 
944   ProcessDataSnapshot process_data;
945   ThreadData::Snapshot(0, &process_data);
946   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName,
947                           kMainThreadName, 2, 2, 4);
948 }
949 
TEST_F(TrackedObjectsTest,DifferentLives)950 TEST_F(TrackedObjectsTest, DifferentLives) {
951   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
952 
953   // Use a well named thread.
954   ThreadData::InitializeThreadContext(kMainThreadName);
955   const char kFunction[] = "DifferentLives";
956   Location location(kFunction, kFile, kLineNumber, NULL);
957 
958   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
959   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
960   // TrackingInfo will call TallyABirth() during construction.
961   base::TrackingInfo pending_task(location, kDelayedStartTime);
962   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
963 
964   const unsigned int kStartOfRun = 5;
965   const unsigned int kEndOfRun = 7;
966   SetTestTime(kStartOfRun);
967   TaskStopwatch stopwatch;
968   stopwatch.Start();
969   SetTestTime(kEndOfRun);
970   stopwatch.Stop();
971 
972   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, stopwatch);
973 
974   const int kSecondFakeLineNumber = 999;
975   Location second_location(kFunction, kFile, kSecondFakeLineNumber, NULL);
976 
977   // TrackingInfo will call TallyABirth() during construction.
978   base::TrackingInfo pending_task2(second_location, kDelayedStartTime);
979   pending_task2.time_posted = kTimePosted;  // Overwrite implied Now().
980 
981   ProcessDataSnapshot process_data;
982   ThreadData::Snapshot(0, &process_data);
983 
984   ASSERT_EQ(1u, process_data.phased_snapshots.size());
985   auto it = process_data.phased_snapshots.find(0);
986   ASSERT_TRUE(it != process_data.phased_snapshots.end());
987   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
988 
989   ASSERT_EQ(2u, process_data_phase.tasks.size());
990 
991   EXPECT_EQ(kFile, process_data_phase.tasks[0].birth.location.file_name);
992   EXPECT_EQ(kFunction,
993             process_data_phase.tasks[0].birth.location.function_name);
994   EXPECT_EQ(kLineNumber,
995             process_data_phase.tasks[0].birth.location.line_number);
996   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[0].birth.thread_name);
997   EXPECT_EQ(1, process_data_phase.tasks[0].death_data.count);
998   EXPECT_EQ(2, process_data_phase.tasks[0].death_data.run_duration_sum);
999   EXPECT_EQ(2, process_data_phase.tasks[0].death_data.run_duration_max);
1000   EXPECT_EQ(2, process_data_phase.tasks[0].death_data.run_duration_sample);
1001   EXPECT_EQ(4, process_data_phase.tasks[0].death_data.queue_duration_sum);
1002   EXPECT_EQ(4, process_data_phase.tasks[0].death_data.queue_duration_max);
1003   EXPECT_EQ(4, process_data_phase.tasks[0].death_data.queue_duration_sample);
1004   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[0].death_thread_name);
1005   EXPECT_EQ(kFile, process_data_phase.tasks[1].birth.location.file_name);
1006   EXPECT_EQ(kFunction,
1007             process_data_phase.tasks[1].birth.location.function_name);
1008   EXPECT_EQ(kSecondFakeLineNumber,
1009             process_data_phase.tasks[1].birth.location.line_number);
1010   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[1].birth.thread_name);
1011   EXPECT_EQ(1, process_data_phase.tasks[1].death_data.count);
1012   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.run_duration_sum);
1013   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.run_duration_max);
1014   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.run_duration_sample);
1015   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.queue_duration_sum);
1016   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.queue_duration_max);
1017   EXPECT_EQ(0, process_data_phase.tasks[1].death_data.queue_duration_sample);
1018   EXPECT_EQ(kStillAlive, process_data_phase.tasks[1].death_thread_name);
1019   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
1020 }
1021 
TEST_F(TrackedObjectsTest,TaskWithNestedExclusion)1022 TEST_F(TrackedObjectsTest, TaskWithNestedExclusion) {
1023   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
1024 
1025   const char kFunction[] = "TaskWithNestedExclusion";
1026   Location location(kFunction, kFile, kLineNumber, NULL);
1027   TallyABirth(location, kMainThreadName);
1028 
1029   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
1030   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
1031   // TrackingInfo will call TallyABirth() during construction.
1032   base::TrackingInfo pending_task(location, kDelayedStartTime);
1033   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
1034 
1035   SetTestTime(5);
1036   TaskStopwatch task_stopwatch;
1037   task_stopwatch.Start();
1038   {
1039     SetTestTime(8);
1040     TaskStopwatch exclusion_stopwatch;
1041     exclusion_stopwatch.Start();
1042     SetTestTime(12);
1043     exclusion_stopwatch.Stop();
1044   }
1045   SetTestTime(15);
1046   task_stopwatch.Stop();
1047 
1048   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, task_stopwatch);
1049 
1050   ProcessDataSnapshot process_data;
1051   ThreadData::Snapshot(0, &process_data);
1052   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName,
1053                           kMainThreadName, 1, 6, 4);
1054 }
1055 
TEST_F(TrackedObjectsTest,TaskWith2NestedExclusions)1056 TEST_F(TrackedObjectsTest, TaskWith2NestedExclusions) {
1057   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
1058 
1059   const char kFunction[] = "TaskWith2NestedExclusions";
1060   Location location(kFunction, kFile, kLineNumber, NULL);
1061   TallyABirth(location, kMainThreadName);
1062 
1063   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
1064   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
1065   // TrackingInfo will call TallyABirth() during construction.
1066   base::TrackingInfo pending_task(location, kDelayedStartTime);
1067   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
1068 
1069   SetTestTime(5);
1070   TaskStopwatch task_stopwatch;
1071   task_stopwatch.Start();
1072   {
1073     SetTestTime(8);
1074     TaskStopwatch exclusion_stopwatch;
1075     exclusion_stopwatch.Start();
1076     SetTestTime(12);
1077     exclusion_stopwatch.Stop();
1078 
1079     SetTestTime(15);
1080     TaskStopwatch exclusion_stopwatch2;
1081     exclusion_stopwatch2.Start();
1082     SetTestTime(18);
1083     exclusion_stopwatch2.Stop();
1084   }
1085   SetTestTime(25);
1086   task_stopwatch.Stop();
1087 
1088   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, task_stopwatch);
1089 
1090   ProcessDataSnapshot process_data;
1091   ThreadData::Snapshot(0, &process_data);
1092   ExpectSimpleProcessData(process_data, kFunction, kMainThreadName,
1093                           kMainThreadName, 1, 13, 4);
1094 }
1095 
TEST_F(TrackedObjectsTest,TaskWithNestedExclusionWithNestedTask)1096 TEST_F(TrackedObjectsTest, TaskWithNestedExclusionWithNestedTask) {
1097   ThreadData::InitializeAndSetTrackingStatus(ThreadData::PROFILING_ACTIVE);
1098 
1099   const char kFunction[] = "TaskWithNestedExclusionWithNestedTask";
1100   Location location(kFunction, kFile, kLineNumber, NULL);
1101 
1102   const int kSecondFakeLineNumber = 999;
1103 
1104   TallyABirth(location, kMainThreadName);
1105 
1106   const TrackedTime kTimePosted = TrackedTime::FromMilliseconds(1);
1107   const base::TimeTicks kDelayedStartTime = base::TimeTicks();
1108   // TrackingInfo will call TallyABirth() during construction.
1109   base::TrackingInfo pending_task(location, kDelayedStartTime);
1110   pending_task.time_posted = kTimePosted;  // Overwrite implied Now().
1111 
1112   SetTestTime(5);
1113   TaskStopwatch task_stopwatch;
1114   task_stopwatch.Start();
1115   {
1116     SetTestTime(8);
1117     TaskStopwatch exclusion_stopwatch;
1118     exclusion_stopwatch.Start();
1119     {
1120       Location second_location(kFunction, kFile, kSecondFakeLineNumber, NULL);
1121       base::TrackingInfo nested_task(second_location, kDelayedStartTime);
1122        // Overwrite implied Now().
1123       nested_task.time_posted = TrackedTime::FromMilliseconds(8);
1124       SetTestTime(9);
1125       TaskStopwatch nested_task_stopwatch;
1126       nested_task_stopwatch.Start();
1127       SetTestTime(11);
1128       nested_task_stopwatch.Stop();
1129       ThreadData::TallyRunOnNamedThreadIfTracking(
1130           nested_task, nested_task_stopwatch);
1131     }
1132     SetTestTime(12);
1133     exclusion_stopwatch.Stop();
1134   }
1135   SetTestTime(15);
1136   task_stopwatch.Stop();
1137 
1138   ThreadData::TallyRunOnNamedThreadIfTracking(pending_task, task_stopwatch);
1139 
1140   ProcessDataSnapshot process_data;
1141   ThreadData::Snapshot(0, &process_data);
1142 
1143   ASSERT_EQ(1u, process_data.phased_snapshots.size());
1144   auto it = process_data.phased_snapshots.find(0);
1145   ASSERT_TRUE(it != process_data.phased_snapshots.end());
1146   const ProcessDataPhaseSnapshot& process_data_phase = it->second;
1147 
1148   // The order in which the two task follow is platform-dependent.
1149   int t0 =
1150       (process_data_phase.tasks[0].birth.location.line_number == kLineNumber)
1151           ? 0
1152           : 1;
1153   int t1 = 1 - t0;
1154 
1155   ASSERT_EQ(2u, process_data_phase.tasks.size());
1156   EXPECT_EQ(kFile, process_data_phase.tasks[t0].birth.location.file_name);
1157   EXPECT_EQ(kFunction,
1158             process_data_phase.tasks[t0].birth.location.function_name);
1159   EXPECT_EQ(kLineNumber,
1160             process_data_phase.tasks[t0].birth.location.line_number);
1161   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[t0].birth.thread_name);
1162   EXPECT_EQ(1, process_data_phase.tasks[t0].death_data.count);
1163   EXPECT_EQ(6, process_data_phase.tasks[t0].death_data.run_duration_sum);
1164   EXPECT_EQ(6, process_data_phase.tasks[t0].death_data.run_duration_max);
1165   EXPECT_EQ(6, process_data_phase.tasks[t0].death_data.run_duration_sample);
1166   EXPECT_EQ(4, process_data_phase.tasks[t0].death_data.queue_duration_sum);
1167   EXPECT_EQ(4, process_data_phase.tasks[t0].death_data.queue_duration_max);
1168   EXPECT_EQ(4, process_data_phase.tasks[t0].death_data.queue_duration_sample);
1169   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[t0].death_thread_name);
1170   EXPECT_EQ(kFile, process_data_phase.tasks[t1].birth.location.file_name);
1171   EXPECT_EQ(kFunction,
1172             process_data_phase.tasks[t1].birth.location.function_name);
1173   EXPECT_EQ(kSecondFakeLineNumber,
1174             process_data_phase.tasks[t1].birth.location.line_number);
1175   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[t1].birth.thread_name);
1176   EXPECT_EQ(1, process_data_phase.tasks[t1].death_data.count);
1177   EXPECT_EQ(2, process_data_phase.tasks[t1].death_data.run_duration_sum);
1178   EXPECT_EQ(2, process_data_phase.tasks[t1].death_data.run_duration_max);
1179   EXPECT_EQ(2, process_data_phase.tasks[t1].death_data.run_duration_sample);
1180   EXPECT_EQ(1, process_data_phase.tasks[t1].death_data.queue_duration_sum);
1181   EXPECT_EQ(1, process_data_phase.tasks[t1].death_data.queue_duration_max);
1182   EXPECT_EQ(1, process_data_phase.tasks[t1].death_data.queue_duration_sample);
1183   EXPECT_EQ(kMainThreadName, process_data_phase.tasks[t1].death_thread_name);
1184   EXPECT_EQ(base::GetCurrentProcId(), process_data.process_id);
1185 }
1186 
1187 }  // namespace tracked_objects
1188