summaryrefslogtreecommitdiff
path: root/layout/style/Loader.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'layout/style/Loader.cpp')
-rw-r--r--layout/style/Loader.cpp2694
1 files changed, 2694 insertions, 0 deletions
diff --git a/layout/style/Loader.cpp b/layout/style/Loader.cpp
new file mode 100644
index 0000000000..a1a0fcfd90
--- /dev/null
+++ b/layout/style/Loader.cpp
@@ -0,0 +1,2694 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: ft=cpp tw=78 sw=2 et ts=2
+ *
+ * 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/.
+ *
+ * This Original Code has been modified by IBM Corporation.
+ * Modifications made by IBM described herein are Copyright (c)
+ * International Business Machines Corporation, 2000. Modifications
+ * to Mozilla code or documentation identified per MPL Section 3.3
+ *
+ * Date Modified by Description of modification
+ * 04/20/2000 IBM Corp. OS/2 VisualAge build.
+ */
+
+/* loading of CSS style sheets using the network APIs */
+
+#include "mozilla/ArrayUtils.h"
+#include "mozilla/LoadInfo.h"
+#include "mozilla/MemoryReporting.h"
+
+#include "mozilla/css/Loader.h"
+#include "mozilla/StyleSheetInlines.h"
+#include "nsIRunnable.h"
+#include "nsIUnicharStreamLoader.h"
+#include "nsSyncLoadService.h"
+#include "nsCOMPtr.h"
+#include "nsString.h"
+#include "nsIContent.h"
+#include "nsIDocument.h"
+#include "nsIDOMNode.h"
+#include "nsIDOMDocument.h"
+#include "nsIURI.h"
+#include "nsNetUtil.h"
+#include "nsIProtocolHandler.h"
+#include "nsContentUtils.h"
+#include "nsIScriptSecurityManager.h"
+#include "nsContentPolicyUtils.h"
+#include "nsIHttpChannel.h"
+#include "nsIHttpChannelInternal.h"
+#include "nsIClassOfService.h"
+#include "nsIScriptError.h"
+#include "nsMimeTypes.h"
+#include "nsIStyleSheetLinkingElement.h"
+#include "nsICSSLoaderObserver.h"
+#include "nsCSSParser.h"
+#include "mozilla/css/ImportRule.h"
+#include "nsThreadUtils.h"
+#include "nsGkAtoms.h"
+#include "nsIThreadInternal.h"
+#include "nsINetworkPredictor.h"
+#include "mozilla/dom/ShadowRoot.h"
+#include "mozilla/dom/URL.h"
+#include "mozilla/AsyncEventDispatcher.h"
+#include "mozilla/StyleSheet.h"
+#include "mozilla/StyleSheetInlines.h"
+#include "mozilla/ConsoleReportCollector.h"
+
+#ifdef MOZ_XUL
+#include "nsXULPrototypeCache.h"
+#endif
+
+#include "nsIMediaList.h"
+#include "nsIDOMStyleSheet.h"
+#include "nsError.h"
+
+#include "nsIContentSecurityPolicy.h"
+#include "mozilla/dom/SRICheck.h"
+
+#include "mozilla/dom/EncodingUtils.h"
+using mozilla::dom::EncodingUtils;
+
+using namespace mozilla::dom;
+
+/**
+ * OVERALL ARCHITECTURE
+ *
+ * The CSS Loader gets requests to load various sorts of style sheets:
+ * inline style from <style> elements, linked style, @import-ed child
+ * sheets, non-document sheets. The loader handles the following tasks:
+ * 1) Creation of the actual style sheet objects: CreateSheet()
+ * 2) setting of the right media, title, enabled state, etc on the
+ * sheet: PrepareSheet()
+ * 3) Insertion of the sheet in the proper cascade order:
+ * InsertSheetInDoc() and InsertChildSheet()
+ * 4) Load of the sheet: LoadSheet() including security checks
+ * 5) Parsing of the sheet: ParseSheet()
+ * 6) Cleanup: SheetComplete()
+ *
+ * The detailed documentation for these functions is found with the
+ * function implementations.
+ *
+ * The following helper object is used:
+ * SheetLoadData -- a small class that is used to store all the
+ * information needed for the loading of a sheet;
+ * this class handles listening for the stream
+ * loader completion and also handles charset
+ * determination.
+ */
+
+namespace mozilla {
+namespace css {
+
+/*********************************************
+ * Data needed to properly load a stylesheet *
+ *********************************************/
+
+static_assert(eAuthorSheetFeatures == 0 &&
+ eUserSheetFeatures == 1 &&
+ eAgentSheetFeatures == 2,
+ "sheet parsing mode constants won't fit "
+ "in SheetLoadData::mParsingMode");
+
+class SheetLoadData final : public nsIRunnable,
+ public nsIUnicharStreamLoaderObserver,
+ public nsIThreadObserver
+{
+protected:
+ virtual ~SheetLoadData(void);
+
+public:
+ // Data for loading a sheet linked from a document
+ SheetLoadData(Loader* aLoader,
+ const nsSubstring& aTitle,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ nsIStyleSheetLinkingElement* aOwningElement,
+ bool aIsAlternate,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode);
+
+ // Data for loading a sheet linked from an @import rule
+ SheetLoadData(Loader* aLoader,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ SheetLoadData* aParentData,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode);
+
+ // Data for loading a non-document sheet
+ SheetLoadData(Loader* aLoader,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ bool aSyncLoad,
+ bool aUseSystemPrincipal,
+ const nsCString& aCharset,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode);
+
+ already_AddRefed<nsIURI> GetReferrerURI();
+
+ void ScheduleLoadEventIfNeeded(nsresult aStatus);
+
+ NS_DECL_ISUPPORTS
+ NS_DECL_NSIRUNNABLE
+ NS_DECL_NSITHREADOBSERVER
+ NS_DECL_NSIUNICHARSTREAMLOADEROBSERVER
+
+ // Hold a ref to the CSSLoader so we can call back to it to let it
+ // know the load finished
+ RefPtr<Loader> mLoader;
+
+ // Title needed to pull datas out of the pending datas table when
+ // the preferred title is changed
+ nsString mTitle;
+
+ // Charset we decided to use for the sheet
+ nsCString mCharset;
+
+ // URI we're loading. Null for inline sheets
+ nsCOMPtr<nsIURI> mURI;
+
+ // Should be 1 for non-inline sheets.
+ uint32_t mLineNumber;
+
+ // The sheet we're loading data for
+ RefPtr<StyleSheet> mSheet;
+
+ // Linked list of datas for the same URI as us
+ SheetLoadData* mNext; // strong ref
+
+ // Load data for the sheet that @import-ed us if we were @import-ed
+ // during the parse
+ RefPtr<SheetLoadData> mParentData;
+
+ // Number of sheets we @import-ed that are still loading
+ uint32_t mPendingChildren;
+
+ // mSyncLoad is true when the load needs to be synchronous -- right
+ // now only for LoadSheetSync and children of sync loads.
+ bool mSyncLoad : 1;
+
+ // mIsNonDocumentSheet is true if the load was triggered by LoadSheetSync or
+ // LoadSheet or an @import from such a sheet. Non-document sheet loads can
+ // proceed even if we have no document.
+ bool mIsNonDocumentSheet : 1;
+
+ // mIsLoading is true from the moment we are placed in the loader's
+ // "loading datas" table (right after the async channel is opened)
+ // to the moment we are removed from said table (due to the load
+ // completing or being cancelled).
+ bool mIsLoading : 1;
+
+ // mIsCancelled is set to true when a sheet load is stopped by
+ // Stop() or StopLoadingSheet() (which was removed in Bug 556446).
+ // SheetLoadData::OnStreamComplete() checks this to avoid parsing
+ // sheets that have been cancelled and such.
+ bool mIsCancelled : 1;
+
+ // mMustNotify is true if the load data is being loaded async and
+ // the original function call that started the load has returned.
+ // This applies only to observer notifications; load/error events
+ // are fired for any SheetLoadData that has a non-null
+ // mOwningElement.
+ bool mMustNotify : 1;
+
+ // mWasAlternate is true if the sheet was an alternate when the load data was
+ // created.
+ bool mWasAlternate : 1;
+
+ // mUseSystemPrincipal is true if the system principal should be used for
+ // this sheet, no matter what the channel principal is. Only true for sync
+ // loads.
+ bool mUseSystemPrincipal : 1;
+
+ // If true, this SheetLoadData is being used as a way to handle
+ // async observer notification for an already-complete sheet.
+ bool mSheetAlreadyComplete : 1;
+
+ // This is the element that imported the sheet. Needed to get the
+ // charset set on it and to fire load/error events.
+ nsCOMPtr<nsIStyleSheetLinkingElement> mOwningElement;
+
+ // The observer that wishes to be notified of load completion
+ nsCOMPtr<nsICSSLoaderObserver> mObserver;
+
+ // The principal that identifies who started loading us.
+ nsCOMPtr<nsIPrincipal> mLoaderPrincipal;
+
+ // The node that identifies who started loading us.
+ nsCOMPtr<nsINode> mRequestingNode;
+
+ // The charset to use if the transport and sheet don't indicate one.
+ // May be empty. Must be empty if mOwningElement is non-null.
+ nsCString mCharsetHint;
+
+ // The status our load ended up with; this determines whether we
+ // should fire error events or load events. This gets initialized
+ // by ScheduleLoadEventIfNeeded, and is only used after that has
+ // been called.
+ MOZ_INIT_OUTSIDE_CTOR nsresult mStatus;
+
+private:
+ void FireLoadEvent(nsIThreadInternal* aThread);
+};
+
+#include "mozilla/Logging.h"
+
+static mozilla::LazyLogModule sCssLoaderLog("nsCSSLoader");
+
+static mozilla::LazyLogModule gSriPRLog("SRI");
+
+#define LOG_ERROR(args) MOZ_LOG(sCssLoaderLog, mozilla::LogLevel::Error, args)
+#define LOG_WARN(args) MOZ_LOG(sCssLoaderLog, mozilla::LogLevel::Warning, args)
+#define LOG_DEBUG(args) MOZ_LOG(sCssLoaderLog, mozilla::LogLevel::Debug, args)
+#define LOG(args) LOG_DEBUG(args)
+
+#define LOG_ERROR_ENABLED() MOZ_LOG_TEST(sCssLoaderLog, mozilla::LogLevel::Error)
+#define LOG_WARN_ENABLED() MOZ_LOG_TEST(sCssLoaderLog, mozilla::LogLevel::Warning)
+#define LOG_DEBUG_ENABLED() MOZ_LOG_TEST(sCssLoaderLog, mozilla::LogLevel::Debug)
+#define LOG_ENABLED() LOG_DEBUG_ENABLED()
+
+#define LOG_URI(format, uri) \
+ PR_BEGIN_MACRO \
+ NS_ASSERTION(uri, "Logging null uri"); \
+ if (LOG_ENABLED()) { \
+ LOG((format, uri->GetSpecOrDefault().get())); \
+ } \
+ PR_END_MACRO
+
+// And some convenience strings...
+static const char* const gStateStrings[] = {
+ "eSheetStateUnknown",
+ "eSheetNeedsParser",
+ "eSheetPending",
+ "eSheetLoading",
+ "eSheetComplete"
+};
+
+/********************************
+ * SheetLoadData implementation *
+ ********************************/
+NS_IMPL_ISUPPORTS(SheetLoadData, nsIUnicharStreamLoaderObserver, nsIRunnable,
+ nsIThreadObserver)
+
+SheetLoadData::SheetLoadData(Loader* aLoader,
+ const nsSubstring& aTitle,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ nsIStyleSheetLinkingElement* aOwningElement,
+ bool aIsAlternate,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode)
+ : mLoader(aLoader),
+ mTitle(aTitle),
+ mURI(aURI),
+ mLineNumber(1),
+ mSheet(aSheet),
+ mNext(nullptr),
+ mPendingChildren(0),
+ mSyncLoad(false),
+ mIsNonDocumentSheet(false),
+ mIsLoading(false),
+ mIsCancelled(false),
+ mMustNotify(false),
+ mWasAlternate(aIsAlternate),
+ mUseSystemPrincipal(false),
+ mSheetAlreadyComplete(false),
+ mOwningElement(aOwningElement),
+ mObserver(aObserver),
+ mLoaderPrincipal(aLoaderPrincipal),
+ mRequestingNode(aRequestingNode)
+{
+ NS_PRECONDITION(mLoader, "Must have a loader!");
+}
+
+SheetLoadData::SheetLoadData(Loader* aLoader,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ SheetLoadData* aParentData,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode)
+ : mLoader(aLoader),
+ mURI(aURI),
+ mLineNumber(1),
+ mSheet(aSheet),
+ mNext(nullptr),
+ mParentData(aParentData),
+ mPendingChildren(0),
+ mSyncLoad(false),
+ mIsNonDocumentSheet(false),
+ mIsLoading(false),
+ mIsCancelled(false),
+ mMustNotify(false),
+ mWasAlternate(false),
+ mUseSystemPrincipal(false),
+ mSheetAlreadyComplete(false),
+ mOwningElement(nullptr),
+ mObserver(aObserver),
+ mLoaderPrincipal(aLoaderPrincipal),
+ mRequestingNode(aRequestingNode)
+{
+ NS_PRECONDITION(mLoader, "Must have a loader!");
+ if (mParentData) {
+ mSyncLoad = mParentData->mSyncLoad;
+ mIsNonDocumentSheet = mParentData->mIsNonDocumentSheet;
+ mUseSystemPrincipal = mParentData->mUseSystemPrincipal;
+ ++(mParentData->mPendingChildren);
+ }
+
+ NS_POSTCONDITION(!mUseSystemPrincipal || mSyncLoad,
+ "Shouldn't use system principal for async loads");
+}
+
+SheetLoadData::SheetLoadData(Loader* aLoader,
+ nsIURI* aURI,
+ StyleSheet* aSheet,
+ bool aSyncLoad,
+ bool aUseSystemPrincipal,
+ const nsCString& aCharset,
+ nsICSSLoaderObserver* aObserver,
+ nsIPrincipal* aLoaderPrincipal,
+ nsINode* aRequestingNode)
+ : mLoader(aLoader),
+ mURI(aURI),
+ mLineNumber(1),
+ mSheet(aSheet),
+ mNext(nullptr),
+ mPendingChildren(0),
+ mSyncLoad(aSyncLoad),
+ mIsNonDocumentSheet(true),
+ mIsLoading(false),
+ mIsCancelled(false),
+ mMustNotify(false),
+ mWasAlternate(false),
+ mUseSystemPrincipal(aUseSystemPrincipal),
+ mSheetAlreadyComplete(false),
+ mOwningElement(nullptr),
+ mObserver(aObserver),
+ mLoaderPrincipal(aLoaderPrincipal),
+ mRequestingNode(aRequestingNode),
+ mCharsetHint(aCharset)
+{
+ NS_PRECONDITION(mLoader, "Must have a loader!");
+ NS_POSTCONDITION(!mUseSystemPrincipal || mSyncLoad,
+ "Shouldn't use system principal for async loads");
+}
+
+SheetLoadData::~SheetLoadData()
+{
+ NS_CSS_NS_RELEASE_LIST_MEMBER(SheetLoadData, this, mNext);
+}
+
+NS_IMETHODIMP
+SheetLoadData::Run()
+{
+ mLoader->HandleLoadEvent(this);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+SheetLoadData::OnDispatchedEvent(nsIThreadInternal* aThread)
+{
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+SheetLoadData::OnProcessNextEvent(nsIThreadInternal* aThread,
+ bool aMayWait)
+{
+ // XXXkhuey this is insane!
+ // We want to fire our load even before or after event processing,
+ // whichever comes first.
+ FireLoadEvent(aThread);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+SheetLoadData::AfterProcessNextEvent(nsIThreadInternal* aThread,
+ bool aEventWasProcessed)
+{
+ // XXXkhuey this too!
+ // We want to fire our load even before or after event processing,
+ // whichever comes first.
+ FireLoadEvent(aThread);
+ return NS_OK;
+}
+
+void
+SheetLoadData::FireLoadEvent(nsIThreadInternal* aThread)
+{
+
+ // First remove ourselves as a thread observer. But we need to keep
+ // ourselves alive while doing that!
+ RefPtr<SheetLoadData> kungFuDeathGrip(this);
+ aThread->RemoveObserver(this);
+
+ // Now fire the event
+ nsCOMPtr<nsINode> node = do_QueryInterface(mOwningElement);
+ NS_ASSERTION(node, "How did that happen???");
+
+ nsContentUtils::DispatchTrustedEvent(node->OwnerDoc(),
+ node,
+ NS_SUCCEEDED(mStatus) ?
+ NS_LITERAL_STRING("load") :
+ NS_LITERAL_STRING("error"),
+ false, false);
+
+ // And unblock onload
+ if (mLoader->mDocument) {
+ mLoader->mDocument->UnblockOnload(true);
+ }
+}
+
+void
+SheetLoadData::ScheduleLoadEventIfNeeded(nsresult aStatus)
+{
+ if (!mOwningElement) {
+ return;
+ }
+
+ mStatus = aStatus;
+
+ nsCOMPtr<nsIThread> thread = do_GetMainThread();
+ nsCOMPtr<nsIThreadInternal> internalThread = do_QueryInterface(thread);
+ if (NS_SUCCEEDED(internalThread->AddObserver(this))) {
+ // Make sure to block onload here
+ if (mLoader->mDocument) {
+ mLoader->mDocument->BlockOnload();
+ }
+ }
+}
+
+/*********************
+ * Style sheet reuse *
+ *********************/
+
+bool
+LoaderReusableStyleSheets::FindReusableStyleSheet(nsIURI* aURL,
+ RefPtr<CSSStyleSheet>& aResult)
+{
+ MOZ_ASSERT(aURL);
+ for (size_t i = mReusableSheets.Length(); i > 0; --i) {
+ size_t index = i - 1;
+ bool sameURI;
+ MOZ_ASSERT(mReusableSheets[index]->GetOriginalURI());
+ nsresult rv = aURL->Equals(mReusableSheets[index]->GetOriginalURI(),
+ &sameURI);
+ if (!NS_FAILED(rv) && sameURI) {
+ aResult = mReusableSheets[index];
+ mReusableSheets.RemoveElementAt(index);
+ return true;
+ }
+ }
+ return false;
+}
+
+/*************************
+ * Loader Implementation *
+ *************************/
+
+Loader::Loader(StyleBackendType aType)
+ : mDocument(nullptr)
+ , mDatasToNotifyOn(0)
+ , mCompatMode(eCompatibility_FullStandards)
+ , mStyleBackendType(Some(aType))
+ , mEnabled(true)
+ , mReporter(new ConsoleReportCollector())
+#ifdef DEBUG
+ , mSyncCallback(false)
+#endif
+{
+}
+
+Loader::Loader(nsIDocument* aDocument)
+ : mDocument(aDocument)
+ , mDatasToNotifyOn(0)
+ , mCompatMode(eCompatibility_FullStandards)
+ , mEnabled(true)
+ , mReporter(new ConsoleReportCollector())
+#ifdef DEBUG
+ , mSyncCallback(false)
+#endif
+{
+ // We can just use the preferred set, since there are no sheets in the
+ // document yet (if there are, how did they get there? _we_ load the sheets!)
+ // and hence the selected set makes no sense at this time.
+ nsCOMPtr<nsIDOMDocument> domDoc = do_QueryInterface(mDocument);
+ if (domDoc) {
+ domDoc->GetPreferredStyleSheetSet(mPreferredSheet);
+ }
+}
+
+Loader::~Loader()
+{
+ NS_ASSERTION(!mSheets || mSheets->mLoadingDatas.Count() == 0,
+ "How did we get destroyed when there are loading data?");
+ NS_ASSERTION(!mSheets || mSheets->mPendingDatas.Count() == 0,
+ "How did we get destroyed when there are pending data?");
+ // Note: no real need to revoke our stylesheet loaded events -- they
+ // hold strong references to us, so if we're going away that means
+ // they're all done.
+}
+
+void
+Loader::DropDocumentReference(void)
+{
+ mDocument = nullptr;
+ // Flush out pending datas just so we don't leak by accident. These
+ // loads should short-circuit through the mDocument check in
+ // LoadSheet and just end up in SheetComplete immediately
+ if (mSheets) {
+ StartAlternateLoads();
+ }
+}
+
+nsresult
+Loader::SetPreferredSheet(const nsAString& aTitle)
+{
+#ifdef DEBUG
+ nsCOMPtr<nsIDOMDocument> doc = do_QueryInterface(mDocument);
+ if (doc) {
+ nsAutoString currentPreferred;
+ doc->GetLastStyleSheetSet(currentPreferred);
+ if (DOMStringIsNull(currentPreferred)) {
+ doc->GetPreferredStyleSheetSet(currentPreferred);
+ }
+ NS_ASSERTION(currentPreferred.Equals(aTitle),
+ "Unexpected argument to SetPreferredSheet");
+ }
+#endif
+
+ mPreferredSheet = aTitle;
+
+ // start any pending alternates that aren't alternates anymore
+ if (mSheets) {
+ LoadDataArray arr(mSheets->mPendingDatas.Count());
+ for (auto iter = mSheets->mPendingDatas.Iter(); !iter.Done(); iter.Next()) {
+ SheetLoadData* data = iter.Data();
+ MOZ_ASSERT(data, "Must have a data");
+
+ // Note that we don't want to affect what the selected style set is, so
+ // use true for aHasAlternateRel.
+ if (!data->mLoader->IsAlternate(data->mTitle, true)) {
+ arr.AppendElement(data);
+ iter.Remove();
+ }
+ }
+
+ mDatasToNotifyOn += arr.Length();
+ for (uint32_t i = 0; i < arr.Length(); ++i) {
+ --mDatasToNotifyOn;
+ LoadSheet(arr[i], eSheetNeedsParser, false);
+ }
+ }
+
+ return NS_OK;
+}
+
+static const char kCharsetSym[] = "@charset \"";
+
+static bool GetCharsetFromData(const char* aStyleSheetData,
+ uint32_t aDataLength,
+ nsACString& aCharset)
+{
+ aCharset.Truncate();
+ if (aDataLength <= sizeof(kCharsetSym) - 1)
+ return false;
+
+ if (strncmp(aStyleSheetData,
+ kCharsetSym,
+ sizeof(kCharsetSym) - 1)) {
+ return false;
+ }
+
+ for (uint32_t i = sizeof(kCharsetSym) - 1; i < aDataLength; ++i) {
+ char c = aStyleSheetData[i];
+ if (c == '"') {
+ ++i;
+ if (i < aDataLength && aStyleSheetData[i] == ';') {
+ return true;
+ }
+ // fail
+ break;
+ }
+ aCharset.Append(c);
+ }
+
+ // Did not see end quote or semicolon
+ aCharset.Truncate();
+ return false;
+}
+
+NS_IMETHODIMP
+SheetLoadData::OnDetermineCharset(nsIUnicharStreamLoader* aLoader,
+ nsISupports* aContext,
+ nsACString const& aSegment,
+ nsACString& aCharset)
+{
+ NS_PRECONDITION(!mOwningElement || mCharsetHint.IsEmpty(),
+ "Can't have element _and_ charset hint");
+
+ LOG_URI("SheetLoadData::OnDetermineCharset for '%s'", mURI);
+
+ // The precedence is (per CSS3 Syntax 2012-11-08 ED):
+ // BOM
+ // Channel
+ // @charset rule
+ // charset attribute on the referrer
+ // encoding of the referrer
+ // UTF-8
+
+ aCharset.Truncate();
+
+ if (nsContentUtils::CheckForBOM((const unsigned char*)aSegment.BeginReading(),
+ aSegment.Length(),
+ aCharset)) {
+ // aCharset is now either "UTF-16BE", "UTF-16BE" or "UTF-8"
+ // which will swallow the BOM.
+ mCharset.Assign(aCharset);
+ LOG((" Setting from BOM to: %s", PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIChannel> channel;
+ nsAutoCString specified;
+ aLoader->GetChannel(getter_AddRefs(channel));
+ if (channel) {
+ channel->GetContentCharset(specified);
+ if (EncodingUtils::FindEncodingForLabel(specified, aCharset)) {
+ mCharset.Assign(aCharset);
+ LOG((" Setting from HTTP to: %s", PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+ }
+
+ if (GetCharsetFromData(aSegment.BeginReading(),
+ aSegment.Length(),
+ specified)) {
+ if (EncodingUtils::FindEncodingForLabel(specified, aCharset)) {
+ // FindEncodingForLabel currently never returns UTF-16LE but will
+ // probably change to never return UTF-16 instead, so check both here
+ // to avoid relying on the exact behavior.
+ if (aCharset.EqualsLiteral("UTF-16") ||
+ aCharset.EqualsLiteral("UTF-16BE") ||
+ aCharset.EqualsLiteral("UTF-16LE")) {
+ // Be consistent with HTML <meta> handling in face of impossibility.
+ // When the @charset rule itself evidently was not UTF-16-encoded,
+ // it saying UTF-16 has to be a lie.
+ aCharset.AssignLiteral("UTF-8");
+ }
+ mCharset.Assign(aCharset);
+ LOG((" Setting from @charset rule to: %s",
+ PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+ }
+
+ // Now try the charset on the <link> or processing instruction
+ // that loaded us
+ if (mOwningElement) {
+ nsAutoString specified16;
+ mOwningElement->GetCharset(specified16);
+ if (EncodingUtils::FindEncodingForLabel(specified16, aCharset)) {
+ mCharset.Assign(aCharset);
+ LOG((" Setting from charset attribute to: %s",
+ PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+ }
+
+ // In the preload case, the value of the charset attribute on <link> comes
+ // in via mCharsetHint instead.
+ if (EncodingUtils::FindEncodingForLabel(mCharsetHint, aCharset)) {
+ mCharset.Assign(aCharset);
+ LOG((" Setting from charset attribute (preload case) to: %s",
+ PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+
+ // Try charset from the parent stylesheet.
+ if (mParentData) {
+ aCharset = mParentData->mCharset;
+ if (!aCharset.IsEmpty()) {
+ mCharset.Assign(aCharset);
+ LOG((" Setting from parent sheet to: %s",
+ PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+ }
+
+ if (mLoader->mDocument) {
+ // no useful data on charset. Try the document charset.
+ aCharset = mLoader->mDocument->GetDocumentCharacterSet();
+ MOZ_ASSERT(!aCharset.IsEmpty());
+ mCharset.Assign(aCharset);
+ LOG((" Setting from document to: %s", PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+ }
+
+ aCharset.AssignLiteral("UTF-8");
+ mCharset = aCharset;
+ LOG((" Setting from default to: %s", PromiseFlatCString(aCharset).get()));
+ return NS_OK;
+}
+
+already_AddRefed<nsIURI>
+SheetLoadData::GetReferrerURI()
+{
+ nsCOMPtr<nsIURI> uri;
+ if (mParentData)
+ uri = mParentData->mSheet->GetSheetURI();
+ if (!uri && mLoader->mDocument)
+ uri = mLoader->mDocument->GetDocumentURI();
+ return uri.forget();
+}
+
+/*
+ * Here we need to check that the load did not give us an http error
+ * page and check the mimetype on the channel to make sure we're not
+ * loading non-text/css data in standards mode.
+ */
+NS_IMETHODIMP
+SheetLoadData::OnStreamComplete(nsIUnicharStreamLoader* aLoader,
+ nsISupports* aContext,
+ nsresult aStatus,
+ const nsAString& aBuffer)
+{
+ LOG(("SheetLoadData::OnStreamComplete"));
+ NS_ASSERTION(!mLoader->mSyncCallback, "Synchronous callback from necko");
+
+ if (mIsCancelled) {
+ // Just return. Don't call SheetComplete -- it's already been
+ // called and calling it again will lead to an extra NS_RELEASE on
+ // this data and a likely crash.
+ return NS_OK;
+ }
+
+ if (!mLoader->mDocument && !mIsNonDocumentSheet) {
+ // Sorry, we don't care about this load anymore
+ LOG_WARN((" No document and not non-document sheet; dropping load"));
+ mLoader->SheetComplete(this, NS_BINDING_ABORTED);
+ return NS_OK;
+ }
+
+ if (NS_FAILED(aStatus)) {
+ LOG_WARN((" Load failed: status 0x%x", aStatus));
+ // Handle sheet not loading error because source was a tracking URL.
+ // We make a note of this sheet node by including it in a dedicated
+ // array of blocked tracking nodes under its parent document.
+ //
+ // Multiple sheet load instances might be tied to this request,
+ // we annotate each one linked to a valid owning element (node).
+ if (aStatus == NS_ERROR_TRACKING_URI) {
+ nsIDocument* doc = mLoader->GetDocument();
+ if (doc) {
+ for (SheetLoadData* data = this; data; data = data->mNext) {
+ // mOwningElement may be null but AddBlockTrackingNode can cope
+ nsCOMPtr<nsIContent> content = do_QueryInterface(data->mOwningElement);
+ doc->AddBlockedTrackingNode(content);
+ }
+ }
+ }
+ mLoader->SheetComplete(this, aStatus);
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIChannel> channel;
+ nsresult result = aLoader->GetChannel(getter_AddRefs(channel));
+ if (NS_FAILED(result)) {
+ LOG_WARN((" No channel from loader"));
+ mLoader->SheetComplete(this, result);
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIURI> originalURI;
+ channel->GetOriginalURI(getter_AddRefs(originalURI));
+
+ // If the channel's original URI is "chrome:", we want that, since
+ // the observer code in nsXULPrototypeCache depends on chrome stylesheets
+ // having a chrome URI. (Whether or not chrome stylesheets come through
+ // this codepath seems nondeterministic.)
+ // Otherwise we want the potentially-HTTP-redirected URI.
+ nsCOMPtr<nsIURI> channelURI;
+ NS_GetFinalChannelURI(channel, getter_AddRefs(channelURI));
+
+ if (!channelURI || !originalURI) {
+ NS_ERROR("Someone just violated the nsIRequest contract");
+ LOG_WARN((" Channel without a URI. Bad!"));
+ mLoader->SheetComplete(this, NS_ERROR_UNEXPECTED);
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIPrincipal> principal;
+ nsIScriptSecurityManager* secMan = nsContentUtils::GetSecurityManager();
+ result = NS_ERROR_NOT_AVAILABLE;
+ if (secMan) { // Could be null if we already shut down
+ if (mUseSystemPrincipal) {
+ result = secMan->GetSystemPrincipal(getter_AddRefs(principal));
+ } else {
+ result = secMan->GetChannelResultPrincipal(channel, getter_AddRefs(principal));
+ }
+ }
+
+ if (NS_FAILED(result)) {
+ LOG_WARN((" Couldn't get principal"));
+ mLoader->SheetComplete(this, result);
+ return NS_OK;
+ }
+
+ mSheet->SetPrincipal(principal);
+
+ // If it's an HTTP channel, we want to make sure this is not an
+ // error document we got.
+ nsCOMPtr<nsIHttpChannel> httpChannel(do_QueryInterface(channel));
+ if (httpChannel) {
+ bool requestSucceeded;
+ result = httpChannel->GetRequestSucceeded(&requestSucceeded);
+ if (NS_SUCCEEDED(result) && !requestSucceeded) {
+ LOG((" Load returned an error page"));
+ mLoader->SheetComplete(this, NS_ERROR_NOT_AVAILABLE);
+ return NS_OK;
+ }
+ }
+
+ nsAutoCString contentType;
+ if (channel) {
+ channel->GetContentType(contentType);
+ }
+
+ // In standards mode, a style sheet must have one of these MIME
+ // types to be processed at all. In quirks mode, we accept any
+ // MIME type, but only if the style sheet is same-origin with the
+ // requesting document or parent sheet. See bug 524223.
+
+ bool validType = contentType.EqualsLiteral("text/css") ||
+ contentType.EqualsLiteral(UNKNOWN_CONTENT_TYPE) ||
+ contentType.IsEmpty();
+
+ if (!validType) {
+ const char *errorMessage;
+ uint32_t errorFlag;
+ bool sameOrigin = true;
+
+ if (mLoaderPrincipal) {
+ bool subsumed;
+ result = mLoaderPrincipal->Subsumes(principal, &subsumed);
+ if (NS_FAILED(result) || !subsumed) {
+ sameOrigin = false;
+ }
+ }
+
+ if (sameOrigin && mLoader->mCompatMode == eCompatibility_NavQuirks) {
+ errorMessage = "MimeNotCssWarn";
+ errorFlag = nsIScriptError::warningFlag;
+ } else {
+ errorMessage = "MimeNotCss";
+ errorFlag = nsIScriptError::errorFlag;
+ }
+
+ const nsAFlatString& specUTF16 =
+ NS_ConvertUTF8toUTF16(channelURI->GetSpecOrDefault());
+ const nsAFlatString& ctypeUTF16 = NS_ConvertASCIItoUTF16(contentType);
+ const char16_t *strings[] = { specUTF16.get(), ctypeUTF16.get() };
+
+ nsCOMPtr<nsIURI> referrer = GetReferrerURI();
+ nsContentUtils::ReportToConsole(errorFlag,
+ NS_LITERAL_CSTRING("CSS Loader"),
+ mLoader->mDocument,
+ nsContentUtils::eCSS_PROPERTIES,
+ errorMessage,
+ strings, ArrayLength(strings),
+ referrer);
+
+ if (errorFlag == nsIScriptError::errorFlag) {
+ LOG_WARN((" Ignoring sheet with improper MIME type %s",
+ contentType.get()));
+ mLoader->SheetComplete(this, NS_ERROR_NOT_AVAILABLE);
+ return NS_OK;
+ }
+ }
+
+ SRIMetadata sriMetadata;
+ mSheet->GetIntegrity(sriMetadata);
+ if (sriMetadata.IsEmpty()) {
+ nsCOMPtr<nsILoadInfo> loadInfo = channel->GetLoadInfo();
+ if (loadInfo->GetEnforceSRI()) {
+ LOG((" Load was blocked by SRI"));
+ MOZ_LOG(gSriPRLog, mozilla::LogLevel::Debug,
+ ("css::Loader::OnStreamComplete, required SRI not found"));
+ mLoader->SheetComplete(this, NS_ERROR_SRI_CORRUPT);
+ // log the failed load to web console
+ nsCOMPtr<nsIContentSecurityPolicy> csp;
+ loadInfo->LoadingPrincipal()->GetCsp(getter_AddRefs(csp));
+ nsAutoCString spec;
+ mLoader->mDocument->GetDocumentURI()->GetAsciiSpec(spec);
+ // line number unknown. mRequestingNode doesn't bear this info.
+ csp->LogViolationDetails(
+ nsIContentSecurityPolicy::VIOLATION_TYPE_REQUIRE_SRI_FOR_STYLE,
+ NS_ConvertUTF8toUTF16(spec), EmptyString(),
+ 0, EmptyString(), EmptyString());
+ return NS_OK;
+ }
+ } else {
+ nsAutoCString sourceUri;
+ if (mLoader->mDocument && mLoader->mDocument->GetDocumentURI()) {
+ mLoader->mDocument->GetDocumentURI()->GetAsciiSpec(sourceUri);
+ }
+ nsresult rv = SRICheck::VerifyIntegrity(sriMetadata, aLoader, aBuffer,
+ sourceUri, mLoader->mReporter);
+ mLoader->mReporter->FlushConsoleReports(mLoader->mDocument);
+ if (NS_FAILED(rv)) {
+ LOG((" Load was blocked by SRI"));
+ MOZ_LOG(gSriPRLog, mozilla::LogLevel::Debug,
+ ("css::Loader::OnStreamComplete, bad metadata"));
+ mLoader->SheetComplete(this, NS_ERROR_SRI_CORRUPT);
+ return NS_OK;
+ }
+ }
+
+ // Enough to set the URIs on mSheet, since any sibling datas we have share
+ // the same mInner as mSheet and will thus get the same URI.
+ mSheet->SetURIs(channelURI, originalURI, channelURI);
+
+ bool completed;
+ result = mLoader->ParseSheet(aBuffer, this, completed);
+ NS_ASSERTION(completed || !mSyncLoad, "sync load did not complete");
+ return result;
+}
+
+bool
+Loader::IsAlternate(const nsAString& aTitle, bool aHasAlternateRel)
+{
+ // A sheet is alternate if it has a nonempty title that doesn't match the
+ // currently selected style set. But if there _is_ no currently selected
+ // style set, the sheet wasn't marked as an alternate explicitly, and aTitle
+ // is nonempty, we should select the style set corresponding to aTitle, since
+ // that's a preferred sheet.
+ if (aTitle.IsEmpty()) {
+ return false;
+ }
+
+ if (!aHasAlternateRel && mDocument && mPreferredSheet.IsEmpty()) {
+ // There's no preferred set yet, and we now have a sheet with a title.
+ // Make that be the preferred set.
+ mDocument->SetHeaderData(nsGkAtoms::headerDefaultStyle, aTitle);
+ // We're definitely not an alternate
+ return false;
+ }
+
+ return !aTitle.Equals(mPreferredSheet);
+}
+
+nsresult
+Loader::ObsoleteSheet(nsIURI* aURI)
+{
+ if (!mSheets) {
+ return NS_OK;
+ }
+ if (!aURI) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ for (auto iter = mSheets->mCompleteSheets.Iter(); !iter.Done(); iter.Next()) {
+ nsIURI* sheetURI = iter.Key()->GetURI();
+ bool areEqual;
+ nsresult rv = sheetURI->Equals(aURI, &areEqual);
+ if (NS_SUCCEEDED(rv) && areEqual) {
+ iter.Remove();
+ }
+ }
+ return NS_OK;
+}
+
+nsresult
+Loader::CheckContentPolicy(nsIPrincipal* aSourcePrincipal,
+ nsIURI* aTargetURI,
+ nsISupports* aContext,
+ bool aIsPreload)
+{
+ // When performing a system load (e.g. aUseSystemPrincipal = true)
+ // then aSourcePrincipal == null; don't consult content policies.
+ if (!aSourcePrincipal) {
+ return NS_OK;
+ }
+
+ nsContentPolicyType contentPolicyType =
+ aIsPreload ? nsIContentPolicy::TYPE_INTERNAL_STYLESHEET_PRELOAD
+ : nsIContentPolicy::TYPE_INTERNAL_STYLESHEET;
+
+ int16_t shouldLoad = nsIContentPolicy::ACCEPT;
+ nsresult rv = NS_CheckContentLoadPolicy(contentPolicyType,
+ aTargetURI,
+ aSourcePrincipal,
+ aContext,
+ NS_LITERAL_CSTRING("text/css"),
+ nullptr, //extra param
+ &shouldLoad,
+ nsContentUtils::GetContentPolicy(),
+ nsContentUtils::GetSecurityManager());
+ if (NS_FAILED(rv) || NS_CP_REJECTED(shouldLoad)) {
+ return NS_ERROR_CONTENT_BLOCKED;
+ }
+ return NS_OK;
+}
+
+/**
+ * CreateSheet() creates a CSSStyleSheet object for the given URI,
+ * if any. If there is no URI given, we just create a new style sheet
+ * object. Otherwise, we check for an existing style sheet object for
+ * that uri in various caches and clone it if we find it. Cloned
+ * sheets will have the title/media/enabled state of the sheet they
+ * are clones off; make sure to call PrepareSheet() on the result of
+ * CreateSheet().
+ */
+nsresult
+Loader::CreateSheet(nsIURI* aURI,
+ nsIContent* aLinkingContent,
+ nsIPrincipal* aLoaderPrincipal,
+ css::SheetParsingMode aParsingMode,
+ CORSMode aCORSMode,
+ ReferrerPolicy aReferrerPolicy,
+ const nsAString& aIntegrity,
+ bool aSyncLoad,
+ bool aHasAlternateRel,
+ const nsAString& aTitle,
+ StyleSheetState& aSheetState,
+ bool *aIsAlternate,
+ RefPtr<StyleSheet>* aSheet)
+{
+ LOG(("css::Loader::CreateSheet"));
+ NS_PRECONDITION(aSheet, "Null out param!");
+
+ if (!mSheets) {
+ mSheets = new Sheets();
+ }
+
+ *aSheet = nullptr;
+ aSheetState = eSheetStateUnknown;
+
+ // Check the alternate state before doing anything else, because it
+ // can mess with our hashtables.
+ *aIsAlternate = IsAlternate(aTitle, aHasAlternateRel);
+
+ // XXXheycam Cached sheets currently must be CSSStyleSheets.
+ if (aURI && GetStyleBackendType() == StyleBackendType::Gecko) {
+ aSheetState = eSheetComplete;
+ RefPtr<StyleSheet> sheet;
+
+ // First, the XUL cache
+#ifdef MOZ_XUL
+ if (IsChromeURI(aURI)) {
+ nsXULPrototypeCache* cache = nsXULPrototypeCache::GetInstance();
+ if (cache) {
+ if (cache->IsEnabled()) {
+ sheet = cache->GetStyleSheet(aURI);
+ LOG((" From XUL cache: %p", sheet.get()));
+ }
+ }
+ }
+#endif
+
+ bool fromCompleteSheets = false;
+ if (!sheet) {
+ // Then our per-document complete sheets.
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aURI, aLoaderPrincipal, aCORSMode, aReferrerPolicy);
+
+ StyleSheet* completeSheet = nullptr;
+ mSheets->mCompleteSheets.Get(&key, &completeSheet);
+ sheet = completeSheet;
+ LOG((" From completed: %p", sheet.get()));
+
+ fromCompleteSheets = !!sheet;
+ }
+
+ if (sheet) {
+ if (sheet->IsServo()) {
+ MOZ_CRASH("stylo: can't clone ServoStyleSheets yet");
+ }
+
+ // This sheet came from the XUL cache or our per-document hashtable; it
+ // better be a complete sheet.
+ NS_ASSERTION(sheet->AsGecko()->IsComplete(),
+ "Sheet thinks it's not complete while we think it is");
+
+ // Make sure it hasn't been modified; if it has, we can't use it
+ if (sheet->AsGecko()->IsModified()) {
+ LOG((" Not cloning completed sheet %p because it's been modified",
+ sheet.get()));
+ sheet = nullptr;
+ fromCompleteSheets = false;
+ }
+ }
+
+ // Then loading sheets
+ if (!sheet && !aSyncLoad) {
+ aSheetState = eSheetLoading;
+ SheetLoadData* loadData = nullptr;
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aURI, aLoaderPrincipal, aCORSMode, aReferrerPolicy);
+ mSheets->mLoadingDatas.Get(&key, &loadData);
+ if (loadData) {
+ sheet = loadData->mSheet;
+ LOG((" From loading: %p", sheet.get()));
+
+#ifdef DEBUG
+ bool debugEqual;
+ NS_ASSERTION((!aLoaderPrincipal && !loadData->mLoaderPrincipal) ||
+ (aLoaderPrincipal && loadData->mLoaderPrincipal &&
+ NS_SUCCEEDED(aLoaderPrincipal->
+ Equals(loadData->mLoaderPrincipal,
+ &debugEqual)) && debugEqual),
+ "Principals should be the same");
+#endif
+ }
+
+ // Then alternate sheets
+ if (!sheet) {
+ aSheetState = eSheetPending;
+ loadData = nullptr;
+ mSheets->mPendingDatas.Get(&key, &loadData);
+ if (loadData) {
+ sheet = loadData->mSheet;
+ LOG((" From pending: %p", sheet.get()));
+
+#ifdef DEBUG
+ bool debugEqual;
+ NS_ASSERTION((!aLoaderPrincipal && !loadData->mLoaderPrincipal) ||
+ (aLoaderPrincipal && loadData->mLoaderPrincipal &&
+ NS_SUCCEEDED(aLoaderPrincipal->
+ Equals(loadData->mLoaderPrincipal,
+ &debugEqual)) && debugEqual),
+ "Principals should be the same");
+#endif
+ }
+ }
+ }
+
+ if (sheet) {
+ // The sheet we have now should be either incomplete or unmodified
+ if (sheet->IsServo()) {
+ MOZ_CRASH("stylo: can't clone ServoStyleSheets yet");
+ }
+ NS_ASSERTION(!sheet->AsGecko()->IsModified() ||
+ !sheet->AsGecko()->IsComplete(),
+ "Unexpected modified complete sheet");
+ NS_ASSERTION(sheet->AsGecko()->IsComplete() ||
+ aSheetState != eSheetComplete,
+ "Sheet thinks it's not complete while we think it is");
+
+ RefPtr<CSSStyleSheet> clonedSheet =
+ sheet->AsGecko()->Clone(nullptr, nullptr, nullptr, nullptr);
+ *aSheet = Move(clonedSheet);
+ if (*aSheet && fromCompleteSheets &&
+ !sheet->AsGecko()->GetOwnerNode() &&
+ !sheet->AsGecko()->GetParentSheet()) {
+ // The sheet we're cloning isn't actually referenced by
+ // anyone. Replace it in the cache, so that if our CSSOM is
+ // later modified we don't end up with two copies of our inner
+ // hanging around.
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aURI, aLoaderPrincipal, aCORSMode, aReferrerPolicy);
+ NS_ASSERTION((*aSheet)->AsGecko()->IsComplete(),
+ "Should only be caching complete sheets");
+ mSheets->mCompleteSheets.Put(&key, *aSheet);
+ }
+ }
+ }
+
+ if (!*aSheet) {
+ aSheetState = eSheetNeedsParser;
+ nsIURI *sheetURI;
+ nsCOMPtr<nsIURI> baseURI;
+ nsIURI* originalURI;
+ if (!aURI) {
+ // Inline style. Use the document's base URL so that @import in
+ // the inline sheet picks up the right base.
+ NS_ASSERTION(aLinkingContent, "Inline stylesheet without linking content?");
+ baseURI = aLinkingContent->GetBaseURI();
+ sheetURI = aLinkingContent->OwnerDoc()->GetDocumentURI();
+ originalURI = nullptr;
+ } else {
+ baseURI = aURI;
+ sheetURI = aURI;
+ originalURI = aURI;
+ }
+
+ SRIMetadata sriMetadata;
+ if (!aIntegrity.IsEmpty()) {
+ MOZ_LOG(gSriPRLog, mozilla::LogLevel::Debug,
+ ("css::Loader::CreateSheet, integrity=%s",
+ NS_ConvertUTF16toUTF8(aIntegrity).get()));
+ nsAutoCString sourceUri;
+ if (mDocument && mDocument->GetDocumentURI()) {
+ mDocument->GetDocumentURI()->GetAsciiSpec(sourceUri);
+ }
+ SRICheck::IntegrityMetadata(aIntegrity, sourceUri, mReporter,
+ &sriMetadata);
+ }
+
+ if (GetStyleBackendType() == StyleBackendType::Gecko) {
+ *aSheet = new CSSStyleSheet(aParsingMode, aCORSMode, aReferrerPolicy, sriMetadata);
+ } else {
+ *aSheet = new ServoStyleSheet(aParsingMode, aCORSMode, aReferrerPolicy, sriMetadata);
+ }
+ (*aSheet)->SetURIs(sheetURI, originalURI, baseURI);
+ }
+
+ NS_ASSERTION(*aSheet, "We should have a sheet by now!");
+ NS_ASSERTION(aSheetState != eSheetStateUnknown, "Have to set a state!");
+ LOG((" State: %s", gStateStrings[aSheetState]));
+
+ return NS_OK;
+}
+
+/**
+ * PrepareSheet() handles setting the media and title on the sheet, as
+ * well as setting the enabled state based on the title and whether
+ * the sheet had "alternate" in its rel.
+ */
+void
+Loader::PrepareSheet(StyleSheet* aSheet,
+ const nsSubstring& aTitle,
+ const nsSubstring& aMediaString,
+ nsMediaList* aMediaList,
+ Element* aScopeElement,
+ bool isAlternate)
+{
+ NS_PRECONDITION(aSheet, "Must have a sheet!");
+
+ // XXXheycam Need to set media, title, etc. on ServoStyleSheets.
+ if (aSheet->IsServo()) {
+ NS_WARNING("stylo: should set metadata on ServoStyleSheets. See bug 1290209.");
+ return;
+ }
+
+ CSSStyleSheet* sheet = aSheet->AsGecko();
+
+ RefPtr<nsMediaList> mediaList(aMediaList);
+
+ if (!aMediaString.IsEmpty()) {
+ NS_ASSERTION(!aMediaList,
+ "must not provide both aMediaString and aMediaList");
+ mediaList = new nsMediaList();
+
+ nsCSSParser mediumParser(this);
+
+ // We have aMediaString only when linked from link elements, style
+ // elements, or PIs, so pass true.
+ mediumParser.ParseMediaList(aMediaString, nullptr, 0, mediaList, true);
+ }
+
+ sheet->SetMedia(mediaList);
+
+ sheet->SetTitle(aTitle);
+ sheet->SetEnabled(!isAlternate);
+ sheet->SetScopeElement(aScopeElement);
+}
+
+/**
+ * InsertSheetInDoc handles ordering of sheets in the document. Here
+ * we have two types of sheets -- those with linking elements and
+ * those without. The latter are loaded by Link: headers.
+ * The following constraints are observed:
+ * 1) Any sheet with a linking element comes after all sheets without
+ * linking elements
+ * 2) Sheets without linking elements are inserted in the order in
+ * which the inserting requests come in, since all of these are
+ * inserted during header data processing in the content sink
+ * 3) Sheets with linking elements are ordered based on document order
+ * as determined by CompareDocumentPosition.
+ */
+nsresult
+Loader::InsertSheetInDoc(StyleSheet* aSheet,
+ nsIContent* aLinkingContent,
+ nsIDocument* aDocument)
+{
+ LOG(("css::Loader::InsertSheetInDoc"));
+ NS_PRECONDITION(aSheet, "Nothing to insert");
+ NS_PRECONDITION(aDocument, "Must have a document to insert into");
+
+ // XXX Need to cancel pending sheet loads for this element, if any
+
+ int32_t sheetCount = aDocument->GetNumberOfStyleSheets();
+
+ /*
+ * Start the walk at the _end_ of the list, since in the typical
+ * case we'll just want to append anyway. We want to break out of
+ * the loop when insertionPoint points to just before the index we
+ * want to insert at. In other words, when we leave the loop
+ * insertionPoint is the index of the stylesheet that immediately
+ * precedes the one we're inserting.
+ */
+ int32_t insertionPoint;
+ for (insertionPoint = sheetCount - 1; insertionPoint >= 0; --insertionPoint) {
+ StyleSheet* curSheet = aDocument->GetStyleSheetAt(insertionPoint);
+ NS_ASSERTION(curSheet, "There must be a sheet here!");
+ nsCOMPtr<nsINode> sheetOwner = curSheet->GetOwnerNode();
+ if (sheetOwner && !aLinkingContent) {
+ // Keep moving; all sheets with a sheetOwner come after all
+ // sheets without a linkingNode
+ continue;
+ }
+
+ if (!sheetOwner) {
+ // Aha! The current sheet has no sheet owner, so we want to
+ // insert after it no matter whether we have a linkingNode
+ break;
+ }
+
+ NS_ASSERTION(aLinkingContent != sheetOwner,
+ "Why do we still have our old sheet?");
+
+ // Have to compare
+ if (nsContentUtils::PositionIsBefore(sheetOwner, aLinkingContent)) {
+ // The current sheet comes before us, and it better be the first
+ // such, because now we break
+ break;
+ }
+ }
+
+ ++insertionPoint; // adjust the index to the spot we want to insert in
+
+ // XXX <meta> elements do not implement nsIStyleSheetLinkingElement;
+ // need to fix this for them to be ordered correctly.
+ nsCOMPtr<nsIStyleSheetLinkingElement>
+ linkingElement = do_QueryInterface(aLinkingContent);
+ if (linkingElement) {
+ linkingElement->SetStyleSheet(aSheet); // This sets the ownerNode on the sheet
+ }
+
+ aDocument->BeginUpdate(UPDATE_STYLE);
+ aDocument->InsertStyleSheetAt(aSheet, insertionPoint);
+ aDocument->EndUpdate(UPDATE_STYLE);
+ LOG((" Inserting into document at position %d", insertionPoint));
+
+ return NS_OK;
+}
+
+/**
+ * InsertChildSheet handles ordering of @import-ed sheet in their
+ * parent sheets. Here we want to just insert based on order of the
+ * @import rules that imported the sheets. In theory we can't just
+ * append to the end because the CSSOM can insert @import rules. In
+ * practice, we get the call to load the child sheet before the CSSOM
+ * has finished inserting the @import rule, so we have no idea where
+ * to put it anyway. So just append for now. (In the future if we
+ * want to insert the sheet at the correct position, we'll need to
+ * restore CSSStyleSheet::InsertStyleSheetAt, which was removed in
+ * bug 1220506.)
+ */
+nsresult
+Loader::InsertChildSheet(StyleSheet* aSheet,
+ StyleSheet* aParentSheet,
+ ImportRule* aParentRule)
+{
+ LOG(("css::Loader::InsertChildSheet"));
+ NS_PRECONDITION(aSheet, "Nothing to insert");
+ NS_PRECONDITION(aParentSheet, "Need a parent to insert into");
+ NS_PRECONDITION(aParentSheet, "How did we get imported?");
+
+ // XXXheycam The InsertChildSheet API doesn't work with ServoStyleSheets,
+ // since they won't have Gecko ImportRules in them.
+ if (aSheet->IsServo()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // child sheets should always start out enabled, even if they got
+ // cloned off of top-level sheets which were disabled
+ aSheet->AsGecko()->SetEnabled(true);
+
+ aParentSheet->AppendStyleSheet(aSheet);
+ aParentRule->SetSheet(aSheet->AsGecko()); // This sets the ownerRule on the sheet
+
+ LOG((" Inserting into parent sheet"));
+ // LOG((" Inserting into parent sheet at position %d", insertionPoint));
+
+ return NS_OK;
+}
+
+/**
+ * LoadSheet handles the actual load of a sheet. If the load is
+ * supposed to be synchronous it just opens a channel synchronously
+ * using the given uri, wraps the resulting stream in a converter
+ * stream and calls ParseSheet. Otherwise it tries to look for an
+ * existing load for this URI and piggyback on it. Failing all that,
+ * a new load is kicked off asynchronously.
+ */
+nsresult
+Loader::LoadSheet(SheetLoadData* aLoadData,
+ StyleSheetState aSheetState,
+ bool aIsPreload)
+{
+ LOG(("css::Loader::LoadSheet"));
+ NS_PRECONDITION(aLoadData, "Need a load data");
+ NS_PRECONDITION(aLoadData->mURI, "Need a URI to load");
+ NS_PRECONDITION(aLoadData->mSheet, "Need a sheet to load into");
+ NS_PRECONDITION(aSheetState != eSheetComplete, "Why bother?");
+ NS_PRECONDITION(!aLoadData->mUseSystemPrincipal || aLoadData->mSyncLoad,
+ "Shouldn't use system principal for async loads");
+ NS_ASSERTION(mSheets, "mLoadingDatas should be initialized by now.");
+
+ LOG_URI(" Load from: '%s'", aLoadData->mURI);
+
+ nsresult rv = NS_OK;
+
+ if (!mDocument && !aLoadData->mIsNonDocumentSheet) {
+ // No point starting the load; just release all the data and such.
+ LOG_WARN((" No document and not non-document sheet; pre-dropping load"));
+ SheetComplete(aLoadData, NS_BINDING_ABORTED);
+ return NS_BINDING_ABORTED;
+ }
+
+ SRIMetadata sriMetadata;
+ aLoadData->mSheet->GetIntegrity(sriMetadata);
+
+ if (aLoadData->mSyncLoad) {
+ LOG((" Synchronous load"));
+ NS_ASSERTION(!aLoadData->mObserver, "Observer for a sync load?");
+ NS_ASSERTION(aSheetState == eSheetNeedsParser,
+ "Sync loads can't reuse existing async loads");
+
+ // Create a nsIUnicharStreamLoader instance to which we will feed
+ // the data from the sync load. Do this before creating the
+ // channel to make error recovery simpler.
+ nsCOMPtr<nsIUnicharStreamLoader> streamLoader;
+ rv = NS_NewUnicharStreamLoader(getter_AddRefs(streamLoader), aLoadData);
+ if (NS_FAILED(rv)) {
+ LOG_ERROR((" Failed to create stream loader for sync load"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ if (mDocument) {
+ mozilla::net::PredictorLearn(aLoadData->mURI, mDocument->GetDocumentURI(),
+ nsINetworkPredictor::LEARN_LOAD_SUBRESOURCE,
+ mDocument);
+ }
+
+ nsSecurityFlags securityFlags =
+ nsILoadInfo::SEC_ALLOW_CROSS_ORIGIN_DATA_INHERITS |
+ nsILoadInfo::SEC_ALLOW_CHROME;
+
+ nsContentPolicyType contentPolicyType =
+ aIsPreload ? nsIContentPolicy::TYPE_INTERNAL_STYLESHEET_PRELOAD
+ : nsIContentPolicy::TYPE_INTERNAL_STYLESHEET;
+
+ // Just load it
+ nsCOMPtr<nsIChannel> channel;
+ // Note that we are calling NS_NewChannelWithTriggeringPrincipal() with both
+ // a node and a principal.
+ // This is because of a case where the node is the document being styled and
+ // the principal is the stylesheet (perhaps from a different origin) that is
+ // applying the styles.
+ if (aLoadData->mRequestingNode && aLoadData->mLoaderPrincipal) {
+ rv = NS_NewChannelWithTriggeringPrincipal(getter_AddRefs(channel),
+ aLoadData->mURI,
+ aLoadData->mRequestingNode,
+ aLoadData->mLoaderPrincipal,
+ securityFlags,
+ contentPolicyType);
+ }
+ else {
+ // either we are loading something inside a document, in which case
+ // we should always have a requestingNode, or we are loading something
+ // outside a document, in which case the loadingPrincipal and the
+ // triggeringPrincipal should always be the systemPrincipal.
+ rv = NS_NewChannel(getter_AddRefs(channel),
+ aLoadData->mURI,
+ nsContentUtils::GetSystemPrincipal(),
+ securityFlags,
+ contentPolicyType);
+ }
+ if (NS_FAILED(rv)) {
+ LOG_ERROR((" Failed to create channel"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ nsCOMPtr<nsIInputStream> stream;
+ rv = channel->Open2(getter_AddRefs(stream));
+
+ if (NS_FAILED(rv)) {
+ LOG_ERROR((" Failed to open URI synchronously"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ // Force UA sheets to be UTF-8.
+ // XXX this is only necessary because the default in
+ // SheetLoadData::OnDetermineCharset is wrong (bug 521039).
+ channel->SetContentCharset(NS_LITERAL_CSTRING("UTF-8"));
+
+ // Manually feed the streamloader the contents of the stream.
+ // This will call back into OnStreamComplete
+ // and thence to ParseSheet. Regardless of whether this fails,
+ // SheetComplete has been called.
+ return nsSyncLoadService::PushSyncStreamToListener(stream,
+ streamLoader,
+ channel);
+ }
+
+ SheetLoadData* existingData = nullptr;
+
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aLoadData->mURI,
+ aLoadData->mLoaderPrincipal,
+ aLoadData->mSheet->GetCORSMode(),
+ aLoadData->mSheet->GetReferrerPolicy());
+ if (aSheetState == eSheetLoading) {
+ mSheets->mLoadingDatas.Get(&key, &existingData);
+ NS_ASSERTION(existingData, "CreateSheet lied about the state");
+ }
+ else if (aSheetState == eSheetPending){
+ mSheets->mPendingDatas.Get(&key, &existingData);
+ NS_ASSERTION(existingData, "CreateSheet lied about the state");
+ }
+
+ if (existingData) {
+ LOG((" Glomming on to existing load"));
+ SheetLoadData* data = existingData;
+ while (data->mNext) {
+ data = data->mNext;
+ }
+ data->mNext = aLoadData; // transfer ownership
+ if (aSheetState == eSheetPending && !aLoadData->mWasAlternate) {
+ // Kick the load off; someone cares about it right away
+
+#ifdef DEBUG
+ SheetLoadData* removedData;
+ NS_ASSERTION(mSheets->mPendingDatas.Get(&key, &removedData) &&
+ removedData == existingData,
+ "Bad pending table.");
+#endif
+
+ mSheets->mPendingDatas.Remove(&key);
+
+ LOG((" Forcing load of pending data"));
+ return LoadSheet(existingData, eSheetNeedsParser, aIsPreload);
+ }
+ // All done here; once the load completes we'll be marked complete
+ // automatically
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsILoadGroup> loadGroup;
+ if (mDocument) {
+ loadGroup = mDocument->GetDocumentLoadGroup();
+ // load for a document with no loadgrup indicates that something is
+ // completely bogus, let's bail out early.
+ if (!loadGroup) {
+ LOG_ERROR((" Failed to query loadGroup from document"));
+ SheetComplete(aLoadData, NS_ERROR_UNEXPECTED);
+ return NS_ERROR_UNEXPECTED;
+ }
+ }
+#ifdef DEBUG
+ mSyncCallback = true;
+#endif
+
+ CORSMode ourCORSMode = aLoadData->mSheet->GetCORSMode();
+ nsSecurityFlags securityFlags =
+ ourCORSMode == CORS_NONE
+ ? nsILoadInfo::SEC_ALLOW_CROSS_ORIGIN_DATA_INHERITS
+ : nsILoadInfo::SEC_REQUIRE_CORS_DATA_INHERITS;
+ if (ourCORSMode == CORS_ANONYMOUS) {
+ securityFlags |= nsILoadInfo::SEC_COOKIES_SAME_ORIGIN;
+ } else if (ourCORSMode == CORS_USE_CREDENTIALS) {
+ securityFlags |= nsILoadInfo::SEC_COOKIES_INCLUDE;
+ }
+ securityFlags |= nsILoadInfo::SEC_ALLOW_CHROME;
+
+ nsContentPolicyType contentPolicyType =
+ aIsPreload ? nsIContentPolicy::TYPE_INTERNAL_STYLESHEET_PRELOAD
+ : nsIContentPolicy::TYPE_INTERNAL_STYLESHEET;
+
+ nsCOMPtr<nsIChannel> channel;
+ // Note we are calling NS_NewChannelWithTriggeringPrincipal here with a node
+ // and a principal. This is because of a case where the node is the document
+ // being styled and the principal is the stylesheet (perhaps from a different
+ // origin) that is applying the styles.
+ if (aLoadData->mRequestingNode && aLoadData->mLoaderPrincipal) {
+ rv = NS_NewChannelWithTriggeringPrincipal(getter_AddRefs(channel),
+ aLoadData->mURI,
+ aLoadData->mRequestingNode,
+ aLoadData->mLoaderPrincipal,
+ securityFlags,
+ contentPolicyType,
+ loadGroup,
+ nullptr, // aCallbacks
+ nsIChannel::LOAD_NORMAL |
+ nsIChannel::LOAD_CLASSIFY_URI);
+ }
+ else {
+ // either we are loading something inside a document, in which case
+ // we should always have a requestingNode, or we are loading something
+ // outside a document, in which case the loadingPrincipal and the
+ // triggeringPrincipal should always be the systemPrincipal.
+ rv = NS_NewChannel(getter_AddRefs(channel),
+ aLoadData->mURI,
+ nsContentUtils::GetSystemPrincipal(),
+ securityFlags,
+ contentPolicyType,
+ loadGroup,
+ nullptr, // aCallbacks
+ nsIChannel::LOAD_NORMAL |
+ nsIChannel::LOAD_CLASSIFY_URI);
+ }
+
+ if (NS_FAILED(rv)) {
+#ifdef DEBUG
+ mSyncCallback = false;
+#endif
+ LOG_ERROR((" Failed to create channel"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ if (!aLoadData->mWasAlternate) {
+ nsCOMPtr<nsIClassOfService> cos(do_QueryInterface(channel));
+ if (cos) {
+ cos->AddClassFlags(nsIClassOfService::Leader);
+ }
+ }
+
+ nsCOMPtr<nsIHttpChannel> httpChannel(do_QueryInterface(channel));
+ if (httpChannel) {
+ // Send a minimal Accept header for text/css
+ httpChannel->SetRequestHeader(NS_LITERAL_CSTRING("Accept"),
+ NS_LITERAL_CSTRING("text/css,*/*;q=0.1"),
+ false);
+ nsCOMPtr<nsIURI> referrerURI = aLoadData->GetReferrerURI();
+ if (referrerURI)
+ httpChannel->SetReferrerWithPolicy(referrerURI,
+ aLoadData->mSheet->GetReferrerPolicy());
+
+ nsCOMPtr<nsIHttpChannelInternal> internalChannel = do_QueryInterface(httpChannel);
+ if (internalChannel) {
+ internalChannel->SetIntegrityMetadata(sriMetadata.GetIntegrityString());
+ }
+
+ // Set the initiator type
+ nsCOMPtr<nsITimedChannel> timedChannel(do_QueryInterface(httpChannel));
+ if (timedChannel) {
+ if (aLoadData->mParentData) {
+ timedChannel->SetInitiatorType(NS_LITERAL_STRING("css"));
+ } else {
+ timedChannel->SetInitiatorType(NS_LITERAL_STRING("link"));
+ }
+ }
+ }
+
+ // Now tell the channel we expect text/css data back.... We do
+ // this before opening it, so it's only treated as a hint.
+ channel->SetContentType(NS_LITERAL_CSTRING("text/css"));
+
+ // We don't have to hold on to the stream loader. The ownership
+ // model is: Necko owns the stream loader, which owns the load data,
+ // which owns us
+ nsCOMPtr<nsIUnicharStreamLoader> streamLoader;
+ rv = NS_NewUnicharStreamLoader(getter_AddRefs(streamLoader), aLoadData);
+ if (NS_FAILED(rv)) {
+#ifdef DEBUG
+ mSyncCallback = false;
+#endif
+ LOG_ERROR((" Failed to create stream loader"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ if (mDocument) {
+ mozilla::net::PredictorLearn(aLoadData->mURI, mDocument->GetDocumentURI(),
+ nsINetworkPredictor::LEARN_LOAD_SUBRESOURCE,
+ mDocument);
+ }
+
+ rv = channel->AsyncOpen2(streamLoader);
+
+#ifdef DEBUG
+ mSyncCallback = false;
+#endif
+
+ if (NS_FAILED(rv)) {
+ LOG_ERROR((" Failed to create stream loader"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ mSheets->mLoadingDatas.Put(&key, aLoadData);
+ aLoadData->mIsLoading = true;
+
+ return NS_OK;
+}
+
+/**
+ * ParseSheet handles parsing the data stream. The main idea here is
+ * to push the current load data onto the parse stack before letting
+ * the CSS parser at the data stream. That lets us handle @import
+ * correctly.
+ */
+nsresult
+Loader::ParseSheet(const nsAString& aInput,
+ SheetLoadData* aLoadData,
+ bool& aCompleted)
+{
+ LOG(("css::Loader::ParseSheet"));
+ NS_PRECONDITION(aLoadData, "Must have load data");
+ NS_PRECONDITION(aLoadData->mSheet, "Must have sheet to parse into");
+
+ aCompleted = false;
+
+ // Push our load data on the stack so any kids can pick it up
+ mParsingDatas.AppendElement(aLoadData);
+ nsIURI* sheetURI = aLoadData->mSheet->GetSheetURI();
+ nsIURI* baseURI = aLoadData->mSheet->GetBaseURI();
+
+ nsresult rv;
+
+ if (aLoadData->mSheet->IsGecko()) {
+ nsCSSParser parser(this, aLoadData->mSheet->AsGecko());
+ rv = parser.ParseSheet(aInput, sheetURI, baseURI,
+ aLoadData->mSheet->Principal(),
+ aLoadData->mLineNumber);
+ } else {
+ rv =
+ aLoadData->mSheet->AsServo()->ParseSheet(aInput, sheetURI, baseURI,
+ aLoadData->mSheet->Principal(),
+ aLoadData->mLineNumber);
+ }
+
+ mParsingDatas.RemoveElementAt(mParsingDatas.Length() - 1);
+
+ if (NS_FAILED(rv)) {
+ LOG_ERROR((" Low-level error in parser!"));
+ SheetComplete(aLoadData, rv);
+ return rv;
+ }
+
+ NS_ASSERTION(aLoadData->mPendingChildren == 0 || !aLoadData->mSyncLoad,
+ "Sync load has leftover pending children!");
+
+ if (aLoadData->mPendingChildren == 0) {
+ LOG((" No pending kids from parse"));
+ aCompleted = true;
+ SheetComplete(aLoadData, NS_OK);
+ }
+ // Otherwise, the children are holding strong refs to the data and
+ // will call SheetComplete() on it when they complete.
+
+ return NS_OK;
+}
+
+/**
+ * SheetComplete is the do-it-all cleanup function. It removes the
+ * load data from the "loading" hashtable, adds the sheet to the
+ * "completed" hashtable, massages the XUL cache, handles siblings of
+ * the load data (other loads for the same URI), handles unblocking
+ * blocked parent loads as needed, and most importantly calls
+ * NS_RELEASE on the load data to destroy the whole mess.
+ */
+void
+Loader::SheetComplete(SheetLoadData* aLoadData, nsresult aStatus)
+{
+ LOG(("css::Loader::SheetComplete"));
+
+ if (aLoadData->mSheet->IsServo() && NS_FAILED(aStatus)) {
+ aLoadData->mSheet->AsServo()->LoadFailed();
+ }
+
+ // 8 is probably big enough for all our common cases. It's not likely that
+ // imports will nest more than 8 deep, and multiple sheets with the same URI
+ // are rare.
+ AutoTArray<RefPtr<SheetLoadData>, 8> datasToNotify;
+ DoSheetComplete(aLoadData, aStatus, datasToNotify);
+
+ // Now it's safe to go ahead and notify observers
+ uint32_t count = datasToNotify.Length();
+ mDatasToNotifyOn += count;
+ for (uint32_t i = 0; i < count; ++i) {
+ --mDatasToNotifyOn;
+
+ SheetLoadData* data = datasToNotify[i];
+ NS_ASSERTION(data && data->mMustNotify, "How did this data get here?");
+ if (data->mObserver) {
+ LOG((" Notifying observer %p for data %p. wasAlternate: %d",
+ data->mObserver.get(), data, data->mWasAlternate));
+ data->mObserver->StyleSheetLoaded(data->mSheet, data->mWasAlternate,
+ aStatus);
+ }
+
+ nsTObserverArray<nsCOMPtr<nsICSSLoaderObserver> >::ForwardIterator iter(mObservers);
+ nsCOMPtr<nsICSSLoaderObserver> obs;
+ while (iter.HasMore()) {
+ obs = iter.GetNext();
+ LOG((" Notifying global observer %p for data %p. wasAlternate: %d",
+ obs.get(), data, data->mWasAlternate));
+ obs->StyleSheetLoaded(data->mSheet, data->mWasAlternate, aStatus);
+ }
+ }
+
+ if (mSheets->mLoadingDatas.Count() == 0 && mSheets->mPendingDatas.Count() > 0) {
+ LOG((" No more loading sheets; starting alternates"));
+ StartAlternateLoads();
+ }
+}
+
+void
+Loader::DoSheetComplete(SheetLoadData* aLoadData, nsresult aStatus,
+ LoadDataArray& aDatasToNotify)
+{
+ LOG(("css::Loader::DoSheetComplete"));
+ NS_PRECONDITION(aLoadData, "Must have a load data!");
+ NS_PRECONDITION(aLoadData->mSheet, "Must have a sheet");
+ NS_ASSERTION(mSheets, "mLoadingDatas should be initialized by now.");
+
+ LOG(("Load completed, status: 0x%x", aStatus));
+
+ // Twiddle the hashtables
+ if (aLoadData->mURI) {
+ LOG_URI(" Finished loading: '%s'", aLoadData->mURI);
+ // Remove the data from the list of loading datas
+ if (aLoadData->mIsLoading) {
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aLoadData->mURI,
+ aLoadData->mLoaderPrincipal,
+ aLoadData->mSheet->GetCORSMode(),
+ aLoadData->mSheet->GetReferrerPolicy());
+#ifdef DEBUG
+ SheetLoadData *loadingData;
+ NS_ASSERTION(mSheets->mLoadingDatas.Get(&key, &loadingData) &&
+ loadingData == aLoadData,
+ "Bad loading table");
+#endif
+
+ mSheets->mLoadingDatas.Remove(&key);
+ aLoadData->mIsLoading = false;
+ }
+ }
+
+ // Go through and deal with the whole linked list.
+ SheetLoadData* data = aLoadData;
+ while (data) {
+ if (!data->mSheetAlreadyComplete) {
+ // If mSheetAlreadyComplete, then the sheet could well be modified between
+ // when we posted the async call to SheetComplete and now, since the sheet
+ // was page-accessible during that whole time.
+ MOZ_ASSERT(!(data->mSheet->IsGecko() &&
+ data->mSheet->AsGecko()->IsModified()),
+ "should not get marked modified during parsing");
+ data->mSheet->SetComplete();
+ data->ScheduleLoadEventIfNeeded(aStatus);
+ }
+ if (data->mMustNotify && (data->mObserver || !mObservers.IsEmpty())) {
+ // Don't notify here so we don't trigger script. Remember the
+ // info we need to notify, then do it later when it's safe.
+ aDatasToNotify.AppendElement(data);
+
+ // On append failure, just press on. We'll fail to notify the observer,
+ // but not much we can do about that....
+ }
+
+ NS_ASSERTION(!data->mParentData ||
+ data->mParentData->mPendingChildren != 0,
+ "Broken pending child count on our parent");
+
+ // If we have a parent, our parent is no longer being parsed, and
+ // we are the last pending child, then our load completion
+ // completes the parent too. Note that the parent _can_ still be
+ // being parsed (eg if the child (us) failed to open the channel
+ // or some such).
+ if (data->mParentData &&
+ --(data->mParentData->mPendingChildren) == 0 &&
+ !mParsingDatas.Contains(data->mParentData)) {
+ DoSheetComplete(data->mParentData, aStatus, aDatasToNotify);
+ }
+
+ data = data->mNext;
+ }
+
+ // Now that it's marked complete, put the sheet in our cache.
+ // If we ever start doing this for failure aStatus, we'll need to
+ // adjust the PostLoadEvent code that thinks anything already
+ // complete must have loaded succesfully.
+ if (NS_SUCCEEDED(aStatus) && aLoadData->mURI) {
+ // Pick our sheet to cache carefully. Ideally, we want to cache
+ // one of the sheets that will be kept alive by a document or
+ // parent sheet anyway, so that if someone then accesses it via
+ // CSSOM we won't have extra clones of the inner lying around.
+ if (aLoadData->mSheet->IsGecko()) {
+ data = aLoadData;
+ CSSStyleSheet* sheet = aLoadData->mSheet->AsGecko();
+ while (data) {
+ if (data->mSheet->GetParentSheet() || data->mSheet->GetOwnerNode()) {
+ sheet = data->mSheet->AsGecko();
+ break;
+ }
+ data = data->mNext;
+ }
+#ifdef MOZ_XUL
+ if (IsChromeURI(aLoadData->mURI)) {
+ nsXULPrototypeCache* cache = nsXULPrototypeCache::GetInstance();
+ if (cache && cache->IsEnabled()) {
+ if (!cache->GetStyleSheet(aLoadData->mURI)) {
+ LOG((" Putting sheet in XUL prototype cache"));
+ NS_ASSERTION(sheet->IsComplete(),
+ "Should only be caching complete sheets");
+ cache->PutStyleSheet(sheet);
+ }
+ }
+ }
+ else {
+#endif
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(aLoadData->mURI,
+ aLoadData->mLoaderPrincipal,
+ aLoadData->mSheet->GetCORSMode(),
+ aLoadData->mSheet->GetReferrerPolicy());
+ NS_ASSERTION(sheet->IsComplete(),
+ "Should only be caching complete sheets");
+ mSheets->mCompleteSheets.Put(&key, sheet);
+#ifdef MOZ_XUL
+ }
+#endif
+ } else {
+ NS_WARNING("stylo: Stylesheet caching not yet supported - see bug 1290218.");
+ }
+ }
+
+ NS_RELEASE(aLoadData); // this will release parents and siblings and all that
+}
+
+nsresult
+Loader::LoadInlineStyle(nsIContent* aElement,
+ const nsAString& aBuffer,
+ uint32_t aLineNumber,
+ const nsAString& aTitle,
+ const nsAString& aMedia,
+ Element* aScopeElement,
+ nsICSSLoaderObserver* aObserver,
+ bool* aCompleted,
+ bool* aIsAlternate)
+{
+ LOG(("css::Loader::LoadInlineStyle"));
+ NS_ASSERTION(mParsingDatas.Length() == 0, "We're in the middle of a parse?");
+
+ *aCompleted = true;
+
+ if (!mEnabled) {
+ LOG_WARN((" Not enabled"));
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_INITIALIZED);
+
+ nsCOMPtr<nsIStyleSheetLinkingElement> owningElement(do_QueryInterface(aElement));
+ NS_ASSERTION(owningElement, "Element is not a style linking element!");
+
+ // Since we're not planning to load a URI, no need to hand a principal to the
+ // load data or to CreateSheet(). Also, OK to use CORS_NONE for the CORS
+ // mode and mDocument's ReferrerPolicy.
+ StyleSheetState state;
+ RefPtr<StyleSheet> sheet;
+ nsresult rv = CreateSheet(nullptr, aElement, nullptr, eAuthorSheetFeatures,
+ CORS_NONE, mDocument->GetReferrerPolicy(),
+ EmptyString(), // no inline integrity checks
+ false, false, aTitle, state, aIsAlternate,
+ &sheet);
+ NS_ENSURE_SUCCESS(rv, rv);
+ NS_ASSERTION(state == eSheetNeedsParser,
+ "Inline sheets should not be cached");
+
+ LOG((" Sheet is alternate: %d", *aIsAlternate));
+
+ PrepareSheet(sheet, aTitle, aMedia, nullptr, aScopeElement, *aIsAlternate);
+
+ if (aElement->HasFlag(NODE_IS_IN_SHADOW_TREE)) {
+ ShadowRoot* containingShadow = aElement->GetContainingShadow();
+ MOZ_ASSERT(containingShadow);
+ containingShadow->InsertSheet(sheet, aElement);
+ } else {
+ rv = InsertSheetInDoc(sheet, aElement, mDocument);
+ NS_ENSURE_SUCCESS(rv, rv);
+ }
+
+ SheetLoadData* data = new SheetLoadData(this, aTitle, nullptr, sheet,
+ owningElement, *aIsAlternate,
+ aObserver, nullptr, static_cast<nsINode*>(aElement));
+
+ // We never actually load this, so just set its principal directly
+ sheet->SetPrincipal(aElement->NodePrincipal());
+
+ NS_ADDREF(data);
+ data->mLineNumber = aLineNumber;
+ // Parse completion releases the load data
+ rv = ParseSheet(aBuffer, data, *aCompleted);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ // If aCompleted is true, |data| may well be deleted by now.
+ if (!*aCompleted) {
+ data->mMustNotify = true;
+ }
+ return rv;
+}
+
+nsresult
+Loader::LoadStyleLink(nsIContent* aElement,
+ nsIURI* aURL,
+ const nsAString& aTitle,
+ const nsAString& aMedia,
+ bool aHasAlternateRel,
+ CORSMode aCORSMode,
+ ReferrerPolicy aReferrerPolicy,
+ const nsAString& aIntegrity,
+ nsICSSLoaderObserver* aObserver,
+ bool* aIsAlternate)
+{
+ LOG(("css::Loader::LoadStyleLink"));
+ NS_PRECONDITION(aURL, "Must have URL to load");
+ NS_ASSERTION(mParsingDatas.Length() == 0, "We're in the middle of a parse?");
+
+ LOG_URI(" Link uri: '%s'", aURL);
+ LOG((" Link title: '%s'", NS_ConvertUTF16toUTF8(aTitle).get()));
+ LOG((" Link media: '%s'", NS_ConvertUTF16toUTF8(aMedia).get()));
+ LOG((" Link alternate rel: %d", aHasAlternateRel));
+
+ if (!mEnabled) {
+ LOG_WARN((" Not enabled"));
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_INITIALIZED);
+
+ nsIPrincipal* principal =
+ aElement ? aElement->NodePrincipal() : mDocument->NodePrincipal();
+
+ nsISupports* context = aElement;
+ if (!context) {
+ context = mDocument;
+ }
+
+ nsresult rv = CheckContentPolicy(principal, aURL, context, false);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ // Don't fire the error event if our document is loaded as data. We're
+ // supposed to not even try to do loads in that case... Unfortunately, we
+ // implement that via nsDataDocumentContentPolicy, which doesn't have a good
+ // way to communicate back to us that _it_ is the thing that blocked the
+ // load.
+ if (aElement && !mDocument->IsLoadedAsData()) {
+ // Fire an async error event on it.
+ RefPtr<AsyncEventDispatcher> loadBlockingAsyncDispatcher =
+ new LoadBlockingAsyncEventDispatcher(aElement,
+ NS_LITERAL_STRING("error"),
+ false, false);
+ loadBlockingAsyncDispatcher->PostDOMEvent();
+ }
+ return rv;
+ }
+
+ StyleSheetState state;
+ RefPtr<StyleSheet> sheet;
+ rv = CreateSheet(aURL, aElement, principal, eAuthorSheetFeatures,
+ aCORSMode, aReferrerPolicy, aIntegrity, false,
+ aHasAlternateRel, aTitle, state, aIsAlternate,
+ &sheet);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ LOG((" Sheet is alternate: %d", *aIsAlternate));
+
+ PrepareSheet(sheet, aTitle, aMedia, nullptr, nullptr, *aIsAlternate);
+
+ rv = InsertSheetInDoc(sheet, aElement, mDocument);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ nsCOMPtr<nsIStyleSheetLinkingElement> owningElement(do_QueryInterface(aElement));
+
+ if (state == eSheetComplete) {
+ LOG((" Sheet already complete: 0x%p", sheet.get()));
+ if (aObserver || !mObservers.IsEmpty() || owningElement) {
+ rv = PostLoadEvent(aURL, sheet, aObserver, *aIsAlternate,
+ owningElement);
+ return rv;
+ }
+
+ return NS_OK;
+ }
+
+ // Now we need to actually load it
+ nsCOMPtr<nsINode> requestingNode = do_QueryInterface(context);
+ SheetLoadData* data = new SheetLoadData(this, aTitle, aURL, sheet,
+ owningElement, *aIsAlternate,
+ aObserver, principal, requestingNode);
+ NS_ADDREF(data);
+
+ // If we have to parse and it's an alternate non-inline, defer it
+ if (aURL && state == eSheetNeedsParser && mSheets->mLoadingDatas.Count() != 0 &&
+ *aIsAlternate) {
+ LOG((" Deferring alternate sheet load"));
+ URIPrincipalReferrerPolicyAndCORSModeHashKey key(data->mURI,
+ data->mLoaderPrincipal,
+ data->mSheet->GetCORSMode(),
+ data->mSheet->GetReferrerPolicy());
+ mSheets->mPendingDatas.Put(&key, data);
+
+ data->mMustNotify = true;
+ return NS_OK;
+ }
+
+ // Load completion will free the data
+ rv = LoadSheet(data, state, false);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ data->mMustNotify = true;
+ return rv;
+}
+
+static bool
+HaveAncestorDataWithURI(SheetLoadData *aData, nsIURI *aURI)
+{
+ if (!aData->mURI) {
+ // Inline style; this won't have any ancestors
+ MOZ_ASSERT(!aData->mParentData,
+ "How does inline style have a parent?");
+ return false;
+ }
+
+ bool equal;
+ if (NS_FAILED(aData->mURI->Equals(aURI, &equal)) || equal) {
+ return true;
+ }
+
+ // Datas down the mNext chain have the same URI as aData, so we
+ // don't have to compare to them. But they might have different
+ // parents, and we have to check all of those.
+ while (aData) {
+ if (aData->mParentData &&
+ HaveAncestorDataWithURI(aData->mParentData, aURI)) {
+ return true;
+ }
+
+ aData = aData->mNext;
+ }
+
+ return false;
+}
+
+nsresult
+Loader::LoadChildSheet(StyleSheet* aParentSheet,
+ nsIURI* aURL,
+ nsMediaList* aMedia,
+ ImportRule* aParentRule,
+ LoaderReusableStyleSheets* aReusableSheets)
+{
+ LOG(("css::Loader::LoadChildSheet"));
+ NS_PRECONDITION(aURL, "Must have a URI to load");
+ NS_PRECONDITION(aParentSheet, "Must have a parent sheet");
+
+ if (!mEnabled) {
+ LOG_WARN((" Not enabled"));
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ LOG_URI(" Child uri: '%s'", aURL);
+
+ nsCOMPtr<nsINode> owningNode;
+
+ // check for an owning document: if none, don't bother walking up the parent
+ // sheets
+ if (aParentSheet->GetOwningDocument()) {
+ StyleSheet* topSheet = aParentSheet;
+ while (StyleSheet* parent = topSheet->GetParentSheet()) {
+ topSheet = parent;
+ }
+ owningNode = topSheet->GetOwnerNode();
+ }
+
+ nsISupports* context = owningNode;
+ if (!context) {
+ context = mDocument;
+ }
+
+ nsIPrincipal* principal = aParentSheet->Principal();
+ nsresult rv = CheckContentPolicy(principal, aURL, context, false);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ SheetLoadData* parentData = nullptr;
+ nsCOMPtr<nsICSSLoaderObserver> observer;
+
+ int32_t count = mParsingDatas.Length();
+ if (count > 0) {
+ LOG((" Have a parent load"));
+ parentData = mParsingDatas.ElementAt(count - 1);
+ // Check for cycles
+ if (HaveAncestorDataWithURI(parentData, aURL)) {
+ // Houston, we have a loop, blow off this child and pretend this never
+ // happened
+ LOG_ERROR((" @import cycle detected, dropping load"));
+ return NS_OK;
+ }
+
+ NS_ASSERTION(parentData->mSheet == aParentSheet,
+ "Unexpected call to LoadChildSheet");
+ } else {
+ LOG((" No parent load; must be CSSOM"));
+ // No parent load data, so the sheet will need to be notified when
+ // we finish, if it can be, if we do the load asynchronously.
+ // XXXheycam ServoStyleSheet doesn't implement nsICSSLoaderObserver yet.
+ MOZ_ASSERT(aParentSheet->IsGecko(),
+ "stylo: ServoStyleSheets don't support child sheet loading yet");
+ observer = aParentSheet->AsGecko();
+ }
+
+ // Now that we know it's safe to load this (passes security check and not a
+ // loop) do so.
+ RefPtr<StyleSheet> sheet;
+ RefPtr<CSSStyleSheet> reusableSheet;
+ StyleSheetState state;
+ if (aReusableSheets && aReusableSheets->FindReusableStyleSheet(aURL, reusableSheet)) {
+ sheet = reusableSheet;
+ aParentRule->SetSheet(reusableSheet);
+ state = eSheetComplete;
+ } else {
+ bool isAlternate;
+ const nsSubstring& empty = EmptyString();
+ // For now, use CORS_NONE for child sheets
+ rv = CreateSheet(aURL, nullptr, principal,
+ aParentSheet->ParsingMode(),
+ CORS_NONE, aParentSheet->GetReferrerPolicy(),
+ EmptyString(), // integrity is only checked on main sheet
+ parentData ? parentData->mSyncLoad : false,
+ false, empty, state, &isAlternate, &sheet);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ PrepareSheet(sheet, empty, empty, aMedia, nullptr, isAlternate);
+ }
+
+ rv = InsertChildSheet(sheet, aParentSheet, aParentRule);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ if (state == eSheetComplete) {
+ LOG((" Sheet already complete"));
+ // We're completely done. No need to notify, even, since the
+ // @import rule addition/modification will trigger the right style
+ // changes automatically.
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsINode> requestingNode = do_QueryInterface(context);
+ SheetLoadData* data = new SheetLoadData(this, aURL, sheet, parentData,
+ observer, principal, requestingNode);
+
+ NS_ADDREF(data);
+ bool syncLoad = data->mSyncLoad;
+
+ // Load completion will release the data
+ rv = LoadSheet(data, state, false);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ // If syncLoad is true, |data| will be deleted by now.
+ if (!syncLoad) {
+ data->mMustNotify = true;
+ }
+ return rv;
+}
+
+nsresult
+Loader::LoadSheetSync(nsIURI* aURL,
+ SheetParsingMode aParsingMode,
+ bool aUseSystemPrincipal,
+ RefPtr<StyleSheet>* aSheet)
+{
+ LOG(("css::Loader::LoadSheetSync"));
+ return InternalLoadNonDocumentSheet(aURL,
+ false, aParsingMode, aUseSystemPrincipal,
+ nullptr, EmptyCString(),
+ aSheet, nullptr);
+}
+
+nsresult
+Loader::LoadSheet(nsIURI* aURL,
+ nsIPrincipal* aOriginPrincipal,
+ const nsCString& aCharset,
+ nsICSSLoaderObserver* aObserver,
+ RefPtr<StyleSheet>* aSheet)
+{
+ LOG(("css::Loader::LoadSheet(aURL, aObserver, aSheet) api call"));
+ NS_PRECONDITION(aSheet, "aSheet is null");
+ return InternalLoadNonDocumentSheet(aURL,
+ false, eAuthorSheetFeatures, false,
+ aOriginPrincipal, aCharset,
+ aSheet, aObserver);
+}
+
+nsresult
+Loader::LoadSheet(nsIURI* aURL,
+ bool aIsPreload,
+ nsIPrincipal* aOriginPrincipal,
+ const nsCString& aCharset,
+ nsICSSLoaderObserver* aObserver,
+ CORSMode aCORSMode,
+ ReferrerPolicy aReferrerPolicy,
+ const nsAString& aIntegrity)
+{
+ LOG(("css::Loader::LoadSheet(aURL, aObserver) api call"));
+ return InternalLoadNonDocumentSheet(aURL,
+ aIsPreload, eAuthorSheetFeatures, false,
+ aOriginPrincipal, aCharset,
+ nullptr, aObserver,
+ aCORSMode, aReferrerPolicy, aIntegrity);
+}
+
+nsresult
+Loader::InternalLoadNonDocumentSheet(nsIURI* aURL,
+ bool aIsPreload,
+ SheetParsingMode aParsingMode,
+ bool aUseSystemPrincipal,
+ nsIPrincipal* aOriginPrincipal,
+ const nsCString& aCharset,
+ RefPtr<StyleSheet>* aSheet,
+ nsICSSLoaderObserver* aObserver,
+ CORSMode aCORSMode,
+ ReferrerPolicy aReferrerPolicy,
+ const nsAString& aIntegrity)
+{
+ NS_PRECONDITION(aURL, "Must have a URI to load");
+ NS_PRECONDITION(aSheet || aObserver, "Sheet and observer can't both be null");
+ NS_PRECONDITION(!aUseSystemPrincipal || !aObserver,
+ "Shouldn't load system-principal sheets async");
+ NS_ASSERTION(mParsingDatas.Length() == 0, "We're in the middle of a parse?");
+
+ LOG_URI(" Non-document sheet uri: '%s'", aURL);
+
+ if (aSheet) {
+ *aSheet = nullptr;
+ }
+
+ if (!mEnabled) {
+ LOG_WARN((" Not enabled"));
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ nsresult rv = CheckContentPolicy(aOriginPrincipal, aURL, mDocument, aIsPreload);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ StyleSheetState state;
+ bool isAlternate;
+ RefPtr<StyleSheet> sheet;
+ bool syncLoad = (aObserver == nullptr);
+ const nsSubstring& empty = EmptyString();
+
+ rv = CreateSheet(aURL, nullptr, aOriginPrincipal, aParsingMode,
+ aCORSMode, aReferrerPolicy, aIntegrity, syncLoad,
+ false, empty, state, &isAlternate, &sheet);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ PrepareSheet(sheet, empty, empty, nullptr, nullptr, isAlternate);
+
+ if (state == eSheetComplete) {
+ LOG((" Sheet already complete"));
+ if (aObserver || !mObservers.IsEmpty()) {
+ rv = PostLoadEvent(aURL, sheet, aObserver, false, nullptr);
+ }
+ if (aSheet) {
+ sheet.swap(*aSheet);
+ }
+ return rv;
+ }
+
+ SheetLoadData* data =
+ new SheetLoadData(this, aURL, sheet, syncLoad,
+ aUseSystemPrincipal, aCharset, aObserver,
+ aOriginPrincipal, mDocument);
+
+ NS_ADDREF(data);
+ rv = LoadSheet(data, state, aIsPreload);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ if (aSheet) {
+ sheet.swap(*aSheet);
+ }
+ if (aObserver) {
+ data->mMustNotify = true;
+ }
+
+ return rv;
+}
+
+nsresult
+Loader::PostLoadEvent(nsIURI* aURI,
+ StyleSheet* aSheet,
+ nsICSSLoaderObserver* aObserver,
+ bool aWasAlternate,
+ nsIStyleSheetLinkingElement* aElement)
+{
+ LOG(("css::Loader::PostLoadEvent"));
+ NS_PRECONDITION(aSheet, "Must have sheet");
+ NS_PRECONDITION(aObserver || !mObservers.IsEmpty() || aElement,
+ "Must have observer or element");
+
+ RefPtr<SheetLoadData> evt =
+ new SheetLoadData(this, EmptyString(), // title doesn't matter here
+ aURI,
+ aSheet,
+ aElement,
+ aWasAlternate,
+ aObserver,
+ nullptr,
+ mDocument);
+ NS_ENSURE_TRUE(evt, NS_ERROR_OUT_OF_MEMORY);
+
+ if (!mPostedEvents.AppendElement(evt)) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ nsresult rv = NS_DispatchToCurrentThread(evt);
+ if (NS_FAILED(rv)) {
+ NS_WARNING("failed to dispatch stylesheet load event");
+ mPostedEvents.RemoveElement(evt);
+ } else {
+ // We'll unblock onload when we handle the event.
+ if (mDocument) {
+ mDocument->BlockOnload();
+ }
+
+ // We want to notify the observer for this data.
+ evt->mMustNotify = true;
+ evt->mSheetAlreadyComplete = true;
+
+ // If we get to this code, aSheet loaded correctly at some point, so
+ // we can just use NS_OK for the status. Note that we do this here
+ // and not from inside our SheetComplete so that we don't end up
+ // running the load event async.
+ evt->ScheduleLoadEventIfNeeded(NS_OK);
+ }
+
+ return rv;
+}
+
+void
+Loader::HandleLoadEvent(SheetLoadData* aEvent)
+{
+ // XXXbz can't assert this yet.... May not have an observer because
+ // we're unblocking the parser
+ // NS_ASSERTION(aEvent->mObserver, "Must have observer");
+ NS_ASSERTION(aEvent->mSheet, "Must have sheet");
+
+ // Very important: this needs to come before the SheetComplete call
+ // below, so that HasPendingLoads() will test true as needed under
+ // notifications we send from that SheetComplete call.
+ mPostedEvents.RemoveElement(aEvent);
+
+ if (!aEvent->mIsCancelled) {
+ // SheetComplete will call Release(), so give it a reference to do
+ // that with.
+ NS_ADDREF(aEvent);
+ SheetComplete(aEvent, NS_OK);
+ }
+
+ if (mDocument) {
+ mDocument->UnblockOnload(true);
+ }
+}
+
+static void
+StopLoadingSheets(
+ nsDataHashtable<URIPrincipalReferrerPolicyAndCORSModeHashKey, SheetLoadData*>& aDatas,
+ Loader::LoadDataArray& aArr)
+{
+ for (auto iter = aDatas.Iter(); !iter.Done(); iter.Next()) {
+ SheetLoadData* data = iter.Data();
+ MOZ_ASSERT(data, "Must have a data!");
+
+ data->mIsLoading = false; // we will handle the removal right here
+ data->mIsCancelled = true;
+
+ aArr.AppendElement(data);
+
+ iter.Remove();
+ }
+}
+
+nsresult
+Loader::Stop()
+{
+ uint32_t pendingCount = mSheets ? mSheets->mPendingDatas.Count() : 0;
+ uint32_t loadingCount = mSheets ? mSheets->mLoadingDatas.Count() : 0;
+ LoadDataArray arr(pendingCount + loadingCount + mPostedEvents.Length());
+
+ if (pendingCount) {
+ StopLoadingSheets(mSheets->mPendingDatas, arr);
+ }
+ if (loadingCount) {
+ StopLoadingSheets(mSheets->mLoadingDatas, arr);
+ }
+
+ uint32_t i;
+ for (i = 0; i < mPostedEvents.Length(); ++i) {
+ SheetLoadData* data = mPostedEvents[i];
+ data->mIsCancelled = true;
+ if (arr.AppendElement(data)) {
+ // SheetComplete() calls Release(), so give this an extra ref.
+ NS_ADDREF(data);
+ }
+#ifdef DEBUG
+ else {
+ NS_NOTREACHED("We preallocated this memory... shouldn't really fail, "
+ "except we never check that preallocation succeeds.");
+ }
+#endif
+ }
+ mPostedEvents.Clear();
+
+ mDatasToNotifyOn += arr.Length();
+ for (i = 0; i < arr.Length(); ++i) {
+ --mDatasToNotifyOn;
+ SheetComplete(arr[i], NS_BINDING_ABORTED);
+ }
+ return NS_OK;
+}
+
+bool
+Loader::HasPendingLoads()
+{
+ return
+ (mSheets && mSheets->mLoadingDatas.Count() != 0) ||
+ (mSheets && mSheets->mPendingDatas.Count() != 0) ||
+ mPostedEvents.Length() != 0 ||
+ mDatasToNotifyOn != 0;
+}
+
+nsresult
+Loader::AddObserver(nsICSSLoaderObserver* aObserver)
+{
+ NS_PRECONDITION(aObserver, "Must have observer");
+ if (mObservers.AppendElementUnlessExists(aObserver)) {
+ return NS_OK;
+ }
+
+ return NS_ERROR_OUT_OF_MEMORY;
+}
+
+void
+Loader::RemoveObserver(nsICSSLoaderObserver* aObserver)
+{
+ mObservers.RemoveElement(aObserver);
+}
+
+void
+Loader::StartAlternateLoads()
+{
+ NS_PRECONDITION(mSheets, "Don't call me!");
+ LoadDataArray arr(mSheets->mPendingDatas.Count());
+ for (auto iter = mSheets->mPendingDatas.Iter(); !iter.Done(); iter.Next()) {
+ arr.AppendElement(iter.Data());
+ iter.Remove();
+ }
+
+ mDatasToNotifyOn += arr.Length();
+ for (uint32_t i = 0; i < arr.Length(); ++i) {
+ --mDatasToNotifyOn;
+ LoadSheet(arr[i], eSheetNeedsParser, false);
+ }
+}
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(Loader)
+
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(Loader)
+ if (tmp->mSheets) {
+ for (auto iter = tmp->mSheets->mCompleteSheets.Iter();
+ !iter.Done();
+ iter.Next()) {
+ NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "Sheet cache nsCSSLoader");
+ if (iter.UserData()->IsGecko()) {
+ CSSStyleSheet* sheet = iter.UserData()->AsGecko();
+ cb.NoteXPCOMChild(NS_ISUPPORTS_CAST(nsIDOMCSSStyleSheet*, sheet));
+ }
+ }
+ }
+ nsTObserverArray<nsCOMPtr<nsICSSLoaderObserver>>::ForwardIterator
+ it(tmp->mObservers);
+ while (it.HasMore()) {
+ ImplCycleCollectionTraverse(cb, it.GetNext(),
+ "mozilla::css::Loader.mObservers");
+ }
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(Loader)
+ if (tmp->mSheets) {
+ tmp->mSheets->mCompleteSheets.Clear();
+ }
+ tmp->mObservers.Clear();
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END
+
+NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(Loader, AddRef)
+NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(Loader, Release)
+
+size_t
+Loader::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = aMallocSizeOf(this);
+
+ if (mSheets) {
+ n += mSheets->mCompleteSheets.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto iter = mSheets->mCompleteSheets.ConstIter();
+ !iter.Done();
+ iter.Next()) {
+ // If aSheet has a parent, then its parent will report it so we don't
+ // have to worry about it here. Likewise, if aSheet has an owning node,
+ // then the document that node is in will report it.
+ const StyleSheet* sheet = iter.UserData();
+ n += (sheet->GetOwnerNode() || sheet->GetParentSheet())
+ ? 0
+ : sheet->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ }
+ n += mObservers.ShallowSizeOfExcludingThis(aMallocSizeOf);
+
+ // Measurement of the following members may be added later if DMD finds it is
+ // worthwhile:
+ // - mLoadingDatas: transient, and should be small
+ // - mPendingDatas: transient, and should be small
+ // - mParsingDatas: transient, and should be small
+ // - mPostedEvents: transient, and should be small
+ //
+ // The following members aren't measured:
+ // - mDocument, because it's a weak backpointer
+ // - mPreferredSheet, because it can be a shared string
+
+ return n;
+}
+
+StyleBackendType
+Loader::GetStyleBackendType() const
+{
+ MOZ_ASSERT(mStyleBackendType || mDocument,
+ "you must construct a Loader with a document or set a "
+ "StyleBackendType on it before calling GetStyleBackendType");
+ if (mStyleBackendType) {
+ return *mStyleBackendType;
+ }
+ return mDocument->GetStyleBackendType();
+}
+
+} // namespace css
+} // namespace mozilla