Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 1 | // Copyright 2019 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 "base/util/memory_pressure/multi_source_memory_pressure_monitor.h" |
| 6 | |
| 7 | #include "base/bind.h" |
Hans Wennborg | c3cffa6 | 2020-04-27 10:09:12 | [diff] [blame] | 8 | #include "base/check_op.h" |
Sebastien Marchand | 6c5354e | 2020-02-07 19:38:25 | [diff] [blame] | 9 | #include "base/metrics/histogram_functions.h" |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 10 | #include "base/metrics/histogram_macros.h" |
Sebastien Marchand | 6c5354e | 2020-02-07 19:38:25 | [diff] [blame] | 11 | #include "base/time/time.h" |
Eric Seckler | f6c544f | 2020-06-02 10:49:21 | [diff] [blame] | 12 | #include "base/trace_event/base_tracing.h" |
ssid | 8734ea7 | 2021-03-17 18:43:58 | [diff] [blame^] | 13 | #include "base/tracing_buildflags.h" |
Ryan Powell | a68d66fc | 2019-08-19 21:32:39 | [diff] [blame] | 14 | #include "base/util/memory_pressure/system_memory_pressure_evaluator.h" |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 15 | |
ssid | 8734ea7 | 2021-03-17 18:43:58 | [diff] [blame^] | 16 | #if BUILDFLAG(ENABLE_BASE_TRACING) |
| 17 | #include "base/trace_event/memory_pressure_level_proto.h" // no-presubmit-check |
| 18 | #endif |
| 19 | |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 20 | namespace util { |
| 21 | |
| 22 | MultiSourceMemoryPressureMonitor::MultiSourceMemoryPressureMonitor() |
| 23 | : current_pressure_level_( |
| 24 | base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_NONE), |
| 25 | dispatch_callback_(base::BindRepeating( |
| 26 | &base::MemoryPressureListener::NotifyMemoryPressure)), |
| 27 | aggregator_(this) { |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 28 | } |
| 29 | |
Jenny Wong | 6592dd4 | 2019-09-26 17:25:07 | [diff] [blame] | 30 | MultiSourceMemoryPressureMonitor::~MultiSourceMemoryPressureMonitor() { |
| 31 | // Destroy system evaluator early while the remaining members of this class |
| 32 | // still exist. MultiSourceMemoryPressureMonitor implements |
| 33 | // MemoryPressureVoteAggregator::Delegate, and |
| 34 | // delegate_->OnMemoryPressureLevelChanged() gets indirectly called during |
| 35 | // ~SystemMemoryPressureEvaluator(). |
| 36 | system_evaluator_.reset(); |
| 37 | } |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 38 | |
Sebastien Marchand | c9d4cd1 | 2019-10-03 22:48:01 | [diff] [blame] | 39 | void MultiSourceMemoryPressureMonitor::Start() { |
| 40 | system_evaluator_ = |
| 41 | SystemMemoryPressureEvaluator::CreateDefaultSystemEvaluator(this); |
| 42 | StartMetricsTimer(); |
| 43 | } |
| 44 | |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 45 | void MultiSourceMemoryPressureMonitor::StartMetricsTimer() { |
| 46 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
Ryan Powell | bdc2759 | 2019-09-04 22:59:39 | [diff] [blame] | 47 | // Unretained is safe here since this task is running on a timer owned by this |
| 48 | // object. |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 49 | metric_timer_.Start( |
| 50 | FROM_HERE, MemoryPressureMonitor::kUMAMemoryPressureLevelPeriod, |
Ryan Powell | bdc2759 | 2019-09-04 22:59:39 | [diff] [blame] | 51 | BindRepeating( |
| 52 | &MultiSourceMemoryPressureMonitor::RecordCurrentPressureLevel, |
| 53 | base::Unretained(this))); |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 54 | } |
| 55 | |
| 56 | void MultiSourceMemoryPressureMonitor::StopMetricsTimer() { |
| 57 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 58 | metric_timer_.Stop(); |
| 59 | } |
| 60 | |
Ryan Powell | bdc2759 | 2019-09-04 22:59:39 | [diff] [blame] | 61 | void MultiSourceMemoryPressureMonitor::RecordCurrentPressureLevel() { |
| 62 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 63 | RecordMemoryPressure(GetCurrentPressureLevel(), /* ticks = */ 1); |
| 64 | } |
| 65 | |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 66 | base::MemoryPressureListener::MemoryPressureLevel |
| 67 | MultiSourceMemoryPressureMonitor::GetCurrentPressureLevel() const { |
| 68 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 69 | return current_pressure_level_; |
| 70 | } |
| 71 | |
| 72 | std::unique_ptr<MemoryPressureVoter> |
| 73 | MultiSourceMemoryPressureMonitor::CreateVoter() { |
| 74 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
Wez | 861e292 | 2020-03-01 21:14:00 | [diff] [blame] | 75 | return aggregator_.CreateVoter(); |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 76 | } |
| 77 | |
| 78 | void MultiSourceMemoryPressureMonitor::SetDispatchCallback( |
| 79 | const DispatchCallback& callback) { |
| 80 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 81 | dispatch_callback_ = callback; |
| 82 | } |
| 83 | |
| 84 | void MultiSourceMemoryPressureMonitor::OnMemoryPressureLevelChanged( |
| 85 | base::MemoryPressureListener::MemoryPressureLevel level) { |
| 86 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
Sebastien Marchand | 6c5354e | 2020-02-07 19:38:25 | [diff] [blame] | 87 | DCHECK_NE(current_pressure_level_, level); |
| 88 | |
ssid | fff7ab4 | 2021-03-04 20:03:11 | [diff] [blame] | 89 | TRACE_EVENT_INSTANT( |
Sebastien Marchand | 9df1f3d | 2020-03-23 18:54:31 | [diff] [blame] | 90 | "base", "MultiSourceMemoryPressureMonitor::OnMemoryPressureLevelChanged", |
ssid | fff7ab4 | 2021-03-04 20:03:11 | [diff] [blame] | 91 | [&](perfetto::EventContext ctx) { |
| 92 | auto* event = ctx.event<perfetto::protos::pbzero::ChromeTrackEvent>(); |
| 93 | auto* data = event->set_chrome_memory_pressure_notification(); |
ssid | 8734ea7 | 2021-03-17 18:43:58 | [diff] [blame^] | 94 | data->set_level( |
| 95 | base::trace_event::MemoryPressureLevelToTraceEnum(level)); |
ssid | fff7ab4 | 2021-03-04 20:03:11 | [diff] [blame] | 96 | }); |
Sebastien Marchand | 9df1f3d | 2020-03-23 18:54:31 | [diff] [blame] | 97 | |
Sebastien Marchand | 6c5354e | 2020-02-07 19:38:25 | [diff] [blame] | 98 | // Records the duration of the latest pressure session, there are 4 |
| 99 | // transitions of interest: |
| 100 | // - Moderate -> None |
| 101 | // - Moderate -> Critical |
| 102 | // - Critical -> Moderate |
| 103 | // - Critical -> None |
| 104 | |
| 105 | base::TimeTicks now = base::TimeTicks::Now(); |
| 106 | |
| 107 | if (current_pressure_level_ != |
| 108 | MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_NONE) { |
| 109 | DCHECK(!last_pressure_change_timestamp_.is_null()); |
| 110 | std::string histogram_name = "Memory.PressureWindowDuration."; |
| 111 | switch (current_pressure_level_) { |
| 112 | // From: |
| 113 | case MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_MODERATE: { |
| 114 | // To: |
| 115 | if (level == MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_NONE) { |
| 116 | histogram_name += "ModerateToNone"; |
| 117 | } else { // MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_CRITICAL |
| 118 | histogram_name += "ModerateToCritical"; |
| 119 | } |
| 120 | break; |
| 121 | } |
| 122 | // From: |
| 123 | case MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_CRITICAL: { |
| 124 | // To: |
| 125 | if (level == MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_NONE) { |
| 126 | histogram_name += "CriticalToNone"; |
| 127 | } else { // MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_MODERATE |
| 128 | histogram_name += "CriticalToModerate"; |
| 129 | } |
| 130 | break; |
| 131 | } |
| 132 | case MemoryPressureLevel::MEMORY_PRESSURE_LEVEL_NONE: |
| 133 | default: |
| 134 | break; |
| 135 | } |
| 136 | |
| 137 | base::UmaHistogramCustomTimes( |
| 138 | histogram_name, now - last_pressure_change_timestamp_, |
| 139 | base::TimeDelta::FromSeconds(1), base::TimeDelta::FromMinutes(10), 50); |
| 140 | } |
| 141 | |
| 142 | last_pressure_change_timestamp_ = now; |
| 143 | |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 144 | current_pressure_level_ = level; |
| 145 | } |
| 146 | |
| 147 | void MultiSourceMemoryPressureMonitor::OnNotifyListenersRequested() { |
| 148 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 149 | dispatch_callback_.Run(current_pressure_level_); |
| 150 | } |
| 151 | |
Ryan Powell | a68d66fc | 2019-08-19 21:32:39 | [diff] [blame] | 152 | void MultiSourceMemoryPressureMonitor::ResetSystemEvaluatorForTesting() { |
| 153 | system_evaluator_.reset(); |
| 154 | } |
| 155 | |
Ryan Powell | 00e61aea | 2019-09-04 18:33:44 | [diff] [blame] | 156 | void MultiSourceMemoryPressureMonitor::SetSystemEvaluator( |
| 157 | std::unique_ptr<SystemMemoryPressureEvaluator> evaluator) { |
| 158 | DCHECK(!system_evaluator_); |
| 159 | system_evaluator_ = std::move(evaluator); |
| 160 | } |
| 161 | |
Ryan Powell | 30287d4 | 2019-08-15 15:48:32 | [diff] [blame] | 162 | } // namespace util |