2016-10-29 01:57:50 +03:00
|
|
|
/* -*- 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/. */
|
|
|
|
|
2016-10-07 00:23:08 +03:00
|
|
|
#include "mozilla/dom/DocGroup.h"
|
2016-10-28 21:25:43 +03:00
|
|
|
#include "mozilla/dom/TabGroup.h"
|
2016-10-07 00:23:08 +03:00
|
|
|
#include "mozilla/Telemetry.h"
|
|
|
|
#include "nsIDocShell.h"
|
2017-12-19 18:14:55 +03:00
|
|
|
#include "nsDOMMutationObserver.h"
|
2016-10-07 00:23:08 +03:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
|
|
|
|
2017-12-19 18:14:55 +03:00
|
|
|
AutoTArray<RefPtr<DocGroup>, 2>* DocGroup::sPendingDocGroups = nullptr;
|
|
|
|
|
2017-01-17 03:10:27 +03:00
|
|
|
/* static */ nsresult
|
2016-10-07 00:23:08 +03:00
|
|
|
DocGroup::GetKey(nsIPrincipal* aPrincipal, nsACString& aKey)
|
|
|
|
{
|
2017-01-27 19:09:20 +03:00
|
|
|
// Use GetBaseDomain() to handle things like file URIs, IP address URIs,
|
|
|
|
// etc. correctly.
|
|
|
|
nsresult rv = aPrincipal->GetBaseDomain(aKey);
|
2017-01-17 03:10:27 +03:00
|
|
|
if (NS_FAILED(rv)) {
|
2017-01-27 19:09:20 +03:00
|
|
|
// We don't really know what to do here. But we should be conservative,
|
|
|
|
// otherwise it would be possible to reorder two events incorrectly in the
|
|
|
|
// future if we interrupt at the DocGroup level, so to be safe, use an
|
|
|
|
// empty string to classify all such documents as belonging to the same
|
|
|
|
// DocGroup.
|
2017-01-17 03:10:27 +03:00
|
|
|
aKey.Truncate();
|
|
|
|
}
|
|
|
|
|
2017-01-27 19:09:20 +03:00
|
|
|
return rv;
|
2016-10-07 00:23:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
DocGroup::RemoveDocument(nsIDocument* aDocument)
|
|
|
|
{
|
2017-03-29 21:37:38 +03:00
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
2016-10-07 00:23:08 +03:00
|
|
|
MOZ_ASSERT(mDocuments.Contains(aDocument));
|
|
|
|
mDocuments.RemoveElement(aDocument);
|
|
|
|
}
|
|
|
|
|
|
|
|
DocGroup::DocGroup(TabGroup* aTabGroup, const nsACString& aKey)
|
|
|
|
: mKey(aKey), mTabGroup(aTabGroup)
|
|
|
|
{
|
|
|
|
// This method does not add itself to mTabGroup->mDocGroups as the caller does it for us.
|
|
|
|
}
|
|
|
|
|
|
|
|
DocGroup::~DocGroup()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(mDocuments.IsEmpty());
|
2017-04-05 07:04:00 +03:00
|
|
|
if (!NS_IsMainThread()) {
|
|
|
|
nsIEventTarget* target = EventTargetFor(TaskCategory::Other);
|
2017-06-14 04:27:17 +03:00
|
|
|
NS_ProxyRelease("DocGroup::mReactionsStack", target, mReactionsStack.forget());
|
2017-04-05 07:04:00 +03:00
|
|
|
}
|
|
|
|
|
2016-10-07 00:23:08 +03:00
|
|
|
mTabGroup->mDocGroups.RemoveEntry(mKey);
|
|
|
|
}
|
|
|
|
|
2016-10-29 01:25:08 +03:00
|
|
|
nsresult
|
2017-07-26 11:13:35 +03:00
|
|
|
DocGroup::Dispatch(TaskCategory aCategory,
|
2016-10-29 01:25:08 +03:00
|
|
|
already_AddRefed<nsIRunnable>&& aRunnable)
|
|
|
|
{
|
2018-02-10 00:17:01 +03:00
|
|
|
return mTabGroup->DispatchWithDocGroup(aCategory, Move(aRunnable), this);
|
2016-11-23 08:14:39 +03:00
|
|
|
}
|
|
|
|
|
2017-06-13 23:40:00 +03:00
|
|
|
nsISerialEventTarget*
|
2016-11-23 08:14:39 +03:00
|
|
|
DocGroup::EventTargetFor(TaskCategory aCategory) const
|
|
|
|
{
|
|
|
|
return mTabGroup->EventTargetFor(aCategory);
|
2016-10-29 01:25:08 +03:00
|
|
|
}
|
|
|
|
|
2016-12-01 13:33:05 +03:00
|
|
|
AbstractThread*
|
2017-04-04 02:28:58 +03:00
|
|
|
DocGroup::AbstractMainThreadFor(TaskCategory aCategory)
|
2016-12-01 13:33:05 +03:00
|
|
|
{
|
|
|
|
MOZ_RELEASE_ASSERT(NS_IsMainThread());
|
|
|
|
return mTabGroup->AbstractMainThreadFor(aCategory);
|
|
|
|
}
|
|
|
|
|
2017-02-02 00:38:33 +03:00
|
|
|
bool*
|
|
|
|
DocGroup::GetValidAccessPtr()
|
|
|
|
{
|
|
|
|
return mTabGroup->GetValidAccessPtr();
|
|
|
|
}
|
|
|
|
|
2017-12-19 18:14:55 +03:00
|
|
|
void
|
|
|
|
DocGroup::SignalSlotChange(const HTMLSlotElement* aSlot)
|
|
|
|
{
|
|
|
|
if (mSignalSlotList.Contains(aSlot)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mSignalSlotList.AppendElement(const_cast<HTMLSlotElement*>(aSlot));
|
|
|
|
|
|
|
|
if (!sPendingDocGroups) {
|
|
|
|
// Queue a mutation observer compound microtask.
|
|
|
|
nsDOMMutationObserver::QueueMutationObserverMicroTask();
|
|
|
|
sPendingDocGroups = new AutoTArray<RefPtr<DocGroup>, 2>;
|
|
|
|
}
|
|
|
|
|
|
|
|
sPendingDocGroups->AppendElement(this);
|
|
|
|
}
|
|
|
|
|
2016-10-07 00:23:08 +03:00
|
|
|
}
|
|
|
|
}
|