OLD | NEW |
1 // Copyright 2011 the V8 project authors. All rights reserved. | 1 // Copyright 2011 the V8 project authors. All rights reserved. |
2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
4 // met: | 4 // met: |
5 // | 5 // |
6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
(...skipping 15 matching lines...) Expand all Loading... |
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
27 | 27 |
28 #include "v8.h" | 28 #include "v8.h" |
29 | 29 |
30 #include "api.h" | 30 #include "api.h" |
31 #include "arguments.h" | 31 #include "arguments.h" |
32 #include "bootstrapper.h" | 32 #include "bootstrapper.h" |
33 #include "builtins.h" | 33 #include "builtins.h" |
34 #include "gdb-jit.h" | 34 #include "gdb-jit.h" |
35 #include "ic-inl.h" | 35 #include "ic-inl.h" |
| 36 #include "mark-compact.h" |
36 #include "vm-state-inl.h" | 37 #include "vm-state-inl.h" |
37 | 38 |
38 namespace v8 { | 39 namespace v8 { |
39 namespace internal { | 40 namespace internal { |
40 | 41 |
41 namespace { | 42 namespace { |
42 | 43 |
43 // Arguments object passed to C++ builtins. | 44 // Arguments object passed to C++ builtins. |
44 template <BuiltinExtraArguments extra_args> | 45 template <BuiltinExtraArguments extra_args> |
45 class BuiltinArguments : public Arguments { | 46 class BuiltinArguments : public Arguments { |
(...skipping 242 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
288 ASSERT(dst != src); // Use MoveElements instead. | 289 ASSERT(dst != src); // Use MoveElements instead. |
289 ASSERT(dst->map() != HEAP->fixed_cow_array_map()); | 290 ASSERT(dst->map() != HEAP->fixed_cow_array_map()); |
290 ASSERT(len > 0); | 291 ASSERT(len > 0); |
291 CopyWords(dst->data_start() + dst_index, | 292 CopyWords(dst->data_start() + dst_index, |
292 src->data_start() + src_index, | 293 src->data_start() + src_index, |
293 len); | 294 len); |
294 WriteBarrierMode mode = dst->GetWriteBarrierMode(*no_gc); | 295 WriteBarrierMode mode = dst->GetWriteBarrierMode(*no_gc); |
295 if (mode == UPDATE_WRITE_BARRIER) { | 296 if (mode == UPDATE_WRITE_BARRIER) { |
296 heap->RecordWrites(dst->address(), dst->OffsetOfElementAt(dst_index), len); | 297 heap->RecordWrites(dst->address(), dst->OffsetOfElementAt(dst_index), len); |
297 } | 298 } |
| 299 heap->incremental_marking()->RecordWrites(dst); |
298 } | 300 } |
299 | 301 |
300 | 302 |
301 static void MoveElements(Heap* heap, | 303 static void MoveElements(Heap* heap, |
302 AssertNoAllocation* no_gc, | 304 AssertNoAllocation* no_gc, |
303 FixedArray* dst, | 305 FixedArray* dst, |
304 int dst_index, | 306 int dst_index, |
305 FixedArray* src, | 307 FixedArray* src, |
306 int src_index, | 308 int src_index, |
307 int len) { | 309 int len) { |
308 ASSERT(dst->map() != HEAP->fixed_cow_array_map()); | 310 ASSERT(dst->map() != HEAP->fixed_cow_array_map()); |
309 memmove(dst->data_start() + dst_index, | 311 memmove(dst->data_start() + dst_index, |
310 src->data_start() + src_index, | 312 src->data_start() + src_index, |
311 len * kPointerSize); | 313 len * kPointerSize); |
312 WriteBarrierMode mode = dst->GetWriteBarrierMode(*no_gc); | 314 WriteBarrierMode mode = dst->GetWriteBarrierMode(*no_gc); |
313 if (mode == UPDATE_WRITE_BARRIER) { | 315 if (mode == UPDATE_WRITE_BARRIER) { |
314 heap->RecordWrites(dst->address(), dst->OffsetOfElementAt(dst_index), len); | 316 heap->RecordWrites(dst->address(), dst->OffsetOfElementAt(dst_index), len); |
315 } | 317 } |
| 318 heap->incremental_marking()->RecordWrites(dst); |
316 } | 319 } |
317 | 320 |
318 | 321 |
319 static void FillWithHoles(Heap* heap, FixedArray* dst, int from, int to) { | 322 static void FillWithHoles(Heap* heap, FixedArray* dst, int from, int to) { |
320 ASSERT(dst->map() != heap->fixed_cow_array_map()); | 323 ASSERT(dst->map() != heap->fixed_cow_array_map()); |
321 MemsetPointer(dst->data_start() + from, heap->the_hole_value(), to - from); | 324 MemsetPointer(dst->data_start() + from, heap->the_hole_value(), to - from); |
322 } | 325 } |
323 | 326 |
324 | 327 |
325 static FixedArray* LeftTrimFixedArray(Heap* heap, | 328 static FixedArray* LeftTrimFixedArray(Heap* heap, |
(...skipping 25 matching lines...) Expand all Loading... |
351 } | 354 } |
352 } | 355 } |
353 // Technically in new space this write might be omitted (except for | 356 // Technically in new space this write might be omitted (except for |
354 // debug mode which iterates through the heap), but to play safer | 357 // debug mode which iterates through the heap), but to play safer |
355 // we still do it. | 358 // we still do it. |
356 heap->CreateFillerObjectAt(elms->address(), to_trim * kPointerSize); | 359 heap->CreateFillerObjectAt(elms->address(), to_trim * kPointerSize); |
357 | 360 |
358 former_start[to_trim] = heap->fixed_array_map(); | 361 former_start[to_trim] = heap->fixed_array_map(); |
359 former_start[to_trim + 1] = Smi::FromInt(len - to_trim); | 362 former_start[to_trim + 1] = Smi::FromInt(len - to_trim); |
360 | 363 |
| 364 // Maintain marking consistency for HeapObjectIterator and |
| 365 // IncrementalMarking. |
| 366 int size_delta = to_trim * kPointerSize; |
| 367 if (heap->marking()->TransferMark(elms->address(), |
| 368 elms->address() + size_delta)) { |
| 369 MemoryChunk::IncrementLiveBytes(elms->address(), -size_delta); |
| 370 } |
| 371 |
361 return FixedArray::cast(HeapObject::FromAddress( | 372 return FixedArray::cast(HeapObject::FromAddress( |
362 elms->address() + to_trim * kPointerSize)); | 373 elms->address() + to_trim * kPointerSize)); |
363 } | 374 } |
364 | 375 |
365 | 376 |
366 static bool ArrayPrototypeHasNoElements(Heap* heap, | 377 static bool ArrayPrototypeHasNoElements(Heap* heap, |
367 Context* global_context, | 378 Context* global_context, |
368 JSObject* array_proto) { | 379 JSObject* array_proto) { |
369 // This method depends on non writability of Object and Array prototype | 380 // This method depends on non writability of Object and Array prototype |
370 // fields. | 381 // fields. |
(...skipping 173 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
544 int len = Smi::cast(array->length())->value(); | 555 int len = Smi::cast(array->length())->value(); |
545 if (len == 0) return heap->undefined_value(); | 556 if (len == 0) return heap->undefined_value(); |
546 | 557 |
547 // Get first element | 558 // Get first element |
548 Object* first = elms->get(0); | 559 Object* first = elms->get(0); |
549 if (first->IsTheHole()) { | 560 if (first->IsTheHole()) { |
550 first = heap->undefined_value(); | 561 first = heap->undefined_value(); |
551 } | 562 } |
552 | 563 |
553 if (!heap->lo_space()->Contains(elms)) { | 564 if (!heap->lo_space()->Contains(elms)) { |
554 // As elms still in the same space they used to be, | 565 array->set_elements(LeftTrimFixedArray(heap, elms, 1)); |
555 // there is no need to update region dirty mark. | |
556 array->set_elements(LeftTrimFixedArray(heap, elms, 1), SKIP_WRITE_BARRIER); | |
557 } else { | 566 } else { |
558 // Shift the elements. | 567 // Shift the elements. |
559 AssertNoAllocation no_gc; | 568 AssertNoAllocation no_gc; |
560 MoveElements(heap, &no_gc, elms, 0, elms, 1, len - 1); | 569 MoveElements(heap, &no_gc, elms, 0, elms, 1, len - 1); |
561 elms->set(len - 1, heap->the_hole_value()); | 570 elms->set(len - 1, heap->the_hole_value()); |
562 } | 571 } |
563 | 572 |
564 // Set the length. | 573 // Set the length. |
565 array->set_length(Smi::FromInt(len - 1)); | 574 array->set_length(Smi::FromInt(len - 1)); |
566 | 575 |
(...skipping 268 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
835 (len - actual_delete_count - actual_start)); | 844 (len - actual_delete_count - actual_start)); |
836 if (trim_array) { | 845 if (trim_array) { |
837 const int delta = actual_delete_count - item_count; | 846 const int delta = actual_delete_count - item_count; |
838 | 847 |
839 if (actual_start > 0) { | 848 if (actual_start > 0) { |
840 AssertNoAllocation no_gc; | 849 AssertNoAllocation no_gc; |
841 MoveElements(heap, &no_gc, elms, delta, elms, 0, actual_start); | 850 MoveElements(heap, &no_gc, elms, delta, elms, 0, actual_start); |
842 } | 851 } |
843 | 852 |
844 elms = LeftTrimFixedArray(heap, elms, delta); | 853 elms = LeftTrimFixedArray(heap, elms, delta); |
845 array->set_elements(elms, SKIP_WRITE_BARRIER); | 854 array->set_elements(elms); |
846 } else { | 855 } else { |
847 AssertNoAllocation no_gc; | 856 AssertNoAllocation no_gc; |
848 MoveElements(heap, &no_gc, | 857 MoveElements(heap, &no_gc, |
849 elms, actual_start + item_count, | 858 elms, actual_start + item_count, |
850 elms, actual_start + actual_delete_count, | 859 elms, actual_start + actual_delete_count, |
851 (len - actual_delete_count - actual_start)); | 860 (len - actual_delete_count - actual_start)); |
852 FillWithHoles(heap, elms, new_length, len); | 861 FillWithHoles(heap, elms, new_length, len); |
853 } | 862 } |
854 } else if (item_count > actual_delete_count) { | 863 } else if (item_count > actual_delete_count) { |
855 // Currently fixed arrays cannot grow too big, so | 864 // Currently fixed arrays cannot grow too big, so |
(...skipping 848 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1704 return Handle<Code>(code_address); \ | 1713 return Handle<Code>(code_address); \ |
1705 } | 1714 } |
1706 BUILTIN_LIST_C(DEFINE_BUILTIN_ACCESSOR_C) | 1715 BUILTIN_LIST_C(DEFINE_BUILTIN_ACCESSOR_C) |
1707 BUILTIN_LIST_A(DEFINE_BUILTIN_ACCESSOR_A) | 1716 BUILTIN_LIST_A(DEFINE_BUILTIN_ACCESSOR_A) |
1708 BUILTIN_LIST_DEBUG_A(DEFINE_BUILTIN_ACCESSOR_A) | 1717 BUILTIN_LIST_DEBUG_A(DEFINE_BUILTIN_ACCESSOR_A) |
1709 #undef DEFINE_BUILTIN_ACCESSOR_C | 1718 #undef DEFINE_BUILTIN_ACCESSOR_C |
1710 #undef DEFINE_BUILTIN_ACCESSOR_A | 1719 #undef DEFINE_BUILTIN_ACCESSOR_A |
1711 | 1720 |
1712 | 1721 |
1713 } } // namespace v8::internal | 1722 } } // namespace v8::internal |
OLD | NEW |