blob: f9c57e696078da0a7e2ab6e644f68ddf0f762861 [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;
161
162#define MAKE_STRUCT_CASE(NAME, Name, name) \
163 case NAME##_TYPE: \
164 Name::cast(this)->Name##Verify(); \
165 break;
166 STRUCT_LIST(MAKE_STRUCT_CASE)
167#undef MAKE_STRUCT_CASE
168
169 default:
170 UNREACHABLE();
171 break;
172 }
173}
174
175
176void HeapObject::VerifyHeapPointer(Object* p) {
177 ASSERT(p->IsHeapObject());
178 ASSERT(Heap::Contains(HeapObject::cast(p)));
179}
180
181
182void HeapNumber::HeapNumberVerify() {
183 ASSERT(IsHeapNumber());
184}
185
186
Steve Blocka7e24c12009-10-30 11:49:00 +0000187void ByteArray::ByteArrayVerify() {
188 ASSERT(IsByteArray());
189}
190
191
192void PixelArray::PixelArrayVerify() {
193 ASSERT(IsPixelArray());
194}
195
196
Steve Block3ce2e202009-11-05 08:53:23 +0000197void ExternalByteArray::ExternalByteArrayVerify() {
198 ASSERT(IsExternalByteArray());
199}
200
201
202void ExternalUnsignedByteArray::ExternalUnsignedByteArrayVerify() {
203 ASSERT(IsExternalUnsignedByteArray());
204}
205
206
207void ExternalShortArray::ExternalShortArrayVerify() {
208 ASSERT(IsExternalShortArray());
209}
210
211
212void ExternalUnsignedShortArray::ExternalUnsignedShortArrayVerify() {
213 ASSERT(IsExternalUnsignedShortArray());
214}
215
216
217void ExternalIntArray::ExternalIntArrayVerify() {
218 ASSERT(IsExternalIntArray());
219}
220
221
222void ExternalUnsignedIntArray::ExternalUnsignedIntArrayVerify() {
223 ASSERT(IsExternalUnsignedIntArray());
224}
225
226
227void ExternalFloatArray::ExternalFloatArrayVerify() {
228 ASSERT(IsExternalFloatArray());
229}
230
231
Steve Blocka7e24c12009-10-30 11:49:00 +0000232void JSObject::JSObjectVerify() {
233 VerifyHeapPointer(properties());
234 VerifyHeapPointer(elements());
235 if (HasFastProperties()) {
236 CHECK_EQ(map()->unused_property_fields(),
237 (map()->inobject_properties() + properties()->length() -
238 map()->NextFreePropertyIndex()));
239 }
Steve Block8defd9f2010-07-08 12:39:36 +0100240 ASSERT(map()->has_fast_elements() ==
Iain Merrick75681382010-08-19 15:07:18 +0100241 (elements()->map() == Heap::fixed_array_map() ||
242 elements()->map() == Heap::fixed_cow_array_map()));
Steve Block8defd9f2010-07-08 12:39:36 +0100243 ASSERT(map()->has_fast_elements() == HasFastElements());
Steve Blocka7e24c12009-10-30 11:49:00 +0000244}
245
246
Steve Blocka7e24c12009-10-30 11:49:00 +0000247void Map::MapVerify() {
248 ASSERT(!Heap::InNewSpace(this));
249 ASSERT(FIRST_TYPE <= instance_type() && instance_type() <= LAST_TYPE);
Steve Block791712a2010-08-27 10:21:07 +0100250 ASSERT(instance_size() == kVariableSizeSentinel ||
251 (kPointerSize <= instance_size() &&
252 instance_size() < Heap::Capacity()));
Steve Blocka7e24c12009-10-30 11:49:00 +0000253 VerifyHeapPointer(prototype());
254 VerifyHeapPointer(instance_descriptors());
255}
256
257
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100258void Map::SharedMapVerify() {
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100259 MapVerify();
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100260 ASSERT(is_shared());
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100261 ASSERT_EQ(Heap::empty_descriptor_array(), instance_descriptors());
262 ASSERT_EQ(Heap::empty_fixed_array(), code_cache());
263 ASSERT_EQ(0, pre_allocated_property_fields());
264 ASSERT_EQ(0, unused_property_fields());
265 ASSERT_EQ(StaticVisitorBase::GetVisitorId(instance_type(), instance_size()),
266 visitor_id());
267}
268
269
Steve Block6ded16b2010-05-10 14:33:55 +0100270void CodeCache::CodeCacheVerify() {
271 VerifyHeapPointer(default_cache());
272 VerifyHeapPointer(normal_type_cache());
273 ASSERT(default_cache()->IsFixedArray());
274 ASSERT(normal_type_cache()->IsUndefined()
275 || normal_type_cache()->IsCodeCacheHashTable());
276}
277
278
Steve Blocka7e24c12009-10-30 11:49:00 +0000279void FixedArray::FixedArrayVerify() {
280 for (int i = 0; i < length(); i++) {
281 Object* e = get(i);
282 if (e->IsHeapObject()) {
283 VerifyHeapPointer(e);
284 } else {
285 e->Verify();
286 }
287 }
288}
289
290
Steve Blocka7e24c12009-10-30 11:49:00 +0000291void JSValue::JSValueVerify() {
292 Object* v = value();
293 if (v->IsHeapObject()) {
294 VerifyHeapPointer(v);
295 }
296}
297
298
Steve Blocka7e24c12009-10-30 11:49:00 +0000299void String::StringVerify() {
300 CHECK(IsString());
301 CHECK(length() >= 0 && length() <= Smi::kMaxValue);
302 if (IsSymbol()) {
303 CHECK(!Heap::InNewSpace(this));
304 }
305}
306
307
Steve Blocka7e24c12009-10-30 11:49:00 +0000308void JSFunction::JSFunctionVerify() {
309 CHECK(IsJSFunction());
310 VerifyObjectField(kPrototypeOrInitialMapOffset);
Ben Murdochb0fe1622011-05-05 13:52:32 +0100311 VerifyObjectField(kNextFunctionLinkOffset);
312 CHECK(next_function_link()->IsUndefined() ||
313 next_function_link()->IsJSFunction());
Steve Blocka7e24c12009-10-30 11:49:00 +0000314}
315
316
Steve Blocka7e24c12009-10-30 11:49:00 +0000317void SharedFunctionInfo::SharedFunctionInfoVerify() {
318 CHECK(IsSharedFunctionInfo());
319 VerifyObjectField(kNameOffset);
320 VerifyObjectField(kCodeOffset);
Ben Murdoch3bec4d22010-07-22 14:51:16 +0100321 VerifyObjectField(kScopeInfoOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +0000322 VerifyObjectField(kInstanceClassNameOffset);
Steve Block6ded16b2010-05-10 14:33:55 +0100323 VerifyObjectField(kFunctionDataOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +0000324 VerifyObjectField(kScriptOffset);
325 VerifyObjectField(kDebugInfoOffset);
326}
327
328
Steve Blocka7e24c12009-10-30 11:49:00 +0000329void JSGlobalProxy::JSGlobalProxyVerify() {
330 CHECK(IsJSGlobalProxy());
331 JSObjectVerify();
332 VerifyObjectField(JSGlobalProxy::kContextOffset);
333 // Make sure that this object has no properties, elements.
334 CHECK_EQ(0, properties()->length());
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +0100335 CHECK(HasFastElements());
336 CHECK_EQ(0, FixedArray::cast(elements())->length());
Steve Blocka7e24c12009-10-30 11:49:00 +0000337}
338
339
Steve Blocka7e24c12009-10-30 11:49:00 +0000340void JSGlobalObject::JSGlobalObjectVerify() {
341 CHECK(IsJSGlobalObject());
342 JSObjectVerify();
343 for (int i = GlobalObject::kBuiltinsOffset;
344 i < JSGlobalObject::kSize;
345 i += kPointerSize) {
346 VerifyObjectField(i);
347 }
348}
349
350
Steve Blocka7e24c12009-10-30 11:49:00 +0000351void JSBuiltinsObject::JSBuiltinsObjectVerify() {
352 CHECK(IsJSBuiltinsObject());
353 JSObjectVerify();
354 for (int i = GlobalObject::kBuiltinsOffset;
355 i < JSBuiltinsObject::kSize;
356 i += kPointerSize) {
357 VerifyObjectField(i);
358 }
359}
360
361
362void Oddball::OddballVerify() {
363 CHECK(IsOddball());
364 VerifyHeapPointer(to_string());
365 Object* number = to_number();
366 if (number->IsHeapObject()) {
367 ASSERT(number == Heap::nan_value());
368 } else {
369 ASSERT(number->IsSmi());
370 int value = Smi::cast(number)->value();
Ben Murdoch086aeea2011-05-13 15:57:08 +0100371 // Hidden oddballs have negative smis.
372 const int kLeastHiddenOddballNumber = -4;
373 ASSERT(value <= 1);
374 ASSERT(value >= kLeastHiddenOddballNumber);
Steve Blocka7e24c12009-10-30 11:49:00 +0000375 }
376}
377
378
379void JSGlobalPropertyCell::JSGlobalPropertyCellVerify() {
380 CHECK(IsJSGlobalPropertyCell());
381 VerifyObjectField(kValueOffset);
382}
383
384
Steve Blocka7e24c12009-10-30 11:49:00 +0000385void Code::CodeVerify() {
386 CHECK(IsAligned(reinterpret_cast<intptr_t>(instruction_start()),
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100387 kCodeAlignment));
Steve Blocka7e24c12009-10-30 11:49:00 +0000388 Address last_gc_pc = NULL;
389 for (RelocIterator it(this); !it.done(); it.next()) {
390 it.rinfo()->Verify();
391 // Ensure that GC will not iterate twice over the same pointer.
392 if (RelocInfo::IsGCRelocMode(it.rinfo()->rmode())) {
393 CHECK(it.rinfo()->pc() != last_gc_pc);
394 last_gc_pc = it.rinfo()->pc();
395 }
396 }
397}
398
399
400void JSArray::JSArrayVerify() {
401 JSObjectVerify();
402 ASSERT(length()->IsNumber() || length()->IsUndefined());
403 ASSERT(elements()->IsUndefined() || elements()->IsFixedArray());
404}
405
406
407void JSRegExp::JSRegExpVerify() {
408 JSObjectVerify();
409 ASSERT(data()->IsUndefined() || data()->IsFixedArray());
410 switch (TypeTag()) {
411 case JSRegExp::ATOM: {
412 FixedArray* arr = FixedArray::cast(data());
413 ASSERT(arr->get(JSRegExp::kAtomPatternIndex)->IsString());
414 break;
415 }
416 case JSRegExp::IRREGEXP: {
417 bool is_native = RegExpImpl::UsesNativeRegExp();
418
419 FixedArray* arr = FixedArray::cast(data());
420 Object* ascii_data = arr->get(JSRegExp::kIrregexpASCIICodeIndex);
421 // TheHole : Not compiled yet.
422 // JSObject: Compilation error.
423 // Code/ByteArray: Compiled code.
424 ASSERT(ascii_data->IsTheHole() || ascii_data->IsJSObject() ||
425 (is_native ? ascii_data->IsCode() : ascii_data->IsByteArray()));
426 Object* uc16_data = arr->get(JSRegExp::kIrregexpUC16CodeIndex);
427 ASSERT(uc16_data->IsTheHole() || ascii_data->IsJSObject() ||
428 (is_native ? uc16_data->IsCode() : uc16_data->IsByteArray()));
429 ASSERT(arr->get(JSRegExp::kIrregexpCaptureCountIndex)->IsSmi());
430 ASSERT(arr->get(JSRegExp::kIrregexpMaxRegisterCountIndex)->IsSmi());
431 break;
432 }
433 default:
434 ASSERT_EQ(JSRegExp::NOT_COMPILED, TypeTag());
435 ASSERT(data()->IsUndefined());
436 break;
437 }
438}
439
440
Steve Blocka7e24c12009-10-30 11:49:00 +0000441void Proxy::ProxyVerify() {
442 ASSERT(IsProxy());
443}
444
445
446void AccessorInfo::AccessorInfoVerify() {
447 CHECK(IsAccessorInfo());
448 VerifyPointer(getter());
449 VerifyPointer(setter());
450 VerifyPointer(name());
451 VerifyPointer(data());
452 VerifyPointer(flag());
453}
454
Ben Murdochb0fe1622011-05-05 13:52:32 +0100455
Steve Blocka7e24c12009-10-30 11:49:00 +0000456void AccessCheckInfo::AccessCheckInfoVerify() {
457 CHECK(IsAccessCheckInfo());
458 VerifyPointer(named_callback());
459 VerifyPointer(indexed_callback());
460 VerifyPointer(data());
461}
462
Ben Murdochb0fe1622011-05-05 13:52:32 +0100463
Steve Blocka7e24c12009-10-30 11:49:00 +0000464void InterceptorInfo::InterceptorInfoVerify() {
465 CHECK(IsInterceptorInfo());
466 VerifyPointer(getter());
467 VerifyPointer(setter());
468 VerifyPointer(query());
469 VerifyPointer(deleter());
470 VerifyPointer(enumerator());
471 VerifyPointer(data());
472}
473
Ben Murdochb0fe1622011-05-05 13:52:32 +0100474
Steve Blocka7e24c12009-10-30 11:49:00 +0000475void CallHandlerInfo::CallHandlerInfoVerify() {
476 CHECK(IsCallHandlerInfo());
477 VerifyPointer(callback());
478 VerifyPointer(data());
479}
480
Ben Murdochb0fe1622011-05-05 13:52:32 +0100481
Steve Blocka7e24c12009-10-30 11:49:00 +0000482void TemplateInfo::TemplateInfoVerify() {
483 VerifyPointer(tag());
484 VerifyPointer(property_list());
485}
486
487void FunctionTemplateInfo::FunctionTemplateInfoVerify() {
488 CHECK(IsFunctionTemplateInfo());
489 TemplateInfoVerify();
490 VerifyPointer(serial_number());
491 VerifyPointer(call_code());
492 VerifyPointer(property_accessors());
493 VerifyPointer(prototype_template());
494 VerifyPointer(parent_template());
495 VerifyPointer(named_property_handler());
496 VerifyPointer(indexed_property_handler());
497 VerifyPointer(instance_template());
498 VerifyPointer(signature());
499 VerifyPointer(access_check_info());
500}
501
Ben Murdochb0fe1622011-05-05 13:52:32 +0100502
Steve Blocka7e24c12009-10-30 11:49:00 +0000503void ObjectTemplateInfo::ObjectTemplateInfoVerify() {
504 CHECK(IsObjectTemplateInfo());
505 TemplateInfoVerify();
506 VerifyPointer(constructor());
507 VerifyPointer(internal_field_count());
508}
509
Ben Murdochb0fe1622011-05-05 13:52:32 +0100510
Steve Blocka7e24c12009-10-30 11:49:00 +0000511void SignatureInfo::SignatureInfoVerify() {
512 CHECK(IsSignatureInfo());
513 VerifyPointer(receiver());
514 VerifyPointer(args());
515}
516
Ben Murdochb0fe1622011-05-05 13:52:32 +0100517
Steve Blocka7e24c12009-10-30 11:49:00 +0000518void TypeSwitchInfo::TypeSwitchInfoVerify() {
519 CHECK(IsTypeSwitchInfo());
520 VerifyPointer(types());
521}
522
Ben Murdochb0fe1622011-05-05 13:52:32 +0100523
Steve Blocka7e24c12009-10-30 11:49:00 +0000524void Script::ScriptVerify() {
525 CHECK(IsScript());
526 VerifyPointer(source());
527 VerifyPointer(name());
528 line_offset()->SmiVerify();
529 column_offset()->SmiVerify();
530 VerifyPointer(data());
531 VerifyPointer(wrapper());
532 type()->SmiVerify();
533 VerifyPointer(line_ends());
534 VerifyPointer(id());
535}
Steve Blocka7e24c12009-10-30 11:49:00 +0000536
537
538#ifdef ENABLE_DEBUGGER_SUPPORT
539void DebugInfo::DebugInfoVerify() {
540 CHECK(IsDebugInfo());
541 VerifyPointer(shared());
542 VerifyPointer(original_code());
543 VerifyPointer(code());
544 VerifyPointer(break_points());
545}
546
547
Steve Blocka7e24c12009-10-30 11:49:00 +0000548void BreakPointInfo::BreakPointInfoVerify() {
549 CHECK(IsBreakPointInfo());
550 code_position()->SmiVerify();
551 source_position()->SmiVerify();
552 statement_position()->SmiVerify();
553 VerifyPointer(break_point_objects());
554}
Ben Murdochb0fe1622011-05-05 13:52:32 +0100555#endif // ENABLE_DEBUGGER_SUPPORT
Steve Blocka7e24c12009-10-30 11:49:00 +0000556
557
558void JSObject::IncrementSpillStatistics(SpillInformation* info) {
559 info->number_of_objects_++;
560 // Named properties
561 if (HasFastProperties()) {
562 info->number_of_objects_with_fast_properties_++;
563 info->number_of_fast_used_fields_ += map()->NextFreePropertyIndex();
564 info->number_of_fast_unused_fields_ += map()->unused_property_fields();
565 } else {
566 StringDictionary* dict = property_dictionary();
567 info->number_of_slow_used_properties_ += dict->NumberOfElements();
568 info->number_of_slow_unused_properties_ +=
569 dict->Capacity() - dict->NumberOfElements();
570 }
571 // Indexed properties
572 switch (GetElementsKind()) {
573 case FAST_ELEMENTS: {
574 info->number_of_objects_with_fast_elements_++;
575 int holes = 0;
576 FixedArray* e = FixedArray::cast(elements());
577 int len = e->length();
578 for (int i = 0; i < len; i++) {
579 if (e->get(i) == Heap::the_hole_value()) holes++;
580 }
581 info->number_of_fast_used_elements_ += len - holes;
582 info->number_of_fast_unused_elements_ += holes;
583 break;
584 }
585 case PIXEL_ELEMENTS: {
586 info->number_of_objects_with_fast_elements_++;
587 PixelArray* e = PixelArray::cast(elements());
588 info->number_of_fast_used_elements_ += e->length();
589 break;
590 }
591 case DICTIONARY_ELEMENTS: {
592 NumberDictionary* dict = element_dictionary();
593 info->number_of_slow_used_elements_ += dict->NumberOfElements();
594 info->number_of_slow_unused_elements_ +=
595 dict->Capacity() - dict->NumberOfElements();
596 break;
597 }
598 default:
599 UNREACHABLE();
600 break;
601 }
602}
603
604
605void JSObject::SpillInformation::Clear() {
606 number_of_objects_ = 0;
607 number_of_objects_with_fast_properties_ = 0;
608 number_of_objects_with_fast_elements_ = 0;
609 number_of_fast_used_fields_ = 0;
610 number_of_fast_unused_fields_ = 0;
611 number_of_slow_used_properties_ = 0;
612 number_of_slow_unused_properties_ = 0;
613 number_of_fast_used_elements_ = 0;
614 number_of_fast_unused_elements_ = 0;
615 number_of_slow_used_elements_ = 0;
616 number_of_slow_unused_elements_ = 0;
617}
618
619void JSObject::SpillInformation::Print() {
620 PrintF("\n JSObject Spill Statistics (#%d):\n", number_of_objects_);
621
622 PrintF(" - fast properties (#%d): %d (used) %d (unused)\n",
623 number_of_objects_with_fast_properties_,
624 number_of_fast_used_fields_, number_of_fast_unused_fields_);
625
626 PrintF(" - slow properties (#%d): %d (used) %d (unused)\n",
627 number_of_objects_ - number_of_objects_with_fast_properties_,
628 number_of_slow_used_properties_, number_of_slow_unused_properties_);
629
630 PrintF(" - fast elements (#%d): %d (used) %d (unused)\n",
631 number_of_objects_with_fast_elements_,
632 number_of_fast_used_elements_, number_of_fast_unused_elements_);
633
634 PrintF(" - slow elements (#%d): %d (used) %d (unused)\n",
635 number_of_objects_ - number_of_objects_with_fast_elements_,
636 number_of_slow_used_elements_, number_of_slow_unused_elements_);
637
638 PrintF("\n");
639}
640
641
Steve Blocka7e24c12009-10-30 11:49:00 +0000642bool DescriptorArray::IsSortedNoDuplicates() {
643 String* current_key = NULL;
644 uint32_t current = 0;
645 for (int i = 0; i < number_of_descriptors(); i++) {
646 String* key = GetKey(i);
647 if (key == current_key) {
648 PrintDescriptors();
649 return false;
650 }
651 current_key = key;
652 uint32_t hash = GetKey(i)->Hash();
653 if (hash < current) {
654 PrintDescriptors();
655 return false;
656 }
657 current = hash;
658 }
659 return true;
660}
661
662
Steve Block6ded16b2010-05-10 14:33:55 +0100663void JSFunctionResultCache::JSFunctionResultCacheVerify() {
664 JSFunction::cast(get(kFactoryIndex))->Verify();
665
666 int size = Smi::cast(get(kCacheSizeIndex))->value();
667 ASSERT(kEntriesIndex <= size);
668 ASSERT(size <= length());
669 ASSERT_EQ(0, size % kEntrySize);
670
671 int finger = Smi::cast(get(kFingerIndex))->value();
672 ASSERT(kEntriesIndex <= finger);
Ben Murdochb8e0da22011-05-16 14:20:40 +0100673 ASSERT((finger < size) || (finger == kEntriesIndex && finger == size));
Steve Block6ded16b2010-05-10 14:33:55 +0100674 ASSERT_EQ(0, finger % kEntrySize);
675
676 if (FLAG_enable_slow_asserts) {
677 for (int i = kEntriesIndex; i < size; i++) {
678 ASSERT(!get(i)->IsTheHole());
679 get(i)->Verify();
680 }
681 for (int i = size; i < length(); i++) {
682 ASSERT(get(i)->IsTheHole());
683 get(i)->Verify();
684 }
685 }
686}
687
688
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100689void NormalizedMapCache::NormalizedMapCacheVerify() {
690 FixedArray::cast(this)->Verify();
691 if (FLAG_enable_slow_asserts) {
692 for (int i = 0; i < length(); i++) {
693 Object* e = get(i);
694 if (e->IsMap()) {
Kristian Monsen0d5e1162010-09-30 15:31:59 +0100695 Map::cast(e)->SharedMapVerify();
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100696 } else {
697 ASSERT(e->IsUndefined());
698 }
699 }
700 }
701}
702
703
Steve Blocka7e24c12009-10-30 11:49:00 +0000704#endif // DEBUG
705
706} } // namespace v8::internal