OLD | NEW |
(Empty) | |
| 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 |
| 5 #include "base/win/memory_pressure_monitor.h" |
| 6 |
| 7 #include "base/basictypes.h" |
| 8 #include "base/memory/memory_pressure_listener.h" |
| 9 #include "base/message_loop/message_loop.h" |
| 10 #include "testing/gmock/include/gmock/gmock.h" |
| 11 #include "testing/gtest/include/gtest/gtest.h" |
| 12 |
| 13 namespace base { |
| 14 namespace win { |
| 15 |
| 16 namespace { |
| 17 |
| 18 struct PressureSettings { |
| 19 int phys_left_mb; |
| 20 MemoryPressureListener::MemoryPressureLevel level; |
| 21 }; |
| 22 |
| 23 } // namespace |
| 24 |
| 25 // This is outside of the anonymous namespace so that it can be seen as a friend |
| 26 // to the monitor class. |
| 27 class TestMemoryPressureMonitor : public MemoryPressureMonitor { |
| 28 public: |
| 29 using MemoryPressureMonitor::CalculateCurrentPressureLevel; |
| 30 using MemoryPressureMonitor::CheckMemoryPressure; |
| 31 |
| 32 static const DWORDLONG kMBBytes = 1024 * 1024; |
| 33 |
| 34 explicit TestMemoryPressureMonitor(bool large_memory) |
| 35 : mem_status_() { |
| 36 // Generate a plausible amount of memory. |
| 37 mem_status_.ullTotalPhys = |
| 38 static_cast<DWORDLONG>(GenerateTotalMemoryMb(large_memory)) * kMBBytes; |
| 39 |
| 40 // Rerun InferThresholds using the test fixture's GetSystemMemoryStatus. |
| 41 InferThresholds(); |
| 42 // Stop the timer. |
| 43 StopObserving(); |
| 44 } |
| 45 |
| 46 TestMemoryPressureMonitor(int system_memory_mb, |
| 47 int moderate_threshold_mb, |
| 48 int critical_threshold_mb) |
| 49 : MemoryPressureMonitor(moderate_threshold_mb, critical_threshold_mb), |
| 50 mem_status_() { |
| 51 // Set the amount of system memory. |
| 52 mem_status_.ullTotalPhys = static_cast<DWORDLONG>( |
| 53 system_memory_mb * kMBBytes); |
| 54 |
| 55 // Stop the timer. |
| 56 StopObserving(); |
| 57 } |
| 58 |
| 59 virtual ~TestMemoryPressureMonitor() {} |
| 60 |
| 61 MOCK_METHOD1(OnMemoryPressure, |
| 62 void(MemoryPressureListener::MemoryPressureLevel level)); |
| 63 |
| 64 // Generates an amount of total memory that is consistent with the requested |
| 65 // memory model. |
| 66 int GenerateTotalMemoryMb(bool large_memory) { |
| 67 int total_mb = 64; |
| 68 while (total_mb < MemoryPressureMonitor::kLargeMemoryThresholdMb) |
| 69 total_mb *= 2; |
| 70 if (large_memory) |
| 71 return total_mb * 2; |
| 72 return total_mb / 2; |
| 73 } |
| 74 |
| 75 // Sets up the memory status to reflect the provided absolute memory left. |
| 76 void SetMemoryFree(int phys_left_mb) { |
| 77 // ullTotalPhys is set in the constructor and not modified. |
| 78 |
| 79 // Set the amount of available memory. |
| 80 mem_status_.ullAvailPhys = |
| 81 static_cast<DWORDLONG>(phys_left_mb) * kMBBytes; |
| 82 DCHECK_LT(mem_status_.ullAvailPhys, mem_status_.ullTotalPhys); |
| 83 |
| 84 // These fields are unused. |
| 85 mem_status_.dwMemoryLoad = 0; |
| 86 mem_status_.ullTotalPageFile = 0; |
| 87 mem_status_.ullAvailPageFile = 0; |
| 88 mem_status_.ullTotalVirtual = 0; |
| 89 mem_status_.ullAvailVirtual = 0; |
| 90 } |
| 91 |
| 92 void SetNone() { |
| 93 SetMemoryFree(moderate_threshold_mb() + 1); |
| 94 } |
| 95 |
| 96 void SetModerate() { |
| 97 SetMemoryFree(moderate_threshold_mb() - 1); |
| 98 } |
| 99 |
| 100 void SetCritical() { |
| 101 SetMemoryFree(critical_threshold_mb() - 1); |
| 102 } |
| 103 |
| 104 private: |
| 105 bool GetSystemMemoryStatus(MEMORYSTATUSEX* mem_status) override { |
| 106 // Simply copy the memory status set by the test fixture. |
| 107 *mem_status = mem_status_; |
| 108 return true; |
| 109 } |
| 110 |
| 111 MEMORYSTATUSEX mem_status_; |
| 112 |
| 113 DISALLOW_COPY_AND_ASSIGN(TestMemoryPressureMonitor); |
| 114 }; |
| 115 |
| 116 class WinMemoryPressureMonitorTest : public testing::Test { |
| 117 protected: |
| 118 void CalculateCurrentMemoryPressureLevelTest( |
| 119 TestMemoryPressureMonitor* monitor) { |
| 120 |
| 121 int mod = monitor->moderate_threshold_mb(); |
| 122 monitor->SetMemoryFree(mod + 1); |
| 123 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_NONE, |
| 124 monitor->CalculateCurrentPressureLevel()); |
| 125 |
| 126 monitor->SetMemoryFree(mod); |
| 127 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 128 monitor->CalculateCurrentPressureLevel()); |
| 129 |
| 130 monitor->SetMemoryFree(mod - 1); |
| 131 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 132 monitor->CalculateCurrentPressureLevel()); |
| 133 |
| 134 int crit = monitor->critical_threshold_mb(); |
| 135 monitor->SetMemoryFree(crit + 1); |
| 136 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 137 monitor->CalculateCurrentPressureLevel()); |
| 138 |
| 139 monitor->SetMemoryFree(crit); |
| 140 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL, |
| 141 monitor->CalculateCurrentPressureLevel()); |
| 142 |
| 143 monitor->SetMemoryFree(crit - 1); |
| 144 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL, |
| 145 monitor->CalculateCurrentPressureLevel()); |
| 146 } |
| 147 |
| 148 base::MessageLoopForUI message_loop_; |
| 149 }; |
| 150 |
| 151 // Tests the fundamental direct calculation of memory pressure with automatic |
| 152 // small-memory thresholds. |
| 153 TEST_F(WinMemoryPressureMonitorTest, CalculateCurrentMemoryPressureLevelSmall) { |
| 154 static const int kModerateMb = |
| 155 MemoryPressureMonitor::kSmallMemoryDefaultModerateThresholdMb; |
| 156 static const int kCriticalMb = |
| 157 MemoryPressureMonitor::kSmallMemoryDefaultCriticalThresholdMb; |
| 158 |
| 159 TestMemoryPressureMonitor monitor(false); // Small-memory model. |
| 160 |
| 161 EXPECT_EQ(kModerateMb, monitor.moderate_threshold_mb()); |
| 162 EXPECT_EQ(kCriticalMb, monitor.critical_threshold_mb()); |
| 163 |
| 164 ASSERT_NO_FATAL_FAILURE(CalculateCurrentMemoryPressureLevelTest(&monitor)); |
| 165 } |
| 166 |
| 167 // Tests the fundamental direct calculation of memory pressure with automatic |
| 168 // large-memory thresholds. |
| 169 TEST_F(WinMemoryPressureMonitorTest, CalculateCurrentMemoryPressureLevelLarge) { |
| 170 static const int kModerateMb = |
| 171 MemoryPressureMonitor::kLargeMemoryDefaultModerateThresholdMb; |
| 172 static const int kCriticalMb = |
| 173 MemoryPressureMonitor::kLargeMemoryDefaultCriticalThresholdMb; |
| 174 |
| 175 TestMemoryPressureMonitor monitor(true); // Large-memory model. |
| 176 |
| 177 EXPECT_EQ(kModerateMb, monitor.moderate_threshold_mb()); |
| 178 EXPECT_EQ(kCriticalMb, monitor.critical_threshold_mb()); |
| 179 |
| 180 ASSERT_NO_FATAL_FAILURE(CalculateCurrentMemoryPressureLevelTest(&monitor)); |
| 181 } |
| 182 |
| 183 // Tests the fundamental direct calculation of memory pressure with manually |
| 184 // specified threshold levels. |
| 185 TEST_F(WinMemoryPressureMonitorTest, |
| 186 CalculateCurrentMemoryPressureLevelCustom) { |
| 187 static const int kSystemMb = 512; |
| 188 static const int kModerateMb = 256; |
| 189 static const int kCriticalMb = 128; |
| 190 |
| 191 TestMemoryPressureMonitor monitor(kSystemMb, kModerateMb, kCriticalMb); |
| 192 |
| 193 EXPECT_EQ(kModerateMb, monitor.moderate_threshold_mb()); |
| 194 EXPECT_EQ(kCriticalMb, monitor.critical_threshold_mb()); |
| 195 |
| 196 ASSERT_NO_FATAL_FAILURE(CalculateCurrentMemoryPressureLevelTest(&monitor)); |
| 197 } |
| 198 |
| 199 // This test tests the various transition states from memory pressure, looking |
| 200 // for the correct behavior on event reposting as well as state updates. |
| 201 TEST_F(WinMemoryPressureMonitorTest, CheckMemoryPressure) { |
| 202 // Large-memory. |
| 203 testing::StrictMock<TestMemoryPressureMonitor> monitor(true); |
| 204 MemoryPressureListener listener( |
| 205 base::Bind(&TestMemoryPressureMonitor::OnMemoryPressure, |
| 206 base::Unretained(&monitor))); |
| 207 |
| 208 // Checking the memory pressure at 0% load should not produce any |
| 209 // events. |
| 210 monitor.SetNone(); |
| 211 monitor.CheckMemoryPressure(); |
| 212 message_loop_.RunUntilIdle(); |
| 213 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_NONE, |
| 214 monitor.GetCurrentPressureLevel()); |
| 215 |
| 216 // Setting the memory level to 80% should produce a moderate pressure level. |
| 217 EXPECT_CALL(monitor, |
| 218 OnMemoryPressure(MemoryPressureListener:: |
| 219 MEMORY_PRESSURE_LEVEL_MODERATE)); |
| 220 monitor.SetModerate(); |
| 221 monitor.CheckMemoryPressure(); |
| 222 message_loop_.RunUntilIdle(); |
| 223 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 224 monitor.GetCurrentPressureLevel()); |
| 225 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 226 |
| 227 // Check that the event gets reposted after a while. |
| 228 for (int i = 0; i < monitor.kModeratePressureCooldownCycles; ++i) { |
| 229 if (i + 1 == monitor.kModeratePressureCooldownCycles) { |
| 230 EXPECT_CALL(monitor, |
| 231 OnMemoryPressure(MemoryPressureListener:: |
| 232 MEMORY_PRESSURE_LEVEL_MODERATE)); |
| 233 } |
| 234 monitor.CheckMemoryPressure(); |
| 235 message_loop_.RunUntilIdle(); |
| 236 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 237 monitor.GetCurrentPressureLevel()); |
| 238 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 239 } |
| 240 |
| 241 // Setting the memory usage to 99% should produce critical levels. |
| 242 EXPECT_CALL(monitor, |
| 243 OnMemoryPressure(MemoryPressureListener:: |
| 244 MEMORY_PRESSURE_LEVEL_CRITICAL)); |
| 245 monitor.SetCritical(); |
| 246 monitor.CheckMemoryPressure(); |
| 247 message_loop_.RunUntilIdle(); |
| 248 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL, |
| 249 monitor.GetCurrentPressureLevel()); |
| 250 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 251 |
| 252 // Calling it again should immediately produce a second call. |
| 253 EXPECT_CALL(monitor, |
| 254 OnMemoryPressure(MemoryPressureListener:: |
| 255 MEMORY_PRESSURE_LEVEL_CRITICAL)); |
| 256 monitor.CheckMemoryPressure(); |
| 257 message_loop_.RunUntilIdle(); |
| 258 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL, |
| 259 monitor.GetCurrentPressureLevel()); |
| 260 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 261 |
| 262 // When lowering the pressure again there should be a notification and the |
| 263 // pressure should go back to moderate. |
| 264 EXPECT_CALL(monitor, |
| 265 OnMemoryPressure(MemoryPressureListener:: |
| 266 MEMORY_PRESSURE_LEVEL_MODERATE)); |
| 267 monitor.SetModerate(); |
| 268 monitor.CheckMemoryPressure(); |
| 269 message_loop_.RunUntilIdle(); |
| 270 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 271 monitor.GetCurrentPressureLevel()); |
| 272 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 273 |
| 274 // Check that the event gets reposted after a while. |
| 275 for (int i = 0; i < monitor.kModeratePressureCooldownCycles; ++i) { |
| 276 if (i + 1 == monitor.kModeratePressureCooldownCycles) { |
| 277 EXPECT_CALL(monitor, |
| 278 OnMemoryPressure(MemoryPressureListener:: |
| 279 MEMORY_PRESSURE_LEVEL_MODERATE)); |
| 280 } |
| 281 monitor.CheckMemoryPressure(); |
| 282 message_loop_.RunUntilIdle(); |
| 283 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE, |
| 284 monitor.GetCurrentPressureLevel()); |
| 285 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 286 } |
| 287 |
| 288 // Going down to no pressure should not produce an notification. |
| 289 monitor.SetNone(); |
| 290 monitor.CheckMemoryPressure(); |
| 291 message_loop_.RunUntilIdle(); |
| 292 EXPECT_EQ(MemoryPressureListener::MEMORY_PRESSURE_LEVEL_NONE, |
| 293 monitor.GetCurrentPressureLevel()); |
| 294 testing::Mock::VerifyAndClearExpectations(&monitor); |
| 295 } |
| 296 |
| 297 } // namespace win |
| 298 } // namespace base |
OLD | NEW |