blob: c1caef2d98b0d79cbd4b6bb2785f9f13842492de [file] [log] [blame]
Steve Block9fac8402011-05-12 15:51:54 +01001// Copyright 2010 the V8 project authors. All rights reserved.
Steve Blocka7e24c12009-10-30 11:49:00 +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
28#include "v8.h"
29
30#include "disassembler.h"
31#include "disasm.h"
Steve Blocka7e24c12009-10-30 11:49:00 +000032#include "jsregexp.h"
Kristian Monsen80d68ea2010-09-08 11:05:35 +010033#include "objects-visiting.h"
Steve Blocka7e24c12009-10-30 11:49:00 +000034
35namespace v8 {
36namespace internal {
37
Ben Murdochb0fe1622011-05-05 13:52:32 +010038#ifdef DEBUG
Steve Block9fac8402011-05-12 15:51:54 +010039
John Reck59135872010-11-02 12:39:01 -070040void MaybeObject::Verify() {
41 Object* this_as_object;
42 if (ToObject(&this_as_object)) {
43 if (this_as_object->IsSmi()) {
44 Smi::cast(this_as_object)->SmiVerify();
45 } else {
46 HeapObject::cast(this_as_object)->HeapObjectVerify();
47 }
Steve Blocka7e24c12009-10-30 11:49:00 +000048 } else {
John Reck59135872010-11-02 12:39:01 -070049 Failure::cast(this)->FailureVerify();
Steve Blocka7e24c12009-10-30 11:49:00 +000050 }
51}
52
53
54void Object::VerifyPointer(Object* p) {
55 if (p->IsHeapObject()) {
56 HeapObject::VerifyHeapPointer(p);
57 } else {
58 ASSERT(p->IsSmi());
59 }
60}
61
62
63void Smi::SmiVerify() {
64 ASSERT(IsSmi());
65}
66
67
68void Failure::FailureVerify() {
69 ASSERT(IsFailure());
70}
71
72
Steve Blocka7e24c12009-10-30 11:49:00 +000073void HeapObject::HeapObjectVerify() {
74 InstanceType instance_type = map()->instance_type();
75
76 if (instance_type < FIRST_NONSTRING_TYPE) {
77 String::cast(this)->StringVerify();
78 return;
79 }
80
81 switch (instance_type) {
82 case MAP_TYPE:
83 Map::cast(this)->MapVerify();
84 break;
85 case HEAP_NUMBER_TYPE:
86 HeapNumber::cast(this)->HeapNumberVerify();
87 break;
88 case FIXED_ARRAY_TYPE:
89 FixedArray::cast(this)->FixedArrayVerify();
90 break;
91 case BYTE_ARRAY_TYPE:
92 ByteArray::cast(this)->ByteArrayVerify();
93 break;
94 case PIXEL_ARRAY_TYPE:
95 PixelArray::cast(this)->PixelArrayVerify();
96 break;
Steve Block3ce2e202009-11-05 08:53:23 +000097 case EXTERNAL_BYTE_ARRAY_TYPE:
98 ExternalByteArray::cast(this)->ExternalByteArrayVerify();
99 break;
100 case EXTERNAL_UNSIGNED_BYTE_ARRAY_TYPE:
101 ExternalUnsignedByteArray::cast(this)->ExternalUnsignedByteArrayVerify();
102 break;
103 case EXTERNAL_SHORT_ARRAY_TYPE:
104 ExternalShortArray::cast(this)->ExternalShortArrayVerify();
105 break;
106 case EXTERNAL_UNSIGNED_SHORT_ARRAY_TYPE:
107 ExternalUnsignedShortArray::cast(this)->
108 ExternalUnsignedShortArrayVerify();
109 break;
110 case EXTERNAL_INT_ARRAY_TYPE:
111 ExternalIntArray::cast(this)->ExternalIntArrayVerify();
112 break;
113 case EXTERNAL_UNSIGNED_INT_ARRAY_TYPE:
114 ExternalUnsignedIntArray::cast(this)->ExternalUnsignedIntArrayVerify();
115 break;
116 case EXTERNAL_FLOAT_ARRAY_TYPE:
117 ExternalFloatArray::cast(this)->ExternalFloatArrayVerify();
118 break;
Steve Blocka7e24c12009-10-30 11:49:00 +0000119 case CODE_TYPE:
120 Code::cast(this)->CodeVerify();
121 break;
122 case ODDBALL_TYPE:
123 Oddball::cast(this)->OddballVerify();
124 break;
125 case JS_OBJECT_TYPE:
126 case JS_CONTEXT_EXTENSION_OBJECT_TYPE:
127 JSObject::cast(this)->JSObjectVerify();
128 break;
129 case JS_VALUE_TYPE:
130 JSValue::cast(this)->JSValueVerify();
131 break;
132 case JS_FUNCTION_TYPE:
133 JSFunction::cast(this)->JSFunctionVerify();
134 break;
135 case JS_GLOBAL_PROXY_TYPE:
136 JSGlobalProxy::cast(this)->JSGlobalProxyVerify();
137 break;
138 case JS_GLOBAL_OBJECT_TYPE:
139 JSGlobalObject::cast(this)->JSGlobalObjectVerify();
140 break;
141 case JS_BUILTINS_OBJECT_TYPE:
142 JSBuiltinsObject::cast(this)->JSBuiltinsObjectVerify();
143 break;
144 case JS_GLOBAL_PROPERTY_CELL_TYPE:
145 JSGlobalPropertyCell::cast(this)->JSGlobalPropertyCellVerify();
146 break;
147 case JS_ARRAY_TYPE:
148 JSArray::cast(this)->JSArrayVerify();
149 break;
150 case JS_REGEXP_TYPE:
151 JSRegExp::cast(this)->JSRegExpVerify();
152 break;
153 case FILLER_TYPE:
154 break;
155 case PROXY_TYPE:
156 Proxy::cast(this)->ProxyVerify();
157 break;
158 case SHARED_FUNCTION_INFO_TYPE:
159 SharedFunctionInfo::cast(this)->SharedFunctionInfoVerify();
160 break;
Steve Block1e0659c2011-05-24 12:43:12 +0100161 case JS_MESSAGE_OBJECT_TYPE:
162 JSMessageObject::cast(this)->JSMessageObjectVerify();
163 break;
Steve Blocka7e24c12009-10-30 11:49:00 +0000164
165#define MAKE_STRUCT_CASE(NAME, Name, name) \
166 case NAME##_TYPE: \
167 Name::cast(this)->Name##Verify(); \
168 break;
169 STRUCT_LIST(MAKE_STRUCT_CASE)
170#undef MAKE_STRUCT_CASE
171
172 default:
173 UNREACHABLE();
174 break;
175 }
176}
177
178
179void HeapObject::VerifyHeapPointer(Object* p) {
180 ASSERT(p->IsHeapObject());
181 ASSERT(Heap::Contains(HeapObject::cast(p)));
182}
183
184
185void HeapNumber::HeapNumberVerify() {
186 ASSERT(IsHeapNumber());
187}
188
189
Steve Blocka7e24c12009-10-30 11:49:00 +0000190void ByteArray::ByteArrayVerify() {
191 ASSERT(IsByteArray());
192}
193
194
195void PixelArray::PixelArrayVerify() {
196 ASSERT(IsPixelArray());
197}
198
199
Steve Block3ce2e202009-11-05 08:53:23 +0000200void ExternalByteArray::ExternalByteArrayVerify() {
201 ASSERT(IsExternalByteArray());
202}
203
204
205void ExternalUnsignedByteArray::ExternalUnsignedByteArrayVerify() {
206 ASSERT(IsExternalUnsignedByteArray());
207}
208
209
210void ExternalShortArray::ExternalShortArrayVerify() {
211 ASSERT(IsExternalShortArray());
212}
213
214
215void ExternalUnsignedShortArray::ExternalUnsignedShortArrayVerify() {
216 ASSERT(IsExternalUnsignedShortArray());
217}
218
219
220void ExternalIntArray::ExternalIntArrayVerify() {
221 ASSERT(IsExternalIntArray());
222}
223
224
225void ExternalUnsignedIntArray::ExternalUnsignedIntArrayVerify() {
226 ASSERT(IsExternalUnsignedIntArray());
227}
228
229
230void ExternalFloatArray::ExternalFloatArrayVerify() {
231 ASSERT(IsExternalFloatArray());
232}
233
234
Steve Blocka7e24c12009-10-30 11:49:00 +0000235void JSObject::JSObjectVerify() {
236 VerifyHeapPointer(properties());
237 VerifyHeapPointer(elements());
238 if (HasFastProperties()) {
239 CHECK_EQ(map()->unused_property_fields(),
240 (map()->inobject_properties() + properties()->length() -
241 map()->NextFreePropertyIndex()));
242 }
Steve Block8defd9f2010-07-08 12:39:36 +0100243 ASSERT(map()->has_fast_elements() ==
Iain Merrick75681382010-08-19 15:07:18 +0100244 (elements()->map() == Heap::fixed_array_map() ||
245 elements()->map() == Heap::fixed_cow_array_map()));
Steve Block8defd9f2010-07-08 12:39:36 +0100246 ASSERT(map()->has_fast_elements() == HasFastElements());
Steve Blocka7e24c12009-10-30 11:49:00 +0000247}
248
249
Steve Blocka7e24c12009-10-30 11:49:00 +0000250void Map::MapVerify() {
251 ASSERT(!Heap::InNewSpace(this));
252 ASSERT(FIRST_TYPE <= instance_type() && instance_type() <= LAST_TYPE);
Steve Block791712a2010-08-27 10:21:07 +0100253 ASSERT(instance_size() == kVariableSizeSentinel ||
254 (kPointerSize <= instance_size() &&
255 instance_size() < Heap::Capacity()));
Steve Blocka7e24c12009-10-30 11:49:00 +0000256 VerifyHeapPointer(prototype());
257 VerifyHeapPointer(instance_descriptors());
258}
259
260
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100261void Map::SharedMapVerify() {
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100262 MapVerify();
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100263 ASSERT(is_shared());
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100264 ASSERT_EQ(Heap::empty_descriptor_array(), instance_descriptors());
265 ASSERT_EQ(Heap::empty_fixed_array(), code_cache());
266 ASSERT_EQ(0, pre_allocated_property_fields());
267 ASSERT_EQ(0, unused_property_fields());
268 ASSERT_EQ(StaticVisitorBase::GetVisitorId(instance_type(), instance_size()),
269 visitor_id());
270}
271
272
Steve Block6ded16b2010-05-10 14:33:55 +0100273void CodeCache::CodeCacheVerify() {
274 VerifyHeapPointer(default_cache());
275 VerifyHeapPointer(normal_type_cache());
276 ASSERT(default_cache()->IsFixedArray());
277 ASSERT(normal_type_cache()->IsUndefined()
278 || normal_type_cache()->IsCodeCacheHashTable());
279}
280
281
Steve Blocka7e24c12009-10-30 11:49:00 +0000282void FixedArray::FixedArrayVerify() {
283 for (int i = 0; i < length(); i++) {
284 Object* e = get(i);
285 if (e->IsHeapObject()) {
286 VerifyHeapPointer(e);
287 } else {
288 e->Verify();
289 }
290 }
291}
292
293
Steve Blocka7e24c12009-10-30 11:49:00 +0000294void JSValue::JSValueVerify() {
295 Object* v = value();
296 if (v->IsHeapObject()) {
297 VerifyHeapPointer(v);
298 }
299}
300
301
Steve Block1e0659c2011-05-24 12:43:12 +0100302void JSMessageObject::JSMessageObjectVerify() {
303 CHECK(IsJSMessageObject());
304 CHECK(type()->IsString());
305 CHECK(arguments()->IsJSArray());
306 VerifyObjectField(kStartPositionOffset);
307 VerifyObjectField(kEndPositionOffset);
308 VerifyObjectField(kArgumentsOffset);
309 VerifyObjectField(kScriptOffset);
310 VerifyObjectField(kStackTraceOffset);
311 VerifyObjectField(kStackFramesOffset);
312}
313
314
Steve Blocka7e24c12009-10-30 11:49:00 +0000315void String::StringVerify() {
316 CHECK(IsString());
317 CHECK(length() >= 0 && length() <= Smi::kMaxValue);
318 if (IsSymbol()) {
319 CHECK(!Heap::InNewSpace(this));
320 }
321}
322
323
Steve Blocka7e24c12009-10-30 11:49:00 +0000324void JSFunction::JSFunctionVerify() {
325 CHECK(IsJSFunction());
326 VerifyObjectField(kPrototypeOrInitialMapOffset);
Ben Murdochb0fe1622011-05-05 13:52:32 +0100327 VerifyObjectField(kNextFunctionLinkOffset);
328 CHECK(next_function_link()->IsUndefined() ||
329 next_function_link()->IsJSFunction());
Steve Blocka7e24c12009-10-30 11:49:00 +0000330}
331
332
Steve Blocka7e24c12009-10-30 11:49:00 +0000333void SharedFunctionInfo::SharedFunctionInfoVerify() {
334 CHECK(IsSharedFunctionInfo());
335 VerifyObjectField(kNameOffset);
336 VerifyObjectField(kCodeOffset);
Ben Murdoch3bec4d22010-07-22 14:51:16 +0100337 VerifyObjectField(kScopeInfoOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +0000338 VerifyObjectField(kInstanceClassNameOffset);
Steve Block6ded16b2010-05-10 14:33:55 +0100339 VerifyObjectField(kFunctionDataOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +0000340 VerifyObjectField(kScriptOffset);
341 VerifyObjectField(kDebugInfoOffset);
342}
343
344
Steve Blocka7e24c12009-10-30 11:49:00 +0000345void JSGlobalProxy::JSGlobalProxyVerify() {
346 CHECK(IsJSGlobalProxy());
347 JSObjectVerify();
348 VerifyObjectField(JSGlobalProxy::kContextOffset);
349 // Make sure that this object has no properties, elements.
350 CHECK_EQ(0, properties()->length());
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +0100351 CHECK(HasFastElements());
352 CHECK_EQ(0, FixedArray::cast(elements())->length());
Steve Blocka7e24c12009-10-30 11:49:00 +0000353}
354
355
Steve Blocka7e24c12009-10-30 11:49:00 +0000356void JSGlobalObject::JSGlobalObjectVerify() {
357 CHECK(IsJSGlobalObject());
358 JSObjectVerify();
359 for (int i = GlobalObject::kBuiltinsOffset;
360 i < JSGlobalObject::kSize;
361 i += kPointerSize) {
362 VerifyObjectField(i);
363 }
364}
365
366
Steve Blocka7e24c12009-10-30 11:49:00 +0000367void JSBuiltinsObject::JSBuiltinsObjectVerify() {
368 CHECK(IsJSBuiltinsObject());
369 JSObjectVerify();
370 for (int i = GlobalObject::kBuiltinsOffset;
371 i < JSBuiltinsObject::kSize;
372 i += kPointerSize) {
373 VerifyObjectField(i);
374 }
375}
376
377
378void Oddball::OddballVerify() {
379 CHECK(IsOddball());
380 VerifyHeapPointer(to_string());
381 Object* number = to_number();
382 if (number->IsHeapObject()) {
383 ASSERT(number == Heap::nan_value());
384 } else {
385 ASSERT(number->IsSmi());
386 int value = Smi::cast(number)->value();
Ben Murdoch086aeea2011-05-13 15:57:08 +0100387 // Hidden oddballs have negative smis.
388 const int kLeastHiddenOddballNumber = -4;
389 ASSERT(value <= 1);
390 ASSERT(value >= kLeastHiddenOddballNumber);
Steve Blocka7e24c12009-10-30 11:49:00 +0000391 }
392}
393
394
395void JSGlobalPropertyCell::JSGlobalPropertyCellVerify() {
396 CHECK(IsJSGlobalPropertyCell());
397 VerifyObjectField(kValueOffset);
398}
399
400
Steve Blocka7e24c12009-10-30 11:49:00 +0000401void Code::CodeVerify() {
402 CHECK(IsAligned(reinterpret_cast<intptr_t>(instruction_start()),
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100403 kCodeAlignment));
Steve Blocka7e24c12009-10-30 11:49:00 +0000404 Address last_gc_pc = NULL;
405 for (RelocIterator it(this); !it.done(); it.next()) {
406 it.rinfo()->Verify();
407 // Ensure that GC will not iterate twice over the same pointer.
408 if (RelocInfo::IsGCRelocMode(it.rinfo()->rmode())) {
409 CHECK(it.rinfo()->pc() != last_gc_pc);
410 last_gc_pc = it.rinfo()->pc();
411 }
412 }
413}
414
415
416void JSArray::JSArrayVerify() {
417 JSObjectVerify();
418 ASSERT(length()->IsNumber() || length()->IsUndefined());
419 ASSERT(elements()->IsUndefined() || elements()->IsFixedArray());
420}
421
422
423void JSRegExp::JSRegExpVerify() {
424 JSObjectVerify();
425 ASSERT(data()->IsUndefined() || data()->IsFixedArray());
426 switch (TypeTag()) {
427 case JSRegExp::ATOM: {
428 FixedArray* arr = FixedArray::cast(data());
429 ASSERT(arr->get(JSRegExp::kAtomPatternIndex)->IsString());
430 break;
431 }
432 case JSRegExp::IRREGEXP: {
433 bool is_native = RegExpImpl::UsesNativeRegExp();
434
435 FixedArray* arr = FixedArray::cast(data());
436 Object* ascii_data = arr->get(JSRegExp::kIrregexpASCIICodeIndex);
437 // TheHole : Not compiled yet.
438 // JSObject: Compilation error.
439 // Code/ByteArray: Compiled code.
440 ASSERT(ascii_data->IsTheHole() || ascii_data->IsJSObject() ||
441 (is_native ? ascii_data->IsCode() : ascii_data->IsByteArray()));
442 Object* uc16_data = arr->get(JSRegExp::kIrregexpUC16CodeIndex);
Steve Block1e0659c2011-05-24 12:43:12 +0100443 ASSERT(uc16_data->IsTheHole() || uc16_data->IsJSObject() ||
Steve Blocka7e24c12009-10-30 11:49:00 +0000444 (is_native ? uc16_data->IsCode() : uc16_data->IsByteArray()));
445 ASSERT(arr->get(JSRegExp::kIrregexpCaptureCountIndex)->IsSmi());
446 ASSERT(arr->get(JSRegExp::kIrregexpMaxRegisterCountIndex)->IsSmi());
447 break;
448 }
449 default:
450 ASSERT_EQ(JSRegExp::NOT_COMPILED, TypeTag());
451 ASSERT(data()->IsUndefined());
452 break;
453 }
454}
455
456
Steve Blocka7e24c12009-10-30 11:49:00 +0000457void Proxy::ProxyVerify() {
458 ASSERT(IsProxy());
459}
460
461
462void AccessorInfo::AccessorInfoVerify() {
463 CHECK(IsAccessorInfo());
464 VerifyPointer(getter());
465 VerifyPointer(setter());
466 VerifyPointer(name());
467 VerifyPointer(data());
468 VerifyPointer(flag());
469}
470
Ben Murdochb0fe1622011-05-05 13:52:32 +0100471
Steve Blocka7e24c12009-10-30 11:49:00 +0000472void AccessCheckInfo::AccessCheckInfoVerify() {
473 CHECK(IsAccessCheckInfo());
474 VerifyPointer(named_callback());
475 VerifyPointer(indexed_callback());
476 VerifyPointer(data());
477}
478
Ben Murdochb0fe1622011-05-05 13:52:32 +0100479
Steve Blocka7e24c12009-10-30 11:49:00 +0000480void InterceptorInfo::InterceptorInfoVerify() {
481 CHECK(IsInterceptorInfo());
482 VerifyPointer(getter());
483 VerifyPointer(setter());
484 VerifyPointer(query());
485 VerifyPointer(deleter());
486 VerifyPointer(enumerator());
487 VerifyPointer(data());
488}
489
Ben Murdochb0fe1622011-05-05 13:52:32 +0100490
Steve Blocka7e24c12009-10-30 11:49:00 +0000491void CallHandlerInfo::CallHandlerInfoVerify() {
492 CHECK(IsCallHandlerInfo());
493 VerifyPointer(callback());
494 VerifyPointer(data());
495}
496
Ben Murdochb0fe1622011-05-05 13:52:32 +0100497
Steve Blocka7e24c12009-10-30 11:49:00 +0000498void TemplateInfo::TemplateInfoVerify() {
499 VerifyPointer(tag());
500 VerifyPointer(property_list());
501}
502
503void FunctionTemplateInfo::FunctionTemplateInfoVerify() {
504 CHECK(IsFunctionTemplateInfo());
505 TemplateInfoVerify();
506 VerifyPointer(serial_number());
507 VerifyPointer(call_code());
508 VerifyPointer(property_accessors());
509 VerifyPointer(prototype_template());
510 VerifyPointer(parent_template());
511 VerifyPointer(named_property_handler());
512 VerifyPointer(indexed_property_handler());
513 VerifyPointer(instance_template());
514 VerifyPointer(signature());
515 VerifyPointer(access_check_info());
516}
517
Ben Murdochb0fe1622011-05-05 13:52:32 +0100518
Steve Blocka7e24c12009-10-30 11:49:00 +0000519void ObjectTemplateInfo::ObjectTemplateInfoVerify() {
520 CHECK(IsObjectTemplateInfo());
521 TemplateInfoVerify();
522 VerifyPointer(constructor());
523 VerifyPointer(internal_field_count());
524}
525
Ben Murdochb0fe1622011-05-05 13:52:32 +0100526
Steve Blocka7e24c12009-10-30 11:49:00 +0000527void SignatureInfo::SignatureInfoVerify() {
528 CHECK(IsSignatureInfo());
529 VerifyPointer(receiver());
530 VerifyPointer(args());
531}
532
Ben Murdochb0fe1622011-05-05 13:52:32 +0100533
Steve Blocka7e24c12009-10-30 11:49:00 +0000534void TypeSwitchInfo::TypeSwitchInfoVerify() {
535 CHECK(IsTypeSwitchInfo());
536 VerifyPointer(types());
537}
538
Ben Murdochb0fe1622011-05-05 13:52:32 +0100539
Steve Blocka7e24c12009-10-30 11:49:00 +0000540void Script::ScriptVerify() {
541 CHECK(IsScript());
542 VerifyPointer(source());
543 VerifyPointer(name());
544 line_offset()->SmiVerify();
545 column_offset()->SmiVerify();
546 VerifyPointer(data());
547 VerifyPointer(wrapper());
548 type()->SmiVerify();
549 VerifyPointer(line_ends());
550 VerifyPointer(id());
551}
Steve Blocka7e24c12009-10-30 11:49:00 +0000552
553
554#ifdef ENABLE_DEBUGGER_SUPPORT
555void DebugInfo::DebugInfoVerify() {
556 CHECK(IsDebugInfo());
557 VerifyPointer(shared());
558 VerifyPointer(original_code());
559 VerifyPointer(code());
560 VerifyPointer(break_points());
561}
562
563
Steve Blocka7e24c12009-10-30 11:49:00 +0000564void BreakPointInfo::BreakPointInfoVerify() {
565 CHECK(IsBreakPointInfo());
566 code_position()->SmiVerify();
567 source_position()->SmiVerify();
568 statement_position()->SmiVerify();
569 VerifyPointer(break_point_objects());
570}
Ben Murdochb0fe1622011-05-05 13:52:32 +0100571#endif // ENABLE_DEBUGGER_SUPPORT
Steve Blocka7e24c12009-10-30 11:49:00 +0000572
573
574void JSObject::IncrementSpillStatistics(SpillInformation* info) {
575 info->number_of_objects_++;
576 // Named properties
577 if (HasFastProperties()) {
578 info->number_of_objects_with_fast_properties_++;
579 info->number_of_fast_used_fields_ += map()->NextFreePropertyIndex();
580 info->number_of_fast_unused_fields_ += map()->unused_property_fields();
581 } else {
582 StringDictionary* dict = property_dictionary();
583 info->number_of_slow_used_properties_ += dict->NumberOfElements();
584 info->number_of_slow_unused_properties_ +=
585 dict->Capacity() - dict->NumberOfElements();
586 }
587 // Indexed properties
588 switch (GetElementsKind()) {
589 case FAST_ELEMENTS: {
590 info->number_of_objects_with_fast_elements_++;
591 int holes = 0;
592 FixedArray* e = FixedArray::cast(elements());
593 int len = e->length();
594 for (int i = 0; i < len; i++) {
595 if (e->get(i) == Heap::the_hole_value()) holes++;
596 }
597 info->number_of_fast_used_elements_ += len - holes;
598 info->number_of_fast_unused_elements_ += holes;
599 break;
600 }
601 case PIXEL_ELEMENTS: {
602 info->number_of_objects_with_fast_elements_++;
603 PixelArray* e = PixelArray::cast(elements());
604 info->number_of_fast_used_elements_ += e->length();
605 break;
606 }
607 case DICTIONARY_ELEMENTS: {
608 NumberDictionary* dict = element_dictionary();
609 info->number_of_slow_used_elements_ += dict->NumberOfElements();
610 info->number_of_slow_unused_elements_ +=
611 dict->Capacity() - dict->NumberOfElements();
612 break;
613 }
614 default:
615 UNREACHABLE();
616 break;
617 }
618}
619
620
621void JSObject::SpillInformation::Clear() {
622 number_of_objects_ = 0;
623 number_of_objects_with_fast_properties_ = 0;
624 number_of_objects_with_fast_elements_ = 0;
625 number_of_fast_used_fields_ = 0;
626 number_of_fast_unused_fields_ = 0;
627 number_of_slow_used_properties_ = 0;
628 number_of_slow_unused_properties_ = 0;
629 number_of_fast_used_elements_ = 0;
630 number_of_fast_unused_elements_ = 0;
631 number_of_slow_used_elements_ = 0;
632 number_of_slow_unused_elements_ = 0;
633}
634
635void JSObject::SpillInformation::Print() {
636 PrintF("\n JSObject Spill Statistics (#%d):\n", number_of_objects_);
637
638 PrintF(" - fast properties (#%d): %d (used) %d (unused)\n",
639 number_of_objects_with_fast_properties_,
640 number_of_fast_used_fields_, number_of_fast_unused_fields_);
641
642 PrintF(" - slow properties (#%d): %d (used) %d (unused)\n",
643 number_of_objects_ - number_of_objects_with_fast_properties_,
644 number_of_slow_used_properties_, number_of_slow_unused_properties_);
645
646 PrintF(" - fast elements (#%d): %d (used) %d (unused)\n",
647 number_of_objects_with_fast_elements_,
648 number_of_fast_used_elements_, number_of_fast_unused_elements_);
649
650 PrintF(" - slow elements (#%d): %d (used) %d (unused)\n",
651 number_of_objects_ - number_of_objects_with_fast_elements_,
652 number_of_slow_used_elements_, number_of_slow_unused_elements_);
653
654 PrintF("\n");
655}
656
657
Steve Blocka7e24c12009-10-30 11:49:00 +0000658bool DescriptorArray::IsSortedNoDuplicates() {
659 String* current_key = NULL;
660 uint32_t current = 0;
661 for (int i = 0; i < number_of_descriptors(); i++) {
662 String* key = GetKey(i);
663 if (key == current_key) {
664 PrintDescriptors();
665 return false;
666 }
667 current_key = key;
668 uint32_t hash = GetKey(i)->Hash();
669 if (hash < current) {
670 PrintDescriptors();
671 return false;
672 }
673 current = hash;
674 }
675 return true;
676}
677
678
Steve Block6ded16b2010-05-10 14:33:55 +0100679void JSFunctionResultCache::JSFunctionResultCacheVerify() {
680 JSFunction::cast(get(kFactoryIndex))->Verify();
681
682 int size = Smi::cast(get(kCacheSizeIndex))->value();
683 ASSERT(kEntriesIndex <= size);
684 ASSERT(size <= length());
685 ASSERT_EQ(0, size % kEntrySize);
686
687 int finger = Smi::cast(get(kFingerIndex))->value();
688 ASSERT(kEntriesIndex <= finger);
Ben Murdochb8e0da22011-05-16 14:20:40 +0100689 ASSERT((finger < size) || (finger == kEntriesIndex && finger == size));
Steve Block6ded16b2010-05-10 14:33:55 +0100690 ASSERT_EQ(0, finger % kEntrySize);
691
692 if (FLAG_enable_slow_asserts) {
693 for (int i = kEntriesIndex; i < size; i++) {
694 ASSERT(!get(i)->IsTheHole());
695 get(i)->Verify();
696 }
697 for (int i = size; i < length(); i++) {
698 ASSERT(get(i)->IsTheHole());
699 get(i)->Verify();
700 }
701 }
702}
703
704
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100705void NormalizedMapCache::NormalizedMapCacheVerify() {
706 FixedArray::cast(this)->Verify();
707 if (FLAG_enable_slow_asserts) {
708 for (int i = 0; i < length(); i++) {
709 Object* e = get(i);
710 if (e->IsMap()) {
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100711 Map::cast(e)->SharedMapVerify();
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100712 } else {
713 ASSERT(e->IsUndefined());
714 }
715 }
716 }
717}
718
719
Steve Blocka7e24c12009-10-30 11:49:00 +0000720#endif // DEBUG
721
722} } // namespace v8::internal