| OLD | NEW | 
|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be | 
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. | 
| 4 | 4 | 
| 5 #include "components/metrics/profiler/tracking_synchronizer.h" | 5 #include "components/metrics/profiler/tracking_synchronizer.h" | 
| 6 | 6 | 
| 7 #include "base/bind.h" | 7 #include "base/bind.h" | 
| 8 #include "base/metrics/histogram.h" | 8 #include "base/metrics/histogram.h" | 
| 9 #include "base/threading/thread.h" | 9 #include "base/threading/thread.h" | 
| 10 #include "base/tracked_objects.h" | 10 #include "base/tracked_objects.h" | 
| 11 #include "components/metrics/profiler/tracking_synchronizer_observer.h" | 11 #include "components/metrics/profiler/tracking_synchronizer_observer.h" | 
|  | 12 #include "components/variations/variations_associated_data.h" | 
| 12 #include "content/public/browser/browser_thread.h" | 13 #include "content/public/browser/browser_thread.h" | 
| 13 #include "content/public/browser/profiler_controller.h" | 14 #include "content/public/browser/profiler_controller.h" | 
| 14 #include "content/public/common/process_type.h" | 15 #include "content/public/common/process_type.h" | 
| 15 | 16 | 
| 16 using base::TimeTicks; | 17 using base::TimeTicks; | 
| 17 using content::BrowserThread; | 18 using content::BrowserThread; | 
| 18 | 19 | 
| 19 namespace metrics { | 20 namespace metrics { | 
| 20 | 21 | 
| 21 namespace { | 22 namespace { | 
| (...skipping 150 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 172 }; | 173 }; | 
| 173 | 174 | 
| 174 // static | 175 // static | 
| 175 base::LazyInstance | 176 base::LazyInstance | 
| 176     <TrackingSynchronizer::RequestContext::RequestContextMap>::Leaky | 177     <TrackingSynchronizer::RequestContext::RequestContextMap>::Leaky | 
| 177         TrackingSynchronizer::RequestContext::outstanding_requests_ = | 178         TrackingSynchronizer::RequestContext::outstanding_requests_ = | 
| 178             LAZY_INSTANCE_INITIALIZER; | 179             LAZY_INSTANCE_INITIALIZER; | 
| 179 | 180 | 
| 180 // TrackingSynchronizer methods and members. | 181 // TrackingSynchronizer methods and members. | 
| 181 | 182 | 
| 182 TrackingSynchronizer::TrackingSynchronizer() | 183 TrackingSynchronizer::TrackingSynchronizer(base::TimeTicks now) | 
| 183     : last_used_sequence_number_(kNeverUsableSequenceNumber) { | 184     : last_used_sequence_number_(kNeverUsableSequenceNumber), start_time_(now) { | 
| 184   DCHECK(!g_tracking_synchronizer); | 185   DCHECK(!g_tracking_synchronizer); | 
| 185   g_tracking_synchronizer = this; | 186   g_tracking_synchronizer = this; | 
|  | 187   phase_start_times_.push_back(now); | 
|  | 188 | 
|  | 189 #if !defined(OS_IOS) | 
|  | 190   // TODO: This ifdef and other ifdefs for OS_IOS in this file are only | 
|  | 191   // short-term hacks to make this compile on iOS, and the proper solution is to | 
|  | 192   // refactor to remove content dependencies from shared code. | 
|  | 193   // See crbug/472210. | 
| 186   content::ProfilerController::GetInstance()->Register(this); | 194   content::ProfilerController::GetInstance()->Register(this); | 
|  | 195 #endif | 
| 187 } | 196 } | 
| 188 | 197 | 
| 189 TrackingSynchronizer::~TrackingSynchronizer() { | 198 TrackingSynchronizer::~TrackingSynchronizer() { | 
|  | 199 #if !defined(OS_IOS) | 
| 190   content::ProfilerController::GetInstance()->Unregister(this); | 200   content::ProfilerController::GetInstance()->Unregister(this); | 
|  | 201 #endif | 
| 191 | 202 | 
| 192   // Just in case we have any pending tasks, clear them out. | 203   // Just in case we have any pending tasks, clear them out. | 
| 193   RequestContext::OnShutdown(); | 204   RequestContext::OnShutdown(); | 
| 194 | 205 | 
| 195   g_tracking_synchronizer = NULL; | 206   g_tracking_synchronizer = NULL; | 
| 196 } | 207 } | 
| 197 | 208 | 
| 198 // static | 209 // static | 
| 199 void TrackingSynchronizer::FetchProfilerDataAsynchronously( | 210 void TrackingSynchronizer::FetchProfilerDataAsynchronously( | 
| 200     const base::WeakPtr<TrackingSynchronizerObserver>& callback_object) { | 211     const base::WeakPtr<TrackingSynchronizerObserver>& callback_object) { | 
| (...skipping 24 matching lines...) Expand all  Loading... | 
| 225   if (!request) | 236   if (!request) | 
| 226     return; | 237     return; | 
| 227   request->AddProcessesPending(pending_processes); | 238   request->AddProcessesPending(pending_processes); | 
| 228   request->SetReceivedProcessGroupCount(end); | 239   request->SetReceivedProcessGroupCount(end); | 
| 229   request->DeleteIfAllDone(); | 240   request->DeleteIfAllDone(); | 
| 230 } | 241 } | 
| 231 | 242 | 
| 232 void TrackingSynchronizer::OnProfilerDataCollected( | 243 void TrackingSynchronizer::OnProfilerDataCollected( | 
| 233     int sequence_number, | 244     int sequence_number, | 
| 234     const tracked_objects::ProcessDataSnapshot& profiler_data, | 245     const tracked_objects::ProcessDataSnapshot& profiler_data, | 
| 235     int process_type) { | 246     content::ProcessType process_type) { | 
| 236   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 247   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 
| 237   DecrementPendingProcessesAndSendData(sequence_number, profiler_data, | 248   DecrementPendingProcessesAndSendData(sequence_number, profiler_data, | 
| 238                                        process_type); | 249                                        process_type); | 
| 239 } | 250 } | 
| 240 | 251 | 
| 241 int TrackingSynchronizer::RegisterAndNotifyAllProcesses( | 252 int TrackingSynchronizer::RegisterAndNotifyAllProcesses( | 
| 242     const base::WeakPtr<TrackingSynchronizerObserver>& callback_object) { | 253     const base::WeakPtr<TrackingSynchronizerObserver>& callback_object) { | 
| 243   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 254   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 
| 244 | 255 | 
| 245   int sequence_number = GetNextAvailableSequenceNumber(); | 256   int sequence_number = GetNextAvailableSequenceNumber(); | 
| 246 | 257 | 
| 247   RequestContext* request = | 258   RequestContext* request = | 
| 248       RequestContext::Register(sequence_number, callback_object); | 259       RequestContext::Register(sequence_number, callback_object); | 
| 249 | 260 | 
| 250   // Increment pending process count for sending browser's profiler data. | 261   // Increment pending process count for sending browser's profiler data. | 
| 251   request->IncrementProcessesPending(); | 262   request->IncrementProcessesPending(); | 
| 252 | 263 | 
|  | 264 #if !defined(OS_IOS) | 
| 253   // Get profiler data from renderer and browser child processes. | 265   // Get profiler data from renderer and browser child processes. | 
| 254   content::ProfilerController::GetInstance()->GetProfilerData(sequence_number); | 266   content::ProfilerController::GetInstance()->GetProfilerData(sequence_number); | 
|  | 267 #endif | 
| 255 | 268 | 
| 256   // Send profiler_data from browser process. | 269   // Send process data snapshot from browser process. | 
| 257   tracked_objects::ProcessDataSnapshot process_data; | 270   tracked_objects::ProcessDataSnapshot process_data_snapshot; | 
| 258   tracked_objects::ThreadData::Snapshot(&process_data); | 271   tracked_objects::ThreadData::Snapshot(&process_data_snapshot); | 
| 259   DecrementPendingProcessesAndSendData(sequence_number, process_data, | 272   DecrementPendingProcessesAndSendData(sequence_number, process_data_snapshot, | 
| 260                                        content::PROCESS_TYPE_BROWSER); | 273                                        content::PROCESS_TYPE_BROWSER); | 
| 261 | 274 | 
| 262   return sequence_number; | 275   return sequence_number; | 
| 263 } | 276 } | 
| 264 | 277 | 
|  | 278 void TrackingSynchronizer::SendData( | 
|  | 279     const tracked_objects::ProcessDataSnapshot& profiler_data, | 
|  | 280     content::ProcessType process_type, | 
|  | 281     base::TimeTicks now, | 
|  | 282     TrackingSynchronizerObserver* observer) const { | 
|  | 283   // We are going to loop though past profiling phases and notify the request | 
|  | 284   // about each phase that is contained in profiler_data. past_events | 
|  | 285   // will track the set of past profiling events as we go. | 
|  | 286   ProfilerEvents past_events; | 
|  | 287 | 
|  | 288   // Go through all completed phases, and through the current one. The current | 
|  | 289   // one is not in phase_completion_events_sequence_, but note the <= | 
|  | 290   // comparison. | 
|  | 291   for (size_t phase = 0; phase <= phase_completion_events_sequence_.size(); | 
|  | 292        ++phase) { | 
|  | 293     auto it = profiler_data.phased_process_data_snapshots.find(phase); | 
|  | 294 | 
|  | 295     if (it != profiler_data.phased_process_data_snapshots.end()) { | 
|  | 296       // If the phase is contained in the received snapshot, notify the | 
|  | 297       // request. | 
|  | 298       const base::TimeDelta phase_start = | 
|  | 299           phase_start_times_[phase] - start_time_; | 
|  | 300       const base::TimeDelta phase_finish = | 
|  | 301           (phase + 1 < phase_start_times_.size() ? phase_start_times_[phase + 1] | 
|  | 302                                                  : now) - | 
|  | 303           start_time_; | 
|  | 304       observer->ReceivedProfilerData(it->second, profiler_data.process_id, | 
|  | 305                                      process_type, phase, phase_start, | 
|  | 306                                      phase_finish, past_events); | 
|  | 307     } | 
|  | 308 | 
|  | 309     if (phase < phase_completion_events_sequence_.size()) { | 
|  | 310       past_events.push_back(phase_completion_events_sequence_[phase]); | 
|  | 311     } | 
|  | 312   } | 
|  | 313 } | 
|  | 314 | 
| 265 void TrackingSynchronizer::DecrementPendingProcessesAndSendData( | 315 void TrackingSynchronizer::DecrementPendingProcessesAndSendData( | 
| 266     int sequence_number, | 316     int sequence_number, | 
| 267     const tracked_objects::ProcessDataSnapshot& profiler_data, | 317     const tracked_objects::ProcessDataSnapshot& profiler_data, | 
| 268     int process_type) { | 318     content::ProcessType process_type) { | 
| 269   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 319   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 
| 270 | 320 | 
| 271   RequestContext* request = RequestContext::GetRequestContext(sequence_number); | 321   RequestContext* request = RequestContext::GetRequestContext(sequence_number); | 
| 272   if (!request) | 322   if (!request) | 
| 273     return; | 323     return; | 
| 274 | 324 | 
| 275   if (request->callback_object_.get()) { | 325   TrackingSynchronizerObserver* observer = request->callback_object_.get(); | 
| 276     request->callback_object_ | 326   if (observer) | 
| 277         ->ReceivedProfilerData(profiler_data, process_type); | 327     SendData(profiler_data, process_type, base::TimeTicks::Now(), observer); | 
| 278   } |  | 
| 279 | 328 | 
| 280   // Delete request if we have heard back from all child processes. | 329   // Delete request if we have heard back from all child processes. | 
| 281   request->DecrementProcessesPending(); | 330   request->DecrementProcessesPending(); | 
| 282   request->DeleteIfAllDone(); | 331   request->DeleteIfAllDone(); | 
| 283 } | 332 } | 
| 284 | 333 | 
| 285 int TrackingSynchronizer::GetNextAvailableSequenceNumber() { | 334 int TrackingSynchronizer::GetNextAvailableSequenceNumber() { | 
| 286   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 335   DCHECK_CURRENTLY_ON(BrowserThread::UI); | 
| 287 | 336 | 
| 288   ++last_used_sequence_number_; | 337   ++last_used_sequence_number_; | 
| 289 | 338 | 
| 290   // Watch out for wrapping to a negative number. | 339   // Watch out for wrapping to a negative number. | 
| 291   if (last_used_sequence_number_ < 0) | 340   if (last_used_sequence_number_ < 0) | 
| 292     last_used_sequence_number_ = 1; | 341     last_used_sequence_number_ = 1; | 
| 293   return last_used_sequence_number_; | 342   return last_used_sequence_number_; | 
| 294 } | 343 } | 
| 295 | 344 | 
| 296 }  // namespace metrics | 345 }  // namespace metrics | 
| OLD | NEW | 
|---|