blob: a87053d37893f93456e73827aad298ba495d6dc3 [file] [log] [blame]
/*
* Copyright (C) 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_
#define ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_
#include "barrier.h"
#include "garbage_collector.h"
#include "immune_region.h"
#include "jni.h"
#include "object_callbacks.h"
#include "offsets.h"
#include "gc/accounting/atomic_stack.h"
#include "gc/accounting/read_barrier_table.h"
#include "gc/accounting/space_bitmap.h"
#include "mirror/object.h"
#include "mirror/object_reference.h"
#include "safe_map.h"
#include <unordered_map>
#include <vector>
namespace art {
class RootInfo;
namespace gc {
namespace accounting {
typedef SpaceBitmap<kObjectAlignment> ContinuousSpaceBitmap;
class HeapBitmap;
} // namespace accounting
namespace space {
class RegionSpace;
} // namespace space
namespace collector {
// Concurrent queue. Used as the mark stack. TODO: use a concurrent
// stack for locality.
class MarkQueue {
public:
explicit MarkQueue(size_t size) : size_(size) {
CHECK(IsPowerOfTwo(size_));
buf_.reset(new Atomic<mirror::Object*>[size_]);
CHECK(buf_.get() != nullptr);
Clear();
}
ALWAYS_INLINE Atomic<mirror::Object*>* GetSlotAddr(size_t index) {
return &(buf_.get()[index & (size_ - 1)]);
}
// Multiple-proceducer enqueue.
bool Enqueue(mirror::Object* to_ref) {
size_t t;
do {
t = tail_.LoadRelaxed();
size_t h = head_.LoadSequentiallyConsistent();
if (t + size_ == h) {
// It's full.
return false;
}
} while (!tail_.CompareExchangeWeakSequentiallyConsistent(t, t + 1));
// We got a slot but its content has not been filled yet at this point.
GetSlotAddr(t)->StoreSequentiallyConsistent(to_ref);
return true;
}
// Thread-unsafe.
bool EnqueueThreadUnsafe(mirror::Object* to_ref) {
size_t t = tail_.LoadRelaxed();
size_t h = head_.LoadRelaxed();
if (t + size_ == h) {
// It's full.
return false;
}
GetSlotAddr(t)->StoreRelaxed(to_ref);
tail_.StoreRelaxed(t + 1);
return true;
}
// Single-consumer dequeue.
mirror::Object* Dequeue() {
size_t h = head_.LoadRelaxed();
size_t t = tail_.LoadSequentiallyConsistent();
if (h == t) {
// it's empty.
return nullptr;
}
Atomic<mirror::Object*>* slot = GetSlotAddr(h);
mirror::Object* ref = slot->LoadSequentiallyConsistent();
while (ref == nullptr) {
// Wait until the slot content becomes visible.
ref = slot->LoadSequentiallyConsistent();
}
slot->StoreRelaxed(nullptr);
head_.StoreSequentiallyConsistent(h + 1);
return ref;
}
bool IsEmpty() {
size_t h = head_.LoadSequentiallyConsistent();
size_t t = tail_.LoadSequentiallyConsistent();
return h == t;
}
void Clear() {
head_.StoreRelaxed(0);
tail_.StoreRelaxed(0);
memset(buf_.get(), 0, size_ * sizeof(Atomic<mirror::Object*>));
}
private:
Atomic<size_t> head_;
Atomic<size_t> tail_;
size_t size_;
std::unique_ptr<Atomic<mirror::Object*>[]> buf_;
};
class ConcurrentCopying : public GarbageCollector {
public:
// TODO: disable thse flags for production use.
// Enable the no-from-space-refs verification at the pause.
static constexpr bool kEnableNoFromSpaceRefsVerification = true;
// Enable the from-space bytes/objects check.
static constexpr bool kEnableFromSpaceAccountingCheck = true;
// Enable verbose mode.
static constexpr bool kVerboseMode = true;
ConcurrentCopying(Heap* heap, const std::string& name_prefix = "");
~ConcurrentCopying();
virtual void RunPhases() OVERRIDE;
void InitializePhase() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void MarkingPhase() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void ReclaimPhase() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void FinishPhase();
void BindBitmaps() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_)
LOCKS_EXCLUDED(Locks::heap_bitmap_lock_);
virtual GcType GetGcType() const OVERRIDE {
return kGcTypePartial;
}
virtual CollectorType GetCollectorType() const OVERRIDE {
return kCollectorTypeCC;
}
virtual void RevokeAllThreadLocalBuffers() OVERRIDE;
void SetRegionSpace(space::RegionSpace* region_space) {
DCHECK(region_space != nullptr);
region_space_ = region_space;
}
space::RegionSpace* RegionSpace() {
return region_space_;
}
void AssertToSpaceInvariant(mirror::Object* obj, MemberOffset offset, mirror::Object* ref)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
bool IsInToSpace(mirror::Object* ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
DCHECK(ref != nullptr);
return IsMarked(ref) == ref;
}
mirror::Object* Mark(mirror::Object* from_ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
bool IsMarking() const {
return is_marking_;
}
bool IsActive() const {
return is_active_;
}
Barrier& GetBarrier() {
return *gc_barrier_;
}
private:
mirror::Object* PopOffMarkStack();
template<bool kThreadSafe>
void PushOntoMarkStack(mirror::Object* obj) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
mirror::Object* Copy(mirror::Object* from_ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void Scan(mirror::Object* to_ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void Process(mirror::Object* obj, MemberOffset offset)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
virtual void VisitRoots(mirror::Object*** roots, size_t count, const RootInfo& info)
OVERRIDE SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
virtual void VisitRoots(mirror::CompressedReference<mirror::Object>** roots, size_t count,
const RootInfo& info)
OVERRIDE SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void VerifyNoFromSpaceReferences() EXCLUSIVE_LOCKS_REQUIRED(Locks::mutator_lock_);
accounting::ObjectStack* GetAllocationStack();
accounting::ObjectStack* GetLiveStack();
bool ProcessMarkStack() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void DelayReferenceReferent(mirror::Class* klass, mirror::Reference* reference)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void ProcessReferences(Thread* self, bool concurrent)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
mirror::Object* IsMarked(mirror::Object* from_ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
static mirror::Object* MarkCallback(mirror::Object* from_ref, void* arg)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
static mirror::Object* IsMarkedCallback(mirror::Object* from_ref, void* arg)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
static bool IsHeapReferenceMarkedCallback(
mirror::HeapReference<mirror::Object>* field, void* arg)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
static void ProcessMarkStackCallback(void* arg)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void SweepSystemWeaks(Thread* self)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) LOCKS_EXCLUDED(Locks::heap_bitmap_lock_);
void Sweep(bool swap_bitmaps)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) EXCLUSIVE_LOCKS_REQUIRED(Locks::heap_bitmap_lock_);
void SweepLargeObjects(bool swap_bitmaps)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) EXCLUSIVE_LOCKS_REQUIRED(Locks::heap_bitmap_lock_);
void ClearBlackPtrs()
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) EXCLUSIVE_LOCKS_REQUIRED(Locks::heap_bitmap_lock_);
void FillWithDummyObject(mirror::Object* dummy_obj, size_t byte_size)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
mirror::Object* AllocateInSkippedBlock(size_t alloc_size)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void CheckEmptyMarkQueue() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void IssueEmptyCheckpoint() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
bool IsOnAllocStack(mirror::Object* ref) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
mirror::Object* GetFwdPtr(mirror::Object* from_ref)
SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void FlipThreadRoots() LOCKS_EXCLUDED(Locks::mutator_lock_);;
void SwapStacks(Thread* self) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_);
void RecordLiveStackFreezeSize(Thread* self);
void ComputeUnevacFromSpaceLiveRatio();
space::RegionSpace* region_space_; // The underlying region space.
std::unique_ptr<Barrier> gc_barrier_;
MarkQueue mark_queue_;
bool is_marking_; // True while marking is ongoing.
bool is_active_; // True while the collection is ongoing.
bool is_asserting_to_space_invariant_; // True while asserting the to-space invariant.
ImmuneRegion immune_region_;
std::unique_ptr<accounting::HeapBitmap> cc_heap_bitmap_;
std::vector<accounting::SpaceBitmap<kObjectAlignment>*> cc_bitmaps_;
accounting::SpaceBitmap<kObjectAlignment>* region_space_bitmap_;
// A cache of Heap::GetMarkBitmap().
accounting::HeapBitmap* heap_mark_bitmap_;
size_t live_stack_freeze_size_;
size_t from_space_num_objects_at_first_pause_;
size_t from_space_num_bytes_at_first_pause_;
Atomic<int> is_mark_queue_push_disallowed_;
// How many objects and bytes we moved. Used for accounting.
Atomic<size_t> bytes_moved_;
Atomic<size_t> objects_moved_;
// The skipped blocks are memory blocks/chucks that were copies of
// objects that were unused due to lost races (cas failures) at
// object copy/forward pointer install. They are reused.
Mutex skipped_blocks_lock_ DEFAULT_MUTEX_ACQUIRED_AFTER;
std::multimap<size_t, uint8_t*> skipped_blocks_map_ GUARDED_BY(skipped_blocks_lock_);
Atomic<size_t> to_space_bytes_skipped_;
Atomic<size_t> to_space_objects_skipped_;
accounting::ReadBarrierTable* rb_table_;
bool force_evacuate_all_; // True if all regions are evacuated.
friend class ConcurrentCopyingRefFieldsVisitor;
friend class ConcurrentCopyingImmuneSpaceObjVisitor;
friend class ConcurrentCopyingVerifyNoFromSpaceRefsVisitor;
friend class ConcurrentCopyingVerifyNoFromSpaceRefsObjectVisitor;
friend class ConcurrentCopyingClearBlackPtrsVisitor;
friend class ConcurrentCopyingLostCopyVisitor;
friend class ThreadFlipVisitor;
friend class FlipCallback;
friend class ConcurrentCopyingComputeUnevacFromSpaceLiveRatioVisitor;
DISALLOW_COPY_AND_ASSIGN(ConcurrentCopying);
};
} // namespace collector
} // namespace gc
} // namespace art
#endif // ART_RUNTIME_GC_COLLECTOR_CONCURRENT_COPYING_H_