Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(294)

Side by Side Diff: src/heap/heap.cc

Issue 1108133003: Shrink new space and uncommit from space in idle notification during long idle times. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Created 5 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « src/heap/heap.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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
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
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
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
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
OLDNEW
« no previous file with comments | « src/heap/heap.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698