зеркало из https://github.com/mozilla/gecko-dev.git
243 строки
6.5 KiB
C++
243 строки
6.5 KiB
C++
/* 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;
|
|
mozilla::ImportScanner mImportScanner;
|
|
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 errDeepTree();
|
|
|
|
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 errBadStartTagInNoscriptInHead(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 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; }
|