Coverage Report

Created: 2018-09-25 14:53

/work/obj-fuzz/dist/include/mozilla/JSEventHandler.h
Line
Count
Source (jump to first uncovered line)
1
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
3
/* This Source Code Form is subject to the terms of the Mozilla Public
4
 * License, v. 2.0. If a copy of the MPL was not distributed with this
5
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
6
7
#ifndef mozilla_JSEventHandler_h_
8
#define mozilla_JSEventHandler_h_
9
10
#include "mozilla/Attributes.h"
11
#include "mozilla/MemoryReporting.h"
12
#include "mozilla/dom/EventHandlerBinding.h"
13
#include "nsCOMPtr.h"
14
#include "nsCycleCollectionParticipant.h"
15
#include "nsAtom.h"
16
#include "nsIDOMEventListener.h"
17
#include "nsIScriptContext.h"
18
19
namespace mozilla {
20
21
class TypedEventHandler
22
{
23
public:
24
  enum HandlerType
25
  {
26
    eUnset = 0,
27
    eNormal = 0x1,
28
    eOnError = 0x2,
29
    eOnBeforeUnload = 0x3,
30
    eTypeBits = 0x3
31
  };
32
33
  TypedEventHandler()
34
    : mBits(0)
35
0
  {
36
0
  }
37
38
  explicit TypedEventHandler(dom::EventHandlerNonNull* aHandler)
39
    : mBits(0)
40
0
  {
41
0
    Assign(aHandler, eNormal);
42
0
  }
43
44
  explicit TypedEventHandler(dom::OnErrorEventHandlerNonNull* aHandler)
45
    : mBits(0)
46
0
  {
47
0
    Assign(aHandler, eOnError);
48
0
  }
49
50
  explicit TypedEventHandler(dom::OnBeforeUnloadEventHandlerNonNull* aHandler)
51
    : mBits(0)
52
0
  {
53
0
    Assign(aHandler, eOnBeforeUnload);
54
0
  }
55
56
  TypedEventHandler(const TypedEventHandler& aOther)
57
0
  {
58
0
    if (aOther.HasEventHandler()) {
59
0
      // Have to make sure we take our own ref
60
0
      Assign(aOther.Ptr(), aOther.Type());
61
0
    } else {
62
0
      mBits = 0;
63
0
    }
64
0
  }
65
66
  ~TypedEventHandler()
67
0
  {
68
0
    ReleaseHandler();
69
0
  }
70
71
  HandlerType Type() const
72
0
  {
73
0
    return HandlerType(mBits & eTypeBits);
74
0
  }
75
76
  bool HasEventHandler() const
77
0
  {
78
0
    return !!Ptr();
79
0
  }
80
81
  void SetHandler(const TypedEventHandler& aHandler)
82
0
  {
83
0
    if (aHandler.HasEventHandler()) {
84
0
      ReleaseHandler();
85
0
      Assign(aHandler.Ptr(), aHandler.Type());
86
0
    } else {
87
0
      ForgetHandler();
88
0
    }
89
0
  }
90
91
  dom::EventHandlerNonNull* NormalEventHandler() const
92
0
  {
93
0
    MOZ_ASSERT(Type() == eNormal && Ptr());
94
0
    return reinterpret_cast<dom::EventHandlerNonNull*>(Ptr());
95
0
  }
96
97
  void SetHandler(dom::EventHandlerNonNull* aHandler)
98
0
  {
99
0
    ReleaseHandler();
100
0
    Assign(aHandler, eNormal);
101
0
  }
102
103
  dom::OnBeforeUnloadEventHandlerNonNull* OnBeforeUnloadEventHandler() const
104
0
  {
105
0
    MOZ_ASSERT(Type() == eOnBeforeUnload);
106
0
    return reinterpret_cast<dom::OnBeforeUnloadEventHandlerNonNull*>(Ptr());
107
0
  }
108
109
  void SetHandler(dom::OnBeforeUnloadEventHandlerNonNull* aHandler)
110
0
  {
111
0
    ReleaseHandler();
112
0
    Assign(aHandler, eOnBeforeUnload);
113
0
  }
114
115
  dom::OnErrorEventHandlerNonNull* OnErrorEventHandler() const
116
0
  {
117
0
    MOZ_ASSERT(Type() == eOnError);
118
0
    return reinterpret_cast<dom::OnErrorEventHandlerNonNull*>(Ptr());
119
0
  }
120
121
  void SetHandler(dom::OnErrorEventHandlerNonNull* aHandler)
122
0
  {
123
0
    ReleaseHandler();
124
0
    Assign(aHandler, eOnError);
125
0
  }
126
127
  dom::CallbackFunction* Ptr() const
128
0
  {
129
0
    // Have to cast eTypeBits so we don't have to worry about
130
0
    // promotion issues after the bitflip.
131
0
    return reinterpret_cast<dom::CallbackFunction*>(mBits &
132
0
                                                      ~uintptr_t(eTypeBits));
133
0
  }
134
135
  void ForgetHandler()
136
0
  {
137
0
    ReleaseHandler();
138
0
    mBits = 0;
139
0
  }
140
141
  bool operator==(const TypedEventHandler& aOther) const
142
0
  {
143
0
    return
144
0
      Ptr() && aOther.Ptr() &&
145
0
      Ptr()->CallbackPreserveColor() == aOther.Ptr()->CallbackPreserveColor();
146
0
  }
147
148
private:
149
  void operator=(const TypedEventHandler&) = delete;
150
151
  void ReleaseHandler()
152
0
  {
153
0
    nsISupports* ptr = Ptr();
154
0
    NS_IF_RELEASE(ptr);
155
0
  }
156
157
  void Assign(nsISupports* aHandler, HandlerType aType)
158
0
  {
159
0
    MOZ_ASSERT(aHandler, "Must have handler");
160
0
    NS_ADDREF(aHandler);
161
0
    mBits = uintptr_t(aHandler) | uintptr_t(aType);
162
0
  }
163
164
  uintptr_t mBits;
165
};
166
167
/**
168
 * Implemented by script event listeners. Used to retrieve the script object
169
 * corresponding to the event target and the handler itself.
170
 *
171
 * Note, mTarget is a raw pointer and the owner of the JSEventHandler object
172
 * is expected to call Disconnect()!
173
 */
174
175
#define NS_JSEVENTHANDLER_IID \
176
{ 0x4f486881, 0x1956, 0x4079, \
177
  { 0x8c, 0xa0, 0xf3, 0xbd, 0x60, 0x5c, 0xc2, 0x79 } }
178
179
class JSEventHandler : public nsIDOMEventListener
180
{
181
public:
182
  NS_DECLARE_STATIC_IID_ACCESSOR(NS_JSEVENTHANDLER_IID)
183
184
  JSEventHandler(nsISupports* aTarget, nsAtom* aType,
185
                 const TypedEventHandler& aTypedHandler);
186
187
  NS_DECL_CYCLE_COLLECTING_ISUPPORTS
188
189
  // nsIDOMEventListener interface
190
  NS_DECL_NSIDOMEVENTLISTENER
191
192
  nsISupports* GetEventTarget() const
193
0
  {
194
0
    return mTarget;
195
0
  }
196
197
  void Disconnect()
198
0
  {
199
0
    mTarget = nullptr;
200
0
  }
201
202
  const TypedEventHandler& GetTypedEventHandler() const
203
0
  {
204
0
    return mTypedHandler;
205
0
  }
206
207
  void ForgetHandler()
208
0
  {
209
0
    mTypedHandler.ForgetHandler();
210
0
  }
211
212
  nsAtom* EventName() const
213
0
  {
214
0
    return mEventName;
215
0
  }
216
217
  // Set a handler for this event listener.  The handler must already
218
  // be bound to the right target.
219
  void SetHandler(const TypedEventHandler& aTypedHandler)
220
0
  {
221
0
    mTypedHandler.SetHandler(aTypedHandler);
222
0
  }
223
  void SetHandler(dom::EventHandlerNonNull* aHandler)
224
0
  {
225
0
    mTypedHandler.SetHandler(aHandler);
226
0
  }
227
  void SetHandler(dom::OnBeforeUnloadEventHandlerNonNull* aHandler)
228
0
  {
229
0
    mTypedHandler.SetHandler(aHandler);
230
0
  }
231
  void SetHandler(dom::OnErrorEventHandlerNonNull* aHandler)
232
0
  {
233
0
    mTypedHandler.SetHandler(aHandler);
234
0
  }
235
236
  size_t SizeOfExcludingThis(MallocSizeOf aMallocSizeOf) const
237
0
  {
238
0
    return 0;
239
0
240
0
    // Measurement of the following members may be added later if DMD finds it
241
0
    // is worthwhile:
242
0
    // - mTarget
243
0
    //
244
0
    // The following members are not measured:
245
0
    // - mTypedHandler: may be shared with others
246
0
    // - mEventName: shared with others
247
0
  }
248
249
  size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf)
250
0
  {
251
0
    return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
252
0
  }
253
254
  NS_DECL_CYCLE_COLLECTION_SKIPPABLE_CLASS(JSEventHandler)
255
256
  bool IsBlackForCC();
257
258
protected:
259
  virtual ~JSEventHandler();
260
261
  nsISupports* mTarget;
262
  RefPtr<nsAtom> mEventName;
263
  TypedEventHandler mTypedHandler;
264
};
265
266
NS_DEFINE_STATIC_IID_ACCESSOR(JSEventHandler, NS_JSEVENTHANDLER_IID)
267
268
} // namespace mozilla
269
270
/**
271
 * Factory function.  aHandler must already be bound to aTarget.
272
 * aContext is allowed to be null if aHandler is already set up.
273
 */
274
nsresult NS_NewJSEventHandler(nsISupports* aTarget,
275
                              nsAtom* aType,
276
                              const mozilla::TypedEventHandler& aTypedHandler,
277
                              mozilla::JSEventHandler** aReturn);
278
279
#endif // mozilla_JSEventHandler_h_
280