blob: 5d21c599e412f4cfeeebe0499cf30c386b79b4e8 [file] [log] [blame]
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -07001/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_
18#define ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_
19
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080020#include "barrier.h"
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070021#include "garbage_collector.h"
Mathieu Chartier763a31e2015-11-16 16:05:55 -080022#include "immune_spaces.h"
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080023#include "jni.h"
24#include "object_callbacks.h"
25#include "offsets.h"
26#include "gc/accounting/atomic_stack.h"
27#include "gc/accounting/read_barrier_table.h"
28#include "gc/accounting/space_bitmap.h"
29#include "mirror/object.h"
30#include "mirror/object_reference.h"
31#include "safe_map.h"
32
33#include <unordered_map>
34#include <vector>
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070035
36namespace art {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080037class RootInfo;
38
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070039namespace gc {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080040
41namespace accounting {
42 typedef SpaceBitmap<kObjectAlignment> ContinuousSpaceBitmap;
43 class HeapBitmap;
44} // namespace accounting
45
46namespace space {
47 class RegionSpace;
48} // namespace space
49
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070050namespace collector {
51
52class ConcurrentCopying : public GarbageCollector {
53 public:
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080054 // TODO: disable thse flags for production use.
55 // Enable the no-from-space-refs verification at the pause.
56 static constexpr bool kEnableNoFromSpaceRefsVerification = true;
57 // Enable the from-space bytes/objects check.
58 static constexpr bool kEnableFromSpaceAccountingCheck = true;
59 // Enable verbose mode.
60 static constexpr bool kVerboseMode = true;
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070061
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080062 ConcurrentCopying(Heap* heap, const std::string& name_prefix = "");
63 ~ConcurrentCopying();
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070064
Mathieu Chartier90443472015-07-16 20:32:27 -070065 virtual void RunPhases() OVERRIDE REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
66 void InitializePhase() SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_);
67 void MarkingPhase() SHARED_REQUIRES(Locks::mutator_lock_)
68 REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
69 void ReclaimPhase() SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_);
70 void FinishPhase() REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080071
Mathieu Chartier90443472015-07-16 20:32:27 -070072 void BindBitmaps() SHARED_REQUIRES(Locks::mutator_lock_)
73 REQUIRES(!Locks::heap_bitmap_lock_);
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -070074 virtual GcType GetGcType() const OVERRIDE {
75 return kGcTypePartial;
76 }
77 virtual CollectorType GetCollectorType() const OVERRIDE {
78 return kCollectorTypeCC;
79 }
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080080 virtual void RevokeAllThreadLocalBuffers() OVERRIDE;
81 void SetRegionSpace(space::RegionSpace* region_space) {
82 DCHECK(region_space != nullptr);
83 region_space_ = region_space;
84 }
85 space::RegionSpace* RegionSpace() {
86 return region_space_;
87 }
88 void AssertToSpaceInvariant(mirror::Object* obj, MemberOffset offset, mirror::Object* ref)
Mathieu Chartier90443472015-07-16 20:32:27 -070089 SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi3f64f252015-06-12 18:35:06 -070090 void AssertToSpaceInvariant(GcRootSource* gc_root_source, mirror::Object* ref)
Mathieu Chartier90443472015-07-16 20:32:27 -070091 SHARED_REQUIRES(Locks::mutator_lock_);
92 bool IsInToSpace(mirror::Object* ref) SHARED_REQUIRES(Locks::mutator_lock_) {
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080093 DCHECK(ref != nullptr);
94 return IsMarked(ref) == ref;
95 }
Hiroshi Yamauchi723e6ce2015-10-28 20:59:47 -070096 ALWAYS_INLINE mirror::Object* Mark(mirror::Object* from_ref) SHARED_REQUIRES(Locks::mutator_lock_)
Mathieu Chartier90443472015-07-16 20:32:27 -070097 REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -080098 bool IsMarking() const {
99 return is_marking_;
100 }
101 bool IsActive() const {
102 return is_active_;
103 }
104 Barrier& GetBarrier() {
105 return *gc_barrier_;
106 }
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700107 bool IsWeakRefAccessEnabled() {
108 return weak_ref_access_enabled_.LoadRelaxed();
109 }
Mathieu Chartier90443472015-07-16 20:32:27 -0700110 void RevokeThreadLocalMarkStack(Thread* thread) SHARED_REQUIRES(Locks::mutator_lock_)
111 REQUIRES(!mark_stack_lock_);
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -0700112
113 private:
Mathieu Chartier90443472015-07-16 20:32:27 -0700114 void PushOntoMarkStack(mirror::Object* obj) SHARED_REQUIRES(Locks::mutator_lock_)
115 REQUIRES(!mark_stack_lock_);
116 mirror::Object* Copy(mirror::Object* from_ref) SHARED_REQUIRES(Locks::mutator_lock_)
117 REQUIRES(!skipped_blocks_lock_, !mark_stack_lock_);
118 void Scan(mirror::Object* to_ref) SHARED_REQUIRES(Locks::mutator_lock_)
119 REQUIRES(!mark_stack_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800120 void Process(mirror::Object* obj, MemberOffset offset)
Mathieu Chartier90443472015-07-16 20:32:27 -0700121 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_ , !skipped_blocks_lock_);
Mathieu Chartierbb87e0f2015-04-03 11:21:55 -0700122 virtual void VisitRoots(mirror::Object*** roots, size_t count, const RootInfo& info)
Mathieu Chartier90443472015-07-16 20:32:27 -0700123 OVERRIDE SHARED_REQUIRES(Locks::mutator_lock_)
124 REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Mathieu Chartierda7c6502015-07-23 16:01:26 -0700125 void MarkRoot(mirror::CompressedReference<mirror::Object>* root)
126 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Mathieu Chartierbb87e0f2015-04-03 11:21:55 -0700127 virtual void VisitRoots(mirror::CompressedReference<mirror::Object>** roots, size_t count,
128 const RootInfo& info)
Mathieu Chartier90443472015-07-16 20:32:27 -0700129 OVERRIDE SHARED_REQUIRES(Locks::mutator_lock_)
130 REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
131 void VerifyNoFromSpaceReferences() REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800132 accounting::ObjectStack* GetAllocationStack();
133 accounting::ObjectStack* GetLiveStack();
Mathieu Chartier90443472015-07-16 20:32:27 -0700134 virtual void ProcessMarkStack() OVERRIDE SHARED_REQUIRES(Locks::mutator_lock_)
135 REQUIRES(!mark_stack_lock_);
136 bool ProcessMarkStackOnce() SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_);
137 void ProcessMarkStackRef(mirror::Object* to_ref) SHARED_REQUIRES(Locks::mutator_lock_)
138 REQUIRES(!mark_stack_lock_);
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700139 size_t ProcessThreadLocalMarkStacks(bool disable_weak_ref_access)
Mathieu Chartier90443472015-07-16 20:32:27 -0700140 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_);
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700141 void RevokeThreadLocalMarkStacks(bool disable_weak_ref_access)
Mathieu Chartier90443472015-07-16 20:32:27 -0700142 SHARED_REQUIRES(Locks::mutator_lock_);
143 void SwitchToSharedMarkStackMode() SHARED_REQUIRES(Locks::mutator_lock_)
144 REQUIRES(!mark_stack_lock_);
145 void SwitchToGcExclusiveMarkStackMode() SHARED_REQUIRES(Locks::mutator_lock_);
Mathieu Chartier97509952015-07-13 14:35:43 -0700146 virtual void DelayReferenceReferent(mirror::Class* klass, mirror::Reference* reference) OVERRIDE
Mathieu Chartier90443472015-07-16 20:32:27 -0700147 SHARED_REQUIRES(Locks::mutator_lock_);
148 void ProcessReferences(Thread* self) SHARED_REQUIRES(Locks::mutator_lock_);
Mathieu Chartier97509952015-07-13 14:35:43 -0700149 virtual mirror::Object* MarkObject(mirror::Object* from_ref) OVERRIDE
Mathieu Chartier90443472015-07-16 20:32:27 -0700150 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Mathieu Chartier97509952015-07-13 14:35:43 -0700151 virtual void MarkHeapReference(mirror::HeapReference<mirror::Object>* from_ref) OVERRIDE
Mathieu Chartier90443472015-07-16 20:32:27 -0700152 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Mathieu Chartier97509952015-07-13 14:35:43 -0700153 virtual mirror::Object* IsMarked(mirror::Object* from_ref) OVERRIDE
Mathieu Chartier90443472015-07-16 20:32:27 -0700154 SHARED_REQUIRES(Locks::mutator_lock_);
Mathieu Chartier97509952015-07-13 14:35:43 -0700155 virtual bool IsMarkedHeapReference(mirror::HeapReference<mirror::Object>* field) OVERRIDE
Mathieu Chartier90443472015-07-16 20:32:27 -0700156 SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800157 void SweepSystemWeaks(Thread* self)
Mathieu Chartier90443472015-07-16 20:32:27 -0700158 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!Locks::heap_bitmap_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800159 void Sweep(bool swap_bitmaps)
Mathieu Chartier90443472015-07-16 20:32:27 -0700160 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_, !mark_stack_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800161 void SweepLargeObjects(bool swap_bitmaps)
Mathieu Chartier90443472015-07-16 20:32:27 -0700162 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800163 void ClearBlackPtrs()
Mathieu Chartier90443472015-07-16 20:32:27 -0700164 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800165 void FillWithDummyObject(mirror::Object* dummy_obj, size_t byte_size)
Mathieu Chartier90443472015-07-16 20:32:27 -0700166 SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800167 mirror::Object* AllocateInSkippedBlock(size_t alloc_size)
Mathieu Chartier90443472015-07-16 20:32:27 -0700168 SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!skipped_blocks_lock_);
169 void CheckEmptyMarkStack() SHARED_REQUIRES(Locks::mutator_lock_) REQUIRES(!mark_stack_lock_);
170 void IssueEmptyCheckpoint() SHARED_REQUIRES(Locks::mutator_lock_);
171 bool IsOnAllocStack(mirror::Object* ref) SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800172 mirror::Object* GetFwdPtr(mirror::Object* from_ref)
Mathieu Chartier90443472015-07-16 20:32:27 -0700173 SHARED_REQUIRES(Locks::mutator_lock_);
174 void FlipThreadRoots() REQUIRES(!Locks::mutator_lock_);
Mathieu Chartiera4f6af92015-08-11 17:35:25 -0700175 void SwapStacks() SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800176 void RecordLiveStackFreezeSize(Thread* self);
177 void ComputeUnevacFromSpaceLiveRatio();
Hiroshi Yamauchi3f64f252015-06-12 18:35:06 -0700178 void LogFromSpaceRefHolder(mirror::Object* obj, MemberOffset offset)
Mathieu Chartier90443472015-07-16 20:32:27 -0700179 SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi3f64f252015-06-12 18:35:06 -0700180 void AssertToSpaceInvariantInNonMovingSpace(mirror::Object* obj, mirror::Object* ref)
Mathieu Chartier90443472015-07-16 20:32:27 -0700181 SHARED_REQUIRES(Locks::mutator_lock_);
182 void ReenableWeakRefAccess(Thread* self) SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi00370822015-08-18 14:47:25 -0700183 void DisableMarking() SHARED_REQUIRES(Locks::mutator_lock_);
184 void IssueDisableMarkingCheckpoint() SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi19eab402015-10-23 19:59:58 -0700185 void ExpandGcMarkStack() SHARED_REQUIRES(Locks::mutator_lock_);
Hiroshi Yamauchi723e6ce2015-10-28 20:59:47 -0700186 mirror::Object* MarkNonMoving(mirror::Object* from_ref) SHARED_REQUIRES(Locks::mutator_lock_)
187 REQUIRES(!mark_stack_lock_, !skipped_blocks_lock_);
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800188
189 space::RegionSpace* region_space_; // The underlying region space.
190 std::unique_ptr<Barrier> gc_barrier_;
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700191 std::unique_ptr<accounting::ObjectStack> gc_mark_stack_;
192 Mutex mark_stack_lock_ DEFAULT_MUTEX_ACQUIRED_AFTER;
193 std::vector<accounting::ObjectStack*> revoked_mark_stacks_
194 GUARDED_BY(mark_stack_lock_);
195 static constexpr size_t kMarkStackSize = kPageSize;
196 static constexpr size_t kMarkStackPoolSize = 256;
197 std::vector<accounting::ObjectStack*> pooled_mark_stacks_
198 GUARDED_BY(mark_stack_lock_);
199 Thread* thread_running_gc_;
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800200 bool is_marking_; // True while marking is ongoing.
201 bool is_active_; // True while the collection is ongoing.
202 bool is_asserting_to_space_invariant_; // True while asserting the to-space invariant.
Mathieu Chartier763a31e2015-11-16 16:05:55 -0800203 ImmuneSpaces immune_spaces_;
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800204 std::unique_ptr<accounting::HeapBitmap> cc_heap_bitmap_;
205 std::vector<accounting::SpaceBitmap<kObjectAlignment>*> cc_bitmaps_;
206 accounting::SpaceBitmap<kObjectAlignment>* region_space_bitmap_;
207 // A cache of Heap::GetMarkBitmap().
208 accounting::HeapBitmap* heap_mark_bitmap_;
209 size_t live_stack_freeze_size_;
210 size_t from_space_num_objects_at_first_pause_;
211 size_t from_space_num_bytes_at_first_pause_;
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700212 Atomic<int> is_mark_stack_push_disallowed_;
213 enum MarkStackMode {
214 kMarkStackModeOff = 0, // Mark stack is off.
215 kMarkStackModeThreadLocal, // All threads except for the GC-running thread push refs onto
216 // thread-local mark stacks. The GC-running thread pushes onto and
217 // pops off the GC mark stack without a lock.
218 kMarkStackModeShared, // All threads share the GC mark stack with a lock.
219 kMarkStackModeGcExclusive // The GC-running thread pushes onto and pops from the GC mark stack
220 // without a lock. Other threads won't access the mark stack.
221 };
222 Atomic<MarkStackMode> mark_stack_mode_;
223 Atomic<bool> weak_ref_access_enabled_;
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800224
225 // How many objects and bytes we moved. Used for accounting.
226 Atomic<size_t> bytes_moved_;
227 Atomic<size_t> objects_moved_;
228
229 // The skipped blocks are memory blocks/chucks that were copies of
230 // objects that were unused due to lost races (cas failures) at
231 // object copy/forward pointer install. They are reused.
232 Mutex skipped_blocks_lock_ DEFAULT_MUTEX_ACQUIRED_AFTER;
233 std::multimap<size_t, uint8_t*> skipped_blocks_map_ GUARDED_BY(skipped_blocks_lock_);
234 Atomic<size_t> to_space_bytes_skipped_;
235 Atomic<size_t> to_space_objects_skipped_;
236
237 accounting::ReadBarrierTable* rb_table_;
238 bool force_evacuate_all_; // True if all regions are evacuated.
239
240 friend class ConcurrentCopyingRefFieldsVisitor;
241 friend class ConcurrentCopyingImmuneSpaceObjVisitor;
242 friend class ConcurrentCopyingVerifyNoFromSpaceRefsVisitor;
243 friend class ConcurrentCopyingVerifyNoFromSpaceRefsObjectVisitor;
244 friend class ConcurrentCopyingClearBlackPtrsVisitor;
245 friend class ConcurrentCopyingLostCopyVisitor;
246 friend class ThreadFlipVisitor;
247 friend class FlipCallback;
248 friend class ConcurrentCopyingComputeUnevacFromSpaceLiveRatioVisitor;
Hiroshi Yamauchi0b713572015-06-16 18:29:23 -0700249 friend class RevokeThreadLocalMarkStackCheckpoint;
Hiroshi Yamauchi2cd334a2015-01-09 14:03:35 -0800250
Mathieu Chartier3130cdf2015-05-03 15:20:23 -0700251 DISALLOW_IMPLICIT_CONSTRUCTORS(ConcurrentCopying);
Hiroshi Yamauchid5307ec2014-03-27 21:07:51 -0700252};
253
254} // namespace collector
255} // namespace gc
256} // namespace art
257
258#endif // ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_