fune/dom/ipc/ProcessPriorityManager.cpp
kriswright 51331a8ca5 Bug 1834629 - Set main thread QoS Priority from another thread. r=mccr8
This bug uses the existing process hang messaging infrastructure to raise priorities on main threads. To ensure reduced latency, we also increase the priority of ProcessHangMon threads. We also address an edge-case bug where flipping the QoS prefs during use might lead to a tab getting stuck at the wrong priority.

Due to increasing the priority of the hang monitor thread, we may see some increase in its utilization on MacOS during high CPU load. I'm not sure the extent of how this may impact the browser, but as it makes the thread "faster" it may be more responsive than some other threads during this case.

I tested thread responsiveness by using the stress tool and dispatching various numbers of worker threads, up to 250. During these tests, even when other parts of firefox were less responsive under stress, tab switching appeared to remain snappy and responsive.

I captured some updated power profiles using the change. Profile where pref flipped off during use: https://share.firefox.dev/46BksO3 (Note that we start with the prefs on, then we flip off the prefs half way and repeat the same behavior to observe the fix to the previous bug that left tabs trapped in the background)

Profile with the pref fully enabled: https://share.firefox.dev/46EBIC7

In regards to the edge case, to avoid spurious tab wakeups, we won't reinstate normal thread priority when pref is disabled until the tab is interacted with again.

Differential Revision: https://phabricator.services.mozilla.com/D182787
2023-07-14 16:13:00 +00:00

1067 lines
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