By having CollectPerformanceInfo traverse the BrowsingContextGroup and tree of BrowsingContexts we can hide the task of getting the DocGroups befind an API in BrowsingContextGroup and ease the removal of TabGroups. Differential Revision: https://phabricator.services.mozilla.com/D64390
143 lines
4.4 KiB
C++
143 lines
4.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 "nsThreadUtils.h"
|
|
#include "mozilla/PerformanceUtils.h"
|
|
#include "mozilla/ResultExtensions.h"
|
|
#include "mozilla/dom/DocGroup.h"
|
|
#include "mozilla/dom/BrowserChild.h"
|
|
#include "mozilla/dom/BrowsingContextGroup.h"
|
|
#include "mozilla/dom/ContentChild.h"
|
|
#include "mozilla/dom/WorkerDebugger.h"
|
|
#include "mozilla/dom/WorkerDebuggerManager.h"
|
|
|
|
#include "MediaDecoder.h"
|
|
#include "XPCJSMemoryReporter.h"
|
|
#include "jsfriendapi.h"
|
|
#include "js/MemoryMetrics.h"
|
|
#include "nsWindowMemoryReporter.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
namespace mozilla {
|
|
|
|
nsTArray<RefPtr<PerformanceInfoPromise>> CollectPerformanceInfo() {
|
|
nsTArray<RefPtr<PerformanceInfoPromise>> promises;
|
|
|
|
// collecting ReportPerformanceInfo from all WorkerDebugger instances
|
|
RefPtr<mozilla::dom::WorkerDebuggerManager> wdm =
|
|
WorkerDebuggerManager::GetOrCreate();
|
|
if (NS_WARN_IF(!wdm)) {
|
|
return promises;
|
|
}
|
|
|
|
for (uint32_t i = 0; i < wdm->GetDebuggersLength(); i++) {
|
|
const RefPtr<WorkerDebugger> debugger = wdm->GetDebuggerAt(i);
|
|
promises.AppendElement(debugger->ReportPerformanceInfo());
|
|
}
|
|
|
|
nsTArray<RefPtr<BrowsingContextGroup>> groups;
|
|
if (XRE_IsContentProcess()) {
|
|
groups.AppendElements(
|
|
ContentChild::GetSingleton()->BrowsingContextGroups());
|
|
} else {
|
|
groups.AppendElements(ContentParent::BrowsingContextGroups());
|
|
}
|
|
|
|
nsTArray<DocGroup*> docGroups;
|
|
for (auto& browsingContextGroup : groups) {
|
|
browsingContextGroup->GetDocGroups(docGroups);
|
|
}
|
|
|
|
for (DocGroup* docGroup : docGroups) {
|
|
promises.AppendElement(docGroup->ReportPerformanceInfo());
|
|
}
|
|
|
|
return promises;
|
|
}
|
|
|
|
void AddWindowTabSizes(nsGlobalWindowOuter* aWindow, nsTabSizes* aSizes) {
|
|
Document* document = aWindow->GetDocument();
|
|
if (document && document->GetCachedSizes(aSizes)) {
|
|
// We got a cached version
|
|
return;
|
|
}
|
|
// We measure the sizes on a fresh nsTabSizes instance
|
|
// because we want to cache the value and aSizes might
|
|
// already have some values from other windows.
|
|
nsTabSizes sizes;
|
|
|
|
// Measure the window.
|
|
SizeOfState state(moz_malloc_size_of);
|
|
nsWindowSizes windowSizes(state);
|
|
aWindow->AddSizeOfIncludingThis(windowSizes);
|
|
// Measure the inner window, if there is one.
|
|
nsGlobalWindowInner* inner = aWindow->GetCurrentInnerWindowInternal();
|
|
if (inner != nullptr) {
|
|
inner->AddSizeOfIncludingThis(windowSizes);
|
|
}
|
|
windowSizes.addToTabSizes(&sizes);
|
|
if (document) {
|
|
document->SetCachedSizes(&sizes);
|
|
}
|
|
aSizes->mDom += sizes.mDom;
|
|
aSizes->mStyle += sizes.mStyle;
|
|
aSizes->mOther += sizes.mOther;
|
|
}
|
|
|
|
nsresult GetTabSizes(nsGlobalWindowOuter* aWindow, nsTabSizes* aSizes) {
|
|
// Add the window (and inner window) sizes. Might be cached.
|
|
AddWindowTabSizes(aWindow, aSizes);
|
|
|
|
BrowsingContext* bc = aWindow->GetBrowsingContext();
|
|
if (!bc) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Measure this window's descendents.
|
|
for (const auto& frame : bc->GetChildren()) {
|
|
if (auto* childWin = nsGlobalWindowOuter::Cast(frame->GetDOMWindow())) {
|
|
MOZ_TRY(GetTabSizes(childWin, aSizes));
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
RefPtr<MemoryPromise> CollectMemoryInfo(
|
|
const nsCOMPtr<nsPIDOMWindowOuter>& aWindow,
|
|
const RefPtr<AbstractThread>& aEventTarget) {
|
|
// Getting Dom sizes. -- XXX should we reimplement GetTabSizes to async here ?
|
|
nsGlobalWindowOuter* window = nsGlobalWindowOuter::Cast(aWindow);
|
|
nsTabSizes sizes;
|
|
nsresult rv = GetTabSizes(window, &sizes);
|
|
if (NS_FAILED(rv)) {
|
|
return MemoryPromise::CreateAndReject(rv, __func__);
|
|
}
|
|
|
|
// Getting GC Heap Usage
|
|
JSObject* obj = window->GetGlobalJSObject();
|
|
uint64_t GCHeapUsage = 0;
|
|
if (obj != nullptr) {
|
|
GCHeapUsage = js::GetGCHeapUsageForObjectZone(obj);
|
|
}
|
|
|
|
// Getting Media sizes.
|
|
return GetMediaMemorySizes()->Then(
|
|
aEventTarget, __func__,
|
|
[GCHeapUsage, sizes](const MediaMemoryInfo& media) {
|
|
return MemoryPromise::CreateAndResolve(
|
|
PerformanceMemoryInfo(media, sizes.mDom, sizes.mStyle, sizes.mOther,
|
|
GCHeapUsage),
|
|
__func__);
|
|
},
|
|
[](const nsresult rv) {
|
|
return MemoryPromise::CreateAndReject(rv, __func__);
|
|
});
|
|
}
|
|
|
|
} // namespace mozilla
|