parser/html/nsHtml5TreeBuilderHSupplement.h
author Masatoshi Kimura <VYV03354@nifty.ne.jp>
Fri, 06 Apr 2018 06:20:28 +0900
changeset 412113 effaf62f3423dfed70794254fa613f3cdfcfe9f4
parent 408600 6af1f5ac596b4aa0a87f8a8151395c692eb81a58
child 448947 6f3709b3878117466168c40affa7bca0b60cf75b
permissions -rw-r--r--
Bug 1445541 - Enable the character encoding menu when the document is decoded as UTF-8 and an error is found, instead of disabling it when no errors are found. r=hsivonen MozReview-Commit-ID: 1e6kQbAxcqm

/* 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/. */

#define NS_HTML5_TREE_BUILDER_HANDLE_ARRAY_LENGTH 512
private:
using Encoding = mozilla::Encoding;
template<typename T>
using NotNull = mozilla::NotNull<T>;

nsHtml5OplessBuilder* mBuilder;
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
// If mBuilder is not null, the tree op machinery is not in use and
// the fields below aren't in use, either.
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
nsHtml5Highlighter* mViewSource;
nsTArray<nsHtml5TreeOperation> mOpQueue;
nsTArray<nsHtml5SpeculativeLoad> mSpeculativeLoadQueue;
nsAHtml5TreeOpSink* mOpSink;
mozilla::UniquePtr<nsIContent* []> mHandles;
int32_t mHandlesUsed;
nsTArray<mozilla::UniquePtr<nsIContent* []>> mOldHandles;
nsHtml5TreeOpStage* mSpeculativeLoadStage;
nsresult mBroken;
bool mCurrentHtmlScriptIsAsyncOrDefer;
bool mPreventScriptExecution;
#ifdef DEBUG
bool mActive;
#endif

// DocumentModeHandler
/**
 * Tree builder uses this to report quirkiness of the document
 */
void
documentMode(nsHtml5DocumentMode m);

nsIContentHandle*
getDocumentFragmentForTemplate(nsIContentHandle* aTemplate);

nsIContentHandle*
getFormPointerForContext(nsIContentHandle* aContext);

/**
 * Using nsIContent** instead of nsIContent* is the parser deals with DOM
 * nodes in a way that works off the main thread. Non-main-thread code
 * can't refcount or otherwise touch nsIContent objects in any way.
 * Yet, the off-the-main-thread code needs to have a way to hold onto a
 * particular node and repeatedly operate on the same node.
 *
 * The way this works is that the off-the-main-thread code has an
 * nsIContent** for each DOM node and a given nsIContent** is only ever
 * actually dereferenced into an actual nsIContent* on the main thread.
 * When the off-the-main-thread code requests a new node, it gets an
 * nsIContent** immediately and a tree op is enqueued for later allocating
 * an actual nsIContent object and writing a pointer to it into the memory
 * location pointed to by the nsIContent**.
 *
 * Since tree ops are in a queue, the node creating tree op will always
 * run before tree ops that try to further operate on the node that the
 * nsIContent** is a handle to.
 *
 * On-the-main-thread parts of the parser use nsIContent* instead of
 * nsIContent**. Since both cases share the same parser core, the parser
 * core casts both to nsIContentHandle*.
 */
nsIContentHandle*
AllocateContentHandle();

void
accumulateCharactersForced(const char16_t* aBuf,
                           int32_t aStart,
                           int32_t aLength)
{
  accumulateCharacters(aBuf, aStart, aLength);
}

void
MarkAsBrokenAndRequestSuspensionWithBuilder(nsresult aRv)
{
  mBuilder->MarkAsBroken(aRv);
  requestSuspension();
}

void
MarkAsBrokenAndRequestSuspensionWithoutBuilder(nsresult aRv)
{
  MarkAsBroken(aRv);
  requestSuspension();
}

void
MarkAsBrokenFromPortability(nsresult aRv);

public:
explicit nsHtml5TreeBuilder(nsHtml5OplessBuilder* aBuilder);

nsHtml5TreeBuilder(nsAHtml5TreeOpSink* aOpSink, nsHtml5TreeOpStage* aStage);

~nsHtml5TreeBuilder();

void
StartPlainTextViewSource(const nsAutoString& aTitle);

void
StartPlainText();

void
StartPlainTextBody();

bool
HasScript();

void
SetOpSink(nsAHtml5TreeOpSink* aOpSink)
{
  mOpSink = aOpSink;
}

void
ClearOps()
{
  mOpQueue.Clear();
}

bool
Flush(bool aDiscretionary = false);

void
FlushLoads();

void
SetDocumentCharset(NotNull<const Encoding*> aEncoding, int32_t aCharsetSource);

void
StreamEnded();

void
NeedsCharsetSwitchTo(NotNull<const Encoding*> aEncoding,
                     int32_t aSource,
                     int32_t aLineNumber);

void
MaybeComplainAboutCharset(const char* aMsgId, bool aError, int32_t aLineNumber);

void
TryToEnableEncodingMenu();

void
AddSnapshotToScript(nsAHtml5TreeBuilderState* aSnapshot, int32_t aLine);

void
DropHandles();

void
SetPreventScriptExecution(bool aPrevent)
{
  mPreventScriptExecution = aPrevent;
}

bool
HasBuilder()
{
  return mBuilder;
}

/**
 * Makes sure the buffers are large enough to be able to tokenize aLength
 * UTF-16 code units before having to make the buffers larger.
 *
 * @param aLength the number of UTF-16 code units to be tokenized before the
 *                next call to this method.
 * @return true if successful; false if out of memory
 */
bool
EnsureBufferSpace(int32_t aLength);

void
EnableViewSource(nsHtml5Highlighter* aHighlighter);

void
errStrayStartTag(nsAtom* aName);

void
errStrayEndTag(nsAtom* aName);

void
errUnclosedElements(int32_t aIndex, nsAtom* aName);

void
errUnclosedElementsImplied(int32_t aIndex, nsAtom* aName);

void
errUnclosedElementsCell(int32_t aIndex);

void
errStrayDoctype();

void
errAlmostStandardsDoctype();

void
errQuirkyDoctype();

void
errNonSpaceInTrailer();

void
errNonSpaceAfterFrameset();

void
errNonSpaceInFrameset();

void
errNonSpaceAfterBody();

void
errNonSpaceInColgroupInFragment();

void
errNonSpaceInNoscriptInHead();

void
errFooBetweenHeadAndBody(nsAtom* aName);

void
errStartTagWithoutDoctype();

void
errNoSelectInTableScope();

void
errStartSelectWhereEndSelectExpected();

void
errStartTagWithSelectOpen(nsAtom* aName);

void
errBadStartTagInHead(nsAtom* aName);

void
errImage();

void
errIsindex();

void
errFooSeenWhenFooOpen(nsAtom* aName);

void
errHeadingWhenHeadingOpen();

void
errFramesetStart();

void
errNoCellToClose();

void
errStartTagInTable(nsAtom* aName);

void
errFormWhenFormOpen();

void
errTableSeenWhileTableOpen();

void
errStartTagInTableBody(nsAtom* aName);

void
errEndTagSeenWithoutDoctype();

void
errEndTagAfterBody();

void
errEndTagSeenWithSelectOpen(nsAtom* aName);

void
errGarbageInColgroup();

void
errEndTagBr();

void
errNoElementToCloseButEndTagSeen(nsAtom* aName);

void
errHtmlStartTagInForeignContext(nsAtom* aName);

void
errTableClosedWhileCaptionOpen();

void
errNoTableRowToClose();

void
errNonSpaceInTable();

void
errUnclosedChildrenInRuby();

void
errStartTagSeenWithoutRuby(nsAtom* aName);

void
errSelfClosing();

void
errNoCheckUnclosedElementsOnStack();

void
errEndTagDidNotMatchCurrentOpenElement(nsAtom* aName, nsAtom* aOther);

void
errEndTagViolatesNestingRules(nsAtom* aName);

void
errEndWithUnclosedElements(nsAtom* aName);

void
MarkAsBroken(nsresult aRv);

/**
 * Checks if this parser is broken. Returns a non-NS_OK (i.e. non-0)
 * value if broken.
 */
nsresult
IsBroken()
{
  return mBroken;
}