Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(103)

Side by Side Diff: Source/heap/Handle.h

Issue 169653003: Allow creating non-thread-local Persistents that can be passed between threads. (Closed) Base URL: svn://svn.chromium.org/blink/trunk
Patch Set: Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | Source/heap/Heap.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2014 Google Inc. All rights reserved. 2 * Copyright (C) 2014 Google Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are 5 * modification, are permitted provided that the following conditions are
6 * met: 6 * met:
7 * 7 *
8 * * Redistributions of source code must retain the above copyright 8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer. 9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above 10 * * Redistributions in binary form must reproduce the above
(...skipping 17 matching lines...) Expand all
28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */ 29 */
30 30
31 #ifndef Handle_h 31 #ifndef Handle_h
32 #define Handle_h 32 #define Handle_h
33 33
34 #include "heap/Heap.h" 34 #include "heap/Heap.h"
35 #include "heap/ThreadState.h" 35 #include "heap/ThreadState.h"
36 #include "heap/Visitor.h" 36 #include "heap/Visitor.h"
37 37
38 #include "wtf/Locker.h"
38 #include "wtf/RawPtr.h" 39 #include "wtf/RawPtr.h"
39 #include "wtf/RefCounted.h" 40 #include "wtf/RefCounted.h"
40 41
41 namespace WebCore { 42 namespace WebCore {
42 43
43 template<typename T> class Member; 44 template<typename T> class Member;
44 45
45 class PersistentNode { 46 class PersistentNode {
46 public: 47 public:
47 explicit PersistentNode(TraceCallback trace) 48 explicit PersistentNode(TraceCallback trace)
(...skipping 21 matching lines...) Expand all
69 m_trace(visitor, this); 70 m_trace(visitor, this);
70 } 71 }
71 72
72 protected: 73 protected:
73 TraceCallback m_trace; 74 TraceCallback m_trace;
74 75
75 private: 76 private:
76 PersistentNode* m_next; 77 PersistentNode* m_next;
77 PersistentNode* m_prev; 78 PersistentNode* m_prev;
78 79
79 template<ThreadAffinity affinity, typename Owner> friend class PersistentBas e; 80 template<typename RootsAccessor, typename Owner> friend class PersistentBase ;
80 friend class PersistentAnchor; 81 friend class PersistentAnchor;
81 friend class ThreadState; 82 friend class ThreadState;
82 }; 83 };
83 84
84 template<ThreadAffinity Affinity, typename Owner> 85 // RootsAccessor for Persistent that provides access to thread-local list
86 // of persistent handles. Can only be used to create handles that
87 // are constructed and destructed on the same thread.
88 template<ThreadAffinity Affinity>
89 class ThreadLocalPersistents {
90 public:
91 static PersistentNode* roots() { return state()->roots(); }
92
93 // No locking requires. Just checks that we are at the right thread.
Mads Ager (chromium) 2014/02/17 15:00:13 required
Vyacheslav Egorov (Chromium) 2014/02/17 15:52:35 Done.
94 class Lock {
95 public:
96 Lock() { state()->checkThread(); }
97 };
98
99 private:
100 static ThreadState* state() { return ThreadStateFor<Affinity>::state(); }
101 };
102
103 // RootsAccessor for Persistent that provides synchronized access to global
104 // list of persistent handles. Can be used for persistent handles that are
105 // passed between threads.
106 class GlobalPersistents {
107 public:
108 static PersistentNode* roots() { return ThreadState::globalRoots(); }
109
110 class Lock {
111 public:
112 Lock() : m_locker(ThreadState::globalRootsMutex()) { }
113 private:
114 MutexLocker m_locker;
115 };
116 };
117
118 // Base class for persistent handles. RootsAccessor specifies which list to
119 // link resulting handle into. Owner specifies the class containing trace
120 // method.
121 template<typename RootsAccessor, typename Owner>
85 class PersistentBase : public PersistentNode { 122 class PersistentBase : public PersistentNode {
86 public: 123 public:
87 ~PersistentBase() 124 ~PersistentBase()
88 { 125 {
89 #ifndef NDEBUG 126 #ifndef NDEBUG
90 ASSERT(isAlive()); 127 ASSERT(isAlive());
91 ASSERT(m_next->isAlive()); 128 ASSERT(m_next->isAlive());
92 ASSERT(m_prev->isAlive()); 129 ASSERT(m_prev->isAlive());
93 m_threadState->checkThread(); 130 ASSERT(m_roots == RootsAccessor::roots()); // Check that the thread is u sing the same roots list.
94 #endif 131 #endif
132 typename RootsAccessor::Lock lock;
Mads Ager (chromium) 2014/02/17 15:00:13 Shouldn't you lock before the debug-mode code? The
Vyacheslav Egorov (Chromium) 2014/02/17 15:52:35 Good catch. Fixed.
95 m_next->m_prev = m_prev; 133 m_next->m_prev = m_prev;
96 m_prev->m_next = m_next; 134 m_prev->m_next = m_next;
97 } 135 }
98 136
99 protected: 137 protected:
100 inline PersistentBase() 138 inline PersistentBase()
101 : PersistentNode(TraceMethodDelegate<Owner, &Owner::trace>::trampoline) 139 : PersistentNode(TraceMethodDelegate<Owner, &Owner::trace>::trampoline)
102 #ifndef NDEBUG 140 #ifndef NDEBUG
103 , m_threadState(state()) 141 , m_roots(RootsAccessor::roots())
104 #endif 142 #endif
105 { 143 {
106 #ifndef NDEBUG 144 typename RootsAccessor::Lock lock;
107 m_threadState->checkThread(); 145 m_prev = RootsAccessor::roots();
108 #endif 146 m_next = m_prev->m_next;
109 ThreadState* threadState = state(); 147 m_prev->m_next = this;
110 m_prev = threadState->roots();
111 m_next = threadState->roots()->m_next;
112 threadState->roots()->m_next = this;
113 m_next->m_prev = this; 148 m_next->m_prev = this;
114 } 149 }
115 150
116 inline explicit PersistentBase(const PersistentBase& otherref) 151 inline explicit PersistentBase(const PersistentBase& otherref)
117 : PersistentNode(otherref.m_trace) 152 : PersistentNode(otherref.m_trace)
118 #ifndef NDEBUG 153 #ifndef NDEBUG
119 , m_threadState(state()) 154 , m_roots(RootsAccessor::roots())
120 #endif 155 #endif
121 { 156 {
122 #ifndef NDEBUG 157 typename RootsAccessor::Lock lock;
123 m_threadState->checkThread(); 158 ASSERT(otherref.m_roots == m_roots); // Handles must belong to the same list.
124 #endif
125 ASSERT(otherref.m_threadState == m_threadState);
126 PersistentBase* other = const_cast<PersistentBase*>(&otherref); 159 PersistentBase* other = const_cast<PersistentBase*>(&otherref);
127 m_prev = other; 160 m_prev = other;
128 m_next = other->m_next; 161 m_next = other->m_next;
129 other->m_next = this; 162 other->m_next = this;
130 m_next->m_prev = this; 163 m_next->m_prev = this;
131 } 164 }
132 165
133 inline PersistentBase& operator=(const PersistentBase& otherref) 166 inline PersistentBase& operator=(const PersistentBase& otherref) { return *t his; }
134 {
135 return *this;
136 }
137
138 static ThreadState* state() { return ThreadStateFor<Affinity>::state(); }
139 167
140 #ifndef NDEBUG 168 #ifndef NDEBUG
141 private: 169 private:
142 ThreadState* m_threadState; 170 PersistentNode* m_roots;
143 #endif 171 #endif
144 }; 172 };
145 173
146 // A dummy Persistent handle that ensures the list of persistents is never null. 174 // A dummy Persistent handle that ensures the list of persistents is never null.
147 // This removes a test from a hot path. 175 // This removes a test from a hot path.
148 class PersistentAnchor : public PersistentNode { 176 class PersistentAnchor : public PersistentNode {
149 public: 177 public:
150 void trace(Visitor*) { } 178 void trace(Visitor* visitor)
179 {
180 for (PersistentNode* current = m_next; current != this; current = curren t->m_next)
181 current->trace(visitor);
182 }
151 183
152 private: 184 private:
153 virtual ~PersistentAnchor() 185 virtual ~PersistentAnchor()
154 { 186 {
155 // FIXME: oilpan: Ideally we should have no left-over persistents at thi s point. However currently there is a 187 // FIXME: oilpan: Ideally we should have no left-over persistents at thi s point. However currently there is a
156 // large number of objects leaked when we tear down the main thread. Sin ce some of these might contain a 188 // large number of objects leaked when we tear down the main thread. Sin ce some of these might contain a
157 // persistent or e.g. be RefCountedGarbageCollected we cannot guarantee there are no remaining Persistents at 189 // persistent or e.g. be RefCountedGarbageCollected we cannot guarantee there are no remaining Persistents at
158 // this point. 190 // this point.
159 } 191 }
160 PersistentAnchor() : PersistentNode(TraceMethodDelegate<PersistentAnchor, &P ersistentAnchor::trace>::trampoline) 192 PersistentAnchor() : PersistentNode(TraceMethodDelegate<PersistentAnchor, &P ersistentAnchor::trace>::trampoline)
161 { 193 {
162 m_next = this; 194 m_next = this;
163 m_prev = this; 195 m_prev = this;
164 } 196 }
165 197
166 friend class ThreadState; 198 friend class ThreadState;
167 }; 199 };
168 200
169 // Persistent handles are used to store pointers into the 201 // Persistent handles are used to store pointers into the
170 // managed heap. As long as the Persistent handle is alive 202 // managed heap. As long as the Persistent handle is alive
171 // the GC will keep the object pointed to alive. Persistent 203 // the GC will keep the object pointed to alive. Persistent
172 // handles can be stored in objects and they are not scoped. 204 // handles can be stored in objects and they are not scoped.
173 // Persistent handles must not be used to contain pointers 205 // Persistent handles must not be used to contain pointers
174 // between objects that are in the managed heap. They are only 206 // between objects that are in the managed heap. They are only
175 // meant to point to managed heap objects from variables/members 207 // meant to point to managed heap objects from variables/members
176 // outside the managed heap. 208 // outside the managed heap.
177 // 209 //
178 // A Persistent is always a GC root from the point of view of 210 // A Persistent is always a GC root from the point of view of
179 // the garbage collector. 211 // the garbage collector.
180 template<typename T> 212 template<typename T, typename RootsAccessor>
Mads Ager (chromium) 2014/02/17 15:03:43 Should we have a default template argument here?
Vyacheslav Egorov (Chromium) 2014/02/17 15:52:35 Forward declaration from Heap.h provides default a
181 class Persistent : public PersistentBase<ThreadingTrait<T>::Affinity, Persistent <T> > { 213 class Persistent : public PersistentBase<RootsAccessor, Persistent<T> > {
182 public: 214 public:
183 Persistent() : m_raw(0) { } 215 Persistent() : m_raw(0) { }
184 216
185 Persistent(T* raw) : m_raw(raw) { } 217 Persistent(T* raw) : m_raw(raw) { }
186 218
187 Persistent(const Persistent& other) : m_raw(other) { } 219 Persistent(const Persistent& other) : m_raw(other) { }
188 220
189 template<typename U> 221 template<typename U>
190 Persistent(const Persistent<U>& other) : m_raw(other) { } 222 Persistent(const Persistent<U>& other) : m_raw(other) { }
191 223
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after
262 T* get() const { return m_raw; } 294 T* get() const { return m_raw; }
263 295
264 private: 296 private:
265 T* m_raw; 297 T* m_raw;
266 }; 298 };
267 299
268 // FIXME: derive affinity based on the collection. 300 // FIXME: derive affinity based on the collection.
269 template<typename Collection, ThreadAffinity Affinity = AnyThread> 301 template<typename Collection, ThreadAffinity Affinity = AnyThread>
270 class PersistentHeapCollectionBase 302 class PersistentHeapCollectionBase
271 : public Collection 303 : public Collection
272 , public PersistentBase<Affinity, PersistentHeapCollectionBase<Collection, A ffinity> > { 304 , public PersistentBase<ThreadLocalPersistents<Affinity>, PersistentHeapColl ectionBase<Collection, Affinity> > {
273 // Never allocate these objects with new. Use Persistent<Collection> instead . 305 // Never allocate these objects with new. Use Persistent<Collection> instead .
274 DISALLOW_ALLOCATION(); 306 DISALLOW_ALLOCATION();
275 public: 307 public:
276 void trace(Visitor* visitor) { visitor->trace(*static_cast<Collection*>(this )); } 308 void trace(Visitor* visitor) { visitor->trace(*static_cast<Collection*>(this )); }
277 }; 309 };
278 310
279 template< 311 template<
280 typename KeyArg, 312 typename KeyArg,
281 typename MappedArg, 313 typename MappedArg,
282 typename HashArg = typename DefaultHash<KeyArg>::Hash, 314 typename HashArg = typename DefaultHash<KeyArg>::Hash,
(...skipping 186 matching lines...) Expand 10 before | Expand all | Expand 10 after
469 template<typename T, size_t inlineCapacity> 501 template<typename T, size_t inlineCapacity>
470 class IndexingBehavior<CollectionPersistent<Vector<T, inlineCapacity, WTF::Defau ltAllocator> > > { 502 class IndexingBehavior<CollectionPersistent<Vector<T, inlineCapacity, WTF::Defau ltAllocator> > > {
471 typedef CollectionPersistent<Vector<T, inlineCapacity, WTF::DefaultAllocator > > CollectionPersistentType; 503 typedef CollectionPersistent<Vector<T, inlineCapacity, WTF::DefaultAllocator > > CollectionPersistentType;
472 public: 504 public:
473 T& operator[] (size_t i) { return (**static_cast<CollectionPersistentType*>( this))[i]; } 505 T& operator[] (size_t i) { return (**static_cast<CollectionPersistentType*>( this))[i]; }
474 const T& operator[] (size_t i) const { return (**static_cast<const Collectio nPersistentType*>(this))[i]; } 506 const T& operator[] (size_t i) const { return (**static_cast<const Collectio nPersistentType*>(this))[i]; }
475 }; 507 };
476 508
477 template<typename Collection, ThreadAffinity Affinity> 509 template<typename Collection, ThreadAffinity Affinity>
478 class CollectionPersistent 510 class CollectionPersistent
479 : public PersistentBase<Affinity, CollectionPersistent<Collection, Affinity> > 511 : public PersistentBase<ThreadLocalPersistents<Affinity>, CollectionPersiste nt<Collection, Affinity> >
480 , public IndexingBehavior<CollectionPersistent<Collection, Affinity> > { 512 , public IndexingBehavior<CollectionPersistent<Collection, Affinity> > {
481 public: 513 public:
482 typedef Collection CollectionType; 514 typedef Collection CollectionType;
483 typedef typename Collection::iterator iterator; 515 typedef typename Collection::iterator iterator;
484 typedef typename Collection::const_iterator const_iterator; 516 typedef typename Collection::const_iterator const_iterator;
485 517
486 CollectionPersistent() { } 518 CollectionPersistent() { }
487 explicit CollectionPersistent(size_t size) : m_collection(Collection(size)) { } 519 explicit CollectionPersistent(size_t size) : m_collection(Collection(size)) { }
488 explicit CollectionPersistent(const Collection& collection) : m_collection(c ollection) { } 520 explicit CollectionPersistent(const Collection& collection) : m_collection(c ollection) { }
489 CollectionPersistent& operator=(const Collection& collection) { m_collection = collection; return *this; } 521 CollectionPersistent& operator=(const Collection& collection) { m_collection = collection; return *this; }
(...skipping 310 matching lines...) Expand 10 before | Expand all | Expand 10 after
800 }; 832 };
801 833
802 template<typename T> inline T* getPtr(const WebCore::Member<T>& p) 834 template<typename T> inline T* getPtr(const WebCore::Member<T>& p)
803 { 835 {
804 return p.get(); 836 return p.get();
805 } 837 }
806 838
807 } // namespace WTF 839 } // namespace WTF
808 840
809 #endif 841 #endif
OLDNEW
« no previous file with comments | « no previous file | Source/heap/Heap.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698