afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 1 | // Copyright 2015 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 | |
avi | 24d693f | 2016-08-06 18:03:52 | [diff] [blame] | 5 | #include "chrome/browser/task_manager/task_manager_interface.h" |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 6 | |
afakhry | 17406972 | 2016-05-24 19:16:16 | [diff] [blame] | 7 | #include "chrome/browser/browser_process.h" |
avi | 24d693f | 2016-08-06 18:03:52 | [diff] [blame] | 8 | #include "chrome/browser/task_manager/sampling/task_manager_impl.h" |
| 9 | #include "chrome/browser/task_manager/sampling/task_manager_io_thread_helper.h" |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 10 | #include "chrome/common/chrome_switches.h" |
afakhry | 17406972 | 2016-05-24 19:16:16 | [diff] [blame] | 11 | #include "chrome/common/pref_names.h" |
| 12 | #include "components/prefs/pref_registry_simple.h" |
| 13 | #include "components/prefs/pref_service.h" |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 14 | #include "content/public/browser/browser_thread.h" |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 15 | #include "content/public/browser/resource_request_info.h" |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 16 | |
afakhry | 17406972 | 2016-05-24 19:16:16 | [diff] [blame] | 17 | #if defined(OS_MACOSX) |
| 18 | #include "chrome/browser/ui/browser_dialogs.h" |
| 19 | #endif // defined(OS_MACOSX) |
| 20 | |
avi | 24d693f | 2016-08-06 18:03:52 | [diff] [blame] | 21 | namespace task_manager { |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 22 | |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 23 | namespace { |
| 24 | BytesTransferredKey KeyForRequest(const net::URLRequest& request) { |
| 25 | // Only net::URLRequestJob instances created by the ResourceDispatcherHost |
| 26 | // have an associated ResourceRequestInfo and a render frame associated. |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 27 | const content::ResourceRequestInfo* info = |
| 28 | content::ResourceRequestInfo::ForRequest(&request); |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 29 | |
Nick Carter | 074b520 | 2017-11-29 00:44:25 | [diff] [blame] | 30 | // Requests without ResourceRequestInfo are attributed to the browser process. |
| 31 | if (!info) |
| 32 | return {content::ChildProcessHost::kInvalidUniqueID, MSG_ROUTING_NONE}; |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 33 | |
Nick Carter | 074b520 | 2017-11-29 00:44:25 | [diff] [blame] | 34 | // Requests from PPAPI instances are proxied through the renderer, and specify |
| 35 | // the plugin_child_id of the plugin process. |
| 36 | if (info->GetPluginChildID() != content::ChildProcessHost::kInvalidUniqueID) |
| 37 | return {info->GetPluginChildID(), MSG_ROUTING_NONE}; |
| 38 | |
| 39 | // Other requests are associated with the child process (and frame, if |
| 40 | // originating from a renderer process). |
| 41 | return {info->GetChildID(), info->GetRenderFrameID()}; |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 42 | } |
| 43 | } // namespace |
| 44 | |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 45 | // static |
afakhry | 17406972 | 2016-05-24 19:16:16 | [diff] [blame] | 46 | void TaskManagerInterface::RegisterPrefs(PrefRegistrySimple* registry) { |
| 47 | registry->RegisterDictionaryPref(prefs::kTaskManagerWindowPlacement); |
| 48 | registry->RegisterDictionaryPref(prefs::kTaskManagerColumnVisibility); |
| 49 | registry->RegisterBooleanPref(prefs::kTaskManagerEndProcessEnabled, true); |
| 50 | } |
| 51 | |
| 52 | // static |
| 53 | bool TaskManagerInterface::IsEndProcessEnabled() { |
| 54 | PrefService* state = g_browser_process->local_state(); |
| 55 | return !state || state->GetBoolean(prefs::kTaskManagerEndProcessEnabled); |
| 56 | } |
| 57 | |
afakhry | 17406972 | 2016-05-24 19:16:16 | [diff] [blame] | 58 | // static |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 59 | TaskManagerInterface* TaskManagerInterface::GetTaskManager() { |
| 60 | DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| 61 | |
| 62 | return TaskManagerImpl::GetInstance(); |
| 63 | } |
| 64 | |
| 65 | // static |
| 66 | void TaskManagerInterface::OnRawBytesRead(const net::URLRequest& request, |
sclittle | ce72c48 | 2015-08-24 20:20:59 | [diff] [blame] | 67 | int64_t bytes_read) { |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 68 | DCHECK_CURRENTLY_ON(content::BrowserThread::IO); |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 69 | BytesTransferredKey key = KeyForRequest(request); |
| 70 | TaskManagerIoThreadHelper::OnRawBytesTransferred(key, bytes_read, |
| 71 | 0 /*bytes_sent*/); |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 72 | } |
| 73 | |
cburn | e9d2f361 | 2017-06-20 22:15:03 | [diff] [blame] | 74 | // static |
| 75 | void TaskManagerInterface::OnRawBytesSent(const net::URLRequest& request, |
| 76 | int64_t bytes_sent) { |
| 77 | DCHECK_CURRENTLY_ON(content::BrowserThread::IO); |
cburn | c0a5953 | 2017-07-06 17:34:39 | [diff] [blame] | 78 | BytesTransferredKey key = KeyForRequest(request); |
| 79 | TaskManagerIoThreadHelper::OnRawBytesTransferred(key, 0 /*bytes_read*/, |
| 80 | bytes_sent); |
cburn | e9d2f361 | 2017-06-20 22:15:03 | [diff] [blame] | 81 | } |
| 82 | |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 83 | void TaskManagerInterface::AddObserver(TaskManagerObserver* observer) { |
| 84 | observers_.AddObserver(observer); |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 85 | observer->observed_task_manager_ = this; |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 86 | |
| 87 | ResourceFlagsAdded(observer->desired_resources_flags()); |
| 88 | |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 89 | base::TimeDelta current_refresh_time = GetCurrentRefreshTime(); |
| 90 | if (current_refresh_time == base::TimeDelta::Max()) { |
| 91 | // This is the first observer to be added. Start updating. |
| 92 | StartUpdating(); |
| 93 | } |
| 94 | |
| 95 | if (observer->desired_refresh_time() > current_refresh_time) |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 96 | return; |
| 97 | |
| 98 | // Reached here, then this is EITHER (not the first observer to be added AND |
| 99 | // it requires a more frequent refresh rate) OR (it's the very first observer |
| 100 | // to be added). |
| 101 | // Reset the refresh timer. |
| 102 | ScheduleRefresh(observer->desired_refresh_time()); |
| 103 | } |
| 104 | |
| 105 | void TaskManagerInterface::RemoveObserver(TaskManagerObserver* observer) { |
| 106 | observers_.RemoveObserver(observer); |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 107 | observer->observed_task_manager_ = nullptr; |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 108 | |
| 109 | // Recalculate the minimum refresh rate and the enabled resource flags. |
avi | 664c07b | 2015-12-26 02:18:31 | [diff] [blame] | 110 | int64_t flags = 0; |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 111 | base::TimeDelta min_time = base::TimeDelta::Max(); |
dcheng | ea68766 | 2016-10-13 16:59:52 | [diff] [blame] | 112 | for (auto& observer : observers_) { |
| 113 | if (observer.desired_refresh_time() < min_time) |
| 114 | min_time = observer.desired_refresh_time(); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 115 | |
dcheng | ea68766 | 2016-10-13 16:59:52 | [diff] [blame] | 116 | flags |= observer.desired_resources_flags(); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | if (min_time == base::TimeDelta::Max()) { |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 120 | // This is the last observer to be removed. Stop updating. |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 121 | SetEnabledResourceFlags(0); |
| 122 | refresh_timer_->Stop(); |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 123 | StopUpdating(); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 124 | } else { |
| 125 | SetEnabledResourceFlags(flags); |
| 126 | ScheduleRefresh(min_time); |
| 127 | } |
| 128 | } |
| 129 | |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 130 | void TaskManagerInterface::RecalculateRefreshFlags() { |
avi | 664c07b | 2015-12-26 02:18:31 | [diff] [blame] | 131 | int64_t flags = 0; |
dcheng | ea68766 | 2016-10-13 16:59:52 | [diff] [blame] | 132 | for (auto& observer : observers_) |
| 133 | flags |= observer.desired_resources_flags(); |
afakhry | 0501503 | 2015-08-14 01:09:56 | [diff] [blame] | 134 | |
| 135 | SetEnabledResourceFlags(flags); |
| 136 | } |
| 137 | |
afakhry | 9824183 | 2016-03-11 19:27:47 | [diff] [blame] | 138 | bool TaskManagerInterface::IsResourceRefreshEnabled(RefreshType type) const { |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 139 | return (enabled_resources_flags_ & type) != 0; |
| 140 | } |
| 141 | |
| 142 | TaskManagerInterface::TaskManagerInterface() |
thestig | 4949a1e | 2016-03-29 00:53:52 | [diff] [blame] | 143 | : refresh_timer_(new base::Timer(true, true)), |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 144 | enabled_resources_flags_(0) { |
| 145 | } |
| 146 | |
| 147 | TaskManagerInterface::~TaskManagerInterface() { |
| 148 | } |
| 149 | |
| 150 | void TaskManagerInterface::NotifyObserversOnTaskAdded(TaskId id) { |
ericwilligers | 58b0e16 | 2016-10-21 07:15:56 | [diff] [blame] | 151 | for (TaskManagerObserver& observer : observers_) |
| 152 | observer.OnTaskAdded(id); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 153 | } |
| 154 | |
| 155 | void TaskManagerInterface::NotifyObserversOnTaskToBeRemoved(TaskId id) { |
ericwilligers | 58b0e16 | 2016-10-21 07:15:56 | [diff] [blame] | 156 | for (TaskManagerObserver& observer : observers_) |
| 157 | observer.OnTaskToBeRemoved(id); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 158 | } |
| 159 | |
| 160 | void TaskManagerInterface::NotifyObserversOnRefresh( |
| 161 | const TaskIdList& task_ids) { |
ericwilligers | 58b0e16 | 2016-10-21 07:15:56 | [diff] [blame] | 162 | for (TaskManagerObserver& observer : observers_) |
| 163 | observer.OnTasksRefreshed(task_ids); |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 164 | } |
| 165 | |
afakhry | 9824183 | 2016-03-11 19:27:47 | [diff] [blame] | 166 | void TaskManagerInterface::NotifyObserversOnRefreshWithBackgroundCalculations( |
| 167 | const TaskIdList& task_ids) { |
ericwilligers | 58b0e16 | 2016-10-21 07:15:56 | [diff] [blame] | 168 | for (TaskManagerObserver& observer : observers_) |
| 169 | observer.OnTasksRefreshedWithBackgroundCalculations(task_ids); |
afakhry | 9824183 | 2016-03-11 19:27:47 | [diff] [blame] | 170 | } |
| 171 | |
| 172 | void TaskManagerInterface::NotifyObserversOnTaskUnresponsive(TaskId id) { |
ericwilligers | 58b0e16 | 2016-10-21 07:15:56 | [diff] [blame] | 173 | for (TaskManagerObserver& observer : observers_) |
| 174 | observer.OnTaskUnresponsive(id); |
afakhry | 9824183 | 2016-03-11 19:27:47 | [diff] [blame] | 175 | } |
| 176 | |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 177 | base::TimeDelta TaskManagerInterface::GetCurrentRefreshTime() const { |
| 178 | return refresh_timer_->IsRunning() ? refresh_timer_->GetCurrentDelay() |
| 179 | : base::TimeDelta::Max(); |
| 180 | } |
| 181 | |
avi | 664c07b | 2015-12-26 02:18:31 | [diff] [blame] | 182 | void TaskManagerInterface::ResourceFlagsAdded(int64_t flags) { |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 183 | enabled_resources_flags_ |= flags; |
| 184 | } |
| 185 | |
avi | 664c07b | 2015-12-26 02:18:31 | [diff] [blame] | 186 | void TaskManagerInterface::SetEnabledResourceFlags(int64_t flags) { |
afakhry | 5448575 | 2015-07-06 17:39:16 | [diff] [blame] | 187 | enabled_resources_flags_ = flags; |
| 188 | } |
| 189 | |
| 190 | void TaskManagerInterface::ScheduleRefresh(base::TimeDelta refresh_time) { |
| 191 | refresh_timer_->Start(FROM_HERE, |
| 192 | refresh_time, |
| 193 | base::Bind(&TaskManagerInterface::Refresh, |
| 194 | base::Unretained(this))); |
| 195 | } |
| 196 | |
avi | 24d693f | 2016-08-06 18:03:52 | [diff] [blame] | 197 | } // namespace task_manager |