зеркало из https://github.com/mozilla/gecko-dev.git
195 строки
5.4 KiB
C++
195 строки
5.4 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/AvailableMemoryTracker.h"
|
|
|
|
#if defined(XP_WIN)
|
|
# include "mozilla/WindowsVersion.h"
|
|
# include "nsIMemoryReporter.h"
|
|
#endif
|
|
|
|
#include "nsIObserver.h"
|
|
#include "nsIObserverService.h"
|
|
#include "nsIRunnable.h"
|
|
#include "nsISupports.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsXULAppAPI.h"
|
|
|
|
#include "mozilla/Mutex.h"
|
|
#include "mozilla/ResultExtensions.h"
|
|
#include "mozilla/Services.h"
|
|
|
|
#if defined(MOZ_MEMORY)
|
|
# include "mozmemory.h"
|
|
#endif // MOZ_MEMORY
|
|
|
|
using namespace mozilla;
|
|
|
|
Atomic<uint32_t, MemoryOrdering::Relaxed> sNumLowPhysicalMemEvents;
|
|
|
|
namespace {
|
|
|
|
#if defined(XP_WIN)
|
|
|
|
# if (NTDDI_VERSION < NTDDI_WINBLUE) || \
|
|
(NTDDI_VERSION == NTDDI_WINBLUE && !defined(WINBLUE_KBSPRING14))
|
|
// Definitions for heap optimization that require the Windows SDK to target the
|
|
// Windows 8.1 Update
|
|
static const HEAP_INFORMATION_CLASS HeapOptimizeResources =
|
|
static_cast<HEAP_INFORMATION_CLASS>(3);
|
|
|
|
static const DWORD HEAP_OPTIMIZE_RESOURCES_CURRENT_VERSION = 1;
|
|
|
|
typedef struct _HEAP_OPTIMIZE_RESOURCES_INFORMATION {
|
|
DWORD Version;
|
|
DWORD Flags;
|
|
} HEAP_OPTIMIZE_RESOURCES_INFORMATION, *PHEAP_OPTIMIZE_RESOURCES_INFORMATION;
|
|
# endif
|
|
|
|
static int64_t LowMemoryEventsPhysicalDistinguishedAmount() {
|
|
return sNumLowPhysicalMemEvents;
|
|
}
|
|
|
|
class LowEventsReporter final : public nsIMemoryReporter {
|
|
~LowEventsReporter() {}
|
|
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
NS_IMETHOD CollectReports(nsIHandleReportCallback* aHandleReport,
|
|
nsISupports* aData, bool aAnonymize) override {
|
|
// clang-format off
|
|
MOZ_COLLECT_REPORT(
|
|
"low-memory-events/physical", KIND_OTHER, UNITS_COUNT_CUMULATIVE,
|
|
LowMemoryEventsPhysicalDistinguishedAmount(),
|
|
"Number of low-physical-memory events fired since startup. We fire such an "
|
|
"event when a windows low memory resource notification is signaled. The "
|
|
"machine will start to page if it runs out of physical memory. This may "
|
|
"cause it to run slowly, but it shouldn't cause it to crash.");
|
|
// clang-format on
|
|
|
|
return NS_OK;
|
|
}
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(LowEventsReporter, nsIMemoryReporter)
|
|
|
|
#endif // defined(XP_WIN)
|
|
|
|
/**
|
|
* This runnable is executed in response to a memory-pressure event; we spin
|
|
* the event-loop when receiving the memory-pressure event in the hope that
|
|
* other observers will synchronously free some memory that we'll be able to
|
|
* purge here.
|
|
*/
|
|
class nsJemallocFreeDirtyPagesRunnable final : public Runnable {
|
|
~nsJemallocFreeDirtyPagesRunnable() = default;
|
|
|
|
#if defined(XP_WIN)
|
|
void OptimizeSystemHeap();
|
|
#endif
|
|
|
|
public:
|
|
NS_DECL_NSIRUNNABLE
|
|
|
|
nsJemallocFreeDirtyPagesRunnable()
|
|
: Runnable("nsJemallocFreeDirtyPagesRunnable") {}
|
|
};
|
|
|
|
NS_IMETHODIMP
|
|
nsJemallocFreeDirtyPagesRunnable::Run() {
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
#if defined(MOZ_MEMORY)
|
|
jemalloc_free_dirty_pages();
|
|
#endif
|
|
|
|
#if defined(XP_WIN)
|
|
OptimizeSystemHeap();
|
|
#endif
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
#if defined(XP_WIN)
|
|
void nsJemallocFreeDirtyPagesRunnable::OptimizeSystemHeap() {
|
|
// HeapSetInformation exists prior to Windows 8.1, but the
|
|
// HeapOptimizeResources information class does not.
|
|
if (IsWin8Point1OrLater()) {
|
|
HEAP_OPTIMIZE_RESOURCES_INFORMATION heapOptInfo = {
|
|
HEAP_OPTIMIZE_RESOURCES_CURRENT_VERSION};
|
|
|
|
::HeapSetInformation(nullptr, HeapOptimizeResources, &heapOptInfo,
|
|
sizeof(heapOptInfo));
|
|
}
|
|
}
|
|
#endif // defined(XP_WIN)
|
|
|
|
/**
|
|
* The memory pressure watcher is used for listening to memory-pressure events
|
|
* and reacting upon them. We use one instance per process currently only for
|
|
* cleaning up dirty unused pages held by jemalloc.
|
|
*/
|
|
class nsMemoryPressureWatcher final : public nsIObserver {
|
|
~nsMemoryPressureWatcher() = default;
|
|
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIOBSERVER
|
|
|
|
void Init();
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(nsMemoryPressureWatcher, nsIObserver)
|
|
|
|
/**
|
|
* Initialize and subscribe to the memory-pressure events. We subscribe to the
|
|
* observer service in this method and not in the constructor because we need
|
|
* to hold a strong reference to 'this' before calling the observer service.
|
|
*/
|
|
void nsMemoryPressureWatcher::Init() {
|
|
nsCOMPtr<nsIObserverService> os = services::GetObserverService();
|
|
|
|
if (os) {
|
|
os->AddObserver(this, "memory-pressure", /* ownsWeak */ false);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reacts to all types of memory-pressure events, launches a runnable to
|
|
* free dirty pages held by jemalloc.
|
|
*/
|
|
NS_IMETHODIMP
|
|
nsMemoryPressureWatcher::Observe(nsISupports* aSubject, const char* aTopic,
|
|
const char16_t* aData) {
|
|
MOZ_ASSERT(!strcmp(aTopic, "memory-pressure"), "Unknown topic");
|
|
|
|
nsCOMPtr<nsIRunnable> runnable = new nsJemallocFreeDirtyPagesRunnable();
|
|
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace mozilla {
|
|
namespace AvailableMemoryTracker {
|
|
|
|
void Init() {
|
|
// The watchers are held alive by the observer service.
|
|
RefPtr<nsMemoryPressureWatcher> watcher = new nsMemoryPressureWatcher();
|
|
watcher->Init();
|
|
|
|
#if defined(XP_WIN)
|
|
RegisterLowMemoryEventsPhysicalDistinguishedAmount(
|
|
LowMemoryEventsPhysicalDistinguishedAmount);
|
|
#endif // defined(XP_WIN)
|
|
}
|
|
|
|
} // namespace AvailableMemoryTracker
|
|
} // namespace mozilla
|