1 /*
2  * Copyright (C) 2012 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 #include "mod_union_table.h"
18 
19 #include <memory>
20 
21 #include "base/logging.h"  // For VLOG
22 #include "base/stl_util.h"
23 #include "bitmap-inl.h"
24 #include "card_table-inl.h"
25 #include "gc/accounting/space_bitmap-inl.h"
26 #include "gc/heap.h"
27 #include "gc/space/image_space.h"
28 #include "gc/space/space.h"
29 #include "mirror/object-inl.h"
30 #include "mirror/object-refvisitor-inl.h"
31 #include "object_callbacks.h"
32 #include "space_bitmap-inl.h"
33 #include "thread-current-inl.h"
34 
35 namespace art HIDDEN {
36 namespace gc {
37 namespace accounting {
38 
39 class ModUnionAddToCardSetVisitor {
40  public:
ModUnionAddToCardSetVisitor(ModUnionTable::CardSet * const cleared_cards)41   explicit ModUnionAddToCardSetVisitor(ModUnionTable::CardSet* const cleared_cards)
42       : cleared_cards_(cleared_cards) {}
43 
operator ()(uint8_t * card,uint8_t expected_value,uint8_t new_value) const44   inline void operator()(uint8_t* card,
45                          uint8_t expected_value,
46                          [[maybe_unused]] uint8_t new_value) const {
47     if (expected_value == CardTable::kCardDirty) {
48       cleared_cards_->insert(card);
49     }
50   }
51 
52  private:
53   ModUnionTable::CardSet* const cleared_cards_;
54 };
55 
56 class ModUnionAddToCardBitmapVisitor {
57  public:
ModUnionAddToCardBitmapVisitor(ModUnionTable::CardBitmap * bitmap,CardTable * card_table)58   ModUnionAddToCardBitmapVisitor(ModUnionTable::CardBitmap* bitmap, CardTable* card_table)
59       : bitmap_(bitmap), card_table_(card_table) {}
60 
operator ()(uint8_t * card,uint8_t expected_value,uint8_t new_value) const61   inline void operator()(uint8_t* card,
62                          uint8_t expected_value,
63                          [[maybe_unused]] uint8_t new_value) const {
64     if (expected_value == CardTable::kCardDirty) {
65       // We want the address the card represents, not the address of the card.
66       bitmap_->Set(reinterpret_cast<uintptr_t>(card_table_->AddrFromCard(card)));
67     }
68   }
69 
70  private:
71   ModUnionTable::CardBitmap* const bitmap_;
72   CardTable* const card_table_;
73 };
74 
75 class ModUnionAddToCardVectorVisitor {
76  public:
ModUnionAddToCardVectorVisitor(std::vector<uint8_t * > * cleared_cards)77   explicit ModUnionAddToCardVectorVisitor(std::vector<uint8_t*>* cleared_cards)
78       : cleared_cards_(cleared_cards) {
79   }
80 
operator ()(uint8_t * card,uint8_t expected_card,uint8_t new_card) const81   void operator()(uint8_t* card, uint8_t expected_card, [[maybe_unused]] uint8_t new_card) const {
82     if (expected_card == CardTable::kCardDirty) {
83       cleared_cards_->push_back(card);
84     }
85   }
86 
87  private:
88   std::vector<uint8_t*>* const cleared_cards_;
89 };
90 
91 class ModUnionUpdateObjectReferencesVisitor {
92  public:
ModUnionUpdateObjectReferencesVisitor(MarkObjectVisitor * visitor,space::ContinuousSpace * from_space,space::ContinuousSpace * immune_space,bool * contains_reference_to_other_space)93   ModUnionUpdateObjectReferencesVisitor(MarkObjectVisitor* visitor,
94                                         space::ContinuousSpace* from_space,
95                                         space::ContinuousSpace* immune_space,
96                                         bool* contains_reference_to_other_space)
97     : visitor_(visitor),
98       from_space_(from_space),
99       immune_space_(immune_space),
100       contains_reference_to_other_space_(contains_reference_to_other_space) {}
101 
102   // Extra parameters are required since we use this same visitor signature for checking objects.
operator ()(mirror::Object * obj,MemberOffset offset,bool is_static) const103   void operator()(mirror::Object* obj, MemberOffset offset, [[maybe_unused]] bool is_static) const
104       REQUIRES_SHARED(Locks::mutator_lock_) {
105     MarkReference(obj->GetFieldObjectReferenceAddr(offset));
106   }
107 
VisitRootIfNonNull(mirror::CompressedReference<mirror::Object> * root) const108   void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
109       REQUIRES_SHARED(Locks::mutator_lock_) {
110     VisitRoot(root);
111   }
112 
VisitRoot(mirror::CompressedReference<mirror::Object> * root) const113   void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
114       REQUIRES_SHARED(Locks::mutator_lock_) {
115     MarkReference(root);
116   }
117 
118  private:
119   template<typename CompressedReferenceType>
MarkReference(CompressedReferenceType * obj_ptr) const120   void MarkReference(CompressedReferenceType* obj_ptr) const
121       REQUIRES_SHARED(Locks::mutator_lock_) {
122     // Only add the reference if it is non null and fits our criteria.
123     mirror::Object* ref = obj_ptr->AsMirrorPtr();
124     if (ref != nullptr && !from_space_->HasAddress(ref) && !immune_space_->HasAddress(ref)) {
125       *contains_reference_to_other_space_ = true;
126       mirror::Object* new_object = visitor_->MarkObject(ref);
127       if (ref != new_object) {
128         obj_ptr->Assign(new_object);
129       }
130     }
131   }
132 
133   MarkObjectVisitor* const visitor_;
134   // Space which we are scanning
135   space::ContinuousSpace* const from_space_;
136   space::ContinuousSpace* const immune_space_;
137   // Set if we have any references to another space.
138   bool* const contains_reference_to_other_space_;
139 };
140 
141 class ModUnionScanImageRootVisitor {
142  public:
143   // Immune space is any other space which we don't care about references to. Currently this is
144   // the image space in the case of the zygote mod union table.
ModUnionScanImageRootVisitor(MarkObjectVisitor * visitor,space::ContinuousSpace * from_space,space::ContinuousSpace * immune_space,bool * contains_reference_to_other_space)145   ModUnionScanImageRootVisitor(MarkObjectVisitor* visitor,
146                                space::ContinuousSpace* from_space,
147                                space::ContinuousSpace* immune_space,
148                                bool* contains_reference_to_other_space)
149       : visitor_(visitor),
150         from_space_(from_space),
151         immune_space_(immune_space),
152         contains_reference_to_other_space_(contains_reference_to_other_space) {}
153 
operator ()(mirror::Object * root) const154   void operator()(mirror::Object* root) const
155       REQUIRES(Locks::heap_bitmap_lock_)
156       REQUIRES_SHARED(Locks::mutator_lock_) {
157     DCHECK(root != nullptr);
158     ModUnionUpdateObjectReferencesVisitor ref_visitor(visitor_,
159                                                       from_space_,
160                                                       immune_space_,
161                                                       contains_reference_to_other_space_);
162     root->VisitReferences(ref_visitor, VoidFunctor());
163   }
164 
165  private:
166   MarkObjectVisitor* const visitor_;
167   // Space which we are scanning
168   space::ContinuousSpace* const from_space_;
169   space::ContinuousSpace* const immune_space_;
170   // Set if we have any references to another space.
171   bool* const contains_reference_to_other_space_;
172 };
173 
ProcessCards()174 void ModUnionTableReferenceCache::ProcessCards() {
175   CardTable* card_table = GetHeap()->GetCardTable();
176   ModUnionAddToCardSetVisitor visitor(&cleared_cards_);
177   // Clear dirty cards in the this space and update the corresponding mod-union bits.
178   card_table->ModifyCardsAtomic(space_->Begin(), space_->End(), AgeCardVisitor(), visitor);
179 }
180 
ClearTable()181 void ModUnionTableReferenceCache::ClearTable() {
182   cleared_cards_.clear();
183   references_.clear();
184 }
185 
186 class AddToReferenceArrayVisitor {
187  public:
AddToReferenceArrayVisitor(ModUnionTableReferenceCache * mod_union_table,MarkObjectVisitor * visitor,std::vector<mirror::HeapReference<mirror::Object> * > * references,bool * has_target_reference)188   AddToReferenceArrayVisitor(ModUnionTableReferenceCache* mod_union_table,
189                              MarkObjectVisitor* visitor,
190                              std::vector<mirror::HeapReference<mirror::Object>*>* references,
191                              bool* has_target_reference)
192       : mod_union_table_(mod_union_table),
193         visitor_(visitor),
194         references_(references),
195         has_target_reference_(has_target_reference) {}
196 
197   // Extra parameters are required since we use this same visitor signature for checking objects.
operator ()(mirror::Object * obj,MemberOffset offset,bool is_static) const198   void operator()(mirror::Object* obj, MemberOffset offset, [[maybe_unused]] bool is_static) const
199       REQUIRES_SHARED(Locks::mutator_lock_) {
200     mirror::HeapReference<mirror::Object>* ref_ptr = obj->GetFieldObjectReferenceAddr(offset);
201     mirror::Object* ref = ref_ptr->AsMirrorPtr();
202     // Only add the reference if it is non null and fits our criteria.
203     if (ref != nullptr && mod_union_table_->ShouldAddReference(ref)) {
204       // Push the adddress of the reference.
205       references_->push_back(ref_ptr);
206     }
207   }
208 
VisitRootIfNonNull(mirror::CompressedReference<mirror::Object> * root) const209   void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
210       REQUIRES_SHARED(Locks::mutator_lock_) {
211     if (!root->IsNull()) {
212       VisitRoot(root);
213     }
214   }
215 
VisitRoot(mirror::CompressedReference<mirror::Object> * root) const216   void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
217       REQUIRES_SHARED(Locks::mutator_lock_) {
218     if (mod_union_table_->ShouldAddReference(root->AsMirrorPtr())) {
219       *has_target_reference_ = true;
220       // TODO: Add MarkCompressedReference callback here.
221       mirror::Object* old_ref = root->AsMirrorPtr();
222       mirror::Object* new_ref = visitor_->MarkObject(old_ref);
223       if (old_ref != new_ref) {
224         root->Assign(new_ref);
225       }
226     }
227   }
228 
229  private:
230   ModUnionTableReferenceCache* const mod_union_table_;
231   MarkObjectVisitor* const visitor_;
232   std::vector<mirror::HeapReference<mirror::Object>*>* const references_;
233   bool* const has_target_reference_;
234 };
235 
236 class ModUnionReferenceVisitor {
237  public:
ModUnionReferenceVisitor(ModUnionTableReferenceCache * const mod_union_table,MarkObjectVisitor * visitor,std::vector<mirror::HeapReference<mirror::Object> * > * references,bool * has_target_reference)238   ModUnionReferenceVisitor(ModUnionTableReferenceCache* const mod_union_table,
239                            MarkObjectVisitor* visitor,
240                            std::vector<mirror::HeapReference<mirror::Object>*>* references,
241                            bool* has_target_reference)
242       : mod_union_table_(mod_union_table),
243         visitor_(visitor),
244         references_(references),
245         has_target_reference_(has_target_reference) {}
246 
operator ()(mirror::Object * obj) const247   void operator()(mirror::Object* obj) const
248       REQUIRES_SHARED(Locks::heap_bitmap_lock_, Locks::mutator_lock_) {
249     // We don't have an early exit since we use the visitor pattern, an early
250     // exit should significantly speed this up.
251     AddToReferenceArrayVisitor visitor(mod_union_table_,
252                                        visitor_,
253                                        references_,
254                                        has_target_reference_);
255     obj->VisitReferences(visitor, VoidFunctor());
256   }
257 
258  private:
259   ModUnionTableReferenceCache* const mod_union_table_;
260   MarkObjectVisitor* const visitor_;
261   std::vector<mirror::HeapReference<mirror::Object>*>* const references_;
262   bool* const has_target_reference_;
263 };
264 
265 class CheckReferenceVisitor {
266  public:
CheckReferenceVisitor(ModUnionTableReferenceCache * mod_union_table,const std::set<mirror::Object * > & references)267   CheckReferenceVisitor(ModUnionTableReferenceCache* mod_union_table,
268                         const std::set<mirror::Object*>& references)
269       : mod_union_table_(mod_union_table),
270         references_(references) {}
271 
272   // Extra parameters are required since we use this same visitor signature for checking objects.
operator ()(mirror::Object * obj,MemberOffset offset,bool is_static) const273   void operator()(mirror::Object* obj, MemberOffset offset, [[maybe_unused]] bool is_static) const
274       REQUIRES_SHARED(Locks::heap_bitmap_lock_, Locks::mutator_lock_) {
275     mirror::Object* ref = obj->GetFieldObject<mirror::Object>(offset);
276     if (ref != nullptr &&
277         mod_union_table_->ShouldAddReference(ref) &&
278         references_.find(ref) == references_.end()) {
279       Heap* heap = mod_union_table_->GetHeap();
280       space::ContinuousSpace* from_space = heap->FindContinuousSpaceFromObject(obj, false);
281       space::ContinuousSpace* to_space = heap->FindContinuousSpaceFromObject(ref, false);
282       LOG(INFO) << "Object " << reinterpret_cast<const void*>(obj) << "(" << obj->PrettyTypeOf()
283                 << ")" << "References "
284                 << reinterpret_cast<const void*>(ref) << "(" << mirror::Object::PrettyTypeOf(ref)
285           << ") without being in mod-union table";
286       LOG(INFO) << "FromSpace " << from_space->GetName() << " type "
287           << from_space->GetGcRetentionPolicy();
288       LOG(INFO) << "ToSpace " << to_space->GetName() << " type "
289           << to_space->GetGcRetentionPolicy();
290       heap->DumpSpaces(LOG_STREAM(INFO));
291       LOG(FATAL) << "FATAL ERROR";
292     }
293   }
294 
VisitRootIfNonNull(mirror::CompressedReference<mirror::Object> * root) const295   void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
296       REQUIRES_SHARED(Locks::mutator_lock_) {
297     if (kIsDebugBuild && !root->IsNull()) {
298       VisitRoot(root);
299     }
300   }
301 
VisitRoot(mirror::CompressedReference<mirror::Object> * root) const302   void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
303       REQUIRES_SHARED(Locks::mutator_lock_) {
304     DCHECK(!mod_union_table_->ShouldAddReference(root->AsMirrorPtr()));
305   }
306 
307  private:
308   ModUnionTableReferenceCache* const mod_union_table_;
309   const std::set<mirror::Object*>& references_;
310 };
311 
312 class ModUnionCheckReferences {
313  public:
ModUnionCheckReferences(ModUnionTableReferenceCache * mod_union_table,const std::set<mirror::Object * > & references)314   ModUnionCheckReferences(ModUnionTableReferenceCache* mod_union_table,
315                           const std::set<mirror::Object*>& references)
316       REQUIRES(Locks::heap_bitmap_lock_)
317       : mod_union_table_(mod_union_table), references_(references) {}
318 
operator ()(mirror::Object * obj) const319   void operator()(mirror::Object* obj) const NO_THREAD_SAFETY_ANALYSIS {
320     Locks::heap_bitmap_lock_->AssertSharedHeld(Thread::Current());
321     CheckReferenceVisitor visitor(mod_union_table_, references_);
322     obj->VisitReferences(visitor, VoidFunctor());
323   }
324 
325  private:
326   ModUnionTableReferenceCache* const mod_union_table_;
327   const std::set<mirror::Object*>& references_;
328 };
329 
330 class EmptyMarkObjectVisitor : public MarkObjectVisitor {
331  public:
MarkObject(mirror::Object * obj)332   mirror::Object* MarkObject(mirror::Object* obj) override {return obj;}
MarkHeapReference(mirror::HeapReference<mirror::Object> *,bool)333   void MarkHeapReference(mirror::HeapReference<mirror::Object>*, bool) override {}
334 };
335 
FilterCards()336 void ModUnionTable::FilterCards() {
337   EmptyMarkObjectVisitor visitor;
338   // Use empty visitor since filtering is automatically done by UpdateAndMarkReferences.
339   UpdateAndMarkReferences(&visitor);
340 }
341 
Verify()342 void ModUnionTableReferenceCache::Verify() {
343   // Start by checking that everything in the mod union table is marked.
344   for (const auto& ref_pair : references_) {
345     for (mirror::HeapReference<mirror::Object>* ref : ref_pair.second) {
346       CHECK(heap_->IsLiveObjectLocked(ref->AsMirrorPtr()));
347     }
348   }
349 
350   // Check the references of each clean card which is also in the mod union table.
351   CardTable* card_table = heap_->GetCardTable();
352   ContinuousSpaceBitmap* live_bitmap = space_->GetLiveBitmap();
353   for (const auto& ref_pair : references_) {
354     const uint8_t* card = ref_pair.first;
355     if (*card == CardTable::kCardClean) {
356       std::set<mirror::Object*> reference_set;
357       for (mirror::HeapReference<mirror::Object>* obj_ptr : ref_pair.second) {
358         reference_set.insert(obj_ptr->AsMirrorPtr());
359       }
360       ModUnionCheckReferences visitor(this, reference_set);
361       uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card));
362       live_bitmap->VisitMarkedRange(start, start + CardTable::kCardSize, visitor);
363     }
364   }
365 }
366 
Dump(std::ostream & os)367 void ModUnionTableReferenceCache::Dump(std::ostream& os) {
368   CardTable* card_table = heap_->GetCardTable();
369   os << "ModUnionTable cleared cards: [";
370   for (uint8_t* card_addr : cleared_cards_) {
371     uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card_addr));
372     uintptr_t end = start + CardTable::kCardSize;
373     os << reinterpret_cast<void*>(start) << "-" << reinterpret_cast<void*>(end) << ",";
374   }
375   os << "]\nModUnionTable references: [";
376   for (const auto& ref_pair : references_) {
377     const uint8_t* card_addr = ref_pair.first;
378     uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card_addr));
379     uintptr_t end = start + CardTable::kCardSize;
380     os << reinterpret_cast<void*>(start) << "-" << reinterpret_cast<void*>(end) << "->{";
381     for (mirror::HeapReference<mirror::Object>* ref : ref_pair.second) {
382       os << reinterpret_cast<const void*>(ref->AsMirrorPtr()) << ",";
383     }
384     os << "},";
385   }
386 }
387 
VisitObjects(ObjectCallback callback,void * arg)388 void ModUnionTableReferenceCache::VisitObjects(ObjectCallback callback, void* arg) {
389   CardTable* const card_table = heap_->GetCardTable();
390   ContinuousSpaceBitmap* live_bitmap = space_->GetLiveBitmap();
391   // Use an unordered_set for constant time search of card in the second loop.
392   // We don't want to change cleared_cards_ to unordered so that traversals are
393   // sequential in address order.
394   // TODO: Optimize this.
395   std::unordered_set<const uint8_t*> card_lookup_map;
396   for (uint8_t* card : cleared_cards_) {
397     uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card));
398     uintptr_t end = start + CardTable::kCardSize;
399     live_bitmap->VisitMarkedRange(start,
400                                   end,
401                                   [callback, arg](mirror::Object* obj) {
402       callback(obj, arg);
403     });
404     card_lookup_map.insert(card);
405   }
406   for (const auto& pair : references_) {
407     const uint8_t* card = pair.first;
408     if (card_lookup_map.find(card) != card_lookup_map.end()) {
409       continue;
410     }
411     uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card));
412     uintptr_t end = start + CardTable::kCardSize;
413     live_bitmap->VisitMarkedRange(start,
414                                   end,
415                                   [callback, arg](mirror::Object* obj) {
416       callback(obj, arg);
417     });
418   }
419 }
420 
UpdateAndMarkReferences(MarkObjectVisitor * visitor)421 void ModUnionTableReferenceCache::UpdateAndMarkReferences(MarkObjectVisitor* visitor) {
422   CardTable* const card_table = heap_->GetCardTable();
423   std::vector<mirror::HeapReference<mirror::Object>*> cards_references;
424   // If has_target_reference is true then there was a GcRoot compressed reference which wasn't
425   // added. In this case we need to keep the card dirty.
426   // We don't know if the GcRoot addresses will remain constant, for example, classloaders have a
427   // hash set of GcRoot which may be resized or modified.
428   bool has_target_reference;
429   ModUnionReferenceVisitor add_visitor(this, visitor, &cards_references, &has_target_reference);
430   CardSet new_cleared_cards;
431   for (uint8_t* card : cleared_cards_) {
432     // Clear and re-compute alloc space references associated with this card.
433     cards_references.clear();
434     has_target_reference = false;
435     uintptr_t start = reinterpret_cast<uintptr_t>(card_table->AddrFromCard(card));
436     uintptr_t end = start + CardTable::kCardSize;
437     space::ContinuousSpace* space =
438         heap_->FindContinuousSpaceFromObject(reinterpret_cast<mirror::Object*>(start), false);
439     DCHECK(space != nullptr);
440     ContinuousSpaceBitmap* live_bitmap = space->GetLiveBitmap();
441     live_bitmap->VisitMarkedRange(start, end, add_visitor);
442     // Update the corresponding references for the card.
443     auto found = references_.find(card);
444     if (found == references_.end()) {
445       // Don't add card for an empty reference array.
446       if (!cards_references.empty()) {
447         references_.Put(card, cards_references);
448       }
449     } else {
450       if (cards_references.empty()) {
451         references_.erase(found);
452       } else {
453         found->second = cards_references;
454       }
455     }
456     if (has_target_reference) {
457       // Keep this card for next time since it contains a GcRoot which matches the
458       // ShouldAddReference criteria. This usually occurs for class loaders.
459       new_cleared_cards.insert(card);
460     }
461   }
462   cleared_cards_ = std::move(new_cleared_cards);
463   size_t count = 0;
464   for (auto it = references_.begin(); it != references_.end();) {
465     std::vector<mirror::HeapReference<mirror::Object>*>& references = it->second;
466     // Since there is no card mark for setting a reference to null, we check each reference.
467     // If all of the references of a card are null then we can remove that card. This is racy
468     // with the mutators, but handled by rescanning dirty cards.
469     bool all_null = true;
470     for (mirror::HeapReference<mirror::Object>* obj_ptr : references) {
471       if (obj_ptr->AsMirrorPtr() != nullptr) {
472         all_null = false;
473         visitor->MarkHeapReference(obj_ptr, /*do_atomic_update=*/ false);
474       }
475     }
476     count += references.size();
477     if (!all_null) {
478       ++it;
479     } else {
480       // All null references, erase the array from the set.
481       it = references_.erase(it);
482     }
483   }
484   if (VLOG_IS_ON(heap)) {
485     VLOG(gc) << "Marked " << count << " references in mod union table";
486   }
487 }
488 
ModUnionTableCardCache(const std::string & name,Heap * heap,space::ContinuousSpace * space)489 ModUnionTableCardCache::ModUnionTableCardCache(const std::string& name,
490                                                Heap* heap,
491                                                space::ContinuousSpace* space)
492     : ModUnionTable(name, heap, space) {
493   // Normally here we could use End() instead of Limit(), but for testing we may want to have a
494   // mod-union table for a space which can still grow.
495   if (!space->IsImageSpace()) {
496     CHECK_ALIGNED(reinterpret_cast<uintptr_t>(space->Limit()), CardTable::kCardSize);
497   }
498   card_bitmap_.reset(CardBitmap::Create(
499       "mod union bitmap", reinterpret_cast<uintptr_t>(space->Begin()),
500       RoundUp(reinterpret_cast<uintptr_t>(space->Limit()), CardTable::kCardSize)));
501 }
502 
503 class CardBitVisitor {
504  public:
CardBitVisitor(MarkObjectVisitor * visitor,space::ContinuousSpace * space,space::ContinuousSpace * immune_space,ModUnionTable::CardBitmap * card_bitmap)505   CardBitVisitor(MarkObjectVisitor* visitor,
506                  space::ContinuousSpace* space,
507                  space::ContinuousSpace* immune_space,
508                  ModUnionTable::CardBitmap* card_bitmap)
509       : visitor_(visitor),
510         space_(space),
511         immune_space_(immune_space),
512         bitmap_(space->GetLiveBitmap()),
513         card_bitmap_(card_bitmap) {
514     DCHECK(immune_space_ != nullptr);
515   }
516 
operator ()(size_t bit_index) const517   void operator()(size_t bit_index) const {
518     const uintptr_t start = card_bitmap_->AddrFromBitIndex(bit_index);
519     DCHECK(space_->HasAddress(reinterpret_cast<mirror::Object*>(start)))
520         << start << " " << *space_;
521     bool reference_to_other_space = false;
522     ModUnionScanImageRootVisitor scan_visitor(visitor_, space_, immune_space_,
523                                               &reference_to_other_space);
524     bitmap_->VisitMarkedRange(start, start + CardTable::kCardSize, scan_visitor);
525     if (!reference_to_other_space) {
526       // No non null reference to another space, clear the bit.
527       card_bitmap_->ClearBit(bit_index);
528     }
529   }
530 
531  private:
532   MarkObjectVisitor* const visitor_;
533   space::ContinuousSpace* const space_;
534   space::ContinuousSpace* const immune_space_;
535   ContinuousSpaceBitmap* const bitmap_;
536   ModUnionTable::CardBitmap* const card_bitmap_;
537 };
538 
ProcessCards()539 void ModUnionTableCardCache::ProcessCards() {
540   CardTable* const card_table = GetHeap()->GetCardTable();
541   ModUnionAddToCardBitmapVisitor visitor(card_bitmap_.get(), card_table);
542   // Clear dirty cards in the this space and update the corresponding mod-union bits.
543   card_table->ModifyCardsAtomic(space_->Begin(), space_->End(), AgeCardVisitor(), visitor);
544 }
545 
ClearTable()546 void ModUnionTableCardCache::ClearTable() {
547   card_bitmap_->Bitmap::Clear();
548 }
549 
550 // Mark all references to the alloc space(s).
UpdateAndMarkReferences(MarkObjectVisitor * visitor)551 void ModUnionTableCardCache::UpdateAndMarkReferences(MarkObjectVisitor* visitor) {
552   // TODO: Needs better support for multi-images? b/26317072
553   space::ImageSpace* image_space =
554       heap_->GetBootImageSpaces().empty() ? nullptr : heap_->GetBootImageSpaces()[0];
555   // If we don't have an image space, just pass in space_ as the immune space. Pass in the same
556   // space_ instead of image_space to avoid a null check in ModUnionUpdateObjectReferencesVisitor.
557   CardBitVisitor bit_visitor(visitor, space_, image_space != nullptr ? image_space : space_,
558       card_bitmap_.get());
559   card_bitmap_->VisitSetBits(
560       0, RoundUp(space_->Size(), CardTable::kCardSize) / CardTable::kCardSize, bit_visitor);
561 }
562 
VisitObjects(ObjectCallback callback,void * arg)563 void ModUnionTableCardCache::VisitObjects(ObjectCallback callback, void* arg) {
564   card_bitmap_->VisitSetBits(
565       0,
566       RoundUp(space_->Size(), CardTable::kCardSize) / CardTable::kCardSize,
567       [this, callback, arg](size_t bit_index) {
568         const uintptr_t start = card_bitmap_->AddrFromBitIndex(bit_index);
569         DCHECK(space_->HasAddress(reinterpret_cast<mirror::Object*>(start)))
570             << start << " " << *space_;
571         space_->GetLiveBitmap()->VisitMarkedRange(start,
572                                                   start + CardTable::kCardSize,
573                                                   [callback, arg](mirror::Object* obj) {
574           callback(obj, arg);
575         });
576       });
577 }
578 
Dump(std::ostream & os)579 void ModUnionTableCardCache::Dump(std::ostream& os) {
580   os << "ModUnionTable dirty cards: [";
581   // TODO: Find cleaner way of doing this.
582   for (uint8_t* addr = space_->Begin(); addr < AlignUp(space_->End(), CardTable::kCardSize);
583       addr += CardTable::kCardSize) {
584     if (card_bitmap_->Test(reinterpret_cast<uintptr_t>(addr))) {
585       os << reinterpret_cast<void*>(addr) << "-"
586          << reinterpret_cast<void*>(addr + CardTable::kCardSize) << "\n";
587     }
588   }
589   os << "]";
590 }
591 
SetCards()592 void ModUnionTableCardCache::SetCards() {
593   // Only clean up to the end since there cannot be any objects past the End() of the space.
594   for (uint8_t* addr = space_->Begin(); addr < AlignUp(space_->End(), CardTable::kCardSize);
595        addr += CardTable::kCardSize) {
596     card_bitmap_->Set(reinterpret_cast<uintptr_t>(addr));
597   }
598 }
599 
ContainsCardFor(uintptr_t addr)600 bool ModUnionTableCardCache::ContainsCardFor(uintptr_t addr) {
601   return card_bitmap_->Test(addr);
602 }
603 
SetCards()604 void ModUnionTableReferenceCache::SetCards() {
605   for (uint8_t* addr = space_->Begin(); addr < AlignUp(space_->End(), CardTable::kCardSize);
606        addr += CardTable::kCardSize) {
607     cleared_cards_.insert(heap_->GetCardTable()->CardFromAddr(reinterpret_cast<void*>(addr)));
608   }
609 }
610 
ContainsCardFor(uintptr_t addr)611 bool ModUnionTableReferenceCache::ContainsCardFor(uintptr_t addr) {
612   auto* card_ptr = heap_->GetCardTable()->CardFromAddr(reinterpret_cast<void*>(addr));
613   return cleared_cards_.find(card_ptr) != cleared_cards_.end() ||
614       references_.find(card_ptr) != references_.end();
615 }
616 
617 }  // namespace accounting
618 }  // namespace gc
619 }  // namespace art
620