| OLD | NEW |
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 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 "content/browser/memory/memory_coordinator_impl.h" | 5 #include "content/browser/memory/memory_coordinator_impl.h" |
| 6 | 6 |
| 7 #include "base/threading/thread_task_runner_handle.h" | 7 #include "base/threading/thread_task_runner_handle.h" |
| 8 #include "content/browser/memory/memory_monitor.h" | 8 #include "content/browser/memory/memory_monitor.h" |
| 9 #include "content/public/browser/notification_service.h" |
| 10 #include "content/public/browser/notification_types.h" |
| 11 #include "content/public/browser/render_process_host.h" |
| 12 #include "content/public/browser/render_widget_host.h" |
| 9 #include "content/public/common/content_features.h" | 13 #include "content/public/common/content_features.h" |
| 10 | 14 |
| 11 namespace content { | 15 namespace content { |
| 12 | 16 |
| 13 namespace { | 17 namespace { |
| 14 | 18 |
| 15 // A expected renderer size. These values come from the median of appropriate | 19 // A expected renderer size. These values come from the median of appropriate |
| 16 // UMA stats. | 20 // UMA stats. |
| 17 #if defined(OS_ANDROID) || defined(OS_IOS) | 21 #if defined(OS_ANDROID) || defined(OS_IOS) |
| 18 const int kDefaultExpectedRendererSizeMB = 40; | 22 const int kDefaultExpectedRendererSizeMB = 40; |
| (...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 87 monitoring_interval_ = | 91 monitoring_interval_ = |
| 88 base::TimeDelta::FromSeconds(kDefaultMonitoringIntervalSeconds); | 92 base::TimeDelta::FromSeconds(kDefaultMonitoringIntervalSeconds); |
| 89 } | 93 } |
| 90 | 94 |
| 91 MemoryCoordinatorImpl::~MemoryCoordinatorImpl() {} | 95 MemoryCoordinatorImpl::~MemoryCoordinatorImpl() {} |
| 92 | 96 |
| 93 void MemoryCoordinatorImpl::Start() { | 97 void MemoryCoordinatorImpl::Start() { |
| 94 DCHECK(CalledOnValidThread()); | 98 DCHECK(CalledOnValidThread()); |
| 95 DCHECK(last_state_change_.is_null()); | 99 DCHECK(last_state_change_.is_null()); |
| 96 DCHECK(ValidateParameters()); | 100 DCHECK(ValidateParameters()); |
| 101 |
| 102 notification_registrar_.Add( |
| 103 this, NOTIFICATION_RENDER_WIDGET_VISIBILITY_CHANGED, |
| 104 NotificationService::AllBrowserContextsAndSources()); |
| 97 ScheduleUpdateState(base::TimeDelta()); | 105 ScheduleUpdateState(base::TimeDelta()); |
| 98 } | 106 } |
| 99 | 107 |
| 100 void MemoryCoordinatorImpl::OnChildAdded(int render_process_id) { | 108 void MemoryCoordinatorImpl::OnChildAdded(int render_process_id) { |
| 101 // Populate the global state as an initial state of a newly created process. | 109 // Populate the global state as an initial state of a newly created process. |
| 102 SetMemoryState(render_process_id, ToMojomMemoryState(current_state_)); | 110 SetMemoryState(render_process_id, ToMojomMemoryState(current_state_)); |
| 103 } | 111 } |
| 104 | 112 |
| 105 base::MemoryState MemoryCoordinatorImpl::GetCurrentMemoryState() const { | 113 base::MemoryState MemoryCoordinatorImpl::GetCurrentMemoryState() const { |
| 106 return current_state_; | 114 return current_state_; |
| 107 } | 115 } |
| 108 | 116 |
| 117 void MemoryCoordinatorImpl::Observe(int type, |
| 118 const NotificationSource& source, |
| 119 const NotificationDetails& details) { |
| 120 DCHECK(type == NOTIFICATION_RENDER_WIDGET_VISIBILITY_CHANGED); |
| 121 RenderWidgetHost* render_widget_host = Source<RenderWidgetHost>(source).ptr(); |
| 122 RenderProcessHost* process = render_widget_host->GetProcess(); |
| 123 if (!process) |
| 124 return; |
| 125 auto iter = children().find(process->GetID()); |
| 126 if (iter == children().end()) |
| 127 return; |
| 128 bool is_visible = *Details<bool>(details).ptr(); |
| 129 // We don't throttle/suspend a visible renderer for now. |
| 130 auto new_state = is_visible ? mojom::MemoryState::NORMAL |
| 131 : ToMojomMemoryState(current_state_); |
| 132 SetMemoryState(iter->first, new_state); |
| 133 } |
| 134 |
| 109 base::MemoryState MemoryCoordinatorImpl::CalculateNextState() { | 135 base::MemoryState MemoryCoordinatorImpl::CalculateNextState() { |
| 110 using MemoryState = base::MemoryState; | 136 using MemoryState = base::MemoryState; |
| 111 | 137 |
| 112 int available = memory_monitor_->GetFreeMemoryUntilCriticalMB(); | 138 int available = memory_monitor_->GetFreeMemoryUntilCriticalMB(); |
| 113 if (available <= 0) | 139 if (available <= 0) |
| 114 return MemoryState::SUSPENDED; | 140 return MemoryState::SUSPENDED; |
| 115 | 141 |
| 116 int expected_renderer_count = available / expected_renderer_size_; | 142 int expected_renderer_count = available / expected_renderer_size_; |
| 117 | 143 |
| 118 switch (current_state_) { | 144 switch (current_state_) { |
| (...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 167 // TODO(bashi): Maybe worth considering another state for the browser. | 193 // TODO(bashi): Maybe worth considering another state for the browser. |
| 168 auto state = current_state_ == MemoryState::SUSPENDED ? MemoryState::THROTTLED | 194 auto state = current_state_ == MemoryState::SUSPENDED ? MemoryState::THROTTLED |
| 169 : current_state_; | 195 : current_state_; |
| 170 base::MemoryCoordinatorClientRegistry::GetInstance()->Notify(state); | 196 base::MemoryCoordinatorClientRegistry::GetInstance()->Notify(state); |
| 171 } | 197 } |
| 172 | 198 |
| 173 void MemoryCoordinatorImpl::NotifyStateToChildren() { | 199 void MemoryCoordinatorImpl::NotifyStateToChildren() { |
| 174 auto mojo_state = ToMojomMemoryState(current_state_); | 200 auto mojo_state = ToMojomMemoryState(current_state_); |
| 175 // It's OK to call SetMemoryState() unconditionally because it checks whether | 201 // It's OK to call SetMemoryState() unconditionally because it checks whether |
| 176 // this state transition is valid. | 202 // this state transition is valid. |
| 177 // TODO(bashi): In SUSPENDED state, we should update children's state | |
| 178 // accordingly when the foreground tab is changed (e.g. resume a renderer | |
| 179 // which becomes foreground and suspend a renderer which goes to background). | |
| 180 for (auto& iter : children()) | 203 for (auto& iter : children()) |
| 181 SetMemoryState(iter.first, mojo_state); | 204 SetMemoryState(iter.first, mojo_state); |
| 182 } | 205 } |
| 183 | 206 |
| 184 void MemoryCoordinatorImpl::ScheduleUpdateState(base::TimeDelta delta) { | 207 void MemoryCoordinatorImpl::ScheduleUpdateState(base::TimeDelta delta) { |
| 185 task_runner_->PostDelayedTask(FROM_HERE, update_state_callback_, delta); | 208 task_runner_->PostDelayedTask(FROM_HERE, update_state_callback_, delta); |
| 186 } | 209 } |
| 187 | 210 |
| 188 bool MemoryCoordinatorImpl::ValidateParameters() { | 211 bool MemoryCoordinatorImpl::ValidateParameters() { |
| 189 return (new_renderers_until_throttled_ > new_renderers_until_suspended_) && | 212 return (new_renderers_until_throttled_ > new_renderers_until_suspended_) && |
| 190 (new_renderers_back_to_normal_ > new_renderers_back_to_throttled_) && | 213 (new_renderers_back_to_normal_ > new_renderers_back_to_throttled_) && |
| 191 (new_renderers_back_to_normal_ > new_renderers_until_throttled_) && | 214 (new_renderers_back_to_normal_ > new_renderers_until_throttled_) && |
| 192 (new_renderers_back_to_throttled_ > new_renderers_until_suspended_); | 215 (new_renderers_back_to_throttled_ > new_renderers_until_suspended_); |
| 193 } | 216 } |
| 194 | 217 |
| 195 } // namespace content | 218 } // namespace content |
| OLD | NEW |