summaryrefslogtreecommitdiffstats
path: root/parser/html/nsHtml5TreeBuilderCppSupplement.h
diff options
context:
space:
mode:
Diffstat (limited to 'parser/html/nsHtml5TreeBuilderCppSupplement.h')
-rw-r--r--parser/html/nsHtml5TreeBuilderCppSupplement.h2032
1 files changed, 2032 insertions, 0 deletions
diff --git a/parser/html/nsHtml5TreeBuilderCppSupplement.h b/parser/html/nsHtml5TreeBuilderCppSupplement.h
new file mode 100644
index 0000000000..ba9c217cb9
--- /dev/null
+++ b/parser/html/nsHtml5TreeBuilderCppSupplement.h
@@ -0,0 +1,2032 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=2 sw=2 et tw=78: */
+/* 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 "ErrorList.h"
+#include "nsError.h"
+#include "nsHtml5AttributeName.h"
+#include "nsHtml5HtmlAttributes.h"
+#include "nsHtml5String.h"
+#include "nsNetUtil.h"
+#include "mozilla/dom/FetchPriority.h"
+#include "mozilla/dom/ShadowRoot.h"
+#include "mozilla/dom/ShadowRootBinding.h"
+#include "mozilla/CheckedInt.h"
+#include "mozilla/Likely.h"
+#include "mozilla/StaticPrefs_dom.h"
+#include "mozilla/StaticPrefs_network.h"
+#include "mozilla/UniquePtr.h"
+#include "mozilla/UniquePtrExtensions.h"
+
+nsHtml5TreeBuilder::nsHtml5TreeBuilder(nsHtml5OplessBuilder* aBuilder)
+ : mode(0),
+ originalMode(0),
+ framesetOk(false),
+ tokenizer(nullptr),
+ scriptingEnabled(false),
+ needToDropLF(false),
+ fragment(false),
+ contextName(nullptr),
+ contextNamespace(kNameSpaceID_None),
+ contextNode(nullptr),
+ templateModePtr(0),
+ stackNodesIdx(0),
+ numStackNodes(0),
+ currentPtr(0),
+ listPtr(0),
+ formPointer(nullptr),
+ headPointer(nullptr),
+ charBufferLen(0),
+ quirks(false),
+ forceNoQuirks(false),
+ allowDeclarativeShadowRoots(false),
+ mBuilder(aBuilder),
+ mViewSource(nullptr),
+ mOpSink(nullptr),
+ mHandles(nullptr),
+ mHandlesUsed(0),
+ mSpeculativeLoadStage(nullptr),
+ mBroken(NS_OK),
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock(false),
+ mPreventScriptExecution(false),
+ mGenerateSpeculativeLoads(false),
+ mHasSeenImportMap(false)
+#ifdef DEBUG
+ ,
+ mActive(false)
+#endif
+{
+ MOZ_COUNT_CTOR(nsHtml5TreeBuilder);
+}
+
+nsHtml5TreeBuilder::nsHtml5TreeBuilder(nsAHtml5TreeOpSink* aOpSink,
+ nsHtml5TreeOpStage* aStage,
+ bool aGenerateSpeculativeLoads)
+ : mode(0),
+ originalMode(0),
+ framesetOk(false),
+ tokenizer(nullptr),
+ scriptingEnabled(false),
+ needToDropLF(false),
+ fragment(false),
+ contextName(nullptr),
+ contextNamespace(kNameSpaceID_None),
+ contextNode(nullptr),
+ templateModePtr(0),
+ stackNodesIdx(0),
+ numStackNodes(0),
+ currentPtr(0),
+ listPtr(0),
+ formPointer(nullptr),
+ headPointer(nullptr),
+ charBufferLen(0),
+ quirks(false),
+ forceNoQuirks(false),
+ allowDeclarativeShadowRoots(false),
+ mBuilder(nullptr),
+ mViewSource(nullptr),
+ mOpSink(aOpSink),
+ mHandles(new nsIContent*[NS_HTML5_TREE_BUILDER_HANDLE_ARRAY_LENGTH]),
+ mHandlesUsed(0),
+ mSpeculativeLoadStage(aStage),
+ mBroken(NS_OK),
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock(false),
+ mPreventScriptExecution(false),
+ mGenerateSpeculativeLoads(aGenerateSpeculativeLoads),
+ mHasSeenImportMap(false)
+#ifdef DEBUG
+ ,
+ mActive(false)
+#endif
+{
+ MOZ_ASSERT(!(!aStage && aGenerateSpeculativeLoads),
+ "Must not generate speculative loads without a stage");
+ MOZ_COUNT_CTOR(nsHtml5TreeBuilder);
+}
+
+nsHtml5TreeBuilder::~nsHtml5TreeBuilder() {
+ MOZ_COUNT_DTOR(nsHtml5TreeBuilder);
+ NS_ASSERTION(!mActive,
+ "nsHtml5TreeBuilder deleted without ever calling end() on it!");
+ mOpQueue.Clear();
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::createElement(
+ int32_t aNamespace, nsAtom* aName, nsHtml5HtmlAttributes* aAttributes,
+ nsIContentHandle* aIntendedParent, nsHtml5ContentCreatorFunction aCreator) {
+ MOZ_ASSERT(aAttributes, "Got null attributes.");
+ MOZ_ASSERT(aName, "Got null name.");
+ MOZ_ASSERT(aNamespace == kNameSpaceID_XHTML ||
+ aNamespace == kNameSpaceID_SVG ||
+ aNamespace == kNameSpaceID_MathML,
+ "Bogus namespace.");
+
+ if (mBuilder) {
+ nsIContent* intendedParent =
+ aIntendedParent ? static_cast<nsIContent*>(aIntendedParent) : nullptr;
+
+ // intendedParent == nullptr is a special case where the
+ // intended parent is the document.
+ nsNodeInfoManager* nodeInfoManager =
+ intendedParent ? intendedParent->OwnerDoc()->NodeInfoManager()
+ : mBuilder->GetNodeInfoManager();
+
+ nsIContent* elem;
+ if (aNamespace == kNameSpaceID_XHTML) {
+ elem = nsHtml5TreeOperation::CreateHTMLElement(
+ aName, aAttributes, mozilla::dom::FROM_PARSER_FRAGMENT,
+ nodeInfoManager, mBuilder, aCreator.html);
+ } else if (aNamespace == kNameSpaceID_SVG) {
+ elem = nsHtml5TreeOperation::CreateSVGElement(
+ aName, aAttributes, mozilla::dom::FROM_PARSER_FRAGMENT,
+ nodeInfoManager, mBuilder, aCreator.svg);
+ } else {
+ MOZ_ASSERT(aNamespace == kNameSpaceID_MathML);
+ elem = nsHtml5TreeOperation::CreateMathMLElement(
+ aName, aAttributes, nodeInfoManager, mBuilder);
+ }
+ if (MOZ_UNLIKELY(aAttributes != tokenizer->GetAttributes() &&
+ aAttributes != nsHtml5HtmlAttributes::EMPTY_ATTRIBUTES)) {
+ delete aAttributes;
+ }
+ return elem;
+ }
+
+ nsIContentHandle* content = AllocateContentHandle();
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+
+ if (aNamespace == kNameSpaceID_XHTML) {
+ opCreateHTMLElement opeation(
+ content, aName, aAttributes, aCreator.html, aIntendedParent,
+ (!!mSpeculativeLoadStage) ? mozilla::dom::FROM_PARSER_NETWORK
+ : mozilla::dom::FROM_PARSER_DOCUMENT_WRITE);
+ treeOp->Init(mozilla::AsVariant(opeation));
+ } else if (aNamespace == kNameSpaceID_SVG) {
+ opCreateSVGElement operation(
+ content, aName, aAttributes, aCreator.svg, aIntendedParent,
+ (!!mSpeculativeLoadStage) ? mozilla::dom::FROM_PARSER_NETWORK
+ : mozilla::dom::FROM_PARSER_DOCUMENT_WRITE);
+ treeOp->Init(mozilla::AsVariant(operation));
+ } else {
+ // kNameSpaceID_MathML
+ opCreateMathMLElement operation(content, aName, aAttributes,
+ aIntendedParent);
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+
+ // mSpeculativeLoadStage is non-null only in the off-the-main-thread
+ // tree builder, which handles the network stream
+
+ // Start wall of code for speculative loading and line numbers
+
+ if (mGenerateSpeculativeLoads && mode != IN_TEMPLATE) {
+ switch (aNamespace) {
+ case kNameSpaceID_XHTML:
+ if (nsGkAtoms::img == aName) {
+ nsHtml5String loading =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_LOADING);
+ if (!loading.LowerCaseEqualsASCII("lazy")) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SRC);
+ nsHtml5String srcset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SRCSET);
+ nsHtml5String crossOrigin =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+ nsHtml5String sizes =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SIZES);
+ mSpeculativeLoadQueue.AppendElement()->InitImage(
+ url, crossOrigin, /* aMedia = */ nullptr, referrerPolicy,
+ srcset, sizes, false);
+ }
+ } else if (nsGkAtoms::source == aName) {
+ nsHtml5String srcset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SRCSET);
+ // Sources without srcset cannot be selected. The source could also be
+ // for a media element, but in that context doesn't use srcset. See
+ // comments in nsHtml5SpeculativeLoad.h about <picture> preloading
+ if (srcset) {
+ nsHtml5String sizes =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SIZES);
+ nsHtml5String type =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_TYPE);
+ nsHtml5String media =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MEDIA);
+ mSpeculativeLoadQueue.AppendElement()->InitPictureSource(
+ srcset, sizes, type, media);
+ }
+ } else if (nsGkAtoms::script == aName) {
+ nsHtml5TreeOperation* treeOp =
+ mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(
+ NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetScriptLineAndColumnNumberAndFreeze operation(
+ content, tokenizer->getLineNumber(),
+ // NOTE: tokenizer->getColumnNumber() points '>'.
+ tokenizer->getColumnNumber() + 1);
+ treeOp->Init(mozilla::AsVariant(operation));
+
+ nsHtml5String type =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_TYPE);
+ nsAutoString typeString;
+ type.ToString(typeString);
+
+ // Since `typeString` after trimming and lowercasing is only checked
+ // for "module" and " importmap", we don't need to remember
+ // pre-trimming emptiness here.
+
+ // ASCII whitespace https://infra.spec.whatwg.org/#ascii-whitespace:
+ // U+0009 TAB, U+000A LF, U+000C FF, U+000D CR, or U+0020 SPACE.
+ static const char kASCIIWhitespace[] = "\t\n\f\r ";
+ typeString.Trim(kASCIIWhitespace);
+
+ bool isModule = typeString.LowerCaseEqualsASCII("module");
+ bool importmap = typeString.LowerCaseEqualsASCII("importmap");
+ bool async = false;
+ bool defer = false;
+ bool nomodule =
+ aAttributes->contains(nsHtml5AttributeName::ATTR_NOMODULE);
+
+ // For microtask semantics, we need to queue either
+ // `opRunScriptThatMayDocumentWriteOrBlock` or
+ // `opRunScriptThatCannotDocumentWriteOrBlock` for every script
+ // element--even ones that we already know won't run.
+ // `mCurrentHtmlScriptCannotDocumentWriteOrBlock` controls which
+ // kind of operation is used for an HTML script, and this is the
+ // place where `mCurrentHtmlScriptCannotDocumentWriteOrBlock`
+ // needs to be set correctly.
+ //
+ // Non-async, non-defer classic scripts that will run MUST use
+ // `opRunScriptThatMayDocumentWriteOrBlock` in order to run
+ // the more complex code that
+ // 1. is able to resume the HTML parse after a parser-blocking
+ // scripts no longer blocks the parser
+ // 2. is able to receive more content to parse on the main thread
+ // via document.write
+ // 3. is able to throw away off-the-main-thread parsing results
+ // if what's document.written on the main thread invalidates
+ // the speculation.
+ //
+ // Async and defer classic scripts as well as module scripts and
+ // importmaps MUST use `opRunScriptThatCannotDocumentWriteOrBlock`.
+ // This is necessary particularly because the relevant main-thread
+ // code assumes it doesn't need to deal with resuming the HTML
+ // parse some time afterwards, so using a tree operation with
+ // mismatching expectations regarding that responsibility may
+ // cause the HTML parse to stall.
+ //
+ // Various scripts that won't actually run work with either type
+ // of tree op in the sense that the HTML parse won't stall.
+ // However, in the case where a script cannot block or insert
+ // data to the HTML parser via document.write, unnecessary use
+ // of `opRunScriptThatMayDocumentWriteOrBlock` instead of
+ // `opRunScriptThatCannotDocumentWriteOrBlock` causes the HTML
+ // parser to enter the speculative mode when doing so isn't
+ // actually required.
+ //
+ // Ideally, we would check for `type`/`language` attribute
+ // combinations that are known to cause non-execution as well as
+ // ScriptLoader::IsScriptEventHandler equivalent. That way, we
+ // wouldn't unnecessarily speculate after scripts that won't
+ // execute. https://bugzilla.mozilla.org/show_bug.cgi?id=1848311
+
+ if (importmap) {
+ // If we see an importmap, we don't want to later start speculative
+ // loads for modulepreloads, since such load might finish before
+ // the importmap is created. This also applies to module scripts so
+ // that any modulepreload integrity checks can be performed before
+ // the modules scripts are loaded.
+ // This state is not part of speculation rollback: If an importmap
+ // is seen speculatively and the speculation is rolled back, the
+ // importmap is still considered seen.
+ // TODO: Sync importmap seenness between the main thread and the
+ // parser thread.
+ // https://bugzilla.mozilla.org/show_bug.cgi?id=1848312
+ mHasSeenImportMap = true;
+ }
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_SRC);
+ if (url) {
+ async = aAttributes->contains(nsHtml5AttributeName::ATTR_ASYNC);
+ defer = aAttributes->contains(nsHtml5AttributeName::ATTR_DEFER);
+ if ((isModule && !mHasSeenImportMap) ||
+ (!isModule && !importmap && !nomodule)) {
+ nsHtml5String charset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CHARSET);
+ nsHtml5String crossOrigin =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String nonce =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_NONCE);
+ nsHtml5String fetchPriority = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_FETCHPRIORITY);
+ nsHtml5String integrity =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_INTEGRITY);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+ mSpeculativeLoadQueue.AppendElement()->InitScript(
+ url, charset, type, crossOrigin, /* aMedia = */ nullptr,
+ nonce, fetchPriority, integrity, referrerPolicy,
+ mode == nsHtml5TreeBuilder::IN_HEAD, async, defer, false);
+ }
+ }
+ // `mCurrentHtmlScriptCannotDocumentWriteOrBlock` MUST be computed to
+ // match the ScriptLoader-perceived kind of the script regardless of
+ // enqueuing a speculative load. Scripts with the `nomodule` attribute
+ // never block or document.write: Either the attribute prevents a
+ // classic script execution or is ignored on a module script or
+ // importmap.
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock =
+ isModule || importmap || async || defer || nomodule;
+ } else if (nsGkAtoms::link == aName) {
+ nsHtml5String rel =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_REL);
+ // Not splitting on space here is bogus but the old parser didn't even
+ // do a case-insensitive check.
+ if (rel) {
+ if (rel.LowerCaseEqualsASCII("stylesheet")) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url) {
+ nsHtml5String charset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CHARSET);
+ nsHtml5String crossOrigin = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String nonce =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_NONCE);
+ nsHtml5String integrity =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_INTEGRITY);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+ nsHtml5String media =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MEDIA);
+ nsHtml5String fetchPriority = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_FETCHPRIORITY);
+ mSpeculativeLoadQueue.AppendElement()->InitStyle(
+ url, charset, crossOrigin, media, referrerPolicy, nonce,
+ integrity, false, fetchPriority);
+ }
+ } else if (rel.LowerCaseEqualsASCII("preconnect")) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url) {
+ nsHtml5String crossOrigin = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ mSpeculativeLoadQueue.AppendElement()->InitPreconnect(
+ url, crossOrigin);
+ }
+ } else if (rel.LowerCaseEqualsASCII("preload")) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url) {
+ nsHtml5String as =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_AS);
+ nsHtml5String charset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CHARSET);
+ nsHtml5String crossOrigin = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String nonce =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_NONCE);
+ nsHtml5String integrity =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_INTEGRITY);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+ nsHtml5String media =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MEDIA);
+ nsHtml5String fetchPriority = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_FETCHPRIORITY);
+
+ // Note that respective speculative loaders for scripts and
+ // styles check all additional attributes to be equal to use the
+ // speculative load. So, if any of them is specified and the
+ // preload has to take the expected effect, those attributes
+ // must also be specified on the actual tag to use the preload.
+ // Omitting an attribute on both will make the values equal
+ // (empty) and thus use the preload.
+ if (as.LowerCaseEqualsASCII("script")) {
+ nsHtml5String type =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_TYPE);
+ mSpeculativeLoadQueue.AppendElement()->InitScript(
+ url, charset, type, crossOrigin, media, nonce,
+ /* aFetchPriority */ fetchPriority, integrity,
+ referrerPolicy, mode == nsHtml5TreeBuilder::IN_HEAD,
+ false, false, true);
+ } else if (as.LowerCaseEqualsASCII("style")) {
+ mSpeculativeLoadQueue.AppendElement()->InitStyle(
+ url, charset, crossOrigin, media, referrerPolicy, nonce,
+ integrity, true, fetchPriority);
+ } else if (as.LowerCaseEqualsASCII("image")) {
+ nsHtml5String srcset = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_IMAGESRCSET);
+ nsHtml5String sizes = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_IMAGESIZES);
+ mSpeculativeLoadQueue.AppendElement()->InitImage(
+ url, crossOrigin, media, referrerPolicy, srcset, sizes,
+ true);
+ } else if (as.LowerCaseEqualsASCII("font")) {
+ mSpeculativeLoadQueue.AppendElement()->InitFont(
+ url, crossOrigin, media, referrerPolicy, fetchPriority);
+ } else if (as.LowerCaseEqualsASCII("fetch")) {
+ mSpeculativeLoadQueue.AppendElement()->InitFetch(
+ url, crossOrigin, media, referrerPolicy, fetchPriority);
+ }
+ // Other "as" values will be supported later.
+ }
+ } else if (mozilla::StaticPrefs::network_modulepreload() &&
+ rel.LowerCaseEqualsASCII("modulepreload") &&
+ !mHasSeenImportMap) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url && url.Length() != 0) {
+ nsHtml5String as =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_AS);
+ nsAutoString asString;
+ as.ToString(asString);
+ if (mozilla::net::IsScriptLikeOrInvalid(asString)) {
+ nsHtml5String charset =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CHARSET);
+ RefPtr<nsAtom> moduleType = nsGkAtoms::_module;
+ nsHtml5String type =
+ nsHtml5String::FromAtom(moduleType.forget());
+ nsHtml5String crossOrigin = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String media =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MEDIA);
+ nsHtml5String nonce =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_NONCE);
+ nsHtml5String integrity = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_INTEGRITY);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+ nsHtml5String fetchPriority = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_FETCHPRIORITY);
+
+ mSpeculativeLoadQueue.AppendElement()->InitScript(
+ url, charset, type, crossOrigin, media, nonce,
+ /* aFetchPriority */ fetchPriority, integrity,
+ referrerPolicy, mode == nsHtml5TreeBuilder::IN_HEAD,
+ false, false, true);
+ }
+ }
+ }
+ }
+ } else if (nsGkAtoms::video == aName) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_POSTER);
+ if (url) {
+ mSpeculativeLoadQueue.AppendElement()->InitImage(
+ url, nullptr, nullptr, nullptr, nullptr, nullptr, false);
+ }
+ } else if (nsGkAtoms::style == aName) {
+ mImportScanner.Start();
+ nsHtml5TreeOperation* treeOp =
+ mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(
+ NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetStyleLineNumber operation(content, tokenizer->getLineNumber());
+ treeOp->Init(mozilla::AsVariant(operation));
+ } else if (nsGkAtoms::html == aName) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MANIFEST);
+ mSpeculativeLoadQueue.AppendElement()->InitManifest(url);
+ } else if (nsGkAtoms::base == aName) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url) {
+ mSpeculativeLoadQueue.AppendElement()->InitBase(url);
+ }
+ } else if (nsGkAtoms::meta == aName) {
+ if (nsHtml5Portability::lowerCaseLiteralEqualsIgnoreAsciiCaseString(
+ "content-security-policy",
+ aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_HTTP_EQUIV))) {
+ nsHtml5String csp =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CONTENT);
+ if (csp) {
+ mSpeculativeLoadQueue.AppendElement()->InitMetaCSP(csp);
+ }
+ } else if (nsHtml5Portability::
+ lowerCaseLiteralEqualsIgnoreAsciiCaseString(
+ "referrer",
+ aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_NAME))) {
+ nsHtml5String referrerPolicy =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CONTENT);
+ if (referrerPolicy) {
+ mSpeculativeLoadQueue.AppendElement()->InitMetaReferrerPolicy(
+ referrerPolicy);
+ }
+ }
+ }
+ break;
+ case kNameSpaceID_SVG:
+ if (nsGkAtoms::image == aName) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (!url) {
+ url = aAttributes->getValue(nsHtml5AttributeName::ATTR_XLINK_HREF);
+ }
+ if (url) {
+ mSpeculativeLoadQueue.AppendElement()->InitImage(
+ url, nullptr, nullptr, nullptr, nullptr, nullptr, false);
+ }
+ } else if (nsGkAtoms::script == aName) {
+ nsHtml5TreeOperation* treeOp =
+ mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(
+ NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetScriptLineAndColumnNumberAndFreeze operation(
+ content, tokenizer->getLineNumber(),
+ // NOTE: tokenizer->getColumnNumber() points '>'.
+ tokenizer->getColumnNumber() + 1);
+ treeOp->Init(mozilla::AsVariant(operation));
+
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (!url) {
+ url = aAttributes->getValue(nsHtml5AttributeName::ATTR_XLINK_HREF);
+ }
+ if (url) {
+ nsHtml5String type =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_TYPE);
+ nsHtml5String crossOrigin =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_CROSSORIGIN);
+ nsHtml5String nonce =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_NONCE);
+ nsHtml5String integrity =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_INTEGRITY);
+ nsHtml5String referrerPolicy = aAttributes->getValue(
+ nsHtml5AttributeName::ATTR_REFERRERPOLICY);
+
+ // Bug 1847712: SVG's `<script>` element doesn't support
+ // `fetchpriority` yet.
+ // Use the empty string and rely on the
+ // "invalid value default" state being used later.
+ // Compared to using a non-empty string, this doesn't
+ // require calling `Release()` for the string.
+ nsHtml5String fetchPriority = nsHtml5String::EmptyString();
+
+ mSpeculativeLoadQueue.AppendElement()->InitScript(
+ url, nullptr, type, crossOrigin, /* aMedia = */ nullptr, nonce,
+ fetchPriority, integrity, referrerPolicy,
+ mode == nsHtml5TreeBuilder::IN_HEAD, false, false, false);
+ }
+ } else if (nsGkAtoms::style == aName) {
+ mImportScanner.Start();
+ nsHtml5TreeOperation* treeOp =
+ mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(
+ NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetStyleLineNumber operation(content, tokenizer->getLineNumber());
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+ break;
+ }
+ } else if (aNamespace != kNameSpaceID_MathML) {
+ // No speculative loader--just line numbers and defer/async check
+ if (nsGkAtoms::style == aName) {
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetStyleLineNumber operation(content, tokenizer->getLineNumber());
+ treeOp->Init(mozilla::AsVariant(operation));
+ } else if (nsGkAtoms::script == aName) {
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetScriptLineAndColumnNumberAndFreeze operation(
+ content, tokenizer->getLineNumber(),
+ // NOTE: tokenizer->getColumnNumber() points '>'.
+ tokenizer->getColumnNumber() + 1);
+ treeOp->Init(mozilla::AsVariant(operation));
+ if (aNamespace == kNameSpaceID_XHTML) {
+ // Although we come here in cases where the value of
+ // `mCurrentHtmlScriptCannotDocumentWriteOrBlock` doesn't actually
+ // matter, we also come here when parsing document.written content on
+ // the main thread. In that case, IT MATTERS that
+ // `mCurrentHtmlScriptCannotDocumentWriteOrBlock` is set correctly,
+ // so let's just always set it correctly even if it a bit of wasted work
+ // in the scenarios where no scripts execute and it doesn't matter.
+ //
+ // See the comments around generating speculative loads for HTML scripts
+ // elements for the details of when
+ // `mCurrentHtmlScriptCannotDocumentWriteOrBlock` needs to be set to
+ // `true` and when to `false`.
+
+ nsHtml5String type =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_TYPE);
+ nsAutoString typeString;
+ type.ToString(typeString);
+
+ // Since `typeString` after trimming and lowercasing is only checked
+ // for "module" and " importmap", we don't need to remember
+ // pre-trimming emptiness here.
+
+ // ASCII whitespace https://infra.spec.whatwg.org/#ascii-whitespace:
+ // U+0009 TAB, U+000A LF, U+000C FF, U+000D CR, or U+0020 SPACE.
+ static const char kASCIIWhitespace[] = "\t\n\f\r ";
+ typeString.Trim(kASCIIWhitespace);
+
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock =
+ typeString.LowerCaseEqualsASCII("module") ||
+ typeString.LowerCaseEqualsASCII("nomodule") ||
+ typeString.LowerCaseEqualsASCII("importmap");
+
+ if (!mCurrentHtmlScriptCannotDocumentWriteOrBlock &&
+ aAttributes->contains(nsHtml5AttributeName::ATTR_SRC)) {
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock =
+ (aAttributes->contains(nsHtml5AttributeName::ATTR_ASYNC) ||
+ aAttributes->contains(nsHtml5AttributeName::ATTR_DEFER));
+ }
+ }
+ } else if (aNamespace == kNameSpaceID_XHTML) {
+ if (nsGkAtoms::html == aName) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_MANIFEST);
+ nsHtml5TreeOperation* treeOp =
+ mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(
+ NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ if (url) {
+ nsString
+ urlString; // Not Auto, because using it to hold nsStringBuffer*
+ url.ToString(urlString);
+ opProcessOfflineManifest operation(ToNewUnicode(urlString));
+ treeOp->Init(mozilla::AsVariant(operation));
+ } else {
+ opProcessOfflineManifest operation(ToNewUnicode(u""_ns));
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+ } else if (nsGkAtoms::base == aName && mViewSource) {
+ nsHtml5String url =
+ aAttributes->getValue(nsHtml5AttributeName::ATTR_HREF);
+ if (url) {
+ mViewSource->AddBase(url);
+ }
+ }
+ }
+ }
+
+ // End wall of code for speculative loading
+
+ return content;
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::createElement(
+ int32_t aNamespace, nsAtom* aName, nsHtml5HtmlAttributes* aAttributes,
+ nsIContentHandle* aFormElement, nsIContentHandle* aIntendedParent,
+ nsHtml5ContentCreatorFunction aCreator) {
+ nsIContentHandle* content =
+ createElement(aNamespace, aName, aAttributes, aIntendedParent, aCreator);
+ if (aFormElement) {
+ if (mBuilder) {
+ nsHtml5TreeOperation::SetFormElement(
+ static_cast<nsIContent*>(content),
+ static_cast<nsIContent*>(aFormElement));
+ } else {
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opSetFormElement operation(content, aFormElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+ }
+ return content;
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::createHtmlElementSetAsRoot(
+ nsHtml5HtmlAttributes* aAttributes) {
+ nsHtml5ContentCreatorFunction creator;
+ // <html> uses NS_NewHTMLSharedElement creator
+ creator.html = NS_NewHTMLSharedElement;
+ nsIContentHandle* content = createElement(kNameSpaceID_XHTML, nsGkAtoms::html,
+ aAttributes, nullptr, creator);
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendToDocument(
+ static_cast<nsIContent*>(content), mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ } else {
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ opAppendToDocument operation(content);
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+ return content;
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::createAndInsertFosterParentedElement(
+ int32_t aNamespace, nsAtom* aName, nsHtml5HtmlAttributes* aAttributes,
+ nsIContentHandle* aFormElement, nsIContentHandle* aTable,
+ nsIContentHandle* aStackParent, nsHtml5ContentCreatorFunction aCreator) {
+ MOZ_ASSERT(aTable, "Null table");
+ MOZ_ASSERT(aStackParent, "Null stack parent");
+
+ if (mBuilder) {
+ // Get the foster parent to use as the intended parent when creating
+ // the child element.
+ nsIContent* fosterParent = nsHtml5TreeOperation::GetFosterParent(
+ static_cast<nsIContent*>(aTable),
+ static_cast<nsIContent*>(aStackParent));
+
+ nsIContentHandle* child = createElement(
+ aNamespace, aName, aAttributes, aFormElement, fosterParent, aCreator);
+
+ insertFosterParentedChild(child, aTable, aStackParent);
+
+ return child;
+ }
+
+ // Tree op to get the foster parent that we use as the intended parent
+ // when creating the child element.
+ nsHtml5TreeOperation* fosterParentTreeOp = mOpQueue.AppendElement();
+ NS_ASSERTION(fosterParentTreeOp, "Tree op allocation failed.");
+ nsIContentHandle* fosterParentHandle = AllocateContentHandle();
+ opGetFosterParent operation(aTable, aStackParent, fosterParentHandle);
+ fosterParentTreeOp->Init(mozilla::AsVariant(operation));
+
+ // Create the element with the correct intended parent.
+ nsIContentHandle* child =
+ createElement(aNamespace, aName, aAttributes, aFormElement,
+ fosterParentHandle, aCreator);
+
+ // Insert the child into the foster parent.
+ insertFosterParentedChild(child, aTable, aStackParent);
+
+ return child;
+}
+
+void nsHtml5TreeBuilder::detachFromParent(nsIContentHandle* aElement) {
+ MOZ_ASSERT(aElement, "Null element");
+
+ if (mBuilder) {
+ nsHtml5TreeOperation::Detach(static_cast<nsIContent*>(aElement), mBuilder);
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opDetach operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::appendElement(nsIContentHandle* aChild,
+ nsIContentHandle* aParent) {
+ MOZ_ASSERT(aChild, "Null child");
+ MOZ_ASSERT(aParent, "Null parent");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::Append(
+ static_cast<nsIContent*>(aChild), static_cast<nsIContent*>(aParent),
+ mozilla::dom::FROM_PARSER_FRAGMENT, mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+
+ opAppend operation(aChild, aParent,
+ (!!mSpeculativeLoadStage)
+ ? mozilla::dom::FROM_PARSER_NETWORK
+ : mozilla::dom::FROM_PARSER_DOCUMENT_WRITE);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::appendChildrenToNewParent(
+ nsIContentHandle* aOldParent, nsIContentHandle* aNewParent) {
+ MOZ_ASSERT(aOldParent, "Null old parent");
+ MOZ_ASSERT(aNewParent, "Null new parent");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendChildrenToNewParent(
+ static_cast<nsIContent*>(aOldParent),
+ static_cast<nsIContent*>(aNewParent), mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAppendChildrenToNewParent operation(aOldParent, aNewParent);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::insertFosterParentedCharacters(
+ char16_t* aBuffer, int32_t aStart, int32_t aLength,
+ nsIContentHandle* aTable, nsIContentHandle* aStackParent) {
+ MOZ_ASSERT(aBuffer, "Null buffer");
+ MOZ_ASSERT(aTable, "Null table");
+ MOZ_ASSERT(aStackParent, "Null stack parent");
+ MOZ_ASSERT(!aStart, "aStart must always be zero.");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::FosterParentText(
+ static_cast<nsIContent*>(aStackParent),
+ aBuffer, // XXX aStart always ignored???
+ aLength, static_cast<nsIContent*>(aTable), mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ auto bufferCopy = mozilla::MakeUniqueFallible<char16_t[]>(aLength);
+ if (!bufferCopy) {
+ // Just assigning mBroken instead of generating tree op. The caller
+ // of tokenizeBuffer() will call MarkAsBroken() as appropriate.
+ mBroken = NS_ERROR_OUT_OF_MEMORY;
+ requestSuspension();
+ return;
+ }
+
+ memcpy(bufferCopy.get(), aBuffer, aLength * sizeof(char16_t));
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opFosterParentText operation(aStackParent, bufferCopy.release(), aTable,
+ aLength);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::insertFosterParentedChild(
+ nsIContentHandle* aChild, nsIContentHandle* aTable,
+ nsIContentHandle* aStackParent) {
+ MOZ_ASSERT(aChild, "Null child");
+ MOZ_ASSERT(aTable, "Null table");
+ MOZ_ASSERT(aStackParent, "Null stack parent");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::FosterParent(
+ static_cast<nsIContent*>(aChild),
+ static_cast<nsIContent*>(aStackParent),
+ static_cast<nsIContent*>(aTable), mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opFosterParent operation(aChild, aStackParent, aTable);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::appendCharacters(nsIContentHandle* aParent,
+ char16_t* aBuffer, int32_t aStart,
+ int32_t aLength) {
+ MOZ_ASSERT(aBuffer, "Null buffer");
+ MOZ_ASSERT(aParent, "Null parent");
+ MOZ_ASSERT(!aStart, "aStart must always be zero.");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendText(
+ aBuffer, // XXX aStart always ignored???
+ aLength, static_cast<nsIContent*>(aParent), mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ auto bufferCopy = mozilla::MakeUniqueFallible<char16_t[]>(aLength);
+ if (!bufferCopy) {
+ // Just assigning mBroken instead of generating tree op. The caller
+ // of tokenizeBuffer() will call MarkAsBroken() as appropriate.
+ mBroken = NS_ERROR_OUT_OF_MEMORY;
+ requestSuspension();
+ return;
+ }
+
+ memcpy(bufferCopy.get(), aBuffer, aLength * sizeof(char16_t));
+
+ if (mImportScanner.ShouldScan()) {
+ nsTArray<nsString> imports =
+ mImportScanner.Scan(mozilla::Span(aBuffer, aLength));
+ for (nsString& url : imports) {
+ mSpeculativeLoadQueue.AppendElement()->InitImportStyle(std::move(url));
+ }
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAppendText operation(aParent, bufferCopy.release(), aLength);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::appendComment(nsIContentHandle* aParent,
+ char16_t* aBuffer, int32_t aStart,
+ int32_t aLength) {
+ MOZ_ASSERT(aBuffer, "Null buffer");
+ MOZ_ASSERT(aParent, "Null parent");
+ MOZ_ASSERT(!aStart, "aStart must always be zero.");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendComment(
+ static_cast<nsIContent*>(aParent),
+ aBuffer, // XXX aStart always ignored???
+ aLength, mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ auto bufferCopy = mozilla::MakeUniqueFallible<char16_t[]>(aLength);
+ if (!bufferCopy) {
+ // Just assigning mBroken instead of generating tree op. The caller
+ // of tokenizeBuffer() will call MarkAsBroken() as appropriate.
+ mBroken = NS_ERROR_OUT_OF_MEMORY;
+ requestSuspension();
+ return;
+ }
+
+ memcpy(bufferCopy.get(), aBuffer, aLength * sizeof(char16_t));
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAppendComment operation(aParent, bufferCopy.release(), aLength);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::appendCommentToDocument(char16_t* aBuffer,
+ int32_t aStart,
+ int32_t aLength) {
+ MOZ_ASSERT(aBuffer, "Null buffer");
+ MOZ_ASSERT(!aStart, "aStart must always be zero.");
+
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendCommentToDocument(
+ aBuffer, // XXX aStart always ignored???
+ aLength, mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ auto bufferCopy = mozilla::MakeUniqueFallible<char16_t[]>(aLength);
+ if (!bufferCopy) {
+ // Just assigning mBroken instead of generating tree op. The caller
+ // of tokenizeBuffer() will call MarkAsBroken() as appropriate.
+ mBroken = NS_ERROR_OUT_OF_MEMORY;
+ requestSuspension();
+ return;
+ }
+
+ memcpy(bufferCopy.get(), aBuffer, aLength * sizeof(char16_t));
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAppendCommentToDocument data(bufferCopy.release(), aLength);
+ treeOp->Init(mozilla::AsVariant(data));
+}
+
+void nsHtml5TreeBuilder::addAttributesToElement(
+ nsIContentHandle* aElement, nsHtml5HtmlAttributes* aAttributes) {
+ MOZ_ASSERT(aElement, "Null element");
+ MOZ_ASSERT(aAttributes, "Null attributes");
+
+ if (aAttributes == nsHtml5HtmlAttributes::EMPTY_ATTRIBUTES) {
+ return;
+ }
+
+ if (mBuilder) {
+ MOZ_ASSERT(
+ aAttributes == tokenizer->GetAttributes(),
+ "Using attribute other than the tokenizer's to add to body or html.");
+ nsresult rv = nsHtml5TreeOperation::AddAttributes(
+ static_cast<nsIContent*>(aElement), aAttributes, mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAddAttributes opeation(aElement, aAttributes);
+ treeOp->Init(mozilla::AsVariant(opeation));
+}
+
+void nsHtml5TreeBuilder::markMalformedIfScript(nsIContentHandle* aElement) {
+ MOZ_ASSERT(aElement, "Null element");
+
+ if (mBuilder) {
+ nsHtml5TreeOperation::MarkMalformedIfScript(
+ static_cast<nsIContent*>(aElement));
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opMarkMalformedIfScript operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::start(bool fragment) {
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock = false;
+#ifdef DEBUG
+ mActive = true;
+#endif
+}
+
+void nsHtml5TreeBuilder::end() {
+ mOpQueue.Clear();
+#ifdef DEBUG
+ mActive = false;
+#endif
+}
+
+void nsHtml5TreeBuilder::appendDoctypeToDocument(nsAtom* aName,
+ nsHtml5String aPublicId,
+ nsHtml5String aSystemId) {
+ MOZ_ASSERT(aName, "Null name");
+ nsString publicId; // Not Auto, because using it to hold nsStringBuffer*
+ nsString systemId; // Not Auto, because using it to hold nsStringBuffer*
+ aPublicId.ToString(publicId);
+ aSystemId.ToString(systemId);
+ if (mBuilder) {
+ nsresult rv = nsHtml5TreeOperation::AppendDoctypeToDocument(
+ aName, publicId, systemId, mBuilder);
+ if (NS_FAILED(rv)) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(rv);
+ }
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opAppendDoctypeToDocument operation(aName, publicId, systemId);
+ treeOp->Init(mozilla::AsVariant(operation));
+ // nsXMLContentSink can flush here, but what's the point?
+ // It can also interrupt here, but we can't.
+}
+
+void nsHtml5TreeBuilder::elementPushed(int32_t aNamespace, nsAtom* aName,
+ nsIContentHandle* aElement) {
+ NS_ASSERTION(aNamespace == kNameSpaceID_XHTML ||
+ aNamespace == kNameSpaceID_SVG ||
+ aNamespace == kNameSpaceID_MathML,
+ "Element isn't HTML, SVG or MathML!");
+ NS_ASSERTION(aName, "Element doesn't have local name!");
+ NS_ASSERTION(aElement, "No element!");
+ /*
+ * The frame constructor uses recursive algorithms, so it can't deal with
+ * arbitrarily deep trees. This is especially a problem on Windows where
+ * the permitted depth of the runtime stack is rather small.
+ *
+ * The following is a protection against author incompetence--not against
+ * malice. There are other ways to make the DOM deep anyway.
+ *
+ * The basic idea is that when the tree builder stack gets too deep,
+ * append operations no longer append to the node that the HTML parsing
+ * algorithm says they should but instead text nodes are append to the last
+ * element that was seen before a magic tree builder stack threshold was
+ * reached and element and comment nodes aren't appended to the DOM at all.
+ *
+ * However, for security reasons, non-child descendant text nodes inside an
+ * SVG script or style element should not become children. Also, non-cell
+ * table elements shouldn't be used as surrogate parents for user experience
+ * reasons.
+ */
+ if (aNamespace != kNameSpaceID_XHTML) {
+ return;
+ }
+ if (aName == nsGkAtoms::body || aName == nsGkAtoms::frameset) {
+ if (mBuilder) {
+ // InnerHTML and DOMParser shouldn't start layout anyway
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ treeOp->Init(mozilla::AsVariant(opStartLayout()));
+ return;
+ }
+ if (nsIContent::RequiresDoneCreatingElement(kNameSpaceID_XHTML, aName)) {
+ if (mBuilder) {
+ nsHtml5TreeOperation::DoneCreatingElement(
+ static_cast<nsIContent*>(aElement));
+ } else {
+ opDoneCreatingElement operation(aElement);
+ mOpQueue.AppendElement()->Init(mozilla::AsVariant(operation));
+ }
+ return;
+ }
+ if (mGenerateSpeculativeLoads && aName == nsGkAtoms::picture) {
+ // See comments in nsHtml5SpeculativeLoad.h about <picture> preloading
+ mSpeculativeLoadQueue.AppendElement()->InitOpenPicture();
+ return;
+ }
+ if (aName == nsGkAtoms::_template) {
+ if (tokenizer->TemplatePushedOrHeadPopped()) {
+ requestSuspension();
+ }
+ }
+}
+
+void nsHtml5TreeBuilder::elementPopped(int32_t aNamespace, nsAtom* aName,
+ nsIContentHandle* aElement) {
+ NS_ASSERTION(aNamespace == kNameSpaceID_XHTML ||
+ aNamespace == kNameSpaceID_SVG ||
+ aNamespace == kNameSpaceID_MathML,
+ "Element isn't HTML, SVG or MathML!");
+ NS_ASSERTION(aName, "Element doesn't have local name!");
+ NS_ASSERTION(aElement, "No element!");
+ if (aNamespace == kNameSpaceID_MathML) {
+ return;
+ }
+ // we now have only SVG and HTML
+ if (aName == nsGkAtoms::script) {
+ if (mPreventScriptExecution) {
+ if (mBuilder) {
+ nsHtml5TreeOperation::PreventScriptExecution(
+ static_cast<nsIContent*>(aElement));
+ return;
+ }
+ opPreventScriptExecution operation(aElement);
+ mOpQueue.AppendElement()->Init(mozilla::AsVariant(operation));
+ return;
+ }
+ if (mBuilder) {
+ return;
+ }
+ if (mCurrentHtmlScriptCannotDocumentWriteOrBlock) {
+ NS_ASSERTION(aNamespace == kNameSpaceID_XHTML,
+ "Only HTML scripts may be async/defer.");
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opRunScriptThatCannotDocumentWriteOrBlock operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+ mCurrentHtmlScriptCannotDocumentWriteOrBlock = false;
+ return;
+ }
+ requestSuspension();
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opRunScriptThatMayDocumentWriteOrBlock operation(aElement, nullptr);
+ treeOp->Init(mozilla::AsVariant(operation));
+ return;
+ }
+ // Some nodes need DoneAddingChildren() called to initialize
+ // properly (e.g. form state restoration).
+ if (nsIContent::RequiresDoneAddingChildren(aNamespace, aName)) {
+ if (mBuilder) {
+ nsHtml5TreeOperation::DoneAddingChildren(
+ static_cast<nsIContent*>(aElement));
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opDoneAddingChildren operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+ if (aNamespace == kNameSpaceID_XHTML && aName == nsGkAtoms::head) {
+ if (tokenizer->TemplatePushedOrHeadPopped()) {
+ requestSuspension();
+ }
+ }
+ return;
+ }
+ if (aName == nsGkAtoms::style ||
+ (aNamespace == kNameSpaceID_XHTML && aName == nsGkAtoms::link)) {
+ if (mBuilder) {
+ MOZ_ASSERT(!nsContentUtils::IsSafeToRunScript(),
+ "Scripts must be blocked.");
+ mBuilder->UpdateStyleSheet(static_cast<nsIContent*>(aElement));
+ return;
+ }
+
+ if (aName == nsGkAtoms::style) {
+ nsTArray<nsString> imports = mImportScanner.Stop();
+ for (nsString& url : imports) {
+ mSpeculativeLoadQueue.AppendElement()->InitImportStyle(std::move(url));
+ }
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opUpdateStyleSheet operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+ return;
+ }
+ if (aNamespace == kNameSpaceID_SVG) {
+ if (aName == nsGkAtoms::svg) {
+ if (!scriptingEnabled || mPreventScriptExecution) {
+ return;
+ }
+ if (mBuilder) {
+ nsHtml5TreeOperation::SvgLoad(static_cast<nsIContent*>(aElement));
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opSvgLoad operation(aElement);
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+ return;
+ }
+
+ if (mGenerateSpeculativeLoads && aName == nsGkAtoms::picture) {
+ // See comments in nsHtml5SpeculativeLoad.h about <picture> preloading
+ mSpeculativeLoadQueue.AppendElement()->InitEndPicture();
+ return;
+ }
+}
+
+void nsHtml5TreeBuilder::accumulateCharacters(const char16_t* aBuf,
+ int32_t aStart, int32_t aLength) {
+ MOZ_RELEASE_ASSERT(charBufferLen + aLength <= charBuffer.length,
+ "About to memcpy past the end of the buffer!");
+ memcpy(charBuffer + charBufferLen, aBuf + aStart, sizeof(char16_t) * aLength);
+ charBufferLen += aLength;
+}
+
+// INT32_MAX is (2^31)-1. Therefore, the highest power-of-two that fits
+// is 2^30. Note that this is counting char16_t units. The underlying
+// bytes will be twice that, but they fit even in 32-bit size_t even
+// if a contiguous chunk of memory of that size is pretty unlikely to
+// be available on a 32-bit system.
+#define MAX_POWER_OF_TWO_IN_INT32 0x40000000
+
+bool nsHtml5TreeBuilder::EnsureBufferSpace(int32_t aLength) {
+ // TODO: Unify nsHtml5Tokenizer::strBuf and nsHtml5TreeBuilder::charBuffer
+ // so that this method becomes unnecessary.
+ mozilla::CheckedInt<int32_t> worstCase(charBufferLen);
+ worstCase += aLength;
+ if (!worstCase.isValid()) {
+ return false;
+ }
+ if (worstCase.value() > MAX_POWER_OF_TWO_IN_INT32) {
+ return false;
+ }
+ if (!charBuffer) {
+ if (worstCase.value() < MAX_POWER_OF_TWO_IN_INT32) {
+ // Add one to round to the next power of two to avoid immediate
+ // reallocation once there are a few characters in the buffer.
+ worstCase += 1;
+ }
+ charBuffer = jArray<char16_t, int32_t>::newFallibleJArray(
+ mozilla::RoundUpPow2(worstCase.value()));
+ if (!charBuffer) {
+ return false;
+ }
+ } else if (worstCase.value() > charBuffer.length) {
+ jArray<char16_t, int32_t> newBuf =
+ jArray<char16_t, int32_t>::newFallibleJArray(
+ mozilla::RoundUpPow2(worstCase.value()));
+ if (!newBuf) {
+ return false;
+ }
+ memcpy(newBuf, charBuffer, sizeof(char16_t) * size_t(charBufferLen));
+ charBuffer = newBuf;
+ }
+ return true;
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::AllocateContentHandle() {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never allocate a handle with builder.");
+ return nullptr;
+ }
+ if (mHandlesUsed == NS_HTML5_TREE_BUILDER_HANDLE_ARRAY_LENGTH) {
+ mOldHandles.AppendElement(std::move(mHandles));
+ mHandles = mozilla::MakeUnique<nsIContent*[]>(
+ NS_HTML5_TREE_BUILDER_HANDLE_ARRAY_LENGTH);
+ mHandlesUsed = 0;
+ }
+#ifdef DEBUG
+ mHandles[mHandlesUsed] = reinterpret_cast<nsIContent*>(uintptr_t(0xC0DEDBAD));
+#endif
+ return &mHandles[mHandlesUsed++];
+}
+
+bool nsHtml5TreeBuilder::HasScriptThatMayDocumentWriteOrBlock() {
+ uint32_t len = mOpQueue.Length();
+ if (!len) {
+ return false;
+ }
+ return mOpQueue.ElementAt(len - 1).IsRunScriptThatMayDocumentWriteOrBlock();
+}
+
+mozilla::Result<bool, nsresult> nsHtml5TreeBuilder::Flush(bool aDiscretionary) {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never flush with builder.");
+ return false;
+ }
+ if (NS_SUCCEEDED(mBroken)) {
+ if (!aDiscretionary || !(charBufferLen && currentPtr >= 0 &&
+ stack[currentPtr]->isFosterParenting())) {
+ // Don't flush text on discretionary flushes if the current element on
+ // the stack is a foster-parenting element and there's pending text,
+ // because flushing in that case would make the tree shape dependent on
+ // where the flush points fall.
+ flushCharacters();
+ }
+ FlushLoads();
+ }
+ if (mOpSink) {
+ bool hasOps = !mOpQueue.IsEmpty();
+ if (hasOps) {
+ // If the builder is broken and mOpQueue is not empty, there must be
+ // one op and it must be eTreeOpMarkAsBroken.
+ if (NS_FAILED(mBroken)) {
+ MOZ_ASSERT(mOpQueue.Length() == 1,
+ "Tree builder is broken with a non-empty op queue whose "
+ "length isn't 1.");
+ MOZ_ASSERT(mOpQueue[0].IsMarkAsBroken(),
+ "Tree builder is broken but the op in queue is not marked "
+ "as broken.");
+ }
+ if (!mOpSink->MoveOpsFrom(mOpQueue)) {
+ return mozilla::Err(NS_ERROR_OUT_OF_MEMORY);
+ }
+ }
+ return hasOps;
+ }
+ // no op sink: throw away ops
+ mOpQueue.Clear();
+ return false;
+}
+
+void nsHtml5TreeBuilder::FlushLoads() {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never flush loads with builder.");
+ return;
+ }
+ if (!mSpeculativeLoadQueue.IsEmpty()) {
+ mSpeculativeLoadStage->MoveSpeculativeLoadsFrom(mSpeculativeLoadQueue);
+ }
+}
+
+void nsHtml5TreeBuilder::SetDocumentCharset(NotNull<const Encoding*> aEncoding,
+ nsCharsetSource aCharsetSource,
+ bool aCommitEncodingSpeculation) {
+ MOZ_ASSERT(!mBuilder, "How did we call this with builder?");
+ MOZ_ASSERT(mSpeculativeLoadStage,
+ "How did we call this without a speculative load stage?");
+ mSpeculativeLoadQueue.AppendElement()->InitSetDocumentCharset(
+ aEncoding, aCharsetSource, aCommitEncodingSpeculation);
+}
+
+void nsHtml5TreeBuilder::UpdateCharsetSource(nsCharsetSource aCharsetSource) {
+ MOZ_ASSERT(!mBuilder, "How did we call this with builder?");
+ MOZ_ASSERT(mSpeculativeLoadStage,
+ "How did we call this without a speculative load stage (even "
+ "though we don't need it right here)?");
+ if (mViewSource) {
+ mViewSource->UpdateCharsetSource(aCharsetSource);
+ return;
+ }
+ opUpdateCharsetSource operation(aCharsetSource);
+ mOpQueue.AppendElement()->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::StreamEnded() {
+ MOZ_ASSERT(!mBuilder, "Must not call StreamEnded with builder.");
+ MOZ_ASSERT(!fragment, "Must not parse fragments off the main thread.");
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ treeOp->Init(mozilla::AsVariant(opStreamEnded()));
+}
+
+void nsHtml5TreeBuilder::NeedsCharsetSwitchTo(
+ NotNull<const Encoding*> aEncoding, int32_t aCharsetSource,
+ int32_t aLineNumber) {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never switch charset with builder.");
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opCharsetSwitchTo opeation(aEncoding, aCharsetSource, aLineNumber);
+ treeOp->Init(mozilla::AsVariant(opeation));
+}
+
+void nsHtml5TreeBuilder::MaybeComplainAboutCharset(const char* aMsgId,
+ bool aError,
+ int32_t aLineNumber) {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never complain about charset with builder.");
+ return;
+ }
+
+ if (mSpeculativeLoadStage) {
+ mSpeculativeLoadQueue.AppendElement()->InitMaybeComplainAboutCharset(
+ aMsgId, aError, aLineNumber);
+ } else {
+ opMaybeComplainAboutCharset opeartion(const_cast<char*>(aMsgId), aError,
+ aLineNumber);
+ mOpQueue.AppendElement()->Init(mozilla::AsVariant(opeartion));
+ }
+}
+
+void nsHtml5TreeBuilder::TryToEnableEncodingMenu() {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never disable encoding menu with builder.");
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement();
+ NS_ASSERTION(treeOp, "Tree op allocation failed.");
+ treeOp->Init(mozilla::AsVariant(opEnableEncodingMenu()));
+}
+
+void nsHtml5TreeBuilder::AddSnapshotToScript(
+ nsAHtml5TreeBuilderState* aSnapshot, int32_t aLine) {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must never use snapshots with builder.");
+ return;
+ }
+ MOZ_ASSERT(HasScriptThatMayDocumentWriteOrBlock(),
+ "No script to add a snapshot to!");
+ MOZ_ASSERT(aSnapshot, "Got null snapshot.");
+ mOpQueue.ElementAt(mOpQueue.Length() - 1).SetSnapshot(aSnapshot, aLine);
+}
+
+void nsHtml5TreeBuilder::DropHandles() {
+ MOZ_ASSERT(!mBuilder, "Must not drop handles with builder.");
+ mOldHandles.Clear();
+ mHandlesUsed = 0;
+}
+
+void nsHtml5TreeBuilder::MarkAsBroken(nsresult aRv) {
+ if (MOZ_UNLIKELY(mBuilder)) {
+ MOZ_ASSERT_UNREACHABLE("Must not call this with builder.");
+ return;
+ }
+ mBroken = aRv;
+ mOpQueue.Clear(); // Previous ops don't matter anymore
+ opMarkAsBroken operation(aRv);
+ mOpQueue.AppendElement()->Init(mozilla::AsVariant(operation));
+}
+
+void nsHtml5TreeBuilder::MarkAsBrokenFromPortability(nsresult aRv) {
+ if (mBuilder) {
+ MarkAsBrokenAndRequestSuspensionWithBuilder(aRv);
+ return;
+ }
+ mBroken = aRv;
+ requestSuspension();
+}
+
+void nsHtml5TreeBuilder::StartPlainTextViewSource(const nsAutoString& aTitle) {
+ MOZ_ASSERT(!mBuilder, "Must not view source with builder.");
+
+ startTag(nsHtml5ElementName::ELT_META,
+ nsHtml5ViewSourceUtils::NewMetaViewportAttributes(), false);
+
+ startTag(nsHtml5ElementName::ELT_TITLE,
+ nsHtml5HtmlAttributes::EMPTY_ATTRIBUTES, false);
+
+ // XUL will add the "Source of: " prefix.
+ uint32_t length = aTitle.Length();
+ if (length > INT32_MAX) {
+ length = INT32_MAX;
+ }
+ characters(aTitle.get(), 0, (int32_t)length);
+ endTag(nsHtml5ElementName::ELT_TITLE);
+
+ startTag(nsHtml5ElementName::ELT_LINK,
+ nsHtml5ViewSourceUtils::NewLinkAttributes(), false);
+
+ startTag(nsHtml5ElementName::ELT_BODY,
+ nsHtml5ViewSourceUtils::NewBodyAttributes(), false);
+
+ StartPlainTextBody();
+}
+
+void nsHtml5TreeBuilder::StartPlainText() {
+ MOZ_ASSERT(!mBuilder, "Must not view source with builder.");
+ setForceNoQuirks(true);
+ startTag(nsHtml5ElementName::ELT_LINK,
+ nsHtml5PlainTextUtils::NewLinkAttributes(), false);
+
+ startTag(nsHtml5ElementName::ELT_BODY,
+ nsHtml5PlainTextUtils::NewBodyAttributes(), false);
+
+ StartPlainTextBody();
+}
+
+void nsHtml5TreeBuilder::StartPlainTextBody() {
+ MOZ_ASSERT(!mBuilder, "Must not view source with builder.");
+ startTag(nsHtml5ElementName::ELT_PRE, nsHtml5HtmlAttributes::EMPTY_ATTRIBUTES,
+ false);
+ needToDropLF = false;
+}
+
+// DocumentModeHandler
+void nsHtml5TreeBuilder::documentMode(nsHtml5DocumentMode m) {
+ if (mBuilder) {
+ mBuilder->SetDocumentMode(m);
+ return;
+ }
+ if (mSpeculativeLoadStage) {
+ mSpeculativeLoadQueue.AppendElement()->InitSetDocumentMode(m);
+ return;
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ treeOp->Init(mozilla::AsVariant(m));
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::getDocumentFragmentForTemplate(
+ nsIContentHandle* aTemplate) {
+ if (mBuilder) {
+ return nsHtml5TreeOperation::GetDocumentFragmentForTemplate(
+ static_cast<nsIContent*>(aTemplate));
+ }
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ nsIContentHandle* fragHandle = AllocateContentHandle();
+ opGetDocumentFragmentForTemplate operation(aTemplate, fragHandle);
+ treeOp->Init(mozilla::AsVariant(operation));
+ return fragHandle;
+}
+
+void nsHtml5TreeBuilder::setDocumentFragmentForTemplate(
+ nsIContentHandle* aTemplate, nsIContentHandle* aFragment) {
+ if (mBuilder) {
+ nsHtml5TreeOperation::SetDocumentFragmentForTemplate(
+ static_cast<nsIContent*>(aTemplate),
+ static_cast<nsIContent*>(aFragment));
+ return;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return;
+ }
+ opSetDocumentFragmentForTemplate operation(aTemplate, aFragment);
+ treeOp->Init(mozilla::AsVariant(operation));
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::getShadowRootFromHost(
+ nsIContentHandle* aHost, nsIContentHandle* aTemplateNode,
+ nsHtml5String aShadowRootMode, bool aShadowRootDelegatesFocus) {
+ mozilla::dom::ShadowRootMode mode;
+ if (aShadowRootMode.LowerCaseEqualsASCII("open")) {
+ mode = mozilla::dom::ShadowRootMode::Open;
+ } else if (aShadowRootMode.LowerCaseEqualsASCII("closed")) {
+ mode = mozilla::dom::ShadowRootMode::Closed;
+ } else {
+ return nullptr;
+ }
+
+ if (mBuilder) {
+ nsIContent* root = nsContentUtils::AttachDeclarativeShadowRoot(
+ static_cast<nsIContent*>(aHost), mode, aShadowRootDelegatesFocus);
+ if (!root) {
+ nsContentUtils::LogSimpleConsoleError(
+ u"Failed to attach Declarative Shadow DOM."_ns, "DOM"_ns,
+ mBuilder->GetDocument()->IsInPrivateBrowsing(),
+ mBuilder->GetDocument()->IsInChromeDocShell());
+ }
+ return root;
+ }
+
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement(mozilla::fallible);
+ if (MOZ_UNLIKELY(!treeOp)) {
+ MarkAsBrokenAndRequestSuspensionWithoutBuilder(NS_ERROR_OUT_OF_MEMORY);
+ return nullptr;
+ }
+ nsIContentHandle* fragHandle = AllocateContentHandle();
+ opGetShadowRootFromHost operation(aHost, fragHandle, aTemplateNode, mode,
+ aShadowRootDelegatesFocus);
+ treeOp->Init(mozilla::AsVariant(operation));
+ return fragHandle;
+}
+
+nsIContentHandle* nsHtml5TreeBuilder::getFormPointerForContext(
+ nsIContentHandle* aContext) {
+ MOZ_ASSERT(mBuilder, "Must have builder.");
+ if (!aContext) {
+ return nullptr;
+ }
+
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // aContext must always be an element that already exists
+ // in the document.
+ nsIContent* contextNode = static_cast<nsIContent*>(aContext);
+ nsIContent* currentAncestor = contextNode;
+
+ // We traverse the ancestors of the context node to find the nearest
+ // form pointer. This traversal is why aContext must not be an emtpy handle.
+ nsIContent* nearestForm = nullptr;
+ while (currentAncestor) {
+ if (currentAncestor->IsHTMLElement(nsGkAtoms::form)) {
+ nearestForm = currentAncestor;
+ break;
+ }
+ currentAncestor = currentAncestor->GetParent();
+ }
+
+ if (!nearestForm) {
+ return nullptr;
+ }
+
+ return nearestForm;
+}
+
+// Error reporting
+
+void nsHtml5TreeBuilder::EnableViewSource(nsHtml5Highlighter* aHighlighter) {
+ MOZ_ASSERT(!mBuilder, "Must not view source with builder.");
+ mViewSource = aHighlighter;
+}
+
+void nsHtml5TreeBuilder::errDeepTree() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errDeepTree");
+ } else if (!mBuilder) {
+ nsHtml5TreeOperation* treeOp = mOpQueue.AppendElement();
+ MOZ_ASSERT(treeOp, "Tree op allocation failed.");
+ opMaybeComplainAboutDeepTree operation(tokenizer->getLineNumber());
+ treeOp->Init(mozilla::AsVariant(operation));
+ }
+}
+
+void nsHtml5TreeBuilder::errStrayStartTag(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStrayStartTag2", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errStrayEndTag(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStrayEndTag", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errUnclosedElements(int32_t aIndex, nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errUnclosedElements", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errUnclosedElementsImplied(int32_t aIndex,
+ nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errUnclosedElementsImplied", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errUnclosedElementsCell(int32_t aIndex) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errUnclosedElementsCell");
+ }
+}
+
+void nsHtml5TreeBuilder::errStrayDoctype() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStrayDoctype");
+ }
+}
+
+void nsHtml5TreeBuilder::errAlmostStandardsDoctype() {
+ if (MOZ_UNLIKELY(mViewSource) && !forceNoQuirks) {
+ mViewSource->AddErrorToCurrentRun("errAlmostStandardsDoctype");
+ }
+}
+
+void nsHtml5TreeBuilder::errQuirkyDoctype() {
+ if (MOZ_UNLIKELY(mViewSource) && !forceNoQuirks) {
+ mViewSource->AddErrorToCurrentRun("errQuirkyDoctype");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceInTrailer() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceInTrailer");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceAfterFrameset() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceAfterFrameset");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceInFrameset() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceInFrameset");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceAfterBody() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceAfterBody");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceInColgroupInFragment() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceInColgroupInFragment");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceInNoscriptInHead() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceInNoscriptInHead");
+ }
+}
+
+void nsHtml5TreeBuilder::errFooBetweenHeadAndBody(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errFooBetweenHeadAndBody", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errStartTagWithoutDoctype() {
+ if (MOZ_UNLIKELY(mViewSource) && !forceNoQuirks) {
+ mViewSource->AddErrorToCurrentRun("errStartTagWithoutDoctype");
+ }
+}
+
+void nsHtml5TreeBuilder::errNoSelectInTableScope() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNoSelectInTableScope");
+ }
+}
+
+void nsHtml5TreeBuilder::errStartSelectWhereEndSelectExpected() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStartSelectWhereEndSelectExpected");
+ }
+}
+
+void nsHtml5TreeBuilder::errStartTagWithSelectOpen(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStartTagWithSelectOpen", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errBadStartTagInNoscriptInHead(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errBadStartTagInNoscriptInHead", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errImage() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errImage");
+ }
+}
+
+void nsHtml5TreeBuilder::errIsindex() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errIsindex");
+ }
+}
+
+void nsHtml5TreeBuilder::errFooSeenWhenFooOpen(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errFooSeenWhenFooOpen2", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errHeadingWhenHeadingOpen() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errHeadingWhenHeadingOpen");
+ }
+}
+
+void nsHtml5TreeBuilder::errFramesetStart() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errFramesetStart");
+ }
+}
+
+void nsHtml5TreeBuilder::errNoCellToClose() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNoCellToClose");
+ }
+}
+
+void nsHtml5TreeBuilder::errStartTagInTable(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStartTagInTable", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errFormWhenFormOpen() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errFormWhenFormOpen");
+ }
+}
+
+void nsHtml5TreeBuilder::errTableSeenWhileTableOpen() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errTableSeenWhileTableOpen");
+ }
+}
+
+void nsHtml5TreeBuilder::errStartTagInTableBody(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStartTagInTableBody", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagSeenWithoutDoctype() {
+ if (MOZ_UNLIKELY(mViewSource) && !forceNoQuirks) {
+ mViewSource->AddErrorToCurrentRun("errEndTagSeenWithoutDoctype");
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagAfterBody() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndTagAfterBody");
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagSeenWithSelectOpen(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndTagSeenWithSelectOpen", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errGarbageInColgroup() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errGarbageInColgroup");
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagBr() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndTagBr");
+ }
+}
+
+void nsHtml5TreeBuilder::errNoElementToCloseButEndTagSeen(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNoElementToCloseButEndTagSeen",
+ aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errHtmlStartTagInForeignContext(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errHtmlStartTagInForeignContext", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errNoTableRowToClose() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNoTableRowToClose");
+ }
+}
+
+void nsHtml5TreeBuilder::errNonSpaceInTable() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNonSpaceInTable");
+ }
+}
+
+void nsHtml5TreeBuilder::errUnclosedChildrenInRuby() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errUnclosedChildrenInRuby");
+ }
+}
+
+void nsHtml5TreeBuilder::errStartTagSeenWithoutRuby(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errStartTagSeenWithoutRuby", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errSelfClosing() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentSlash("errSelfClosing");
+ }
+}
+
+void nsHtml5TreeBuilder::errNoCheckUnclosedElementsOnStack() {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errNoCheckUnclosedElementsOnStack");
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagDidNotMatchCurrentOpenElement(
+ nsAtom* aName, nsAtom* aOther) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndTagDidNotMatchCurrentOpenElement",
+ aName, aOther);
+ }
+}
+
+void nsHtml5TreeBuilder::errEndTagViolatesNestingRules(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndTagViolatesNestingRules", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errEndWithUnclosedElements(nsAtom* aName) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errEndWithUnclosedElements", aName);
+ }
+}
+
+void nsHtml5TreeBuilder::errListUnclosedStartTags(int32_t aIgnored) {
+ if (MOZ_UNLIKELY(mViewSource)) {
+ mViewSource->AddErrorToCurrentRun("errListUnclosedStartTags");
+ }
+}