mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 05:37:11 +00:00
69d1f32ff7
- Bug 1268085 - Remove unused post barrier callbacks r=terrence (0ab13411c9) - Bug 1267699 - Move some public types to the right namespace; r=sfink (3d5008e610) - Bug 1267550 (part 1) - Rename MOZ_MUST_USE as MOZ_MUST_USE_TYPE. r=ehsan. (6f47375796) - Bug 1259021 - Rename Vector::extractRawBuffer to extractOrCopyRawBuffer r=Waldo (97ca94495b) - Bug 1259021 - Add Vector::extractRawBuffer method that doesn't copy the buffer r=Waldo (e58deec48f) - Bug 1265892 - Change Vector to use Impl::new_ consistently. r=Waldo (7a52d21b29) - Bug 1267912 - Rename nsNetUtil.inl as nsNetUtilInlines.h. r=valentin. (548a41b293) - Bug 1265690 part 1 - Mark StringBuffer methods WARN_UNUSED_RESULT, fix OOM issues. r=jonco (0d7e6837e3) - Bug 1265690 part 2 - Fix some more OOM issues in TypedObject code. r=jonco (b60902453e) - Bug 1263490 - Part 2: Add GetFirstDollarIndex intrinsic and use it inRegExpReplace. r=till (4ba19db8c4) - Bug 1263490 - Part 3: Inline GetFirstDollarIndex intrinsic. r=h4writer (e7d9b5d1cc) - Bug 1263490 - Part 4: Fold GetFirstDollarIndex into a integer constant. r=h4writer (3479c7d1af) - Bug 1267269 - Make MIRType an enum class. r=bbouvier (d580ef372a) - Bug 1259295 - BaldrMonkey: Postorder (r=luke) (6ef7a77663) - Bug 1254142: BaldrMonkey: make br_table yield (r=luke) (80e7635e58) - Bug 1263202 - BaldrMonkey: switch to arities on branches, calls and return (r=bbouvier) (f5a0358634) - Bug 1236358 - Improper reading of string16 in Pickle::ReadString16. r=jld (8370ba6a0b) - Bug 1263205 - BaldrMonkey: Update section headers for proposed spec changes (r=luke) (0def2e6bc2) - Bug 1263205 - BaldrMonkey: Update for proposed new section names (r=luke) (e57f0e3367) - Bug 1263205 - BaldrMonkey: Add 'form' field to types section (r=bbouvier) (794edc890f) - Bug 1259021 - Use in-place storage in AutoStableStringChars to avoid allocation for short strings r=jandem r=Waldo (ffb53cbcf4) - Bug 1267550 (part 2) - Rename MOZ_WARN_UNUSED_RESULT as MOZ_MUST_USE. r=froydnj. (47bc674b86) - Bug 1268518: Baldr: implement int32/int64 rotations; r=luke (0d5eedccce) - Bug 1255008: IonMonkey - Add a by default disabled flow sensitive alias analysis pass, r=jandem (521c585d75) - Bug 1266781: Baldr: implement proper checked truncations to integer types; r=sunfish (46078fb3d3) - Bug 1266781: Rename MTruncateToInt64 into MWasmTruncateInt64; r=sunfish (c7d7d1ac11) - Bug 1266781: Add new traps; r=luke (b7ed3d44e6) - Bug 1268024: Pass the atomic attribute down to EmitHeapAccess; r=luke (6195f7d7a3) - Bug 1268024: A few cleanups related to loads/stores; r=luke (88141e3a01) - Bug 1258312 - Make Pickle::Resize infallible r=jld (241ee9b60d) - Bug 1162772, part 1 - Allow CompartmentCreationOptions to store Secure Context state. r=jorendorff (ff666384cf) - Bug 1162772, part 2 - Expose whether SEC_FORCE_INHERIT_PRINCIPAL was dropped from an nsILoadInfo. r=bz (ada46f86bf) - Bug 1162772, part 3 - Add a getChannelResultPrincipalIfNotSandboxed method to nsIScriptSecurityManager. r=bz (5b1d9f6807) - Bug 1162772, part 4 - Implement nsGlobalWindow::IsSecureContext. r=bz (f392f439c9) - Bug 1162772, part 5 - Expose Window.isSecureContext to content. r=bz (e7296e2cf1) - Bug 1267509 - Make nsContentSecurityManager::IsURIPotentiallyTrustworthy act on an nsIPrincipal. r=bz (83de80350a) - Bug 1219098 - Use UniquePtr in UncompressedSourceCache, for it is good (r=jandem) (b68769c729) - Bug 1244279 - Part 1: Take a bit in ObjectElements::Flags to indicate whether the object is in the whole cell store buffer. r=terrence (968cf373f9) - Bug 1244279 - Part 0: Add a GC ubench for large arrays with both elements and properties. r=terrence (ec76b48323) - Bug 1255925 - Give a name to getters/setters and integer-named methods. r=efaust (f978cc6916) - Bug 888969 - Make the getPrototypeOf/setPrototypeOf traps scriptable. r=efaust, r=bholley (eb2325a9ea) - Bug 1267557 part 0 - Move JS poison constants to jsutil.h. r=jonco (65afc690d2) - Bug 1267557 part 1 - Also poison bytes allocated before the actual jitcode. r=nbp (70f0b327d3) - Bug 1267557 part 2 - Use different jitcode poison values. r=nbp (08008ab9dc) - Bug 1267557 part 3 - Define JS_SWEPT_CODE_PATTERN for mips. r=nbp (17e894d59d) - Bug 1267449 - Do not infinite loop in js_fputs; r=jimb (67f961b6cd) - Bug 1219098 - Reenable compression on large sources, but revert to uncompressed if decompression happens (r=jandem) (b44ee8d77d) - Bug 1267551 (part 1) - Use MOZ_MUST_USE more in jsnum.h. r=jonco. (d2476bf8f4) - Bug 1267551 (part 2) - Use MOZ_MUST_USE more in js/src/ds/. r=jonco. (4ff5d9aa88) - Bug 1267412 - Use MutableHandleValue instead of pointer-to-AutoValueVector; r=sfink (3f6dd284bb) - Bug 1266406 - Use EnumSet<AllocKind> to simplify GC sweeping phase information r=terrence (64811500e7) - Bug 1266457 - Update pointers in GC things in two phases when compacting r=terrence (f6f5bc4e4d) - Bug 1266457 - Simplify typed object trace hook r=terence (3b06c8d1e5) - Bug 1268541 - Compact arenas containing base shapes r=terrence (b458b92eea) - Bug 1268805 - Implement PrivateGCThingValue. (r=terrence) (deec9a83ae) - Bug 1268415: Initialize members in UpdatePointerTasks; r=jonco (6cb219005a) - Bug 1268501 - Release the GC lock periodically when releasing arenas on the backgound thread r=terrence (37f0997682) - Bug 1263572 - Wait for background sweeping to finish before checking base shapes r=terrence (354801a411) - Bug 1266887 - Store Rooted heads on the Zone; r=sfink (91c0101ee3) - Bug 1266402 - Add iteration to EnumSet<T> so that it can be used in range-based for loops r=Waldo (e9507a2524) - Bug 1266404 - Allow construction of an EnumSet<T> using an initializer list r=Waldo (1b6d340e99) - Bug 1254020 - Always compute theme scaling factor when per-monitor dpi aware, even if only a single display is currently present. r=emk (a00cda21f4) - Bug 1263525 - Add dedicated function for std_Array self-hosted intrinsic. r=efaust (449d8bb7eb) - Bug 1255925 - Change JSFunction::name to return a JSAtom. r=efaust (5ab396ce83) - Bug 888969 - Make our tree's sole implementation of nsIRemoteTagService.getRemoteObjectTag not depend upon the infallibility of [[GetPrototypeOf]] on the object provided to it. r=bz (f388f4bf1f) - Bug 1264896 - Kill off nsIRemoteTagService and do what it does, in its sole caller, in far-faster C++. r=billm (5ed3fb103d) - Bug 1268246 - Add a simple Poison class lifetime checker. r=froydnj (7b237bc70e) - Bug 1249496 - Don't apply dpi-based scaling for window titlebar dimensions when on a secondary display, because windows doesn't scale it. r=emk (64dd706dbc) - Bug 1164518 - Avoid unnecessary DB updates when caching Safe Browsing results. r=gcp (3cafd9a4df) - Bug 1264472 - Use nsRunnables in FIDO U2F. r=keeler (3aa9570132) - Bug 1236060 - Dispatch error should advance queue. r=smaug (74155b75dd) - Bug 1251697 part 1. Thread an ErrorResult reference through the worker XHR WorkerThreadProxySyncRunnable implementations. r=khuey (77804cbb7c) - Bug 1251697 part 2. Have WorkerThreadProxySyncRunnable hand the ErrorResult reference it holds to its ResponseRunnable so it can report exceptions on there instead of on a JSContext. r=khuey (355c9ee313) - Bug 1251697 part 3. Remove the JSContext argument of StopSyncLoopRunnable::MaybeSetException. r=khuey (010f5b1058) - Bug 1155328. r=smaug (e1f8dac304) - Bug 1265927: Move nsRunnable to mozilla::Runnable, CancelableRunnable to mozilla::CancelableRunnable. r=froydnj (f83bfcae02) - Bug 1239946 - Change test to return error on Speak. r=eeejay (1d402beb02) - Bug 1254378 - Update synth tests and introduce no voiceschanged test. r=smaug (f5823bb70e) - Bug 1251627. Fix XMLHttpRequest.send() to follow the spec better in terms of the exceptions it throws. r=khuey (cd0e321948) - Bug 1268868: [MSE] P1. Re-enable gap detection within a media segment. r=gerald (b8b8df4bc2) - Bug 1268868: [MSE] P2. Reset longest duration after keyframe is seen. r=gerald (2b1401465c) - Bug 1268868: [MSE] P3. Prevent crash should gap be detected in content. r=gerald (063d9376fc) - Bug 1254378 - Implement nsISynthVoiceRegistry.notifyVoicesChanged. r=smaug (4b63b1c360) - Bug 1266804 - Un-inline js::Unbox(); r=jorendorff (0f288b6173) - Bug 1268863 - Report ScriptSources that are only reachable via AsmJSModule (r=njn) (5ba40acb64) - bump version to 45.1b1 (1414db0ca8) - Bug 1262062 - remove old futex names. r=bbouvier (62662bdd2e) - memory: build fix after renaming MOZ_WARN_UNUSED_RESULT (7254dc8d53) - import from mozilla: - Bug 1268725 - BaldrMonkey: Refactor away the internal storage from ExprIter. r=luke (1931bd636f17) - Bug 1268725 - BaldrMonkey: Convert default arguments into explicit arguments. r=luke (c8a11b8b6bbd) (867ec715d6)
746 lines
22 KiB
C++
746 lines
22 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 "ImportManager.h"
|
|
|
|
#include "mozilla/EventListenerManager.h"
|
|
#include "HTMLLinkElement.h"
|
|
#include "nsContentPolicyUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsIChannel.h"
|
|
#include "nsIContentPolicy.h"
|
|
#include "nsIContentSecurityPolicy.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsIDOMDocument.h"
|
|
#include "nsIDOMEvent.h"
|
|
#include "nsIPrincipal.h"
|
|
#include "nsIScriptObjectPrincipal.h"
|
|
#include "nsScriptLoader.h"
|
|
#include "nsNetUtil.h"
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// AutoError
|
|
//-----------------------------------------------------------------------------
|
|
|
|
class AutoError {
|
|
public:
|
|
explicit AutoError(mozilla::dom::ImportLoader* loader, bool scriptsBlocked = true)
|
|
: mLoader(loader)
|
|
, mPassed(false)
|
|
, mScriptsBlocked(scriptsBlocked)
|
|
{}
|
|
|
|
~AutoError()
|
|
{
|
|
if (!mPassed) {
|
|
mLoader->Error(mScriptsBlocked);
|
|
}
|
|
}
|
|
|
|
void Pass() { mPassed = true; }
|
|
|
|
private:
|
|
mozilla::dom::ImportLoader* mLoader;
|
|
bool mPassed;
|
|
bool mScriptsBlocked;
|
|
};
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// ImportLoader::Updater
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void
|
|
ImportLoader::Updater::GetReferrerChain(nsINode* aNode,
|
|
nsTArray<nsINode*>& aResult)
|
|
{
|
|
// We fill up the array backward. First the last link: aNode.
|
|
MOZ_ASSERT(mLoader->mLinks.Contains(aNode));
|
|
|
|
aResult.AppendElement(aNode);
|
|
nsINode* node = aNode;
|
|
RefPtr<ImportManager> manager = mLoader->Manager();
|
|
for (ImportLoader* referrersLoader = manager->Find(node->OwnerDoc());
|
|
referrersLoader;
|
|
referrersLoader = manager->Find(node->OwnerDoc()))
|
|
{
|
|
// Then walking up the main referrer chain and append each link
|
|
// to the array.
|
|
node = referrersLoader->GetMainReferrer();
|
|
MOZ_ASSERT(node);
|
|
aResult.AppendElement(node);
|
|
}
|
|
|
|
// The reversed order is more useful for consumers.
|
|
// XXX: This should probably go to nsTArray or some generic utility
|
|
// lib for our containers that we don't have... I would really like to
|
|
// get rid of this part...
|
|
uint32_t l = aResult.Length();
|
|
for (uint32_t i = 0; i < l / 2; i++) {
|
|
Swap(aResult[i], aResult[l - i - 1]);
|
|
}
|
|
}
|
|
|
|
bool
|
|
ImportLoader::Updater::ShouldUpdate(nsTArray<nsINode*>& aNewPath)
|
|
{
|
|
if (mLoader->Manager()->GetNearestPredecessor(mLoader->GetMainReferrer()) !=
|
|
mLoader->mBlockingPredecessor) {
|
|
return true;
|
|
}
|
|
// Let's walk down on the main referrer chains of both the current main and
|
|
// the new link, and find the last pair of links that are from the same
|
|
// document. This is the junction point between the two referrer chain. Their
|
|
// order in the subimport list of that document will determine if we have to
|
|
// update the spanning tree or this new edge changes nothing in the script
|
|
// execution order.
|
|
nsTArray<nsINode*> oldPath;
|
|
GetReferrerChain(mLoader->mLinks[mLoader->mMainReferrer], oldPath);
|
|
uint32_t max = std::min(oldPath.Length(), aNewPath.Length());
|
|
MOZ_ASSERT(max > 0);
|
|
uint32_t lastCommonImportAncestor = 0;
|
|
|
|
for (uint32_t i = 0;
|
|
i < max && oldPath[i]->OwnerDoc() == aNewPath[i]->OwnerDoc();
|
|
i++)
|
|
{
|
|
lastCommonImportAncestor = i;
|
|
}
|
|
|
|
MOZ_ASSERT(lastCommonImportAncestor < max);
|
|
nsINode* oldLink = oldPath[lastCommonImportAncestor];
|
|
nsINode* newLink = aNewPath[lastCommonImportAncestor];
|
|
|
|
if ((lastCommonImportAncestor == max - 1) &&
|
|
newLink == oldLink ) {
|
|
// If one chain contains the other entirely, then this is a simple cycle,
|
|
// nothing to be done here.
|
|
MOZ_ASSERT(oldPath.Length() != aNewPath.Length(),
|
|
"This would mean that new link == main referrer link");
|
|
return false;
|
|
}
|
|
|
|
MOZ_ASSERT(aNewPath != oldPath,
|
|
"How could this happen?");
|
|
nsIDocument* doc = oldLink->OwnerDoc();
|
|
MOZ_ASSERT(doc->HasSubImportLink(newLink));
|
|
MOZ_ASSERT(doc->HasSubImportLink(oldLink));
|
|
|
|
return doc->IndexOfSubImportLink(newLink) < doc->IndexOfSubImportLink(oldLink);
|
|
}
|
|
|
|
void
|
|
ImportLoader::Updater::UpdateMainReferrer(uint32_t aNewIdx)
|
|
{
|
|
MOZ_ASSERT(aNewIdx < mLoader->mLinks.Length());
|
|
nsINode* newMainReferrer = mLoader->mLinks[aNewIdx];
|
|
|
|
// This new link means we have to execute our scripts sooner...
|
|
// Let's make sure that unblocking a loader does not trigger a script execution.
|
|
// So we start with placing the new blockers and only then will we remove any
|
|
// blockers.
|
|
if (mLoader->IsBlocking()) {
|
|
// Our import parent is changed, let's block the new one and later unblock
|
|
// the old one.
|
|
newMainReferrer->OwnerDoc()->ScriptLoader()->AddExecuteBlocker();
|
|
newMainReferrer->OwnerDoc()->BlockDOMContentLoaded();
|
|
}
|
|
|
|
if (mLoader->mDocument) {
|
|
// Our nearest predecessor has changed. So let's add the ScriptLoader to the
|
|
// new one if there is any. And remove it from the old one.
|
|
RefPtr<ImportManager> manager = mLoader->Manager();
|
|
nsScriptLoader* loader = mLoader->mDocument->ScriptLoader();
|
|
ImportLoader*& pred = mLoader->mBlockingPredecessor;
|
|
ImportLoader* newPred = manager->GetNearestPredecessor(newMainReferrer);
|
|
if (pred) {
|
|
if (newPred) {
|
|
newPred->AddBlockedScriptLoader(loader);
|
|
}
|
|
pred->RemoveBlockedScriptLoader(loader);
|
|
}
|
|
}
|
|
|
|
if (mLoader->IsBlocking()) {
|
|
mLoader->mImportParent->ScriptLoader()->RemoveExecuteBlocker();
|
|
mLoader->mImportParent->UnblockDOMContentLoaded();
|
|
}
|
|
|
|
// Finally update mMainReferrer to point to the newly added link.
|
|
mLoader->mMainReferrer = aNewIdx;
|
|
mLoader->mImportParent = newMainReferrer->OwnerDoc();
|
|
}
|
|
|
|
nsINode*
|
|
ImportLoader::Updater::NextDependant(nsINode* aCurrentLink,
|
|
nsTArray<nsINode*>& aPath,
|
|
NodeTable& aVisitedNodes, bool aSkipChildren)
|
|
{
|
|
// Depth first graph traversal.
|
|
if (!aSkipChildren) {
|
|
// "first child"
|
|
ImportLoader* loader = mLoader->Manager()->Find(aCurrentLink);
|
|
if (loader && loader->GetDocument()) {
|
|
nsINode* firstSubImport = loader->GetDocument()->GetSubImportLink(0);
|
|
if (firstSubImport && !aVisitedNodes.Contains(firstSubImport)) {
|
|
aPath.AppendElement(aCurrentLink);
|
|
aVisitedNodes.PutEntry(firstSubImport);
|
|
return firstSubImport;
|
|
}
|
|
}
|
|
}
|
|
|
|
aPath.AppendElement(aCurrentLink);
|
|
// "(parent's) next sibling"
|
|
while(aPath.Length() > 1) {
|
|
aCurrentLink = aPath[aPath.Length() - 1];
|
|
aPath.RemoveElementAt(aPath.Length() - 1);
|
|
|
|
// Let's find the next "sibling"
|
|
ImportLoader* loader = mLoader->Manager()->Find(aCurrentLink->OwnerDoc());
|
|
MOZ_ASSERT(loader && loader->GetDocument(), "How can this happend?");
|
|
nsIDocument* doc = loader->GetDocument();
|
|
MOZ_ASSERT(doc->HasSubImportLink(aCurrentLink));
|
|
uint32_t idx = doc->IndexOfSubImportLink(aCurrentLink);
|
|
nsINode* next = doc->GetSubImportLink(idx + 1);
|
|
if (next) {
|
|
// Note: If we found an already visited link that means the parent links has
|
|
// closed the circle it's always the "first child" section that should find
|
|
// the first already visited node. Let's just assert that.
|
|
MOZ_ASSERT(!aVisitedNodes.Contains(next));
|
|
aVisitedNodes.PutEntry(next);
|
|
return next;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
ImportLoader::Updater::UpdateDependants(nsINode* aNode,
|
|
nsTArray<nsINode*>& aPath)
|
|
{
|
|
NodeTable visitedNodes;
|
|
nsINode* current = aNode;
|
|
uint32_t initialLength = aPath.Length();
|
|
bool neededUpdate = true;
|
|
while ((current = NextDependant(current, aPath, visitedNodes, !neededUpdate))) {
|
|
if (!current || aPath.Length() <= initialLength) {
|
|
break;
|
|
}
|
|
ImportLoader* loader = mLoader->Manager()->Find(current);
|
|
if (!loader) {
|
|
continue;
|
|
}
|
|
Updater& updater = loader->mUpdater;
|
|
neededUpdate = updater.ShouldUpdate(aPath);
|
|
if (neededUpdate) {
|
|
updater.UpdateMainReferrer(loader->mLinks.IndexOf(current));
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
ImportLoader::Updater::UpdateSpanningTree(nsINode* aNode)
|
|
{
|
|
if (mLoader->mReady || mLoader->mStopped) {
|
|
// Scripts already executed, nothing to be done here.
|
|
return;
|
|
}
|
|
|
|
if (mLoader->mLinks.Length() == 1) {
|
|
// If this is the first referrer, let's mark it.
|
|
mLoader->mMainReferrer = 0;
|
|
return;
|
|
}
|
|
|
|
nsTArray<nsINode*> newReferrerChain;
|
|
GetReferrerChain(aNode, newReferrerChain);
|
|
if (ShouldUpdate(newReferrerChain)) {
|
|
UpdateMainReferrer(mLoader->mLinks.Length() - 1);
|
|
UpdateDependants(aNode, newReferrerChain);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// ImportLoader
|
|
//-----------------------------------------------------------------------------
|
|
|
|
NS_INTERFACE_MAP_BEGIN(ImportLoader)
|
|
NS_INTERFACE_MAP_ENTRY(nsIStreamListener)
|
|
NS_INTERFACE_MAP_ENTRY(nsIRequestObserver)
|
|
NS_INTERFACE_MAP_ENTRIES_CYCLE_COLLECTION(ImportLoader)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(ImportLoader)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(ImportLoader)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION(ImportLoader,
|
|
mDocument,
|
|
mImportParent,
|
|
mLinks)
|
|
|
|
ImportLoader::ImportLoader(nsIURI* aURI, nsIDocument* aImportParent)
|
|
: mURI(aURI)
|
|
, mImportParent(aImportParent)
|
|
, mBlockingPredecessor(nullptr)
|
|
, mReady(false)
|
|
, mStopped(false)
|
|
, mBlockingScripts(false)
|
|
, mUpdater(this)
|
|
{
|
|
}
|
|
|
|
void
|
|
ImportLoader::BlockScripts()
|
|
{
|
|
MOZ_ASSERT(!mBlockingScripts);
|
|
mImportParent->ScriptLoader()->AddExecuteBlocker();
|
|
mImportParent->BlockDOMContentLoaded();
|
|
mBlockingScripts = true;
|
|
}
|
|
|
|
void
|
|
ImportLoader::UnblockScripts()
|
|
{
|
|
MOZ_ASSERT(mBlockingScripts);
|
|
mImportParent->ScriptLoader()->RemoveExecuteBlocker();
|
|
mImportParent->UnblockDOMContentLoaded();
|
|
for (uint32_t i = 0; i < mBlockedScriptLoaders.Length(); i++) {
|
|
mBlockedScriptLoaders[i]->RemoveExecuteBlocker();
|
|
}
|
|
mBlockedScriptLoaders.Clear();
|
|
mBlockingScripts = false;
|
|
}
|
|
|
|
void
|
|
ImportLoader::SetBlockingPredecessor(ImportLoader* aLoader)
|
|
{
|
|
mBlockingPredecessor = aLoader;
|
|
}
|
|
|
|
void
|
|
ImportLoader::DispatchEventIfFinished(nsINode* aNode)
|
|
{
|
|
MOZ_ASSERT(!(mReady && mStopped));
|
|
if (mReady) {
|
|
DispatchLoadEvent(aNode);
|
|
}
|
|
if (mStopped) {
|
|
DispatchErrorEvent(aNode);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImportLoader::AddBlockedScriptLoader(nsScriptLoader* aScriptLoader)
|
|
{
|
|
if (mBlockedScriptLoaders.Contains(aScriptLoader)) {
|
|
return;
|
|
}
|
|
|
|
aScriptLoader->AddExecuteBlocker();
|
|
|
|
// Let's keep track of the pending script loaders.
|
|
mBlockedScriptLoaders.AppendElement(aScriptLoader);
|
|
}
|
|
|
|
bool
|
|
ImportLoader::RemoveBlockedScriptLoader(nsScriptLoader* aScriptLoader)
|
|
{
|
|
aScriptLoader->RemoveExecuteBlocker();
|
|
return mBlockedScriptLoaders.RemoveElement(aScriptLoader);
|
|
}
|
|
|
|
void
|
|
ImportLoader::AddLinkElement(nsINode* aNode)
|
|
{
|
|
// If a new link element is added to the import tree that
|
|
// refers to an import that is already finished loading or
|
|
// stopped trying, we need to fire the corresponding event
|
|
// on it.
|
|
mLinks.AppendElement(aNode);
|
|
mUpdater.UpdateSpanningTree(aNode);
|
|
DispatchEventIfFinished(aNode);
|
|
}
|
|
|
|
void
|
|
ImportLoader::RemoveLinkElement(nsINode* aNode)
|
|
{
|
|
mLinks.RemoveElement(aNode);
|
|
}
|
|
|
|
// Events has to be fired with a script runner, so mImport can
|
|
// be set on the link element before the load event is fired even
|
|
// if ImportLoader::Get returns an already loaded import and we
|
|
// fire the load event immediately on the new referring link element.
|
|
class AsyncEvent : public Runnable {
|
|
public:
|
|
AsyncEvent(nsINode* aNode, bool aSuccess)
|
|
: mNode(aNode)
|
|
, mSuccess(aSuccess)
|
|
{
|
|
MOZ_ASSERT(mNode);
|
|
}
|
|
|
|
NS_IMETHOD Run() {
|
|
return nsContentUtils::DispatchTrustedEvent(mNode->OwnerDoc(),
|
|
mNode,
|
|
mSuccess ? NS_LITERAL_STRING("load")
|
|
: NS_LITERAL_STRING("error"),
|
|
/* aCanBubble = */ false,
|
|
/* aCancelable = */ false);
|
|
}
|
|
|
|
private:
|
|
nsCOMPtr<nsINode> mNode;
|
|
bool mSuccess;
|
|
};
|
|
|
|
void
|
|
ImportLoader::DispatchErrorEvent(nsINode* aNode)
|
|
{
|
|
nsContentUtils::AddScriptRunner(new AsyncEvent(aNode, /* aSuccess = */ false));
|
|
}
|
|
|
|
void
|
|
ImportLoader::DispatchLoadEvent(nsINode* aNode)
|
|
{
|
|
nsContentUtils::AddScriptRunner(new AsyncEvent(aNode, /* aSuccess = */ true));
|
|
}
|
|
|
|
void
|
|
ImportLoader::Done()
|
|
{
|
|
mReady = true;
|
|
uint32_t l = mLinks.Length();
|
|
for (uint32_t i = 0; i < l; i++) {
|
|
DispatchLoadEvent(mLinks[i]);
|
|
}
|
|
UnblockScripts();
|
|
ReleaseResources();
|
|
}
|
|
|
|
void
|
|
ImportLoader::Error(bool aUnblockScripts)
|
|
{
|
|
mDocument = nullptr;
|
|
mStopped = true;
|
|
uint32_t l = mLinks.Length();
|
|
for (uint32_t i = 0; i < l; i++) {
|
|
DispatchErrorEvent(mLinks[i]);
|
|
}
|
|
if (aUnblockScripts) {
|
|
UnblockScripts();
|
|
}
|
|
ReleaseResources();
|
|
}
|
|
|
|
// Release all the resources we don't need after there is no more
|
|
// data available on the channel, and the parser is done.
|
|
void ImportLoader::ReleaseResources()
|
|
{
|
|
mParserStreamListener = nullptr;
|
|
mImportParent = nullptr;
|
|
}
|
|
|
|
nsIPrincipal*
|
|
ImportLoader::Principal()
|
|
{
|
|
MOZ_ASSERT(mImportParent);
|
|
nsCOMPtr<nsIDocument> master = mImportParent->MasterDocument();
|
|
nsCOMPtr<nsIScriptObjectPrincipal> sop = do_QueryInterface(master);
|
|
MOZ_ASSERT(sop);
|
|
return sop->GetPrincipal();
|
|
}
|
|
|
|
void
|
|
ImportLoader::Open()
|
|
{
|
|
AutoError ae(this, false);
|
|
|
|
nsCOMPtr<nsILoadGroup> loadGroup =
|
|
mImportParent->MasterDocument()->GetDocumentLoadGroup();
|
|
|
|
nsCOMPtr<nsIChannel> channel;
|
|
nsresult rv = NS_NewChannel(getter_AddRefs(channel),
|
|
mURI,
|
|
mImportParent,
|
|
nsILoadInfo::SEC_REQUIRE_CORS_DATA_INHERITS,
|
|
nsIContentPolicy::TYPE_SUBDOCUMENT,
|
|
loadGroup,
|
|
nullptr, // aCallbacks
|
|
nsIRequest::LOAD_BACKGROUND);
|
|
|
|
NS_ENSURE_SUCCESS_VOID(rv);
|
|
rv = channel->AsyncOpen2(this);
|
|
NS_ENSURE_SUCCESS_VOID(rv);
|
|
|
|
BlockScripts();
|
|
ae.Pass();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImportLoader::OnDataAvailable(nsIRequest* aRequest,
|
|
nsISupports* aContext,
|
|
nsIInputStream* aStream,
|
|
uint64_t aOffset,
|
|
uint32_t aCount)
|
|
{
|
|
MOZ_ASSERT(mParserStreamListener);
|
|
|
|
AutoError ae(this);
|
|
nsresult rv;
|
|
nsCOMPtr<nsIChannel> channel = do_QueryInterface(aRequest, &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
rv = mParserStreamListener->OnDataAvailable(channel, aContext,
|
|
aStream, aOffset,
|
|
aCount);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
ae.Pass();
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImportLoader::HandleEvent(nsIDOMEvent *aEvent)
|
|
{
|
|
#ifdef DEBUG
|
|
nsAutoString type;
|
|
aEvent->GetType(type);
|
|
MOZ_ASSERT(type.EqualsLiteral("DOMContentLoaded"));
|
|
#endif
|
|
Done();
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImportLoader::OnStopRequest(nsIRequest* aRequest,
|
|
nsISupports* aContext,
|
|
nsresult aStatus)
|
|
{
|
|
// OnStartRequest throws a special error code to let us know that we
|
|
// shouldn't do anything else.
|
|
if (aStatus == NS_ERROR_DOM_ABORT_ERR) {
|
|
// We failed in OnStartRequest, nothing more to do (we've already
|
|
// dispatched an error event) just return here.
|
|
MOZ_ASSERT(mStopped);
|
|
return NS_OK;
|
|
}
|
|
|
|
if (mParserStreamListener) {
|
|
mParserStreamListener->OnStopRequest(aRequest, aContext, aStatus);
|
|
}
|
|
|
|
if (!mDocument) {
|
|
// If at this point we don't have a document, then the error was
|
|
// already reported.
|
|
return NS_ERROR_DOM_ABORT_ERR;
|
|
}
|
|
|
|
nsCOMPtr<EventTarget> eventTarget = do_QueryInterface(mDocument);
|
|
EventListenerManager* manager = eventTarget->GetOrCreateListenerManager();
|
|
manager->AddEventListenerByType(this,
|
|
NS_LITERAL_STRING("DOMContentLoaded"),
|
|
TrustedEventsAtSystemGroupBubble());
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
ImportLoader::OnStartRequest(nsIRequest* aRequest, nsISupports* aContext)
|
|
{
|
|
AutoError ae(this);
|
|
nsIPrincipal* principal = Principal();
|
|
|
|
nsCOMPtr<nsIChannel> channel = do_QueryInterface(aRequest);
|
|
if (!channel) {
|
|
return NS_ERROR_DOM_ABORT_ERR;
|
|
}
|
|
|
|
if (nsContentUtils::IsSystemPrincipal(principal)) {
|
|
// We should never import non-system documents and run their scripts with system principal!
|
|
nsCOMPtr<nsIPrincipal> channelPrincipal;
|
|
nsContentUtils::GetSecurityManager()->GetChannelResultPrincipal(channel,
|
|
getter_AddRefs(channelPrincipal));
|
|
if (!nsContentUtils::IsSystemPrincipal(channelPrincipal)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
}
|
|
channel->SetOwner(principal);
|
|
|
|
nsAutoCString type;
|
|
channel->GetContentType(type);
|
|
if (!type.EqualsLiteral("text/html")) {
|
|
NS_WARNING("ImportLoader wrong content type");
|
|
return NS_ERROR_DOM_ABORT_ERR;
|
|
}
|
|
|
|
// The scope object is same for all the imports in an import tree,
|
|
// let's get it form the import parent.
|
|
nsCOMPtr<nsIGlobalObject> global = mImportParent->GetScopeObject();
|
|
nsCOMPtr<nsIDOMDocument> importDoc;
|
|
nsCOMPtr<nsIURI> baseURI = mImportParent->GetBaseURI();
|
|
const nsAString& emptyStr = EmptyString();
|
|
nsresult rv = NS_NewDOMDocument(getter_AddRefs(importDoc),
|
|
emptyStr, emptyStr, nullptr, mURI,
|
|
baseURI, principal, false, global,
|
|
DocumentFlavorHTML);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
|
|
// The imported document must know which master document it belongs to.
|
|
mDocument = do_QueryInterface(importDoc);
|
|
nsCOMPtr<nsIDocument> master = mImportParent->MasterDocument();
|
|
mDocument->SetMasterDocument(master);
|
|
|
|
// We want to inherit the sandbox flags from the master document.
|
|
mDocument->SetSandboxFlags(master->GetSandboxFlags());
|
|
|
|
// We have to connect the blank document we created with the channel we opened,
|
|
// and create its own LoadGroup for it.
|
|
nsCOMPtr<nsIStreamListener> listener;
|
|
nsCOMPtr<nsILoadGroup> loadGroup;
|
|
channel->GetLoadGroup(getter_AddRefs(loadGroup));
|
|
nsCOMPtr<nsILoadGroup> newLoadGroup = do_CreateInstance(NS_LOADGROUP_CONTRACTID);
|
|
NS_ENSURE_TRUE(newLoadGroup, NS_ERROR_OUT_OF_MEMORY);
|
|
newLoadGroup->SetLoadGroup(loadGroup);
|
|
rv = mDocument->StartDocumentLoad("import", channel, newLoadGroup,
|
|
nullptr, getter_AddRefs(listener),
|
|
true);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
|
|
nsCOMPtr<nsIURI> originalURI;
|
|
rv = channel->GetOriginalURI(getter_AddRefs(originalURI));
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
|
|
nsCOMPtr<nsIURI> URI;
|
|
rv = channel->GetURI(getter_AddRefs(URI));
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
MOZ_ASSERT(URI, "URI of a channel should never be null");
|
|
|
|
bool equals;
|
|
rv = URI->Equals(originalURI, &equals);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
|
|
if (!equals) {
|
|
// In case of a redirection we must add the new URI to the import map.
|
|
Manager()->AddLoaderWithNewURI(this, URI);
|
|
}
|
|
|
|
// Let's start the parser.
|
|
mParserStreamListener = listener;
|
|
rv = listener->OnStartRequest(aRequest, aContext);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_ABORT_ERR);
|
|
|
|
ae.Pass();
|
|
return NS_OK;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// ImportManager
|
|
//-----------------------------------------------------------------------------
|
|
|
|
NS_IMPL_CYCLE_COLLECTION(ImportManager,
|
|
mImports)
|
|
|
|
NS_INTERFACE_MAP_BEGIN(ImportManager)
|
|
NS_INTERFACE_MAP_ENTRIES_CYCLE_COLLECTION(ImportManager)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(ImportManager)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(ImportManager)
|
|
|
|
already_AddRefed<ImportLoader>
|
|
ImportManager::Get(nsIURI* aURI, nsINode* aNode, nsIDocument* aOrigDocument)
|
|
{
|
|
// Check if we have a loader for that URI, if not create one,
|
|
// and start it up.
|
|
RefPtr<ImportLoader> loader;
|
|
mImports.Get(aURI, getter_AddRefs(loader));
|
|
bool needToStart = false;
|
|
if (!loader) {
|
|
loader = new ImportLoader(aURI, aOrigDocument);
|
|
mImports.Put(aURI, loader);
|
|
needToStart = true;
|
|
}
|
|
|
|
MOZ_ASSERT(loader);
|
|
// Let's keep track of the sub imports links in each document. It will
|
|
// be used later for scrip execution order calculation. (see UpdateSpanningTree)
|
|
// NOTE: removing and adding back the link to the tree somewhere else will
|
|
// NOT have an effect on script execution order.
|
|
if (!aOrigDocument->HasSubImportLink(aNode)) {
|
|
aOrigDocument->AddSubImportLink(aNode);
|
|
}
|
|
|
|
loader->AddLinkElement(aNode);
|
|
|
|
if (needToStart) {
|
|
loader->Open();
|
|
}
|
|
|
|
return loader.forget();
|
|
}
|
|
|
|
ImportLoader*
|
|
ImportManager::Find(nsIDocument* aImport)
|
|
{
|
|
return mImports.GetWeak(aImport->GetDocumentURIObject());
|
|
}
|
|
|
|
ImportLoader*
|
|
ImportManager::Find(nsINode* aLink)
|
|
{
|
|
HTMLLinkElement* linkElement = static_cast<HTMLLinkElement*>(aLink);
|
|
nsCOMPtr<nsIURI> uri = linkElement->GetHrefURI();
|
|
return mImports.GetWeak(uri);
|
|
}
|
|
|
|
void
|
|
ImportManager::AddLoaderWithNewURI(ImportLoader* aLoader, nsIURI* aNewURI)
|
|
{
|
|
mImports.Put(aNewURI, aLoader);
|
|
}
|
|
|
|
ImportLoader* ImportManager::GetNearestPredecessor(nsINode* aNode)
|
|
{
|
|
// Return the previous link if there is any in the same document.
|
|
nsIDocument* doc = aNode->OwnerDoc();
|
|
int32_t idx = doc->IndexOfSubImportLink(aNode);
|
|
MOZ_ASSERT(idx != -1, "aNode must be a sub import link of its owner document");
|
|
|
|
for (; idx > 0; idx--) {
|
|
HTMLLinkElement* link =
|
|
static_cast<HTMLLinkElement*>(doc->GetSubImportLink(idx - 1));
|
|
nsCOMPtr<nsIURI> uri = link->GetHrefURI();
|
|
RefPtr<ImportLoader> ret;
|
|
mImports.Get(uri, getter_AddRefs(ret));
|
|
// Only main referrer links are interesting.
|
|
if (ret->GetMainReferrer() == link) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
if (idx == 0) {
|
|
if (doc->IsMasterDocument()) {
|
|
// If there is no previous one, and it was the master document, then
|
|
// there is no predecessor.
|
|
return nullptr;
|
|
}
|
|
// Else we find the main referrer of the import parent of the link's document.
|
|
// And do a recursion.
|
|
ImportLoader* owner = Find(doc);
|
|
MOZ_ASSERT(owner);
|
|
nsCOMPtr<nsINode> mainReferrer = owner->GetMainReferrer();
|
|
return GetNearestPredecessor(mainReferrer);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|