blob: 6b41d29c982db314d3ff9f4bdc67bea69872e423 (
plain)
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
124
125
|
/* -*- Mode: C++; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/* vim: set sts=2 ts=8 sw=2 tw=99 et: */
/* 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 "gfxVars.h"
#include "gfxVarReceiver.h"
#include "mozilla/dom/ContentChild.h"
namespace mozilla {
namespace gfx {
StaticAutoPtr<gfxVars> gfxVars::sInstance;
StaticAutoPtr<nsTArray<gfxVars::VarBase*>> gfxVars::sVarList;
void
gfxVars::Initialize()
{
if (sInstance) {
return;
}
// sVarList must be initialized first since it's used in the constructor for
// sInstance.
sVarList = new nsTArray<gfxVars::VarBase*>();
sInstance = new gfxVars;
// Like Preferences, we want content to synchronously get initial data on
// init. Note the GPU process is not handled here - it cannot send sync
// messages, so instead the initial data is pushed down.
if (XRE_IsContentProcess()) {
InfallibleTArray<GfxVarUpdate> vars;
dom::ContentChild::GetSingleton()->SendGetGfxVars(&vars);
for (const auto& var : vars) {
ApplyUpdate(var);
}
}
}
gfxVars::gfxVars()
{
}
void
gfxVars::Shutdown()
{
sInstance = nullptr;
sVarList = nullptr;
}
/* static */ void
gfxVars::ApplyUpdate(const GfxVarUpdate& aUpdate)
{
// Only subprocesses receive updates and apply them locally.
MOZ_ASSERT(!XRE_IsParentProcess());
sVarList->ElementAt(aUpdate.index())->SetValue(aUpdate.value());
}
/* static */ void
gfxVars::AddReceiver(gfxVarReceiver* aReceiver)
{
MOZ_ASSERT(NS_IsMainThread());
// Don't double-add receivers, in case a broken content process sends two
// init messages.
if (!sInstance->mReceivers.Contains(aReceiver)) {
sInstance->mReceivers.AppendElement(aReceiver);
}
}
/* static */ void
gfxVars::RemoveReceiver(gfxVarReceiver* aReceiver)
{
MOZ_ASSERT(NS_IsMainThread());
if (sInstance) {
sInstance->mReceivers.RemoveElement(aReceiver);
}
}
/* static */ nsTArray<GfxVarUpdate>
gfxVars::FetchNonDefaultVars()
{
MOZ_ASSERT(NS_IsMainThread());
MOZ_ASSERT(sVarList);
nsTArray<GfxVarUpdate> updates;
for (size_t i = 0; i < sVarList->Length(); i++) {
VarBase* var = sVarList->ElementAt(i);
if (var->HasDefaultValue()) {
continue;
}
GfxVarValue value;
var->GetValue(&value);
updates.AppendElement(GfxVarUpdate(i, value));
}
return updates;
}
gfxVars::VarBase::VarBase()
{
mIndex = gfxVars::sVarList->Length();
gfxVars::sVarList->AppendElement(this);
}
void
gfxVars::NotifyReceivers(VarBase* aVar)
{
MOZ_ASSERT(NS_IsMainThread());
GfxVarValue value;
aVar->GetValue(&value);
GfxVarUpdate update(aVar->Index(), value);
for (auto& receiver : mReceivers) {
receiver->OnVarChanged(update);
}
}
} // namespace gfx
} // namespace mozilla
|