blob: f7758fdf4fcfed374698e35c8f68515e03c35cbb [file] [log] [blame]
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +00001// Copyright 2012 the V8 project authors. All rights reserved.
ager@chromium.orgea4f62e2010-08-16 16:28:43 +00002// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
ricow@chromium.orgeb7c1442010-10-04 08:54:21 +000028#ifndef V8_OBJECTS_VISITING_H_
29#define V8_OBJECTS_VISITING_H_
ager@chromium.orgea4f62e2010-08-16 16:28:43 +000030
lrn@chromium.org1c092762011-05-09 09:42:16 +000031#include "allocation.h"
32
ager@chromium.orgea4f62e2010-08-16 16:28:43 +000033// This file provides base classes and auxiliary methods for defining
34// static object visitors used during GC.
35// Visiting HeapObject body with a normal ObjectVisitor requires performing
36// two switches on object's instance type to determine object size and layout
37// and one or more virtual method calls on visitor itself.
38// Static visitor is different: it provides a dispatch table which contains
39// pointers to specialized visit functions. Each map has the visitor_id
40// field which contains an index of specialized visitor to use.
41
42namespace v8 {
43namespace internal {
44
45
46// Base class for all static visitors.
47class StaticVisitorBase : public AllStatic {
48 public:
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000049#define VISITOR_ID_LIST(V) \
yangguo@chromium.orgfb377212012-11-16 14:43:43 +000050 V(SeqOneByteString) \
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000051 V(SeqTwoByteString) \
52 V(ShortcutCandidate) \
53 V(ByteArray) \
54 V(FreeSpace) \
55 V(FixedArray) \
56 V(FixedDoubleArray) \
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +000057 V(ConstantPoolArray) \
yangguo@chromium.org46839fb2012-08-28 09:06:19 +000058 V(NativeContext) \
jkummerow@chromium.orgba72ec82013-07-22 09:21:20 +000059 V(AllocationSite) \
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000060 V(DataObject2) \
61 V(DataObject3) \
62 V(DataObject4) \
63 V(DataObject5) \
64 V(DataObject6) \
65 V(DataObject7) \
66 V(DataObject8) \
67 V(DataObject9) \
68 V(DataObjectGeneric) \
69 V(JSObject2) \
70 V(JSObject3) \
71 V(JSObject4) \
72 V(JSObject5) \
73 V(JSObject6) \
74 V(JSObject7) \
75 V(JSObject8) \
76 V(JSObject9) \
77 V(JSObjectGeneric) \
78 V(Struct2) \
79 V(Struct3) \
80 V(Struct4) \
81 V(Struct5) \
82 V(Struct6) \
83 V(Struct7) \
84 V(Struct8) \
85 V(Struct9) \
86 V(StructGeneric) \
87 V(ConsString) \
88 V(SlicedString) \
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000089 V(Symbol) \
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000090 V(Oddball) \
91 V(Code) \
92 V(Map) \
danno@chromium.org41728482013-06-12 22:31:22 +000093 V(Cell) \
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000094 V(PropertyCell) \
95 V(SharedFunctionInfo) \
96 V(JSFunction) \
97 V(JSWeakMap) \
jkummerow@chromium.orgba72ec82013-07-22 09:21:20 +000098 V(JSWeakSet) \
danno@chromium.org1fd77d52013-06-07 16:01:45 +000099 V(JSArrayBuffer) \
100 V(JSTypedArray) \
mstarzinger@chromium.org1510d582013-06-28 14:00:48 +0000101 V(JSDataView) \
jkummerow@chromium.org28583c92012-07-16 11:31:55 +0000102 V(JSRegExp)
103
104 // For data objects, JS objects and structs along with generic visitor which
105 // can visit object of any size we provide visitors specialized by
106 // object size in words.
107 // Ids of specialized visitors are declared in a linear order (without
108 // holes) starting from the id of visitor specialized for 2 words objects
109 // (base visitor id) and ending with the id of generic visitor.
110 // Method GetVisitorIdForSize depends on this ordering to calculate visitor
111 // id of specialized visitor from given instance size, base visitor id and
112 // generic visitor's id.
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000113 enum VisitorId {
jkummerow@chromium.org28583c92012-07-16 11:31:55 +0000114#define VISITOR_ID_ENUM_DECL(id) kVisit##id,
115 VISITOR_ID_LIST(VISITOR_ID_ENUM_DECL)
116#undef VISITOR_ID_ENUM_DECL
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000117 kVisitorIdCount,
jkummerow@chromium.org28583c92012-07-16 11:31:55 +0000118 kVisitDataObject = kVisitDataObject2,
119 kVisitJSObject = kVisitJSObject2,
120 kVisitStruct = kVisitStruct2,
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000121 kMinObjectSizeInWords = 2
122 };
123
ager@chromium.org5b2fbee2010-09-08 06:38:15 +0000124 // Visitor ID should fit in one byte.
125 STATIC_ASSERT(kVisitorIdCount <= 256);
126
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000127 // Determine which specialized visitor should be used for given instance type
128 // and instance type.
129 static VisitorId GetVisitorId(int instance_type, int instance_size);
130
131 static VisitorId GetVisitorId(Map* map) {
132 return GetVisitorId(map->instance_type(), map->instance_size());
133 }
134
135 // For visitors that allow specialization by size calculate VisitorId based
136 // on size, base visitor id and generic visitor id.
137 static VisitorId GetVisitorIdForSize(VisitorId base,
138 VisitorId generic,
139 int object_size) {
140 ASSERT((base == kVisitDataObject) ||
141 (base == kVisitStruct) ||
142 (base == kVisitJSObject));
143 ASSERT(IsAligned(object_size, kPointerSize));
144 ASSERT(kMinObjectSizeInWords * kPointerSize <= object_size);
danno@chromium.org59400602013-08-13 17:09:37 +0000145 ASSERT(object_size <= Page::kMaxNonCodeHeapObjectSize);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000146
147 const VisitorId specialization = static_cast<VisitorId>(
148 base + (object_size >> kPointerSizeLog2) - kMinObjectSizeInWords);
149
150 return Min(specialization, generic);
151 }
152};
153
154
155template<typename Callback>
156class VisitorDispatchTable {
157 public:
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000158 void CopyFrom(VisitorDispatchTable* other) {
159 // We are not using memcpy to guarantee that during update
160 // every element of callbacks_ array will remain correct
161 // pointer (memcpy might be implemented as a byte copying loop).
162 for (int i = 0; i < StaticVisitorBase::kVisitorIdCount; i++) {
163 NoBarrier_Store(&callbacks_[i], other->callbacks_[i]);
164 }
165 }
166
svenpanne@chromium.orga8bb4d92011-10-10 13:20:40 +0000167 inline Callback GetVisitorById(StaticVisitorBase::VisitorId id) {
168 return reinterpret_cast<Callback>(callbacks_[id]);
169 }
170
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000171 inline Callback GetVisitor(Map* map) {
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000172 return reinterpret_cast<Callback>(callbacks_[map->visitor_id()]);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000173 }
174
175 void Register(StaticVisitorBase::VisitorId id, Callback callback) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000176 ASSERT(id < StaticVisitorBase::kVisitorIdCount); // id is unsigned.
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000177 callbacks_[id] = reinterpret_cast<AtomicWord>(callback);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000178 }
179
180 template<typename Visitor,
181 StaticVisitorBase::VisitorId base,
182 StaticVisitorBase::VisitorId generic,
183 int object_size_in_words>
184 void RegisterSpecialization() {
185 static const int size = object_size_in_words * kPointerSize;
186 Register(StaticVisitorBase::GetVisitorIdForSize(base, generic, size),
187 &Visitor::template VisitSpecialized<size>);
188 }
189
190
191 template<typename Visitor,
192 StaticVisitorBase::VisitorId base,
193 StaticVisitorBase::VisitorId generic>
194 void RegisterSpecializations() {
195 STATIC_ASSERT(
196 (generic - base + StaticVisitorBase::kMinObjectSizeInWords) == 10);
197 RegisterSpecialization<Visitor, base, generic, 2>();
198 RegisterSpecialization<Visitor, base, generic, 3>();
199 RegisterSpecialization<Visitor, base, generic, 4>();
200 RegisterSpecialization<Visitor, base, generic, 5>();
201 RegisterSpecialization<Visitor, base, generic, 6>();
202 RegisterSpecialization<Visitor, base, generic, 7>();
203 RegisterSpecialization<Visitor, base, generic, 8>();
204 RegisterSpecialization<Visitor, base, generic, 9>();
205 Register(generic, &Visitor::Visit);
206 }
207
208 private:
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000209 AtomicWord callbacks_[StaticVisitorBase::kVisitorIdCount];
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000210};
211
212
213template<typename StaticVisitor>
214class BodyVisitorBase : public AllStatic {
215 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000216 INLINE(static void IteratePointers(Heap* heap,
217 HeapObject* object,
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000218 int start_offset,
fschneider@chromium.org9e3e0b62011-01-03 10:16:46 +0000219 int end_offset)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000220 Object** start_slot = reinterpret_cast<Object**>(object->address() +
221 start_offset);
222 Object** end_slot = reinterpret_cast<Object**>(object->address() +
223 end_offset);
danno@chromium.org2ab0c3b2012-10-05 08:50:56 +0000224 StaticVisitor::VisitPointers(heap, start_slot, end_slot);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000225 }
226};
227
228
229template<typename StaticVisitor, typename BodyDescriptor, typename ReturnType>
230class FlexibleBodyVisitor : public BodyVisitorBase<StaticVisitor> {
231 public:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000232 INLINE(static ReturnType Visit(Map* map, HeapObject* object)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000233 int object_size = BodyDescriptor::SizeOf(map, object);
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000234 BodyVisitorBase<StaticVisitor>::IteratePointers(
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000235 map->GetHeap(),
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000236 object,
237 BodyDescriptor::kStartOffset,
238 object_size);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000239 return static_cast<ReturnType>(object_size);
240 }
241
242 template<int object_size>
243 static inline ReturnType VisitSpecialized(Map* map, HeapObject* object) {
ricow@chromium.org0b9f8502010-08-18 07:45:01 +0000244 ASSERT(BodyDescriptor::SizeOf(map, object) == object_size);
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000245 BodyVisitorBase<StaticVisitor>::IteratePointers(
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000246 map->GetHeap(),
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000247 object,
248 BodyDescriptor::kStartOffset,
249 object_size);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000250 return static_cast<ReturnType>(object_size);
251 }
252};
253
254
255template<typename StaticVisitor, typename BodyDescriptor, typename ReturnType>
256class FixedBodyVisitor : public BodyVisitorBase<StaticVisitor> {
257 public:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000258 INLINE(static ReturnType Visit(Map* map, HeapObject* object)) {
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000259 BodyVisitorBase<StaticVisitor>::IteratePointers(
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000260 map->GetHeap(),
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000261 object,
262 BodyDescriptor::kStartOffset,
263 BodyDescriptor::kEndOffset);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000264 return static_cast<ReturnType>(BodyDescriptor::kSize);
265 }
266};
267
268
269// Base class for visitors used for a linear new space iteration.
270// IterateBody returns size of visited object.
271// Certain types of objects (i.e. Code objects) are not handled
272// by dispatch table of this visitor because they cannot appear
273// in the new space.
274//
275// This class is intended to be used in the following way:
276//
277// class SomeVisitor : public StaticNewSpaceVisitor<SomeVisitor> {
278// ...
279// }
280//
281// This is an example of Curiously recurring template pattern
282// (see http://en.wikipedia.org/wiki/Curiously_recurring_template_pattern).
283// We use CRTP to guarantee aggressive compile time optimizations (i.e.
284// inlining and specialization of StaticVisitor::VisitPointers methods).
285template<typename StaticVisitor>
286class StaticNewSpaceVisitor : public StaticVisitorBase {
287 public:
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000288 static void Initialize();
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000289
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000290 INLINE(static int IterateBody(Map* map, HeapObject* obj)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000291 return table_.GetVisitor(map)(map, obj);
292 }
293
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000294 INLINE(static void VisitPointers(Heap* heap, Object** start, Object** end)) {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000295 for (Object** p = start; p < end; p++) StaticVisitor::VisitPointer(heap, p);
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000296 }
297
298 private:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000299 INLINE(static int VisitJSFunction(Map* map, HeapObject* object)) {
erik.corry@gmail.comed49e962012-04-17 11:57:53 +0000300 Heap* heap = map->GetHeap();
301 VisitPointers(heap,
danno@chromium.org2ab0c3b2012-10-05 08:50:56 +0000302 HeapObject::RawField(object, JSFunction::kPropertiesOffset),
erik.corry@gmail.comed49e962012-04-17 11:57:53 +0000303 HeapObject::RawField(object, JSFunction::kCodeEntryOffset));
304
305 // Don't visit code entry. We are using this visitor only during scavenges.
306
307 VisitPointers(
308 heap,
309 HeapObject::RawField(object,
310 JSFunction::kCodeEntryOffset + kPointerSize),
311 HeapObject::RawField(object,
312 JSFunction::kNonWeakFieldsEndOffset));
313 return JSFunction::kSize;
314 }
315
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000316 INLINE(static int VisitByteArray(Map* map, HeapObject* object)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000317 return reinterpret_cast<ByteArray*>(object)->ByteArraySize();
318 }
319
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000320 INLINE(static int VisitFixedDoubleArray(Map* map, HeapObject* object)) {
svenpanne@chromium.org6d786c92011-06-15 10:58:27 +0000321 int length = reinterpret_cast<FixedDoubleArray*>(object)->length();
322 return FixedDoubleArray::SizeFor(length);
323 }
324
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000325 INLINE(static int VisitJSObject(Map* map, HeapObject* object)) {
kmillikin@chromium.org7c2628c2011-08-10 11:27:35 +0000326 return JSObjectVisitor::Visit(map, object);
327 }
328
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000329 INLINE(static int VisitSeqOneByteString(Map* map, HeapObject* object)) {
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000330 return SeqOneByteString::cast(object)->
331 SeqOneByteStringSize(map->instance_type());
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000332 }
333
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000334 INLINE(static int VisitSeqTwoByteString(Map* map, HeapObject* object)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000335 return SeqTwoByteString::cast(object)->
336 SeqTwoByteStringSize(map->instance_type());
337 }
338
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000339 INLINE(static int VisitFreeSpace(Map* map, HeapObject* object)) {
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000340 return FreeSpace::cast(object)->Size();
341 }
342
danno@chromium.org1fd77d52013-06-07 16:01:45 +0000343 INLINE(static int VisitJSArrayBuffer(Map* map, HeapObject* object));
344 INLINE(static int VisitJSTypedArray(Map* map, HeapObject* object));
mstarzinger@chromium.org1510d582013-06-28 14:00:48 +0000345 INLINE(static int VisitJSDataView(Map* map, HeapObject* object));
danno@chromium.org1fd77d52013-06-07 16:01:45 +0000346
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000347 class DataObjectVisitor {
348 public:
349 template<int object_size>
350 static inline int VisitSpecialized(Map* map, HeapObject* object) {
351 return object_size;
352 }
353
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000354 INLINE(static int Visit(Map* map, HeapObject* object)) {
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000355 return map->instance_size();
356 }
357 };
358
359 typedef FlexibleBodyVisitor<StaticVisitor,
360 StructBodyDescriptor,
361 int> StructVisitor;
362
363 typedef FlexibleBodyVisitor<StaticVisitor,
364 JSObject::BodyDescriptor,
365 int> JSObjectVisitor;
366
367 typedef int (*Callback)(Map* map, HeapObject* object);
368
369 static VisitorDispatchTable<Callback> table_;
370};
371
372
373template<typename StaticVisitor>
374VisitorDispatchTable<typename StaticNewSpaceVisitor<StaticVisitor>::Callback>
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000375 StaticNewSpaceVisitor<StaticVisitor>::table_;
376
377
378// Base class for visitors used to transitively mark the entire heap.
379// IterateBody returns nothing.
380// Certain types of objects might not be handled by this base class and
381// no visitor function is registered by the generic initialization. A
382// specialized visitor function needs to be provided by the inheriting
383// class itself for those cases.
384//
385// This class is intended to be used in the following way:
386//
387// class SomeVisitor : public StaticMarkingVisitor<SomeVisitor> {
388// ...
389// }
390//
391// This is an example of Curiously recurring template pattern.
392template<typename StaticVisitor>
393class StaticMarkingVisitor : public StaticVisitorBase {
394 public:
395 static void Initialize();
396
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000397 INLINE(static void IterateBody(Map* map, HeapObject* obj)) {
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000398 table_.GetVisitor(map)(map, obj);
399 }
400
mstarzinger@chromium.org1510d582013-06-28 14:00:48 +0000401 INLINE(static void VisitPropertyCell(Map* map, HeapObject* object));
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000402 INLINE(static void VisitAllocationSite(Map* map, HeapObject* object));
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000403 INLINE(static void VisitCodeEntry(Heap* heap, Address entry_address));
404 INLINE(static void VisitEmbeddedPointer(Heap* heap, RelocInfo* rinfo));
danno@chromium.org41728482013-06-12 22:31:22 +0000405 INLINE(static void VisitCell(Heap* heap, RelocInfo* rinfo));
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000406 INLINE(static void VisitDebugTarget(Heap* heap, RelocInfo* rinfo));
407 INLINE(static void VisitCodeTarget(Heap* heap, RelocInfo* rinfo));
408 INLINE(static void VisitCodeAgeSequence(Heap* heap, RelocInfo* rinfo));
409 INLINE(static void VisitExternalReference(RelocInfo* rinfo)) { }
410 INLINE(static void VisitRuntimeEntry(RelocInfo* rinfo)) { }
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000411
412 // TODO(mstarzinger): This should be made protected once refactoring is done.
svenpanne@chromium.orgc859c4f2012-10-15 11:51:39 +0000413 // Mark non-optimize code for functions inlined into the given optimized
414 // code. This will prevent it from being flushed.
415 static void MarkInlinedFunctionsCode(Heap* heap, Code* code);
416
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000417 protected:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000418 INLINE(static void VisitMap(Map* map, HeapObject* object));
419 INLINE(static void VisitCode(Map* map, HeapObject* object));
420 INLINE(static void VisitSharedFunctionInfo(Map* map, HeapObject* object));
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +0000421 INLINE(static void VisitConstantPoolArray(Map* map, HeapObject* object));
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000422 INLINE(static void VisitJSFunction(Map* map, HeapObject* object));
423 INLINE(static void VisitJSRegExp(Map* map, HeapObject* object));
danno@chromium.org1fd77d52013-06-07 16:01:45 +0000424 INLINE(static void VisitJSArrayBuffer(Map* map, HeapObject* object));
425 INLINE(static void VisitJSTypedArray(Map* map, HeapObject* object));
mstarzinger@chromium.org1510d582013-06-28 14:00:48 +0000426 INLINE(static void VisitJSDataView(Map* map, HeapObject* object));
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000427 INLINE(static void VisitNativeContext(Map* map, HeapObject* object));
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000428
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000429 // Mark pointers in a Map and its TransitionArray together, possibly
430 // treating transitions or back pointers weak.
431 static void MarkMapContents(Heap* heap, Map* map);
432 static void MarkTransitionArray(Heap* heap, TransitionArray* transitions);
433
svenpanne@chromium.orgc859c4f2012-10-15 11:51:39 +0000434 // Code flushing support.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000435 INLINE(static bool IsFlushable(Heap* heap, JSFunction* function));
436 INLINE(static bool IsFlushable(Heap* heap, SharedFunctionInfo* shared_info));
svenpanne@chromium.orgc859c4f2012-10-15 11:51:39 +0000437
438 // Helpers used by code flushing support that visit pointer fields and treat
439 // references to code objects either strongly or weakly.
440 static void VisitSharedFunctionInfoStrongCode(Heap* heap, HeapObject* object);
441 static void VisitSharedFunctionInfoWeakCode(Heap* heap, HeapObject* object);
442 static void VisitJSFunctionStrongCode(Heap* heap, HeapObject* object);
443 static void VisitJSFunctionWeakCode(Heap* heap, HeapObject* object);
444
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000445 class DataObjectVisitor {
446 public:
447 template<int size>
448 static inline void VisitSpecialized(Map* map, HeapObject* object) {
449 }
450
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000451 INLINE(static void Visit(Map* map, HeapObject* object)) {
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000452 }
453 };
454
455 typedef FlexibleBodyVisitor<StaticVisitor,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000456 FixedArray::BodyDescriptor,
457 void> FixedArrayVisitor;
458
459 typedef FlexibleBodyVisitor<StaticVisitor,
verwaest@chromium.orgb6d052d2012-07-27 08:03:27 +0000460 JSObject::BodyDescriptor,
461 void> JSObjectVisitor;
462
463 typedef FlexibleBodyVisitor<StaticVisitor,
464 StructBodyDescriptor,
465 void> StructObjectVisitor;
466
467 typedef void (*Callback)(Map* map, HeapObject* object);
468
469 static VisitorDispatchTable<Callback> table_;
470};
471
472
473template<typename StaticVisitor>
474VisitorDispatchTable<typename StaticMarkingVisitor<StaticVisitor>::Callback>
475 StaticMarkingVisitor<StaticVisitor>::table_;
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000476
477
ager@chromium.orgea4f62e2010-08-16 16:28:43 +0000478} } // namespace v8::internal
479
ricow@chromium.orgeb7c1442010-10-04 08:54:21 +0000480#endif // V8_OBJECTS_VISITING_H_