зеркало из https://github.com/mozilla/gecko-dev.git
219 строки
6.7 KiB
C++
219 строки
6.7 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/dom/TabGroup.h"
|
|
|
|
#include "mozilla/dom/ContentChild.h"
|
|
#include "mozilla/dom/BrowserChild.h"
|
|
#include "mozilla/dom/DocGroup.h"
|
|
#include "mozilla/dom/TimeoutManager.h"
|
|
#include "mozilla/AbstractThread.h"
|
|
#include "mozilla/ClearOnShutdown.h"
|
|
#include "mozilla/StaticPtr.h"
|
|
#include "mozilla/Telemetry.h"
|
|
#include "mozilla/ThrottledEventQueue.h"
|
|
#include "nsIDocShell.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
static StaticRefPtr<TabGroup> sChromeTabGroup;
|
|
|
|
LinkedList<TabGroup>* TabGroup::sTabGroups = nullptr;
|
|
|
|
TabGroup::TabGroup(bool aIsChrome)
|
|
: mLastWindowLeft(false),
|
|
mThrottledQueuesInitialized(false),
|
|
mIsChrome(aIsChrome) {
|
|
if (!sTabGroups) {
|
|
sTabGroups = new LinkedList<TabGroup>();
|
|
}
|
|
sTabGroups->insertBack(this);
|
|
|
|
CreateEventTargets(/* aNeedValidation = */ !aIsChrome);
|
|
|
|
// Do not throttle runnables from chrome windows. In theory we should
|
|
// not have abuse issues from these windows and many browser chrome
|
|
// tests have races that fail if we do throttle chrome runnables.
|
|
if (aIsChrome) {
|
|
MOZ_ASSERT(!sChromeTabGroup);
|
|
return;
|
|
}
|
|
|
|
// This constructor can be called from the IPC I/O thread. In that case, we
|
|
// won't actually use the TabGroup on the main thread until GetFromWindowActor
|
|
// is called, so we initialize the throttled queues there.
|
|
if (NS_IsMainThread()) {
|
|
EnsureThrottledEventQueues();
|
|
}
|
|
}
|
|
|
|
TabGroup::~TabGroup() {
|
|
MOZ_ASSERT(mDocGroups.IsEmpty());
|
|
MOZ_ASSERT(mWindows.IsEmpty());
|
|
MOZ_RELEASE_ASSERT(mLastWindowLeft || mIsChrome);
|
|
|
|
LinkedListElement<TabGroup>* listElement =
|
|
static_cast<LinkedListElement<TabGroup>*>(this);
|
|
listElement->remove();
|
|
|
|
if (sTabGroups->isEmpty()) {
|
|
delete sTabGroups;
|
|
sTabGroups = nullptr;
|
|
}
|
|
}
|
|
|
|
void TabGroup::EnsureThrottledEventQueues() {
|
|
if (mThrottledQueuesInitialized) {
|
|
return;
|
|
}
|
|
|
|
mThrottledQueuesInitialized = true;
|
|
|
|
for (size_t i = 0; i < size_t(TaskCategory::Count); i++) {
|
|
TaskCategory category = static_cast<TaskCategory>(i);
|
|
if (category == TaskCategory::Worker) {
|
|
mEventTargets[i] = ThrottledEventQueue::Create(mEventTargets[i],
|
|
"TabGroup worker queue");
|
|
} else if (category == TaskCategory::Timer) {
|
|
mEventTargets[i] =
|
|
ThrottledEventQueue::Create(mEventTargets[i], "TabGroup timer queue");
|
|
}
|
|
}
|
|
}
|
|
|
|
/* static */
|
|
TabGroup* TabGroup::GetChromeTabGroup() {
|
|
if (!sChromeTabGroup) {
|
|
sChromeTabGroup = new TabGroup(true /* chrome tab group */);
|
|
ClearOnShutdown(&sChromeTabGroup);
|
|
}
|
|
return sChromeTabGroup;
|
|
}
|
|
|
|
/* static */
|
|
TabGroup* TabGroup::GetFromWindow(mozIDOMWindowProxy* aWindow) {
|
|
if (BrowserChild* browserChild = BrowserChild::GetFrom(aWindow)) {
|
|
return browserChild->TabGroup();
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
/* static */
|
|
TabGroup* TabGroup::GetFromActor(BrowserChild* aBrowserChild) {
|
|
MOZ_RELEASE_ASSERT(NS_IsMainThread());
|
|
|
|
// Middleman processes do not assign event targets to their tab children.
|
|
if (recordreplay::IsMiddleman()) {
|
|
return GetChromeTabGroup();
|
|
}
|
|
|
|
nsCOMPtr<nsIEventTarget> target =
|
|
aBrowserChild->Manager()->GetEventTargetFor(aBrowserChild);
|
|
if (!target) {
|
|
return nullptr;
|
|
}
|
|
|
|
// We have an event target. We assume the IPC code created it via
|
|
// TabGroup::CreateEventTarget.
|
|
RefPtr<SchedulerGroup> group = SchedulerGroup::FromEventTarget(target);
|
|
MOZ_RELEASE_ASSERT(group);
|
|
auto tabGroup = group->AsTabGroup();
|
|
MOZ_RELEASE_ASSERT(tabGroup);
|
|
|
|
// We delay creating the event targets until now since the TabGroup
|
|
// constructor ran off the main thread.
|
|
tabGroup->EnsureThrottledEventQueues();
|
|
|
|
return tabGroup;
|
|
}
|
|
|
|
already_AddRefed<DocGroup> TabGroup::GetDocGroup(const nsACString& aKey) {
|
|
RefPtr<DocGroup> docGroup(mDocGroups.GetEntry(aKey)->mDocGroup);
|
|
return docGroup.forget();
|
|
}
|
|
|
|
already_AddRefed<DocGroup> TabGroup::AddDocument(const nsACString& aKey,
|
|
Document* aDocument) {
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
HashEntry* entry = mDocGroups.PutEntry(aKey);
|
|
RefPtr<DocGroup> docGroup;
|
|
if (entry->mDocGroup) {
|
|
docGroup = entry->mDocGroup;
|
|
} else {
|
|
const nsID agentClusterId = nsContentUtils::GenerateUUID();
|
|
|
|
docGroup = new DocGroup(this, aKey, agentClusterId);
|
|
entry->mDocGroup = docGroup;
|
|
}
|
|
|
|
// Make sure that the hashtable was updated and now contains the correct value
|
|
MOZ_ASSERT(RefPtr<DocGroup>(GetDocGroup(aKey)) == docGroup);
|
|
|
|
docGroup->mDocuments.AppendElement(aDocument);
|
|
|
|
return docGroup.forget();
|
|
}
|
|
|
|
/* static */
|
|
already_AddRefed<TabGroup> TabGroup::Join(nsPIDOMWindowOuter* aWindow,
|
|
TabGroup* aTabGroup) {
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
RefPtr<TabGroup> tabGroup = aTabGroup;
|
|
if (!tabGroup) {
|
|
tabGroup = new TabGroup();
|
|
}
|
|
MOZ_RELEASE_ASSERT(!tabGroup->mLastWindowLeft);
|
|
MOZ_ASSERT(!tabGroup->mWindows.Contains(aWindow));
|
|
tabGroup->mWindows.AppendElement(aWindow);
|
|
|
|
return tabGroup.forget();
|
|
}
|
|
|
|
void TabGroup::Leave(nsPIDOMWindowOuter* aWindow) {
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
MOZ_ASSERT(mWindows.Contains(aWindow));
|
|
mWindows.RemoveElement(aWindow);
|
|
|
|
MaybeDestroy();
|
|
}
|
|
|
|
void TabGroup::MaybeDestroy() {
|
|
// The Chrome TabGroup doesn't have cyclical references through mEventTargets
|
|
// to itself, meaning that we don't have to worry about nulling mEventTargets
|
|
// out after the last window leaves.
|
|
if (!mIsChrome && !mLastWindowLeft && mWindows.IsEmpty()) {
|
|
mLastWindowLeft = true;
|
|
Shutdown(false);
|
|
}
|
|
}
|
|
|
|
TabGroup::HashEntry::HashEntry(const nsACString* aKey)
|
|
: nsCStringHashKey(aKey), mDocGroup(nullptr) {}
|
|
|
|
nsISerialEventTarget* TabGroup::EventTargetFor(TaskCategory aCategory) const {
|
|
if (aCategory == TaskCategory::Worker || aCategory == TaskCategory::Timer) {
|
|
MOZ_RELEASE_ASSERT(mThrottledQueuesInitialized || mIsChrome);
|
|
}
|
|
return SchedulerGroup::EventTargetFor(aCategory);
|
|
}
|
|
|
|
AbstractThread* TabGroup::AbstractMainThreadForImpl(TaskCategory aCategory) {
|
|
// The mEventTargets of the chrome TabGroup are all set to do_GetMainThread().
|
|
// We could just return AbstractThread::MainThread() without a wrapper.
|
|
// Once we've disconnected everything, we still allow people to dispatch.
|
|
// We'll just go directly to the main thread.
|
|
if (this == sChromeTabGroup || NS_WARN_IF(mLastWindowLeft)) {
|
|
return AbstractThread::MainThread();
|
|
}
|
|
|
|
return SchedulerGroup::AbstractMainThreadForImpl(aCategory);
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|