зеркало из https://github.com/mozilla/gecko-dev.git
1068 строки
35 KiB
C++
1068 строки
35 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 "ProcessPriorityManager.h"
|
|
#include "mozilla/ClearOnShutdown.h"
|
|
#include "mozilla/dom/CanonicalBrowsingContext.h"
|
|
#include "mozilla/dom/ContentParent.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/BrowserHost.h"
|
|
#include "mozilla/dom/BrowserParent.h"
|
|
#include "mozilla/Hal.h"
|
|
#include "mozilla/IntegerPrintfMacros.h"
|
|
#include "mozilla/Preferences.h"
|
|
#include "mozilla/ProfilerMarkers.h"
|
|
#include "mozilla/ProfilerState.h"
|
|
#include "mozilla/Services.h"
|
|
#include "mozilla/StaticPrefs_dom.h"
|
|
#include "mozilla/StaticPrefs_threads.h"
|
|
#include "mozilla/Telemetry.h"
|
|
#include "mozilla/Unused.h"
|
|
#include "mozilla/Logging.h"
|
|
#include "nsPrintfCString.h"
|
|
#include "nsXULAppAPI.h"
|
|
#include "nsFrameLoader.h"
|
|
#include "nsINamed.h"
|
|
#include "nsIObserverService.h"
|
|
#include "StaticPtr.h"
|
|
#include "nsIObserver.h"
|
|
#include "nsITimer.h"
|
|
#include "nsIPropertyBag2.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsCRT.h"
|
|
#include "nsTHashSet.h"
|
|
#include "nsQueryObject.h"
|
|
#include "nsTHashMap.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
using namespace mozilla::hal;
|
|
|
|
#ifdef XP_WIN
|
|
# include <process.h>
|
|
# define getpid _getpid
|
|
#else
|
|
# include <unistd.h>
|
|
#endif
|
|
|
|
#ifdef LOG
|
|
# undef LOG
|
|
#endif
|
|
|
|
// Use LOGP inside a ParticularProcessPriorityManager method; use LOG
|
|
// everywhere else. LOGP prints out information about the particular process
|
|
// priority manager.
|
|
//
|
|
// (Wow, our logging story is a huge mess.)
|
|
|
|
// #define ENABLE_LOGGING 1
|
|
|
|
#if defined(ANDROID) && defined(ENABLE_LOGGING)
|
|
# include <android/log.h>
|
|
# define LOG(fmt, ...) \
|
|
__android_log_print(ANDROID_LOG_INFO, "Gecko:ProcessPriorityManager", fmt, \
|
|
##__VA_ARGS__)
|
|
# define LOGP(fmt, ...) \
|
|
__android_log_print( \
|
|
ANDROID_LOG_INFO, "Gecko:ProcessPriorityManager", \
|
|
"[%schild-id=%" PRIu64 ", pid=%d] " fmt, NameWithComma().get(), \
|
|
static_cast<uint64_t>(ChildID()), Pid(), ##__VA_ARGS__)
|
|
|
|
#elif defined(ENABLE_LOGGING)
|
|
# define LOG(fmt, ...) \
|
|
printf("ProcessPriorityManager - " fmt "\n", ##__VA_ARGS__)
|
|
# define LOGP(fmt, ...) \
|
|
printf("ProcessPriorityManager[%schild-id=%" PRIu64 ", pid=%d] - " fmt \
|
|
"\n", \
|
|
NameWithComma().get(), static_cast<uint64_t>(ChildID()), Pid(), \
|
|
##__VA_ARGS__)
|
|
#else
|
|
static LogModule* GetPPMLog() {
|
|
static LazyLogModule sLog("ProcessPriorityManager");
|
|
return sLog;
|
|
}
|
|
# define LOG(fmt, ...) \
|
|
MOZ_LOG(GetPPMLog(), LogLevel::Debug, \
|
|
("ProcessPriorityManager - " fmt, ##__VA_ARGS__))
|
|
# define LOGP(fmt, ...) \
|
|
MOZ_LOG(GetPPMLog(), LogLevel::Debug, \
|
|
("ProcessPriorityManager[%schild-id=%" PRIu64 ", pid=%d] - " fmt, \
|
|
NameWithComma().get(), static_cast<uint64_t>(ChildID()), Pid(), \
|
|
##__VA_ARGS__))
|
|
#endif
|
|
|
|
namespace geckoprofiler::markers {
|
|
struct SubProcessPriorityChange {
|
|
static constexpr Span<const char> MarkerTypeName() {
|
|
return MakeStringSpan("subprocessprioritychange");
|
|
}
|
|
static void StreamJSONMarkerData(baseprofiler::SpliceableJSONWriter& aWriter,
|
|
int32_t aPid,
|
|
const ProfilerString8View& aPreviousPriority,
|
|
const ProfilerString8View& aNewPriority) {
|
|
aWriter.IntProperty("pid", aPid);
|
|
aWriter.StringProperty("Before", aPreviousPriority);
|
|
aWriter.StringProperty("After", aNewPriority);
|
|
}
|
|
static MarkerSchema MarkerTypeDisplay() {
|
|
using MS = MarkerSchema;
|
|
MS schema{MS::Location::MarkerChart, MS::Location::MarkerTable};
|
|
schema.AddKeyFormat("pid", MS::Format::Integer);
|
|
schema.AddKeyFormat("Before", MS::Format::String);
|
|
schema.AddKeyFormat("After", MS::Format::String);
|
|
schema.SetAllLabels(
|
|
"priority of child {marker.data.pid}:"
|
|
" {marker.data.Before} -> {marker.data.After}");
|
|
return schema;
|
|
}
|
|
};
|
|
|
|
struct SubProcessPriority {
|
|
static constexpr Span<const char> MarkerTypeName() {
|
|
return MakeStringSpan("subprocesspriority");
|
|
}
|
|
static void StreamJSONMarkerData(baseprofiler::SpliceableJSONWriter& aWriter,
|
|
int32_t aPid,
|
|
const ProfilerString8View& aPriority,
|
|
const ProfilingState& aProfilingState) {
|
|
aWriter.IntProperty("pid", aPid);
|
|
aWriter.StringProperty("Priority", aPriority);
|
|
aWriter.StringProperty("Marker cause",
|
|
ProfilerString8View::WrapNullTerminatedString(
|
|
ProfilingStateToString(aProfilingState)));
|
|
}
|
|
static MarkerSchema MarkerTypeDisplay() {
|
|
using MS = MarkerSchema;
|
|
MS schema{MS::Location::MarkerChart, MS::Location::MarkerTable};
|
|
schema.AddKeyFormat("pid", MS::Format::Integer);
|
|
schema.AddKeyFormat("Priority", MS::Format::String);
|
|
schema.AddKeyFormat("Marker cause", MS::Format::String);
|
|
schema.SetAllLabels(
|
|
"priority of child {marker.data.pid}: {marker.data.Priority}");
|
|
return schema;
|
|
}
|
|
};
|
|
} // namespace geckoprofiler::markers
|
|
|
|
namespace {
|
|
|
|
class ParticularProcessPriorityManager;
|
|
|
|
/**
|
|
* This singleton class does the work to implement the process priority manager
|
|
* in the main process. This class may not be used in child processes. (You
|
|
* can call StaticInit, but it won't do anything, and GetSingleton() will
|
|
* return null.)
|
|
*
|
|
* ProcessPriorityManager::CurrentProcessIsForeground() and
|
|
* ProcessPriorityManager::AnyProcessHasHighPriority() which can be called in
|
|
* any process, are handled separately, by the ProcessPriorityManagerChild
|
|
* class.
|
|
*/
|
|
class ProcessPriorityManagerImpl final : public nsIObserver,
|
|
public nsSupportsWeakReference {
|
|
public:
|
|
/**
|
|
* If we're in the main process, get the ProcessPriorityManagerImpl
|
|
* singleton. If we're in a child process, return null.
|
|
*/
|
|
static ProcessPriorityManagerImpl* GetSingleton();
|
|
|
|
static void StaticInit();
|
|
static bool PrefsEnabled();
|
|
static void SetProcessPriorityIfEnabled(int aPid, ProcessPriority aPriority);
|
|
static bool TestMode();
|
|
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIOBSERVER
|
|
|
|
/**
|
|
* This function implements ProcessPriorityManager::SetProcessPriority.
|
|
*/
|
|
void SetProcessPriority(ContentParent* aContentParent,
|
|
ProcessPriority aPriority);
|
|
|
|
/**
|
|
* If a magic testing-only pref is set, notify the observer service on the
|
|
* given topic with the given data. This is used for testing
|
|
*/
|
|
void FireTestOnlyObserverNotification(const char* aTopic,
|
|
const nsACString& aData);
|
|
|
|
/**
|
|
* This must be called by a ParticularProcessPriorityManager when it changes
|
|
* its priority.
|
|
*/
|
|
void NotifyProcessPriorityChanged(
|
|
ParticularProcessPriorityManager* aParticularManager,
|
|
hal::ProcessPriority aOldPriority);
|
|
|
|
void BrowserPriorityChanged(CanonicalBrowsingContext* aBC, bool aPriority);
|
|
void BrowserPriorityChanged(BrowserParent* aBrowserParent, bool aPriority);
|
|
|
|
void ResetPriority(ContentParent* aContentParent);
|
|
|
|
private:
|
|
static bool sPrefListenersRegistered;
|
|
static bool sInitialized;
|
|
static StaticRefPtr<ProcessPriorityManagerImpl> sSingleton;
|
|
|
|
static void PrefChangedCallback(const char* aPref, void* aClosure);
|
|
|
|
ProcessPriorityManagerImpl();
|
|
~ProcessPriorityManagerImpl();
|
|
ProcessPriorityManagerImpl(const ProcessPriorityManagerImpl&) = delete;
|
|
|
|
const ProcessPriorityManagerImpl& operator=(
|
|
const ProcessPriorityManagerImpl&) = delete;
|
|
|
|
void Init();
|
|
|
|
already_AddRefed<ParticularProcessPriorityManager>
|
|
GetParticularProcessPriorityManager(ContentParent* aContentParent);
|
|
|
|
void ObserveContentParentDestroyed(nsISupports* aSubject);
|
|
|
|
nsTHashMap<uint64_t, RefPtr<ParticularProcessPriorityManager> >
|
|
mParticularManagers;
|
|
|
|
/** Contains the PIDs of child processes holding high-priority wakelocks */
|
|
nsTHashSet<uint64_t> mHighPriorityChildIDs;
|
|
};
|
|
|
|
/**
|
|
* This singleton class implements the parts of the process priority manager
|
|
* that are available from all processes.
|
|
*/
|
|
class ProcessPriorityManagerChild final : public nsIObserver {
|
|
public:
|
|
static void StaticInit();
|
|
static ProcessPriorityManagerChild* Singleton();
|
|
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIOBSERVER
|
|
|
|
bool CurrentProcessIsForeground();
|
|
|
|
private:
|
|
static StaticRefPtr<ProcessPriorityManagerChild> sSingleton;
|
|
|
|
ProcessPriorityManagerChild();
|
|
~ProcessPriorityManagerChild() = default;
|
|
ProcessPriorityManagerChild(const ProcessPriorityManagerChild&) = delete;
|
|
|
|
const ProcessPriorityManagerChild& operator=(
|
|
const ProcessPriorityManagerChild&) = delete;
|
|
|
|
void Init();
|
|
|
|
hal::ProcessPriority mCachedPriority;
|
|
};
|
|
|
|
/**
|
|
* This class manages the priority of one particular process. It is
|
|
* main-process only.
|
|
*/
|
|
class ParticularProcessPriorityManager final : public WakeLockObserver,
|
|
public nsITimerCallback,
|
|
public nsINamed,
|
|
public nsSupportsWeakReference {
|
|
~ParticularProcessPriorityManager();
|
|
|
|
public:
|
|
explicit ParticularProcessPriorityManager(ContentParent* aContentParent);
|
|
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSITIMERCALLBACK
|
|
|
|
virtual void Notify(const WakeLockInformation& aInfo) override;
|
|
void Init();
|
|
|
|
int32_t Pid() const;
|
|
uint64_t ChildID() const;
|
|
|
|
/**
|
|
* Used in logging, this method returns the ContentParent's name followed by
|
|
* ", ". If we can't get the ContentParent's name for some reason, it
|
|
* returns an empty string.
|
|
*
|
|
* The reference returned here is guaranteed to be live until the next call
|
|
* to NameWithComma() or until the ParticularProcessPriorityManager is
|
|
* destroyed, whichever comes first.
|
|
*/
|
|
const nsAutoCString& NameWithComma();
|
|
|
|
ProcessPriority CurrentPriority();
|
|
ProcessPriority ComputePriority();
|
|
|
|
enum TimeoutPref {
|
|
BACKGROUND_PERCEIVABLE_GRACE_PERIOD,
|
|
BACKGROUND_GRACE_PERIOD,
|
|
};
|
|
|
|
void ScheduleResetPriority(TimeoutPref aTimeoutPref);
|
|
void ResetPriority();
|
|
void ResetPriorityNow();
|
|
void SetPriorityNow(ProcessPriority aPriority);
|
|
|
|
void BrowserPriorityChanged(BrowserParent* aBrowserParent, bool aPriority);
|
|
|
|
void ShutDown();
|
|
|
|
NS_IMETHOD GetName(nsACString& aName) override {
|
|
aName.AssignLiteral("ParticularProcessPriorityManager");
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
void FireTestOnlyObserverNotification(const char* aTopic, const char* aData);
|
|
|
|
bool IsHoldingWakeLock(const nsAString& aTopic);
|
|
|
|
ContentParent* mContentParent;
|
|
uint64_t mChildID;
|
|
ProcessPriority mPriority;
|
|
bool mHoldsCPUWakeLock;
|
|
bool mHoldsHighPriorityWakeLock;
|
|
bool mHoldsPlayingAudioWakeLock;
|
|
bool mHoldsPlayingVideoWakeLock;
|
|
|
|
/**
|
|
* Used to implement NameWithComma().
|
|
*/
|
|
nsAutoCString mNameWithComma;
|
|
|
|
nsCOMPtr<nsITimer> mResetPriorityTimer;
|
|
|
|
// This hashtable contains the list of high priority TabIds for this process.
|
|
nsTHashSet<uint64_t> mHighPriorityBrowserParents;
|
|
};
|
|
|
|
/* static */
|
|
bool ProcessPriorityManagerImpl::sInitialized = false;
|
|
/* static */
|
|
bool ProcessPriorityManagerImpl::sPrefListenersRegistered = false;
|
|
/* static */
|
|
StaticRefPtr<ProcessPriorityManagerImpl> ProcessPriorityManagerImpl::sSingleton;
|
|
|
|
NS_IMPL_ISUPPORTS(ProcessPriorityManagerImpl, nsIObserver,
|
|
nsISupportsWeakReference);
|
|
|
|
/* static */
|
|
void ProcessPriorityManagerImpl::PrefChangedCallback(const char* aPref,
|
|
void* aClosure) {
|
|
StaticInit();
|
|
if (!PrefsEnabled() && sSingleton) {
|
|
sSingleton = nullptr;
|
|
sInitialized = false;
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
bool ProcessPriorityManagerImpl::PrefsEnabled() {
|
|
return StaticPrefs::dom_ipc_processPriorityManager_enabled();
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManagerImpl::SetProcessPriorityIfEnabled(
|
|
int aPid, ProcessPriority aPriority) {
|
|
// The preference doesn't disable the process priority manager, but only its
|
|
// effect. This way the IPCs still happen and can be used to collect telemetry
|
|
// about CPU use.
|
|
if (PrefsEnabled()) {
|
|
hal::SetProcessPriority(aPid, aPriority);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
bool ProcessPriorityManagerImpl::TestMode() {
|
|
return StaticPrefs::dom_ipc_processPriorityManager_testMode();
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManagerImpl::StaticInit() {
|
|
if (sInitialized) {
|
|
return;
|
|
}
|
|
|
|
// The process priority manager is main-process only.
|
|
if (!XRE_IsParentProcess()) {
|
|
sInitialized = true;
|
|
return;
|
|
}
|
|
|
|
// Run StaticInit() again if the pref changes. We don't expect this to
|
|
// happen in normal operation, but it happens during testing.
|
|
if (!sPrefListenersRegistered) {
|
|
sPrefListenersRegistered = true;
|
|
Preferences::RegisterCallback(PrefChangedCallback,
|
|
"dom.ipc.processPriorityManager.enabled");
|
|
}
|
|
|
|
sInitialized = true;
|
|
|
|
sSingleton = new ProcessPriorityManagerImpl();
|
|
sSingleton->Init();
|
|
ClearOnShutdown(&sSingleton);
|
|
}
|
|
|
|
/* static */
|
|
ProcessPriorityManagerImpl* ProcessPriorityManagerImpl::GetSingleton() {
|
|
if (!sSingleton) {
|
|
StaticInit();
|
|
}
|
|
|
|
return sSingleton;
|
|
}
|
|
|
|
ProcessPriorityManagerImpl::ProcessPriorityManagerImpl() {
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
}
|
|
|
|
ProcessPriorityManagerImpl::~ProcessPriorityManagerImpl() = default;
|
|
|
|
void ProcessPriorityManagerImpl::Init() {
|
|
LOG("Starting up. This is the parent process.");
|
|
|
|
// The parent process's priority never changes; set it here and then forget
|
|
// about it. We'll manage only subprocesses' priorities using the process
|
|
// priority manager.
|
|
SetProcessPriorityIfEnabled(getpid(), PROCESS_PRIORITY_PARENT_PROCESS);
|
|
|
|
nsCOMPtr<nsIObserverService> os = services::GetObserverService();
|
|
if (os) {
|
|
os->AddObserver(this, "ipc:content-shutdown", /* ownsWeak */ true);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ProcessPriorityManagerImpl::Observe(nsISupports* aSubject, const char* aTopic,
|
|
const char16_t* aData) {
|
|
nsDependentCString topic(aTopic);
|
|
if (topic.EqualsLiteral("ipc:content-shutdown")) {
|
|
ObserveContentParentDestroyed(aSubject);
|
|
} else {
|
|
MOZ_ASSERT(false);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<ParticularProcessPriorityManager>
|
|
ProcessPriorityManagerImpl::GetParticularProcessPriorityManager(
|
|
ContentParent* aContentParent) {
|
|
// If this content parent is already being shut down, there's no
|
|
// need to adjust its priority.
|
|
if (aContentParent->IsDead()) {
|
|
return nullptr;
|
|
}
|
|
|
|
const uint64_t cpId = aContentParent->ChildID();
|
|
return mParticularManagers.WithEntryHandle(cpId, [&](auto&& entry) {
|
|
if (!entry) {
|
|
entry.Insert(new ParticularProcessPriorityManager(aContentParent));
|
|
entry.Data()->Init();
|
|
}
|
|
return do_AddRef(entry.Data());
|
|
});
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::SetProcessPriority(
|
|
ContentParent* aContentParent, ProcessPriority aPriority) {
|
|
MOZ_ASSERT(aContentParent);
|
|
if (RefPtr pppm = GetParticularProcessPriorityManager(aContentParent)) {
|
|
pppm->SetPriorityNow(aPriority);
|
|
}
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::ObserveContentParentDestroyed(
|
|
nsISupports* aSubject) {
|
|
nsCOMPtr<nsIPropertyBag2> props = do_QueryInterface(aSubject);
|
|
NS_ENSURE_TRUE_VOID(props);
|
|
|
|
uint64_t childID = CONTENT_PROCESS_ID_UNKNOWN;
|
|
props->GetPropertyAsUint64(u"childID"_ns, &childID);
|
|
NS_ENSURE_TRUE_VOID(childID != CONTENT_PROCESS_ID_UNKNOWN);
|
|
|
|
if (auto entry = mParticularManagers.Lookup(childID)) {
|
|
entry.Data()->ShutDown();
|
|
mHighPriorityChildIDs.Remove(childID);
|
|
entry.Remove();
|
|
}
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::NotifyProcessPriorityChanged(
|
|
ParticularProcessPriorityManager* aParticularManager,
|
|
ProcessPriority aOldPriority) {
|
|
ProcessPriority newPriority = aParticularManager->CurrentPriority();
|
|
|
|
if (newPriority >= PROCESS_PRIORITY_FOREGROUND_HIGH &&
|
|
aOldPriority < PROCESS_PRIORITY_FOREGROUND_HIGH) {
|
|
mHighPriorityChildIDs.Insert(aParticularManager->ChildID());
|
|
} else if (newPriority < PROCESS_PRIORITY_FOREGROUND_HIGH &&
|
|
aOldPriority >= PROCESS_PRIORITY_FOREGROUND_HIGH) {
|
|
mHighPriorityChildIDs.Remove(aParticularManager->ChildID());
|
|
}
|
|
}
|
|
|
|
static nsCString BCToString(dom::CanonicalBrowsingContext* aBC) {
|
|
nsCOMPtr<nsIURI> uri = aBC->GetCurrentURI();
|
|
return nsPrintfCString("id=%" PRIu64 " uri=%s active=%d pactive=%d",
|
|
aBC->Id(),
|
|
uri ? uri->GetSpecOrDefault().get() : "(no uri)",
|
|
aBC->IsActive(), aBC->IsPriorityActive());
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::BrowserPriorityChanged(
|
|
dom::CanonicalBrowsingContext* aBC, bool aPriority) {
|
|
MOZ_ASSERT(aBC->IsTop());
|
|
|
|
LOG("BrowserPriorityChanged(%s, %d)\n", BCToString(aBC).get(), aPriority);
|
|
|
|
bool alreadyActive = aBC->IsPriorityActive();
|
|
if (alreadyActive == aPriority) {
|
|
return;
|
|
}
|
|
|
|
Telemetry::ScalarAdd(
|
|
Telemetry::ScalarID::DOM_CONTENTPROCESS_OS_PRIORITY_CHANGE_CONSIDERED, 1);
|
|
|
|
aBC->SetPriorityActive(aPriority);
|
|
|
|
aBC->PreOrderWalk([&](BrowsingContext* aContext) {
|
|
CanonicalBrowsingContext* canonical = aContext->Canonical();
|
|
LOG("PreOrderWalk for %p: %p -> %p, %p\n", aBC, canonical,
|
|
canonical->GetContentParent(), canonical->GetBrowserParent());
|
|
if (ContentParent* cp = canonical->GetContentParent()) {
|
|
if (RefPtr pppm = GetParticularProcessPriorityManager(cp)) {
|
|
if (auto* bp = canonical->GetBrowserParent()) {
|
|
pppm->BrowserPriorityChanged(bp, aPriority);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::BrowserPriorityChanged(
|
|
BrowserParent* aBrowserParent, bool aPriority) {
|
|
LOG("BrowserPriorityChanged(bp=%p, %d)\n", aBrowserParent, aPriority);
|
|
|
|
if (RefPtr pppm =
|
|
GetParticularProcessPriorityManager(aBrowserParent->Manager())) {
|
|
Telemetry::ScalarAdd(
|
|
Telemetry::ScalarID::DOM_CONTENTPROCESS_OS_PRIORITY_CHANGE_CONSIDERED,
|
|
1);
|
|
pppm->BrowserPriorityChanged(aBrowserParent, aPriority);
|
|
}
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::ResetPriority(ContentParent* aContentParent) {
|
|
if (RefPtr pppm = GetParticularProcessPriorityManager(aContentParent)) {
|
|
pppm->ResetPriority();
|
|
}
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS(ParticularProcessPriorityManager, nsITimerCallback,
|
|
nsISupportsWeakReference, nsINamed);
|
|
|
|
ParticularProcessPriorityManager::ParticularProcessPriorityManager(
|
|
ContentParent* aContentParent)
|
|
: mContentParent(aContentParent),
|
|
mChildID(aContentParent->ChildID()),
|
|
mPriority(PROCESS_PRIORITY_UNKNOWN),
|
|
mHoldsCPUWakeLock(false),
|
|
mHoldsHighPriorityWakeLock(false),
|
|
mHoldsPlayingAudioWakeLock(false),
|
|
mHoldsPlayingVideoWakeLock(false) {
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
MOZ_RELEASE_ASSERT(!aContentParent->IsDead());
|
|
LOGP("Creating ParticularProcessPriorityManager.");
|
|
// Our static analysis doesn't allow capturing ref-counted pointers in
|
|
// lambdas, so we need to hide it in a uintptr_t. This is safe because this
|
|
// lambda will be destroyed in ~ParticularProcessPriorityManager().
|
|
uintptr_t self = reinterpret_cast<uintptr_t>(this);
|
|
profiler_add_state_change_callback(
|
|
AllProfilingStates(),
|
|
[self](ProfilingState aProfilingState) {
|
|
const ParticularProcessPriorityManager* selfPtr =
|
|
reinterpret_cast<const ParticularProcessPriorityManager*>(self);
|
|
PROFILER_MARKER("Subprocess Priority", OTHER,
|
|
MarkerThreadId::MainThread(), SubProcessPriority,
|
|
selfPtr->Pid(),
|
|
ProfilerString8View::WrapNullTerminatedString(
|
|
ProcessPriorityToString(selfPtr->mPriority)),
|
|
aProfilingState);
|
|
},
|
|
self);
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::Init() {
|
|
RegisterWakeLockObserver(this);
|
|
|
|
// This process may already hold the CPU lock; for example, our parent may
|
|
// have acquired it on our behalf.
|
|
mHoldsCPUWakeLock = IsHoldingWakeLock(u"cpu"_ns);
|
|
mHoldsHighPriorityWakeLock = IsHoldingWakeLock(u"high-priority"_ns);
|
|
mHoldsPlayingAudioWakeLock = IsHoldingWakeLock(u"audio-playing"_ns);
|
|
mHoldsPlayingVideoWakeLock = IsHoldingWakeLock(u"video-playing"_ns);
|
|
|
|
LOGP(
|
|
"Done starting up. mHoldsCPUWakeLock=%d, "
|
|
"mHoldsHighPriorityWakeLock=%d, mHoldsPlayingAudioWakeLock=%d, "
|
|
"mHoldsPlayingVideoWakeLock=%d",
|
|
mHoldsCPUWakeLock, mHoldsHighPriorityWakeLock, mHoldsPlayingAudioWakeLock,
|
|
mHoldsPlayingVideoWakeLock);
|
|
}
|
|
|
|
bool ParticularProcessPriorityManager::IsHoldingWakeLock(
|
|
const nsAString& aTopic) {
|
|
WakeLockInformation info;
|
|
GetWakeLockInfo(aTopic, &info);
|
|
return info.lockingProcesses().Contains(ChildID());
|
|
}
|
|
|
|
ParticularProcessPriorityManager::~ParticularProcessPriorityManager() {
|
|
LOGP("Destroying ParticularProcessPriorityManager.");
|
|
|
|
profiler_remove_state_change_callback(reinterpret_cast<uintptr_t>(this));
|
|
|
|
ShutDown();
|
|
}
|
|
|
|
/* virtual */
|
|
void ParticularProcessPriorityManager::Notify(
|
|
const WakeLockInformation& aInfo) {
|
|
if (!mContentParent) {
|
|
// We've been shut down.
|
|
return;
|
|
}
|
|
|
|
bool* dest = nullptr;
|
|
if (aInfo.topic().EqualsLiteral("cpu")) {
|
|
dest = &mHoldsCPUWakeLock;
|
|
} else if (aInfo.topic().EqualsLiteral("high-priority")) {
|
|
dest = &mHoldsHighPriorityWakeLock;
|
|
} else if (aInfo.topic().EqualsLiteral("audio-playing")) {
|
|
dest = &mHoldsPlayingAudioWakeLock;
|
|
} else if (aInfo.topic().EqualsLiteral("video-playing")) {
|
|
dest = &mHoldsPlayingVideoWakeLock;
|
|
}
|
|
|
|
if (dest) {
|
|
bool thisProcessLocks = aInfo.lockingProcesses().Contains(ChildID());
|
|
if (thisProcessLocks != *dest) {
|
|
*dest = thisProcessLocks;
|
|
LOGP(
|
|
"Got wake lock changed event. "
|
|
"Now mHoldsCPUWakeLock=%d, mHoldsHighPriorityWakeLock=%d, "
|
|
"mHoldsPlayingAudioWakeLock=%d, mHoldsPlayingVideoWakeLock=%d",
|
|
mHoldsCPUWakeLock, mHoldsHighPriorityWakeLock,
|
|
mHoldsPlayingAudioWakeLock, mHoldsPlayingVideoWakeLock);
|
|
ResetPriority();
|
|
}
|
|
}
|
|
}
|
|
|
|
uint64_t ParticularProcessPriorityManager::ChildID() const {
|
|
// We have to cache mContentParent->ChildID() instead of getting it from the
|
|
// ContentParent each time because after ShutDown() is called, mContentParent
|
|
// is null. If we didn't cache ChildID(), then we wouldn't be able to run
|
|
// LOGP() after ShutDown().
|
|
return mChildID;
|
|
}
|
|
|
|
int32_t ParticularProcessPriorityManager::Pid() const {
|
|
return mContentParent ? mContentParent->Pid() : -1;
|
|
}
|
|
|
|
const nsAutoCString& ParticularProcessPriorityManager::NameWithComma() {
|
|
mNameWithComma.Truncate();
|
|
if (!mContentParent) {
|
|
return mNameWithComma; // empty string
|
|
}
|
|
|
|
nsAutoString name;
|
|
mContentParent->FriendlyName(name);
|
|
if (name.IsEmpty()) {
|
|
return mNameWithComma; // empty string
|
|
}
|
|
|
|
CopyUTF16toUTF8(name, mNameWithComma);
|
|
mNameWithComma.AppendLiteral(", ");
|
|
return mNameWithComma;
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::ResetPriority() {
|
|
ProcessPriority processPriority = ComputePriority();
|
|
if (mPriority == PROCESS_PRIORITY_UNKNOWN || mPriority > processPriority) {
|
|
// Apps set at a perceivable background priority are often playing media.
|
|
// Most media will have short gaps while changing tracks between songs,
|
|
// switching videos, etc. Give these apps a longer grace period so they
|
|
// can get their next track started, if there is one, before getting
|
|
// downgraded.
|
|
if (mPriority == PROCESS_PRIORITY_BACKGROUND_PERCEIVABLE) {
|
|
ScheduleResetPriority(BACKGROUND_PERCEIVABLE_GRACE_PERIOD);
|
|
} else {
|
|
ScheduleResetPriority(BACKGROUND_GRACE_PERIOD);
|
|
}
|
|
return;
|
|
}
|
|
|
|
SetPriorityNow(processPriority);
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::ResetPriorityNow() {
|
|
SetPriorityNow(ComputePriority());
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::ScheduleResetPriority(
|
|
TimeoutPref aTimeoutPref) {
|
|
if (mResetPriorityTimer) {
|
|
LOGP("ScheduleResetPriority bailing; the timer is already running.");
|
|
return;
|
|
}
|
|
|
|
uint32_t timeout = 0;
|
|
switch (aTimeoutPref) {
|
|
case BACKGROUND_PERCEIVABLE_GRACE_PERIOD:
|
|
timeout = StaticPrefs::
|
|
dom_ipc_processPriorityManager_backgroundPerceivableGracePeriodMS();
|
|
break;
|
|
case BACKGROUND_GRACE_PERIOD:
|
|
timeout =
|
|
StaticPrefs::dom_ipc_processPriorityManager_backgroundGracePeriodMS();
|
|
break;
|
|
default:
|
|
MOZ_ASSERT(false, "Unrecognized timeout pref");
|
|
break;
|
|
}
|
|
|
|
LOGP("Scheduling reset timer to fire in %dms.", timeout);
|
|
NS_NewTimerWithCallback(getter_AddRefs(mResetPriorityTimer), this, timeout,
|
|
nsITimer::TYPE_ONE_SHOT);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ParticularProcessPriorityManager::Notify(nsITimer* aTimer) {
|
|
LOGP("Reset priority timer callback; about to ResetPriorityNow.");
|
|
ResetPriorityNow();
|
|
mResetPriorityTimer = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
ProcessPriority ParticularProcessPriorityManager::CurrentPriority() {
|
|
return mPriority;
|
|
}
|
|
|
|
ProcessPriority ParticularProcessPriorityManager::ComputePriority() {
|
|
if (!mHighPriorityBrowserParents.IsEmpty() ||
|
|
mContentParent->GetRemoteType() == EXTENSION_REMOTE_TYPE ||
|
|
mHoldsPlayingAudioWakeLock) {
|
|
return PROCESS_PRIORITY_FOREGROUND;
|
|
}
|
|
|
|
if (mHoldsCPUWakeLock || mHoldsHighPriorityWakeLock ||
|
|
mHoldsPlayingVideoWakeLock) {
|
|
return PROCESS_PRIORITY_BACKGROUND_PERCEIVABLE;
|
|
}
|
|
|
|
return PROCESS_PRIORITY_BACKGROUND;
|
|
}
|
|
|
|
#ifdef XP_MACOSX
|
|
// Method used for setting QoS levels on background main threads.
|
|
static bool PriorityUsesLowPowerMainThread(
|
|
const hal::ProcessPriority& aPriority) {
|
|
return aPriority == hal::PROCESS_PRIORITY_BACKGROUND ||
|
|
aPriority == hal::PROCESS_PRIORITY_PREALLOC;
|
|
}
|
|
// Method reduces redundancy in pref check while addressing the edge case
|
|
// where a pref is flipped to false during active browser use.
|
|
static bool PrefsUseLowPriorityThreads() {
|
|
return StaticPrefs::threads_use_low_power_enabled() &&
|
|
StaticPrefs::threads_lower_mainthread_priority_in_background_enabled();
|
|
}
|
|
#endif
|
|
|
|
void ParticularProcessPriorityManager::SetPriorityNow(
|
|
ProcessPriority aPriority) {
|
|
if (aPriority == PROCESS_PRIORITY_UNKNOWN) {
|
|
MOZ_ASSERT(false);
|
|
return;
|
|
}
|
|
|
|
LOGP("Changing priority from %s to %s (cp=%p).",
|
|
ProcessPriorityToString(mPriority), ProcessPriorityToString(aPriority),
|
|
mContentParent);
|
|
|
|
if (!mContentParent || mPriority == aPriority) {
|
|
return;
|
|
}
|
|
|
|
PROFILER_MARKER(
|
|
"Subprocess Priority", OTHER,
|
|
MarkerOptions(MarkerThreadId::MainThread(), MarkerStack::Capture()),
|
|
SubProcessPriorityChange, this->Pid(),
|
|
ProfilerString8View::WrapNullTerminatedString(
|
|
ProcessPriorityToString(mPriority)),
|
|
ProfilerString8View::WrapNullTerminatedString(
|
|
ProcessPriorityToString(aPriority)));
|
|
|
|
ProcessPriority oldPriority = mPriority;
|
|
|
|
mPriority = aPriority;
|
|
|
|
// We skip incrementing the DOM_CONTENTPROCESS_OS_PRIORITY_RAISED if we're
|
|
// transitioning from the PROCESS_PRIORITY_UNKNOWN level, which is where
|
|
// we initialize at.
|
|
if (oldPriority < mPriority && oldPriority != PROCESS_PRIORITY_UNKNOWN) {
|
|
Telemetry::ScalarAdd(
|
|
Telemetry::ScalarID::DOM_CONTENTPROCESS_OS_PRIORITY_RAISED, 1);
|
|
} else if (oldPriority > mPriority) {
|
|
Telemetry::ScalarAdd(
|
|
Telemetry::ScalarID::DOM_CONTENTPROCESS_OS_PRIORITY_LOWERED, 1);
|
|
}
|
|
|
|
ProcessPriorityManagerImpl::SetProcessPriorityIfEnabled(Pid(), mPriority);
|
|
|
|
if (oldPriority != mPriority) {
|
|
ProcessPriorityManagerImpl::GetSingleton()->NotifyProcessPriorityChanged(
|
|
this, oldPriority);
|
|
|
|
#ifdef XP_MACOSX
|
|
// In cases where we have low-power threads enabled (such as on MacOS) we
|
|
// can go ahead and put the main thread in the background here. If the new
|
|
// priority is the background priority, we can tell the OS to put the main
|
|
// thread on low-power cores. Alternately, if we are changing from the
|
|
// background to a higher priority, we change the main thread back to its
|
|
// normal state.
|
|
//
|
|
// The messages for this will be relayed using the ProcessHangMonitor such
|
|
// that the priority can be raised even if the main thread is unresponsive.
|
|
if (PriorityUsesLowPowerMainThread(mPriority) !=
|
|
(PriorityUsesLowPowerMainThread(oldPriority))) {
|
|
if (PriorityUsesLowPowerMainThread(mPriority) &&
|
|
PrefsUseLowPriorityThreads()) {
|
|
mContentParent->SetMainThreadQoSPriority(nsIThread::QOS_PRIORITY_LOW);
|
|
} else if (PriorityUsesLowPowerMainThread(oldPriority)) {
|
|
// In the event that the user changes prefs while tabs are in the
|
|
// background, we still want to have the ability to put the main thread
|
|
// back in the foreground to keep tabs from being stuck in the
|
|
// background priority.
|
|
mContentParent->SetMainThreadQoSPriority(
|
|
nsIThread::QOS_PRIORITY_NORMAL);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
Unused << mContentParent->SendNotifyProcessPriorityChanged(mPriority);
|
|
}
|
|
|
|
FireTestOnlyObserverNotification("process-priority-set",
|
|
ProcessPriorityToString(mPriority));
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::BrowserPriorityChanged(
|
|
BrowserParent* aBrowserParent, bool aPriority) {
|
|
MOZ_ASSERT(aBrowserParent);
|
|
|
|
if (!aPriority) {
|
|
mHighPriorityBrowserParents.Remove(aBrowserParent->GetTabId());
|
|
} else {
|
|
mHighPriorityBrowserParents.Insert(aBrowserParent->GetTabId());
|
|
}
|
|
|
|
ResetPriority();
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::ShutDown() {
|
|
LOGP("shutdown for %p (mContentParent %p)", this, mContentParent);
|
|
|
|
// Unregister our wake lock observer if ShutDown hasn't been called. (The
|
|
// wake lock observer takes raw refs, so we don't want to take chances here!)
|
|
// We don't call UnregisterWakeLockObserver unconditionally because the code
|
|
// will print a warning if it's called unnecessarily.
|
|
if (mContentParent) {
|
|
UnregisterWakeLockObserver(this);
|
|
}
|
|
|
|
if (mResetPriorityTimer) {
|
|
mResetPriorityTimer->Cancel();
|
|
mResetPriorityTimer = nullptr;
|
|
}
|
|
|
|
mContentParent = nullptr;
|
|
}
|
|
|
|
void ProcessPriorityManagerImpl::FireTestOnlyObserverNotification(
|
|
const char* aTopic, const nsACString& aData) {
|
|
if (!TestMode()) {
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsIObserverService> os = services::GetObserverService();
|
|
NS_ENSURE_TRUE_VOID(os);
|
|
|
|
nsPrintfCString topic("process-priority-manager:TEST-ONLY:%s", aTopic);
|
|
|
|
LOG("Notifying observer %s, data %s", topic.get(),
|
|
PromiseFlatCString(aData).get());
|
|
os->NotifyObservers(nullptr, topic.get(), NS_ConvertUTF8toUTF16(aData).get());
|
|
}
|
|
|
|
void ParticularProcessPriorityManager::FireTestOnlyObserverNotification(
|
|
const char* aTopic, const char* aData) {
|
|
MOZ_ASSERT(aData, "Pass in data");
|
|
|
|
if (!ProcessPriorityManagerImpl::TestMode()) {
|
|
return;
|
|
}
|
|
|
|
nsAutoCString data(nsPrintfCString("%" PRIu64, ChildID()));
|
|
data.Append(':');
|
|
data.AppendASCII(aData);
|
|
|
|
// ProcessPriorityManagerImpl::GetSingleton() is guaranteed not to return
|
|
// null, since ProcessPriorityManagerImpl is the only class which creates
|
|
// ParticularProcessPriorityManagers.
|
|
|
|
ProcessPriorityManagerImpl::GetSingleton()->FireTestOnlyObserverNotification(
|
|
aTopic, data);
|
|
}
|
|
|
|
StaticRefPtr<ProcessPriorityManagerChild>
|
|
ProcessPriorityManagerChild::sSingleton;
|
|
|
|
/* static */
|
|
void ProcessPriorityManagerChild::StaticInit() {
|
|
if (!sSingleton) {
|
|
sSingleton = new ProcessPriorityManagerChild();
|
|
sSingleton->Init();
|
|
ClearOnShutdown(&sSingleton);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
ProcessPriorityManagerChild* ProcessPriorityManagerChild::Singleton() {
|
|
StaticInit();
|
|
return sSingleton;
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS(ProcessPriorityManagerChild, nsIObserver)
|
|
|
|
ProcessPriorityManagerChild::ProcessPriorityManagerChild() {
|
|
if (XRE_IsParentProcess()) {
|
|
mCachedPriority = PROCESS_PRIORITY_PARENT_PROCESS;
|
|
} else {
|
|
mCachedPriority = PROCESS_PRIORITY_UNKNOWN;
|
|
}
|
|
}
|
|
|
|
void ProcessPriorityManagerChild::Init() {
|
|
// The process priority should only be changed in child processes; don't even
|
|
// bother listening for changes if we're in the main process.
|
|
if (!XRE_IsParentProcess()) {
|
|
nsCOMPtr<nsIObserverService> os = services::GetObserverService();
|
|
NS_ENSURE_TRUE_VOID(os);
|
|
os->AddObserver(this, "ipc:process-priority-changed", /* weak = */ false);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ProcessPriorityManagerChild::Observe(nsISupports* aSubject, const char* aTopic,
|
|
const char16_t* aData) {
|
|
MOZ_ASSERT(!strcmp(aTopic, "ipc:process-priority-changed"));
|
|
|
|
nsCOMPtr<nsIPropertyBag2> props = do_QueryInterface(aSubject);
|
|
NS_ENSURE_TRUE(props, NS_OK);
|
|
|
|
int32_t priority = static_cast<int32_t>(PROCESS_PRIORITY_UNKNOWN);
|
|
props->GetPropertyAsInt32(u"priority"_ns, &priority);
|
|
NS_ENSURE_TRUE(ProcessPriority(priority) != PROCESS_PRIORITY_UNKNOWN, NS_OK);
|
|
|
|
mCachedPriority = static_cast<ProcessPriority>(priority);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
bool ProcessPriorityManagerChild::CurrentProcessIsForeground() {
|
|
return mCachedPriority == PROCESS_PRIORITY_UNKNOWN ||
|
|
mCachedPriority >= PROCESS_PRIORITY_FOREGROUND;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace mozilla {
|
|
|
|
/* static */
|
|
void ProcessPriorityManager::Init() {
|
|
ProcessPriorityManagerImpl::StaticInit();
|
|
ProcessPriorityManagerChild::StaticInit();
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManager::SetProcessPriority(ContentParent* aContentParent,
|
|
ProcessPriority aPriority) {
|
|
MOZ_ASSERT(aContentParent);
|
|
MOZ_ASSERT(aContentParent->Pid() != -1);
|
|
|
|
ProcessPriorityManagerImpl* singleton =
|
|
ProcessPriorityManagerImpl::GetSingleton();
|
|
if (singleton) {
|
|
singleton->SetProcessPriority(aContentParent, aPriority);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
bool ProcessPriorityManager::CurrentProcessIsForeground() {
|
|
return ProcessPriorityManagerChild::Singleton()->CurrentProcessIsForeground();
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManager::BrowserPriorityChanged(
|
|
CanonicalBrowsingContext* aBC, bool aPriority) {
|
|
if (auto* singleton = ProcessPriorityManagerImpl::GetSingleton()) {
|
|
singleton->BrowserPriorityChanged(aBC, aPriority);
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManager::BrowserPriorityChanged(
|
|
BrowserParent* aBrowserParent, bool aPriority) {
|
|
MOZ_ASSERT(aBrowserParent);
|
|
|
|
ProcessPriorityManagerImpl* singleton =
|
|
ProcessPriorityManagerImpl::GetSingleton();
|
|
if (!singleton) {
|
|
return;
|
|
}
|
|
singleton->BrowserPriorityChanged(aBrowserParent, aPriority);
|
|
}
|
|
|
|
/* static */
|
|
void ProcessPriorityManager::RemoteBrowserFrameShown(
|
|
nsFrameLoader* aFrameLoader) {
|
|
ProcessPriorityManagerImpl* singleton =
|
|
ProcessPriorityManagerImpl::GetSingleton();
|
|
if (!singleton) {
|
|
return;
|
|
}
|
|
|
|
BrowserParent* bp = BrowserParent::GetFrom(aFrameLoader);
|
|
NS_ENSURE_TRUE_VOID(bp);
|
|
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
|
|
// Ignore calls that aren't from a Browser.
|
|
if (!aFrameLoader->OwnerIsMozBrowserFrame()) {
|
|
return;
|
|
}
|
|
|
|
singleton->ResetPriority(bp->Manager());
|
|
}
|
|
|
|
} // namespace mozilla
|