Commit bc8b104d authored by peria@chromium.org's avatar peria@chromium.org

Remove a raw pointer to a PlatformEventController instance in...

Remove a raw pointer to a PlatformEventController instance in PlatformEventDispatcher on Oilpan build

BUG=509911

Review URL: https://codereview.chromium.org/1301543002

git-svn-id: svn://svn.chromium.org/blink/trunk@201313 bbb929c8-8fbe-4397-9dbb-9b2b20218538
parent c3fb0b63
...@@ -11,95 +11,62 @@ ...@@ -11,95 +11,62 @@
namespace blink { namespace blink {
PlatformEventDispatcher::PlatformEventDispatcher() PlatformEventDispatcher::PlatformEventDispatcher()
: m_needsPurge(false) : m_isDispatching(false)
, m_isDispatching(false) , m_isListening(false)
{ {
} }
void PlatformEventDispatcher::addController(PlatformEventController* controller) void PlatformEventDispatcher::addController(PlatformEventController* controller)
{ {
bool wasEmpty = m_controllers.isEmpty(); ASSERT(controller);
if (!m_controllers.contains(controller)) ASSERT(!m_controllers.contains(controller));
m_controllers.append(controller);
if (wasEmpty) m_controllers.add(controller);
if (!m_isListening) {
startListening(); startListening();
m_isListening = true;
}
} }
void PlatformEventDispatcher::removeController(PlatformEventController* controller) void PlatformEventDispatcher::removeController(PlatformEventController* controller)
{ {
// Do not actually remove the controller from the vector, instead zero them out. ASSERT(m_controllers.contains(controller));
// The zeros are removed in these two cases:
// 1. either immediately if we are not dispatching any events,
// 2. or after events to all controllers have dispatched (see notifyControllers()).
// This is to correctly handle the re-entrancy case when a controller is destroyed
// while the events are still being dispatched.
size_t index = m_controllers.find(controller);
if (index == kNotFound)
return;
m_controllers[index] = nullptr;
m_needsPurge = true;
if (!m_isDispatching) m_controllers.remove(controller);
purgeControllers(); if (!m_isDispatching && m_controllers.isEmpty()) {
}
void PlatformEventDispatcher::purgeControllers()
{
ASSERT(m_needsPurge);
size_t i = 0;
while (i < m_controllers.size()) {
if (!m_controllers[i]) {
m_controllers[i] = m_controllers.last();
m_controllers.removeLast();
} else {
++i;
}
}
m_needsPurge = false;
if (m_controllers.isEmpty())
stopListening(); stopListening();
m_isListening = false;
}
} }
void PlatformEventDispatcher::notifyControllers() void PlatformEventDispatcher::notifyControllers()
{ {
if (m_controllers.isEmpty())
return;
{ {
TemporaryChange<bool> changeIsDispatching(m_isDispatching, true); TemporaryChange<bool> changeIsDispatching(m_isDispatching, true);
// Don't notify controllers removed or added during event dispatch. // HashSet m_controllers can be updated during an iteration, and it stops the iteration.
size_t size = m_controllers.size(); // Thus we store it into a Vector to access all elements.
for (size_t i = 0; i < size; ++i) { WillBeHeapVector<RawPtrWillBeMember<PlatformEventController>> snapshotVector;
if (m_controllers[i]) copyToVector(m_controllers, snapshotVector);
m_controllers[i]->didUpdateData(); for (PlatformEventController* controller : snapshotVector) {
if (m_controllers.contains(controller))
controller->didUpdateData();
} }
} }
if (m_needsPurge) if (m_controllers.isEmpty()) {
purgeControllers(); stopListening();
m_isListening = false;
}
} }
DEFINE_TRACE(PlatformEventDispatcher) DEFINE_TRACE(PlatformEventDispatcher)
{ {
#if ENABLE(OILPAN) #if ENABLE(OILPAN)
// Trace the backing store, the weak(&bare) element references won't be.
visitor->trace(m_controllers); visitor->trace(m_controllers);
visitor->template registerWeakMembers<PlatformEventDispatcher, &PlatformEventDispatcher::clearWeakMembers>(this);
#endif #endif
} }
#if ENABLE(OILPAN)
void PlatformEventDispatcher::clearWeakMembers(Visitor* visitor)
{
for (size_t i = 0; i < m_controllers.size(); ++i) {
if (!Heap::isHeapObjectAlive(m_controllers[i])) {
m_controllers[i] = nullptr;
m_needsPurge = true;
}
}
// Next notification will purge the empty slots.
}
#endif
} // namespace blink } // namespace blink
...@@ -30,13 +30,9 @@ protected: ...@@ -30,13 +30,9 @@ protected:
private: private:
void purgeControllers(); void purgeControllers();
#if ENABLE(OILPAN) WillBeHeapHashSet<RawPtrWillBeWeakMember<PlatformEventController>> m_controllers;
void clearWeakMembers(Visitor*);
#endif
WillBeHeapVector<PlatformEventController*> m_controllers;
bool m_needsPurge;
bool m_isDispatching; bool m_isDispatching;
bool m_isListening;
}; };
} // namespace blink } // namespace blink
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment