1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
|
/* 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 "nsHtml5Module.h"
#include "mozilla/AlreadyAddRefed.h"
#include "mozilla/Attributes.h"
#include "mozilla/Preferences.h"
#include "mozilla/Services.h"
#include "mozilla/StaticPrefs_html5.h"
#include "nsCOMPtr.h"
#include "nsHtml5AttributeName.h"
#include "nsHtml5ElementName.h"
#include "nsHtml5HtmlAttributes.h"
#include "nsHtml5NamedCharacters.h"
#include "nsHtml5Portability.h"
#include "nsHtml5StackNode.h"
#include "nsHtml5Tokenizer.h"
#include "nsHtml5TreeBuilder.h"
#include "nsHtml5UTF16Buffer.h"
#include "nsIObserverService.h"
using namespace mozilla;
// static
nsIThread* nsHtml5Module::sStreamParserThread = nullptr;
class nsHtml5ParserThreadTerminator final : public nsIObserver {
public:
NS_DECL_ISUPPORTS
explicit nsHtml5ParserThreadTerminator(nsIThread* aThread)
: mThread(aThread) {}
NS_IMETHOD Observe(nsISupports*, const char* topic,
const char16_t*) override {
NS_ASSERTION(!strcmp(topic, "xpcom-shutdown-threads"), "Unexpected topic");
mThread->Shutdown();
mThread = nullptr;
NS_IF_RELEASE(nsHtml5Module::sStreamParserThread);
nsHtml5Module::sStreamParserThread = nullptr;
return NS_OK;
}
private:
~nsHtml5ParserThreadTerminator() = default;
nsCOMPtr<nsIThread> mThread;
};
NS_IMPL_ISUPPORTS(nsHtml5ParserThreadTerminator, nsIObserver)
// static
void nsHtml5Module::InitializeStatics() {
nsHtml5AttributeName::initializeStatics();
nsHtml5ElementName::initializeStatics();
nsHtml5HtmlAttributes::initializeStatics();
nsHtml5NamedCharacters::initializeStatics();
nsHtml5Portability::initializeStatics();
nsHtml5StackNode::initializeStatics();
nsHtml5Tokenizer::initializeStatics();
nsHtml5TreeBuilder::initializeStatics();
nsHtml5UTF16Buffer::initializeStatics();
NS_NewNamedThread("HTML5 Parser", &sStreamParserThread);
if (sStreamParserThread) {
nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService();
if (os) {
os->AddObserver(new nsHtml5ParserThreadTerminator(sStreamParserThread),
"xpcom-shutdown-threads", false);
} else {
MOZ_ASSERT(false,
"How come we failed to create get the observer service?");
}
} else {
MOZ_ASSERT(false, "How come we failed to create the parser thread?");
}
#ifdef DEBUG
sNsHtml5ModuleInitialized = true;
#endif
}
// static
void nsHtml5Module::ReleaseStatics() {
#ifdef DEBUG
sNsHtml5ModuleInitialized = false;
#endif
nsHtml5AttributeName::releaseStatics();
nsHtml5ElementName::releaseStatics();
nsHtml5HtmlAttributes::releaseStatics();
nsHtml5NamedCharacters::releaseStatics();
nsHtml5Portability::releaseStatics();
nsHtml5StackNode::releaseStatics();
nsHtml5Tokenizer::releaseStatics();
nsHtml5TreeBuilder::releaseStatics();
nsHtml5UTF16Buffer::releaseStatics();
NS_IF_RELEASE(sStreamParserThread);
}
// static
already_AddRefed<nsHtml5Parser> nsHtml5Module::NewHtml5Parser() {
MOZ_ASSERT(sNsHtml5ModuleInitialized, "nsHtml5Module not initialized.");
RefPtr<nsHtml5Parser> rv = new nsHtml5Parser();
return rv.forget();
}
// static
already_AddRefed<nsISerialEventTarget>
nsHtml5Module::GetStreamParserEventTarget() {
MOZ_ASSERT(sNsHtml5ModuleInitialized, "nsHtml5Module not initialized.");
if (sStreamParserThread) {
nsCOMPtr<nsISerialEventTarget> target = sStreamParserThread;
return target.forget();
}
nsCOMPtr<nsIThread> mainThread;
NS_GetMainThread(getter_AddRefs(mainThread));
MOZ_RELEASE_ASSERT(mainThread); // Unrecoverable situation
nsCOMPtr<nsISerialEventTarget> target = mainThread;
return target.forget();
}
#ifdef DEBUG
bool nsHtml5Module::sNsHtml5ModuleInitialized = false;
#endif
|