OLD | NEW |
---|---|
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #include "src/accessors.h" | 7 #include "src/accessors.h" |
8 #include "src/api.h" | 8 #include "src/api.h" |
9 #include "src/base/bits.h" | 9 #include "src/base/bits.h" |
10 #include "src/base/once.h" | 10 #include "src/base/once.h" |
(...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
123 total_gc_time_ms_(0.0), | 123 total_gc_time_ms_(0.0), |
124 max_alive_after_gc_(0), | 124 max_alive_after_gc_(0), |
125 min_in_mutator_(kMaxInt), | 125 min_in_mutator_(kMaxInt), |
126 marking_time_(0.0), | 126 marking_time_(0.0), |
127 sweeping_time_(0.0), | 127 sweeping_time_(0.0), |
128 last_idle_notification_time_(0.0), | 128 last_idle_notification_time_(0.0), |
129 mark_compact_collector_(this), | 129 mark_compact_collector_(this), |
130 store_buffer_(this), | 130 store_buffer_(this), |
131 marking_(this), | 131 marking_(this), |
132 incremental_marking_(this), | 132 incremental_marking_(this), |
133 gc_count_at_last_idle_gc_(0), | |
134 full_codegen_bytes_generated_(0), | 133 full_codegen_bytes_generated_(0), |
135 crankshaft_codegen_bytes_generated_(0), | 134 crankshaft_codegen_bytes_generated_(0), |
136 gcs_since_last_deopt_(0), | 135 gcs_since_last_deopt_(0), |
137 allocation_sites_scratchpad_length_(0), | 136 allocation_sites_scratchpad_length_(0), |
138 promotion_queue_(this), | 137 promotion_queue_(this), |
139 configured_(false), | 138 configured_(false), |
140 external_string_table_(this), | 139 external_string_table_(this), |
141 chunks_queued_for_free_(NULL), | 140 chunks_queued_for_free_(NULL), |
142 gc_callbacks_depth_(0), | 141 gc_callbacks_depth_(0), |
143 deserialization_complete_(false), | 142 deserialization_complete_(false), |
(...skipping 4353 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
4497 if (!IsHeapIterable()) { | 4496 if (!IsHeapIterable()) { |
4498 CollectAllGarbage(kMakeHeapIterableMask, "Heap::MakeHeapIterable"); | 4497 CollectAllGarbage(kMakeHeapIterableMask, "Heap::MakeHeapIterable"); |
4499 } | 4498 } |
4500 if (mark_compact_collector()->sweeping_in_progress()) { | 4499 if (mark_compact_collector()->sweeping_in_progress()) { |
4501 mark_compact_collector()->EnsureSweepingCompleted(); | 4500 mark_compact_collector()->EnsureSweepingCompleted(); |
4502 } | 4501 } |
4503 DCHECK(IsHeapIterable()); | 4502 DCHECK(IsHeapIterable()); |
4504 } | 4503 } |
4505 | 4504 |
4506 | 4505 |
4507 void Heap::IdleMarkCompact(const char* message) { | 4506 void Heap::ReduceNewSpaceSize(bool long_idle_time) { |
4508 bool uncommit = false; | 4507 if (long_idle_time) { |
4509 if (gc_count_at_last_idle_gc_ == gc_count_) { | |
4510 // No GC since the last full GC, the mutator is probably not active. | |
4511 isolate_->compilation_cache()->Clear(); | |
4512 uncommit = true; | |
4513 } | |
4514 CollectAllGarbage(kReduceMemoryFootprintMask, message); | |
4515 gc_idle_time_handler_.NotifyIdleMarkCompact(); | |
4516 gc_count_at_last_idle_gc_ = gc_count_; | |
4517 if (uncommit) { | |
4518 new_space_.Shrink(); | 4508 new_space_.Shrink(); |
4519 UncommitFromSpace(); | 4509 UncommitFromSpace(); |
4520 } | 4510 } |
4521 } | 4511 } |
4522 | 4512 |
4523 | 4513 |
4524 bool Heap::TryFinalizeIdleIncrementalMarking( | 4514 bool Heap::TryFinalizeIdleIncrementalMarking( |
4525 double idle_time_in_ms, size_t size_of_objects, | 4515 bool long_idle_time, double idle_time_in_ms, size_t size_of_objects, |
4526 size_t final_incremental_mark_compact_speed_in_bytes_per_ms) { | 4516 size_t final_incremental_mark_compact_speed_in_bytes_per_ms) { |
4527 if (FLAG_overapproximate_weak_closure && | 4517 if (FLAG_overapproximate_weak_closure && |
4528 (incremental_marking()->IsReadyToOverApproximateWeakClosure() || | 4518 (incremental_marking()->IsReadyToOverApproximateWeakClosure() || |
4529 (!incremental_marking()->weak_closure_was_overapproximated() && | 4519 (!incremental_marking()->weak_closure_was_overapproximated() && |
4530 mark_compact_collector_.marking_deque()->IsEmpty() && | 4520 mark_compact_collector_.marking_deque()->IsEmpty() && |
4531 gc_idle_time_handler_.ShouldDoOverApproximateWeakClosure( | 4521 gc_idle_time_handler_.ShouldDoOverApproximateWeakClosure( |
4532 static_cast<size_t>(idle_time_in_ms))))) { | 4522 static_cast<size_t>(idle_time_in_ms))))) { |
4533 OverApproximateWeakClosure( | 4523 OverApproximateWeakClosure( |
4534 "Idle notification: overapproximate weak closure"); | 4524 "Idle notification: overapproximate weak closure"); |
4535 return true; | 4525 return true; |
4536 } else if (incremental_marking()->IsComplete() || | 4526 } else if (incremental_marking()->IsComplete() || |
4537 (mark_compact_collector_.marking_deque()->IsEmpty() && | 4527 (mark_compact_collector_.marking_deque()->IsEmpty() && |
4538 gc_idle_time_handler_.ShouldDoFinalIncrementalMarkCompact( | 4528 gc_idle_time_handler_.ShouldDoFinalIncrementalMarkCompact( |
4539 static_cast<size_t>(idle_time_in_ms), size_of_objects, | 4529 static_cast<size_t>(idle_time_in_ms), size_of_objects, |
4540 final_incremental_mark_compact_speed_in_bytes_per_ms))) { | 4530 final_incremental_mark_compact_speed_in_bytes_per_ms))) { |
4541 CollectAllGarbage(kNoGCFlags, "idle notification: finalize incremental"); | 4531 CollectAllGarbage(kNoGCFlags, "idle notification: finalize incremental"); |
4532 ReduceNewSpaceSize(long_idle_time); | |
4542 return true; | 4533 return true; |
4543 } | 4534 } |
4544 return false; | 4535 return false; |
4545 } | 4536 } |
4546 | 4537 |
4547 | 4538 |
4548 static double MonotonicallyIncreasingTimeInMs() { | 4539 static double MonotonicallyIncreasingTimeInMs() { |
4549 return V8::GetCurrentPlatform()->MonotonicallyIncreasingTime() * | 4540 return V8::GetCurrentPlatform()->MonotonicallyIncreasingTime() * |
4550 static_cast<double>(base::Time::kMillisecondsPerSecond); | 4541 static_cast<double>(base::Time::kMillisecondsPerSecond); |
4551 } | 4542 } |
4552 | 4543 |
4553 | 4544 |
4554 bool Heap::IdleNotification(int idle_time_in_ms) { | 4545 bool Heap::IdleNotification(int idle_time_in_ms) { |
4555 return IdleNotification( | 4546 return IdleNotification( |
4556 V8::GetCurrentPlatform()->MonotonicallyIncreasingTime() + | 4547 V8::GetCurrentPlatform()->MonotonicallyIncreasingTime() + |
4557 (static_cast<double>(idle_time_in_ms) / | 4548 (static_cast<double>(idle_time_in_ms) / |
4558 static_cast<double>(base::Time::kMillisecondsPerSecond))); | 4549 static_cast<double>(base::Time::kMillisecondsPerSecond))); |
4559 } | 4550 } |
4560 | 4551 |
4561 | 4552 |
4562 bool Heap::IdleNotification(double deadline_in_seconds) { | 4553 bool Heap::IdleNotification(double deadline_in_seconds) { |
4563 CHECK(HasBeenSetUp()); // http://crbug.com/425035 | 4554 CHECK(HasBeenSetUp()); // http://crbug.com/425035 |
4564 double deadline_in_ms = | 4555 double deadline_in_ms = |
4565 deadline_in_seconds * | 4556 deadline_in_seconds * |
4566 static_cast<double>(base::Time::kMillisecondsPerSecond); | 4557 static_cast<double>(base::Time::kMillisecondsPerSecond); |
4567 HistogramTimerScope idle_notification_scope( | 4558 HistogramTimerScope idle_notification_scope( |
4568 isolate_->counters()->gc_idle_notification()); | 4559 isolate_->counters()->gc_idle_notification()); |
4569 double idle_time_in_ms = deadline_in_ms - MonotonicallyIncreasingTimeInMs(); | 4560 double idle_time_in_ms = deadline_in_ms - MonotonicallyIncreasingTimeInMs(); |
4561 bool long_idle_time = static_cast<size_t>(idle_time_in_ms) > | |
4562 GCIdleTimeHandler::kMaxFrameRenderingIdleTime; | |
4570 | 4563 |
4571 GCIdleTimeHandler::HeapState heap_state; | 4564 GCIdleTimeHandler::HeapState heap_state; |
4572 heap_state.contexts_disposed = contexts_disposed_; | 4565 heap_state.contexts_disposed = contexts_disposed_; |
4573 heap_state.contexts_disposal_rate = | 4566 heap_state.contexts_disposal_rate = |
4574 tracer()->ContextDisposalRateInMilliseconds(); | 4567 tracer()->ContextDisposalRateInMilliseconds(); |
4575 heap_state.size_of_objects = static_cast<size_t>(SizeOfObjects()); | 4568 heap_state.size_of_objects = static_cast<size_t>(SizeOfObjects()); |
4576 heap_state.incremental_marking_stopped = incremental_marking()->IsStopped(); | 4569 heap_state.incremental_marking_stopped = incremental_marking()->IsStopped(); |
4577 // TODO(ulan): Start incremental marking only for large heaps. | 4570 // TODO(ulan): Start incremental marking only for large heaps. |
4578 intptr_t limit = old_generation_allocation_limit_; | 4571 intptr_t limit = old_generation_allocation_limit_; |
4579 if (static_cast<size_t>(idle_time_in_ms) > | 4572 if (long_idle_time) { |
4580 GCIdleTimeHandler::kMaxFrameRenderingIdleTime) { | |
4581 limit = idle_old_generation_allocation_limit_; | 4573 limit = idle_old_generation_allocation_limit_; |
4582 } | 4574 } |
4583 | 4575 |
4584 heap_state.can_start_incremental_marking = | 4576 heap_state.can_start_incremental_marking = |
4585 incremental_marking()->CanBeActivated() && | 4577 incremental_marking()->CanBeActivated() && |
4586 HeapIsFullEnoughToStartIncrementalMarking(limit) && | 4578 HeapIsFullEnoughToStartIncrementalMarking(limit) && |
4587 !mark_compact_collector()->sweeping_in_progress(); | 4579 !mark_compact_collector()->sweeping_in_progress(); |
4588 heap_state.sweeping_in_progress = | 4580 heap_state.sweeping_in_progress = |
4589 mark_compact_collector()->sweeping_in_progress(); | 4581 mark_compact_collector()->sweeping_in_progress(); |
4590 heap_state.sweeping_completed = | 4582 heap_state.sweeping_completed = |
(...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
4625 IncrementalMarking::FORCE_MARKING, | 4617 IncrementalMarking::FORCE_MARKING, |
4626 IncrementalMarking::DO_NOT_FORCE_COMPLETION); | 4618 IncrementalMarking::DO_NOT_FORCE_COMPLETION); |
4627 remaining_idle_time_in_ms = | 4619 remaining_idle_time_in_ms = |
4628 deadline_in_ms - MonotonicallyIncreasingTimeInMs(); | 4620 deadline_in_ms - MonotonicallyIncreasingTimeInMs(); |
4629 } while (remaining_idle_time_in_ms >= | 4621 } while (remaining_idle_time_in_ms >= |
4630 2.0 * GCIdleTimeHandler::kIncrementalMarkingStepTimeInMs && | 4622 2.0 * GCIdleTimeHandler::kIncrementalMarkingStepTimeInMs && |
4631 !incremental_marking()->IsComplete() && | 4623 !incremental_marking()->IsComplete() && |
4632 !mark_compact_collector_.marking_deque()->IsEmpty()); | 4624 !mark_compact_collector_.marking_deque()->IsEmpty()); |
4633 if (remaining_idle_time_in_ms > 0.0) { | 4625 if (remaining_idle_time_in_ms > 0.0) { |
4634 action.additional_work = TryFinalizeIdleIncrementalMarking( | 4626 action.additional_work = TryFinalizeIdleIncrementalMarking( |
4635 remaining_idle_time_in_ms, heap_state.size_of_objects, | 4627 long_idle_time, remaining_idle_time_in_ms, |
4628 heap_state.size_of_objects, | |
4636 heap_state.final_incremental_mark_compact_speed_in_bytes_per_ms); | 4629 heap_state.final_incremental_mark_compact_speed_in_bytes_per_ms); |
4637 } | 4630 } |
4638 break; | 4631 break; |
4639 } | 4632 } |
4640 case DO_FULL_GC: { | 4633 case DO_FULL_GC: { |
4634 if (long_idle_time) isolate_->compilation_cache()->Clear(); | |
ulan
2015/04/28 10:26:47
Seems too aggressive. This should happen only if m
Hannes Payer (out of office)
2015/04/28 11:11:12
Done, guarded again with the gc count. But warning
| |
4641 if (contexts_disposed_) { | 4635 if (contexts_disposed_) { |
4642 HistogramTimerScope scope(isolate_->counters()->gc_context()); | 4636 HistogramTimerScope scope(isolate_->counters()->gc_context()); |
4643 CollectAllGarbage(kNoGCFlags, "idle notification: contexts disposed"); | 4637 CollectAllGarbage(kNoGCFlags, "idle notification: contexts disposed"); |
4644 gc_idle_time_handler_.NotifyIdleMarkCompact(); | |
4645 gc_count_at_last_idle_gc_ = gc_count_; | |
4646 } else { | 4638 } else { |
4647 IdleMarkCompact("idle notification: finalize idle round"); | 4639 CollectAllGarbage(kReduceMemoryFootprintMask, |
4640 "idle notification: finalize idle round"); | |
4648 } | 4641 } |
4642 ReduceNewSpaceSize(long_idle_time); | |
4643 gc_idle_time_handler_.NotifyIdleMarkCompact(); | |
4649 break; | 4644 break; |
4650 } | 4645 } |
4651 case DO_SCAVENGE: | 4646 case DO_SCAVENGE: |
4652 CollectGarbage(NEW_SPACE, "idle notification: scavenge"); | 4647 CollectGarbage(NEW_SPACE, "idle notification: scavenge"); |
4648 ReduceNewSpaceSize(idle_time_in_ms); | |
4653 break; | 4649 break; |
4654 case DO_FINALIZE_SWEEPING: | 4650 case DO_FINALIZE_SWEEPING: |
4655 mark_compact_collector()->EnsureSweepingCompleted(); | 4651 mark_compact_collector()->EnsureSweepingCompleted(); |
4656 break; | 4652 break; |
4657 case DO_NOTHING: | 4653 case DO_NOTHING: |
4658 break; | 4654 break; |
4659 } | 4655 } |
4660 | 4656 |
4661 double current_time = MonotonicallyIncreasingTimeInMs(); | 4657 double current_time = MonotonicallyIncreasingTimeInMs(); |
4662 last_idle_notification_time_ = current_time; | 4658 last_idle_notification_time_ = current_time; |
(...skipping 1718 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
6381 static_cast<int>(object_sizes_last_time_[index])); | 6377 static_cast<int>(object_sizes_last_time_[index])); |
6382 CODE_AGE_LIST_COMPLETE(ADJUST_LAST_TIME_OBJECT_COUNT) | 6378 CODE_AGE_LIST_COMPLETE(ADJUST_LAST_TIME_OBJECT_COUNT) |
6383 #undef ADJUST_LAST_TIME_OBJECT_COUNT | 6379 #undef ADJUST_LAST_TIME_OBJECT_COUNT |
6384 | 6380 |
6385 MemCopy(object_counts_last_time_, object_counts_, sizeof(object_counts_)); | 6381 MemCopy(object_counts_last_time_, object_counts_, sizeof(object_counts_)); |
6386 MemCopy(object_sizes_last_time_, object_sizes_, sizeof(object_sizes_)); | 6382 MemCopy(object_sizes_last_time_, object_sizes_, sizeof(object_sizes_)); |
6387 ClearObjectStats(); | 6383 ClearObjectStats(); |
6388 } | 6384 } |
6389 } | 6385 } |
6390 } // namespace v8::internal | 6386 } // namespace v8::internal |
OLD | NEW |