• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 #include "content/browser/profiler_controller_impl.h"
6 
7 #include "base/bind.h"
8 #include "base/command_line.h"
9 #include "base/tracked_objects.h"
10 #include "content/common/child_process_messages.h"
11 #include "content/public/browser/browser_child_process_host_iterator.h"
12 #include "content/public/browser/browser_thread.h"
13 #include "content/public/browser/child_process_data.h"
14 #include "content/public/browser/profiler_subscriber.h"
15 #include "content/public/browser/render_process_host.h"
16 #include "content/public/common/content_switches.h"
17 
18 namespace content {
19 
GetInstance()20 ProfilerController* ProfilerController::GetInstance() {
21   return ProfilerControllerImpl::GetInstance();
22 }
23 
GetInstance()24 ProfilerControllerImpl* ProfilerControllerImpl::GetInstance() {
25   return Singleton<ProfilerControllerImpl>::get();
26 }
27 
ProfilerControllerImpl()28 ProfilerControllerImpl::ProfilerControllerImpl() : subscriber_(NULL) {
29 }
30 
~ProfilerControllerImpl()31 ProfilerControllerImpl::~ProfilerControllerImpl() {
32 }
33 
OnPendingProcesses(int sequence_number,int pending_processes,bool end)34 void ProfilerControllerImpl::OnPendingProcesses(int sequence_number,
35                                                 int pending_processes,
36                                                 bool end) {
37   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
38   if (subscriber_)
39     subscriber_->OnPendingProcesses(sequence_number, pending_processes, end);
40 }
41 
OnProfilerDataCollected(int sequence_number,const tracked_objects::ProcessDataSnapshot & profiler_data,int process_type)42 void ProfilerControllerImpl::OnProfilerDataCollected(
43     int sequence_number,
44     const tracked_objects::ProcessDataSnapshot& profiler_data,
45     int process_type) {
46   if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) {
47     BrowserThread::PostTask(
48         BrowserThread::UI, FROM_HERE,
49         base::Bind(&ProfilerControllerImpl::OnProfilerDataCollected,
50                    base::Unretained(this),
51                    sequence_number,
52                    profiler_data,
53                    process_type));
54     return;
55   }
56 
57   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
58   if (subscriber_) {
59     subscriber_->OnProfilerDataCollected(sequence_number, profiler_data,
60                                          process_type);
61   }
62 }
63 
Register(ProfilerSubscriber * subscriber)64 void ProfilerControllerImpl::Register(ProfilerSubscriber* subscriber) {
65   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
66   DCHECK(!subscriber_);
67   subscriber_ = subscriber;
68 }
69 
Unregister(const ProfilerSubscriber * subscriber)70 void ProfilerControllerImpl::Unregister(const ProfilerSubscriber* subscriber) {
71   DCHECK_EQ(subscriber_, subscriber);
72   subscriber_ = NULL;
73 }
74 
GetProfilerDataFromChildProcesses(int sequence_number)75 void ProfilerControllerImpl::GetProfilerDataFromChildProcesses(
76     int sequence_number) {
77   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
78 
79   int pending_processes = 0;
80   for (BrowserChildProcessHostIterator iter; !iter.Done(); ++iter) {
81     // Skips requesting profiler data from the "GPU Process" if we are using in
82     // process GPU. Those stats should be in the Browser-process's GPU thread.
83     if (iter.GetData().process_type == PROCESS_TYPE_GPU &&
84         CommandLine::ForCurrentProcess()->HasSwitch(switches::kInProcessGPU)) {
85       continue;
86     }
87 
88     ++pending_processes;
89     if (!iter.Send(new ChildProcessMsg_GetChildProfilerData(sequence_number)))
90       --pending_processes;
91   }
92 
93   BrowserThread::PostTask(
94       BrowserThread::UI,
95       FROM_HERE,
96       base::Bind(
97           &ProfilerControllerImpl::OnPendingProcesses,
98           base::Unretained(this),
99           sequence_number,
100           pending_processes,
101           true));
102 }
103 
GetProfilerData(int sequence_number)104 void ProfilerControllerImpl::GetProfilerData(int sequence_number) {
105   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
106 
107   int pending_processes = 0;
108   for (RenderProcessHost::iterator it(RenderProcessHost::AllHostsIterator());
109        !it.IsAtEnd(); it.Advance()) {
110     ++pending_processes;
111     if (!it.GetCurrentValue()->Send(
112             new ChildProcessMsg_GetChildProfilerData(sequence_number))) {
113       --pending_processes;
114     }
115   }
116   OnPendingProcesses(sequence_number, pending_processes, false);
117 
118   BrowserThread::PostTask(
119       BrowserThread::IO,
120       FROM_HERE,
121       base::Bind(&ProfilerControllerImpl::GetProfilerDataFromChildProcesses,
122                  base::Unretained(this),
123                  sequence_number));
124 }
125 
126 }  // namespace content
127