Index: athena/resource_manager/memory_pressure_notifier.cc |
diff --git a/athena/resource_manager/memory_pressure_notifier.cc b/athena/resource_manager/memory_pressure_notifier.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..4df752ab2e3f611d6476aa3c50e99684a507c4f7 |
--- /dev/null |
+++ b/athena/resource_manager/memory_pressure_notifier.cc |
@@ -0,0 +1,118 @@ |
+// Copyright (c) 2014 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "athena/resource_manager/memory_pressure_notifier.h" |
+ |
+#include "athena/resource_manager/public/resource_manager_delegate.h" |
+#include "base/memory/scoped_ptr.h" |
+#include "base/time/time.h" |
+#include "base/timer/timer.h" |
+ |
+ |
+namespace athena { |
+ |
+//////////////////////////////////////////////////////////////////////////////// |
+// MemoryPressureNotifierImpl |
+// |
+// Does the actual work of observing. Ideally we would need to have at least one |
+// event driven margin window which can monitor the memory size. However, this |
+// functionality does not yet exist and for the time being we poll instead the |
+// available memory amount. Note that on the bright side this will also work on |
+// a non ChromeOS kernel. |
+// Note: Any calls to the resource manager will take some time to show effect. |
+class MemoryPressureNotifierImpl |
Jun Mukai
2014/08/27 01:21:18
You don't have to introduce pimpl pattern to this
Mr4D (OOO till 08-26)
2014/08/27 16:12:12
Done. (Did that because of the thread safe releas
|
+ : public base::RefCountedThreadSafe<MemoryPressureNotifierImpl> { |
+ public: |
+ explicit MemoryPressureNotifierImpl(MemoryPressureObserver* listener) |
+ : listener_(listener), |
+ current_pressure_(MemoryPressureObserver::MEMORY_PRESSURE_UNKNOWN) { |
+ StartObserving(); |
+ } |
+ |
+ private: |
+ friend class base::RefCountedThreadSafe<MemoryPressureNotifierImpl>; |
+ |
+ virtual ~MemoryPressureNotifierImpl() { |
+ StopObserving(); |
+ } |
+ |
+ // Starts observing the memory fill level. |
+ // Calls to StartObserving should always be matched with calls to |
+ // StopObserving. |
+ void StartObserving(); |
+ |
+ // Stop observing the memory fill level. |
+ // May be safely called if StartObserving has not been called. |
+ void StopObserving(); |
+ |
+ // The function which gets periodically be called to check any changes in the |
+ // memory pressure. |
+ void CheckMemoryPressure(); |
+ |
+ // Converts free percent of memory into a memory pressure value. |
+ MemoryPressureObserver::MemoryPressure GetMemoryPressureLevelFromFillLevel( |
+ int memory_fill_level); |
+ |
+ base::RepeatingTimer<MemoryPressureNotifierImpl> timer_; |
+ |
+ // The listener which needs to be informed about memory pressure. |
+ MemoryPressureObserver* listener_; |
+ |
+ // Our current memory pressure. |
+ MemoryPressureObserver::MemoryPressure current_pressure_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(MemoryPressureNotifierImpl); |
+}; |
+ |
+void MemoryPressureNotifierImpl::StartObserving() { |
+ int time_in_ms = listener_->GetDelegate()->MemoryPressureIntervalInMS(); |
+ timer_.Start(FROM_HERE, |
+ base::TimeDelta::FromMilliseconds(time_in_ms), |
+ base::Bind(&MemoryPressureNotifierImpl::CheckMemoryPressure, |
+ base::Unretained(this))); |
+} |
+ |
+void MemoryPressureNotifierImpl::StopObserving() { |
+ // If StartObserving failed, StopObserving will still get called. |
+ timer_.Stop(); |
+} |
+ |
+void MemoryPressureNotifierImpl::CheckMemoryPressure() { |
+ MemoryPressureObserver::MemoryPressure pressure = |
+ GetMemoryPressureLevelFromFillLevel( |
+ listener_->GetDelegate()->GetUsedMemoryInPercent()); |
+ if (current_pressure_ != pressure || |
+ (pressure != MemoryPressureObserver::MEMORY_PRESSURE_LOW && |
+ pressure != MemoryPressureObserver::MEMORY_PRESSURE_UNKNOWN)) { |
+ // If we are anything worse than |MEMORY_PRESSURE_LOW|, we notify the |
+ // listener. |
+ current_pressure_ = pressure; |
+ listener_->OnMemoryPressure(current_pressure_); |
+ } |
+} |
+ |
+MemoryPressureObserver::MemoryPressure |
+MemoryPressureNotifierImpl::GetMemoryPressureLevelFromFillLevel( |
+ int memory_fill_level) { |
+ if (memory_fill_level == 0) |
+ return MemoryPressureObserver::MEMORY_PRESSURE_UNKNOWN; |
+ if (memory_fill_level < 50) |
+ return MemoryPressureObserver::MEMORY_PRESSURE_LOW; |
+ if (memory_fill_level < 75) |
+ return MemoryPressureObserver::MEMORY_PRESSURE_MODERATE; |
+ if (memory_fill_level < 90) |
+ return MemoryPressureObserver::MEMORY_PRESSURE_HIGH; |
+ return MemoryPressureObserver::MEMORY_PRESSURE_CRITICAL; |
+} |
+ |
+//////////////////////////////////////////////////////////////////////////////// |
+// MemoryPressureNotifier |
+ |
+MemoryPressureNotifier::MemoryPressureNotifier(MemoryPressureObserver* listener) |
+ : observer_(new MemoryPressureNotifierImpl(listener)) { |
+} |
+ |
+MemoryPressureNotifier::~MemoryPressureNotifier() {} |
+ |
+} // namespace athena |