blob: cf80840a1ab51a3a5e2c2f697a1f70d1333f8f82 [file] [log] [blame]
/*
* Copyright 2021 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.
*/
#include <fcntl.h>
// Glibc v2.19 doesn't include these in fcntl.h so host builds will fail without.
#if !defined(FALLOC_FL_PUNCH_HOLE) || !defined(FALLOC_FL_KEEP_SIZE)
#include <linux/falloc.h>
#endif
#include <linux/userfaultfd.h>
#include <poll.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
#include <sys/resource.h>
#include <sys/stat.h>
#include <unistd.h>
#include <fstream>
#include <numeric>
#include <string>
#include <string_view>
#include <vector>
#include "android-base/file.h"
#include "android-base/parsebool.h"
#include "android-base/parseint.h"
#include "android-base/properties.h"
#include "android-base/strings.h"
#include "base/file_utils.h"
#include "base/memfd.h"
#include "base/quasi_atomic.h"
#include "base/systrace.h"
#include "base/utils.h"
#include "gc/accounting/mod_union_table-inl.h"
#include "gc/collector_type.h"
#include "gc/reference_processor.h"
#include "gc/space/bump_pointer_space.h"
#include "gc/task_processor.h"
#include "gc/verification-inl.h"
#include "jit/jit_code_cache.h"
#include "mark_compact-inl.h"
#include "mirror/object-refvisitor-inl.h"
#include "read_barrier_config.h"
#include "scoped_thread_state_change-inl.h"
#include "sigchain.h"
#include "thread_list.h"
#ifdef ART_TARGET_ANDROID
#include "com_android_art.h"
#endif
#ifndef __BIONIC__
#ifndef MREMAP_DONTUNMAP
#define MREMAP_DONTUNMAP 4
#endif
#ifndef MAP_FIXED_NOREPLACE
#define MAP_FIXED_NOREPLACE 0x100000
#endif
#ifndef __NR_userfaultfd
#if defined(__x86_64__)
#define __NR_userfaultfd 323
#elif defined(__i386__)
#define __NR_userfaultfd 374
#elif defined(__aarch64__)
#define __NR_userfaultfd 282
#elif defined(__arm__)
#define __NR_userfaultfd 388
#else
#error "__NR_userfaultfd undefined"
#endif
#endif // __NR_userfaultfd
#endif // __BIONIC__
#ifdef ART_TARGET_ANDROID
namespace {
using ::android::base::GetBoolProperty;
using ::android::base::ParseBool;
using ::android::base::ParseBoolResult;
} // namespace
#endif
namespace art {
static bool HaveMremapDontunmap() {
void* old = mmap(nullptr, gPageSize, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_SHARED, -1, 0);
CHECK_NE(old, MAP_FAILED);
void* addr = mremap(old, gPageSize, gPageSize, MREMAP_MAYMOVE | MREMAP_DONTUNMAP, nullptr);
CHECK_EQ(munmap(old, gPageSize), 0);
if (addr != MAP_FAILED) {
CHECK_EQ(munmap(addr, gPageSize), 0);
return true;
} else {
return false;
}
}
// We require MREMAP_DONTUNMAP functionality of the mremap syscall, which was
// introduced in 5.13 kernel version. But it was backported to GKI kernels.
static bool gHaveMremapDontunmap = IsKernelVersionAtLeast(5, 13) || HaveMremapDontunmap();
// Bitmap of features supported by userfaultfd. This is obtained via uffd API ioctl.
static uint64_t gUffdFeatures = 0;
// Both, missing and minor faults on shmem are needed only for minor-fault mode.
static constexpr uint64_t kUffdFeaturesForMinorFault =
UFFD_FEATURE_MISSING_SHMEM | UFFD_FEATURE_MINOR_SHMEM;
static constexpr uint64_t kUffdFeaturesForSigbus = UFFD_FEATURE_SIGBUS;
// We consider SIGBUS feature necessary to enable this GC as it's superior than
// threading-based implementation for janks. However, since we have the latter
// already implemented, for testing purposes, we allow choosing either of the
// two at boot time in the constructor below.
// We may want minor-fault in future to be available for making jit-code-cache
// updation concurrent, which uses shmem.
bool KernelSupportsUffd() {
#ifdef __linux__
if (gHaveMremapDontunmap) {
int fd = syscall(__NR_userfaultfd, O_CLOEXEC | UFFD_USER_MODE_ONLY);
// On non-android devices we may not have the kernel patches that restrict
// userfaultfd to user mode. But that is not a security concern as we are
// on host. Therefore, attempt one more time without UFFD_USER_MODE_ONLY.
if (!kIsTargetAndroid && fd == -1 && errno == EINVAL) {
fd = syscall(__NR_userfaultfd, O_CLOEXEC);
}
if (fd >= 0) {
// We are only fetching the available features, which is returned by the
// ioctl.
struct uffdio_api api = {.api = UFFD_API, .features = 0, .ioctls = 0};
CHECK_EQ(ioctl(fd, UFFDIO_API, &api), 0) << "ioctl_userfaultfd : API:" << strerror(errno);
gUffdFeatures = api.features;
close(fd);
// Minimum we need is sigbus feature for using userfaultfd.
return (api.features & kUffdFeaturesForSigbus) == kUffdFeaturesForSigbus;
}
}
#endif
return false;
}
// The other cases are defined as constexpr in runtime/read_barrier_config.h
#if !defined(ART_FORCE_USE_READ_BARRIER) && defined(ART_USE_READ_BARRIER)
// Returns collector type asked to be used on the cmdline.
static gc::CollectorType FetchCmdlineGcType() {
std::string argv;
gc::CollectorType gc_type = gc::CollectorType::kCollectorTypeNone;
if (android::base::ReadFileToString("/proc/self/cmdline", &argv)) {
if (argv.find("-Xgc:CMC") != std::string::npos) {
gc_type = gc::CollectorType::kCollectorTypeCMC;
} else if (argv.find("-Xgc:CC") != std::string::npos) {
gc_type = gc::CollectorType::kCollectorTypeCC;
}
}
return gc_type;
}
#ifdef ART_TARGET_ANDROID
static int GetOverrideCacheInfoFd() {
std::string args_str;
if (!android::base::ReadFileToString("/proc/self/cmdline", &args_str)) {
LOG(WARNING) << "Failed to load /proc/self/cmdline";
return -1;
}
std::vector<std::string_view> args;
Split(std::string_view(args_str), /*separator=*/'\0', &args);
for (std::string_view arg : args) {
if (android::base::ConsumePrefix(&arg, "--cache-info-fd=")) { // This is a dex2oat flag.
int fd;
if (!android::base::ParseInt(std::string(arg), &fd)) {
LOG(ERROR) << "Failed to parse --cache-info-fd (value: '" << arg << "')";
return -1;
}
return fd;
}
}
return -1;
}
static std::unordered_map<std::string, std::string> GetCachedProperties() {
// For simplicity, we don't handle multiple calls because otherwise we would have to reset the fd.
static bool called = false;
CHECK(!called) << "GetCachedBoolProperty can be called only once";
called = true;
std::string cache_info_contents;
int fd = GetOverrideCacheInfoFd();
if (fd >= 0) {
if (!android::base::ReadFdToString(fd, &cache_info_contents)) {
PLOG(ERROR) << "Failed to read cache-info from fd " << fd;
return {};
}
} else {
std::string path = GetApexDataDalvikCacheDirectory(InstructionSet::kNone) + "/cache-info.xml";
if (!android::base::ReadFileToString(path, &cache_info_contents)) {
// If the file is not found, then we are in chroot or in a standalone runtime process (e.g.,
// IncidentHelper), or odsign/odrefresh failed to generate and sign the cache info. There's
// nothing we can do.
if (errno != ENOENT) {
PLOG(ERROR) << "Failed to read cache-info from the default path";
}
return {};
}
}
std::optional<com::android::art::CacheInfo> cache_info =
com::android::art::parse(cache_info_contents.c_str());
if (!cache_info.has_value()) {
// This should never happen.
LOG(ERROR) << "Failed to parse cache-info";
return {};
}
const com::android::art::KeyValuePairList* list = cache_info->getFirstSystemProperties();
if (list == nullptr) {
// This should never happen.
LOG(ERROR) << "Missing system properties from cache-info";
return {};
}
const std::vector<com::android::art::KeyValuePair>& properties = list->getItem();
std::unordered_map<std::string, std::string> result;
for (const com::android::art::KeyValuePair& pair : properties) {
result[pair.getK()] = pair.getV();
}
return result;
}
static bool GetCachedBoolProperty(
const std::unordered_map<std::string, std::string>& cached_properties,
const std::string& key,
bool default_value) {
auto it = cached_properties.find(key);
if (it == cached_properties.end()) {
return default_value;
}
ParseBoolResult result = ParseBool(it->second);
switch (result) {
case ParseBoolResult::kTrue:
return true;
case ParseBoolResult::kFalse:
return false;
case ParseBoolResult::kError:
return default_value;
}
}
static bool SysPropSaysUffdGc() {
// The phenotype flag can change at time time after boot, but it shouldn't take effect until a
// reboot. Therefore, we read the phenotype flag from the cache info, which is generated on boot.
std::unordered_map<std::string, std::string> cached_properties = GetCachedProperties();
return !GetCachedBoolProperty(
cached_properties, "persist.device_config.runtime_native_boot.force_disable_uffd_gc", false);
}
#else
// Never called.
static bool SysPropSaysUffdGc() { return true; }
#endif
static bool ShouldUseUserfaultfd() {
static_assert(kUseBakerReadBarrier || kUseTableLookupReadBarrier);
#ifdef __linux__
// Use CMC/CC if that is being explicitly asked for on cmdline. Otherwise,
// always use CC on host. On target, use CMC only if system properties says so
// and the kernel supports it.
gc::CollectorType gc_type = FetchCmdlineGcType();
return gc_type == gc::CollectorType::kCollectorTypeCMC ||
(gc_type == gc::CollectorType::kCollectorTypeNone &&
kIsTargetAndroid &&
SysPropSaysUffdGc() &&
KernelSupportsUffd());
#else
return false;
#endif
}
const bool gUseUserfaultfd = ShouldUseUserfaultfd();
const bool gUseReadBarrier = !gUseUserfaultfd;
#endif
namespace gc {
namespace collector {
// Turn off kCheckLocks when profiling the GC as it slows down the GC
// significantly.
static constexpr bool kCheckLocks = kDebugLocking;
static constexpr bool kVerifyRootsMarked = kIsDebugBuild;
// Two threads should suffice on devices.
static constexpr size_t kMaxNumUffdWorkers = 2;
// Number of compaction buffers reserved for mutator threads in SIGBUS feature
// case. It's extremely unlikely that we will ever have more than these number
// of mutator threads trying to access the moving-space during one compaction
// phase.
static constexpr size_t kMutatorCompactionBufferCount = 2048;
// Minimum from-space chunk to be madvised (during concurrent compaction) in one go.
static constexpr ssize_t kMinFromSpaceMadviseSize = 1 * MB;
// Concurrent compaction termination logic is different (and slightly more efficient) if the
// kernel has the fault-retry feature (allowing repeated faults on the same page), which was
// introduced in 5.7 (https://android-review.git.corp.google.com/c/kernel/common/+/1540088).
// This allows a single page fault to be handled, in turn, by each worker thread, only waking
// up the GC thread at the end.
static const bool gKernelHasFaultRetry = IsKernelVersionAtLeast(5, 7);
std::pair<bool, bool> MarkCompact::GetUffdAndMinorFault() {
bool uffd_available;
// In most cases the gUffdFeatures will already be initialized at boot time
// when libart is loaded. On very old kernels we may get '0' from the kernel,
// in which case we would be doing the syscalls each time this function is
// called. But that's very unlikely case. There are no correctness issues as
// the response from kernel never changes after boot.
if (UNLIKELY(gUffdFeatures == 0)) {
uffd_available = KernelSupportsUffd();
} else {
// We can have any uffd features only if uffd exists.
uffd_available = true;
}
bool minor_fault_available =
(gUffdFeatures & kUffdFeaturesForMinorFault) == kUffdFeaturesForMinorFault;
return std::pair<bool, bool>(uffd_available, minor_fault_available);
}
bool MarkCompact::CreateUserfaultfd(bool post_fork) {
if (post_fork || uffd_ == kFdUnused) {
// Check if we have MREMAP_DONTUNMAP here for cases where
// 'ART_USE_READ_BARRIER=false' is used. Additionally, this check ensures
// that userfaultfd isn't used on old kernels, which cause random ioctl
// failures.
if (gHaveMremapDontunmap) {
// Don't use O_NONBLOCK as we rely on read waiting on uffd_ if there isn't
// any read event available. We don't use poll.
uffd_ = syscall(__NR_userfaultfd, O_CLOEXEC | UFFD_USER_MODE_ONLY);
// On non-android devices we may not have the kernel patches that restrict
// userfaultfd to user mode. But that is not a security concern as we are
// on host. Therefore, attempt one more time without UFFD_USER_MODE_ONLY.
if (!kIsTargetAndroid && UNLIKELY(uffd_ == -1 && errno == EINVAL)) {
uffd_ = syscall(__NR_userfaultfd, O_CLOEXEC);
}
if (UNLIKELY(uffd_ == -1)) {
uffd_ = kFallbackMode;
LOG(WARNING) << "Userfaultfd isn't supported (reason: " << strerror(errno)
<< ") and therefore falling back to stop-the-world compaction.";
} else {
DCHECK(IsValidFd(uffd_));
// Initialize uffd with the features which are required and available.
// Using private anonymous mapping in threading mode is the default,
// for which we don't need to ask for any features. Note: this mode
// is not used in production.
struct uffdio_api api = {.api = UFFD_API, .features = 0, .ioctls = 0};
if (use_uffd_sigbus_) {
// We should add SIGBUS feature only if we plan on using it as
// requesting it here will mean threading mode will not work.
CHECK_EQ(gUffdFeatures & kUffdFeaturesForSigbus, kUffdFeaturesForSigbus);
api.features |= kUffdFeaturesForSigbus;
}
if (uffd_minor_fault_supported_) {
// NOTE: This option is currently disabled.
CHECK_EQ(gUffdFeatures & kUffdFeaturesForMinorFault, kUffdFeaturesForMinorFault);
api.features |= kUffdFeaturesForMinorFault;
}
CHECK_EQ(ioctl(uffd_, UFFDIO_API, &api), 0)
<< "ioctl_userfaultfd: API: " << strerror(errno);
}
} else {
uffd_ = kFallbackMode;
}
}
uffd_initialized_ = !post_fork || uffd_ == kFallbackMode;
return IsValidFd(uffd_);
}
template <size_t kAlignment>
MarkCompact::LiveWordsBitmap<kAlignment>* MarkCompact::LiveWordsBitmap<kAlignment>::Create(
uintptr_t begin, uintptr_t end) {
return static_cast<LiveWordsBitmap<kAlignment>*>(
MemRangeBitmap::Create("Concurrent Mark Compact live words bitmap", begin, end));
}
static bool IsSigbusFeatureAvailable() {
MarkCompact::GetUffdAndMinorFault();
return (gUffdFeatures & kUffdFeaturesForSigbus) == kUffdFeaturesForSigbus;
}
size_t MarkCompact::InitializeInfoMap(uint8_t* p, size_t moving_space_sz) {
size_t nr_moving_pages = moving_space_sz / gPageSize;
chunk_info_vec_ = reinterpret_cast<uint32_t*>(p);
vector_length_ = moving_space_sz / kOffsetChunkSize;
size_t total = vector_length_ * sizeof(uint32_t);
first_objs_non_moving_space_ = reinterpret_cast<ObjReference*>(p + total);
total += heap_->GetNonMovingSpace()->Capacity() / gPageSize * sizeof(ObjReference);
first_objs_moving_space_ = reinterpret_cast<ObjReference*>(p + total);
total += nr_moving_pages * sizeof(ObjReference);
pre_compact_offset_moving_space_ = reinterpret_cast<uint32_t*>(p + total);
total += nr_moving_pages * sizeof(uint32_t);
return total;
}
MarkCompact::MarkCompact(Heap* heap)
: GarbageCollector(heap, "concurrent mark compact"),
gc_barrier_(0),
lock_("mark compact lock", kGenericBottomLock),
bump_pointer_space_(heap->GetBumpPointerSpace()),
moving_space_bitmap_(bump_pointer_space_->GetMarkBitmap()),
moving_space_begin_(bump_pointer_space_->Begin()),
moving_space_end_(bump_pointer_space_->Limit()),
moving_to_space_fd_(kFdUnused),
moving_from_space_fd_(kFdUnused),
uffd_(kFdUnused),
sigbus_in_progress_count_(kSigbusCounterCompactionDoneMask),
compaction_in_progress_count_(0),
thread_pool_counter_(0),
compacting_(false),
uffd_initialized_(false),
uffd_minor_fault_supported_(false),
use_uffd_sigbus_(IsSigbusFeatureAvailable()),
minor_fault_initialized_(false),
map_linear_alloc_shared_(false),
clamp_info_map_status_(ClampInfoStatus::kClampInfoNotDone) {
if (kIsDebugBuild) {
updated_roots_.reset(new std::unordered_set<void*>());
}
// TODO: When using minor-fault feature, the first GC after zygote-fork
// requires mapping the linear-alloc again with MAP_SHARED. This leaves a
// gap for suspended threads to access linear-alloc when it's empty (after
// mremap) and not yet userfaultfd registered. This cannot be fixed by merely
// doing uffd registration first. For now, just assert that we are not using
// minor-fault. Eventually, a cleanup of linear-alloc update logic to only
// use private anonymous would be ideal.
CHECK(!uffd_minor_fault_supported_);
// TODO: Depending on how the bump-pointer space move is implemented. If we
// switch between two virtual memories each time, then we will have to
// initialize live_words_bitmap_ accordingly.
live_words_bitmap_.reset(LiveWordsBitmap<kAlignment>::Create(
reinterpret_cast<uintptr_t>(bump_pointer_space_->Begin()),
reinterpret_cast<uintptr_t>(bump_pointer_space_->Limit())));
// Create one MemMap for all the data structures
size_t moving_space_size = bump_pointer_space_->Capacity();
size_t chunk_info_vec_size = moving_space_size / kOffsetChunkSize;
size_t nr_moving_pages = moving_space_size / gPageSize;
size_t nr_non_moving_pages = heap->GetNonMovingSpace()->Capacity() / gPageSize;
std::string err_msg;
info_map_ = MemMap::MapAnonymous("Concurrent mark-compact chunk-info vector",
chunk_info_vec_size * sizeof(uint32_t)
+ nr_non_moving_pages * sizeof(ObjReference)
+ nr_moving_pages * sizeof(ObjReference)
+ nr_moving_pages * sizeof(uint32_t),
PROT_READ | PROT_WRITE,
/*low_4gb=*/ false,
&err_msg);
if (UNLIKELY(!info_map_.IsValid())) {
LOG(FATAL) << "Failed to allocate concurrent mark-compact chunk-info vector: " << err_msg;
} else {
size_t total = InitializeInfoMap(info_map_.Begin(), moving_space_size);
DCHECK_EQ(total, info_map_.Size());
}
size_t moving_space_alignment = BestPageTableAlignment(moving_space_size);
// The moving space is created at a fixed address, which is expected to be
// PMD-size aligned.
if (!IsAlignedParam(bump_pointer_space_->Begin(), moving_space_alignment)) {
LOG(WARNING) << "Bump pointer space is not aligned to " << PrettySize(moving_space_alignment)
<< ". This can lead to longer stop-the-world pauses for compaction";
}
// NOTE: PROT_NONE is used here as these mappings are for address space reservation
// only and will be used only after appropriately remapping them.
from_space_map_ = MemMap::MapAnonymousAligned("Concurrent mark-compact from-space",
moving_space_size,
PROT_NONE,
/*low_4gb=*/kObjPtrPoisoning,
moving_space_alignment,
&err_msg);
if (UNLIKELY(!from_space_map_.IsValid())) {
LOG(FATAL) << "Failed to allocate concurrent mark-compact from-space" << err_msg;
} else {
from_space_begin_ = from_space_map_.Begin();
}
// In some cases (32-bit or kObjPtrPoisoning) it's too much to ask for 3
// heap-sized mappings in low-4GB. So tolerate failure here by attempting to
// mmap again right before the compaction pause. And if even that fails, then
// running the GC cycle in copy-mode rather than minor-fault.
//
// This map doesn't have to be aligned to 2MB as we don't mremap on it.
if (!kObjPtrPoisoning && uffd_minor_fault_supported_) {
// We need this map only if minor-fault feature is supported. But in that case
// don't create the mapping if obj-ptr poisoning is enabled as then the mapping
// has to be created in low_4gb. Doing this here rather than later causes the
// Dex2oatImageTest.TestExtension gtest to fail in 64-bit platforms.
shadow_to_space_map_ = MemMap::MapAnonymous("Concurrent mark-compact moving-space shadow",
moving_space_size,
PROT_NONE,
/*low_4gb=*/false,
&err_msg);
if (!shadow_to_space_map_.IsValid()) {
LOG(WARNING) << "Failed to allocate concurrent mark-compact moving-space shadow: " << err_msg;
}
}
const size_t num_pages =
1 + (use_uffd_sigbus_ ? kMutatorCompactionBufferCount :
std::min(heap_->GetParallelGCThreadCount(), kMaxNumUffdWorkers));
compaction_buffers_map_ = MemMap::MapAnonymous("Concurrent mark-compact compaction buffers",
gPageSize * num_pages,
PROT_READ | PROT_WRITE,
/*low_4gb=*/kObjPtrPoisoning,
&err_msg);
if (UNLIKELY(!compaction_buffers_map_.IsValid())) {
LOG(FATAL) << "Failed to allocate concurrent mark-compact compaction buffers" << err_msg;
}
// We also use the first page-sized buffer for the purpose of terminating concurrent compaction.
conc_compaction_termination_page_ = compaction_buffers_map_.Begin();
// Touch the page deliberately to avoid userfaults on it. We madvise it in
// CompactionPhase() before using it to terminate concurrent compaction.
ForceRead(conc_compaction_termination_page_);
// In most of the cases, we don't expect more than one LinearAlloc space.
linear_alloc_spaces_data_.reserve(1);
// Initialize GC metrics.
metrics::ArtMetrics* metrics = GetMetrics();
// The mark-compact collector supports only full-heap collections at the moment.
gc_time_histogram_ = metrics->FullGcCollectionTime();
metrics_gc_count_ = metrics->FullGcCount();
metrics_gc_count_delta_ = metrics->FullGcCountDelta();
gc_throughput_histogram_ = metrics->FullGcThroughput();
gc_tracing_throughput_hist_ = metrics->FullGcTracingThroughput();
gc_throughput_avg_ = metrics->FullGcThroughputAvg();
gc_tracing_throughput_avg_ = metrics->FullGcTracingThroughputAvg();
gc_scanned_bytes_ = metrics->FullGcScannedBytes();
gc_scanned_bytes_delta_ = metrics->FullGcScannedBytesDelta();
gc_freed_bytes_ = metrics->FullGcFreedBytes();
gc_freed_bytes_delta_ = metrics->FullGcFreedBytesDelta();
gc_duration_ = metrics->FullGcDuration();
gc_duration_delta_ = metrics->FullGcDurationDelta();
are_metrics_initialized_ = true;
}
void MarkCompact::AddLinearAllocSpaceData(uint8_t* begin, size_t len) {
DCHECK_ALIGNED_PARAM(begin, gPageSize);
DCHECK_ALIGNED_PARAM(len, gPageSize);
DCHECK_GE(len, gPMDSize);
size_t alignment = BestPageTableAlignment(len);
bool is_shared = false;
// We use MAP_SHARED on non-zygote processes for leveraging userfaultfd's minor-fault feature.
if (map_linear_alloc_shared_) {
void* ret = mmap(begin,
len,
PROT_READ | PROT_WRITE,
MAP_ANONYMOUS | MAP_SHARED | MAP_FIXED,
/*fd=*/-1,
/*offset=*/0);
CHECK_EQ(ret, begin) << "mmap failed: " << strerror(errno);
is_shared = true;
}
std::string err_msg;
MemMap shadow(MemMap::MapAnonymousAligned("linear-alloc shadow map",
len,
PROT_NONE,
/*low_4gb=*/false,
alignment,
&err_msg));
if (!shadow.IsValid()) {
LOG(FATAL) << "Failed to allocate linear-alloc shadow map: " << err_msg;
UNREACHABLE();
}
MemMap page_status_map(MemMap::MapAnonymous("linear-alloc page-status map",
len / gPageSize,
PROT_READ | PROT_WRITE,
/*low_4gb=*/false,
&err_msg));
if (!page_status_map.IsValid()) {
LOG(FATAL) << "Failed to allocate linear-alloc page-status shadow map: " << err_msg;
UNREACHABLE();
}
linear_alloc_spaces_data_.emplace_back(std::forward<MemMap>(shadow),
std::forward<MemMap>(page_status_map),
begin,
begin + len,
is_shared);
}
void MarkCompact::ClampGrowthLimit(size_t new_capacity) {
// From-space is the same size as moving-space in virtual memory.
// However, if it's in >4GB address space then we don't need to do it
// synchronously.
#if defined(__LP64__)
constexpr bool kClampFromSpace = kObjPtrPoisoning;
#else
constexpr bool kClampFromSpace = true;
#endif
size_t old_capacity = bump_pointer_space_->Capacity();
new_capacity = bump_pointer_space_->ClampGrowthLimit(new_capacity);
if (new_capacity < old_capacity) {
CHECK(from_space_map_.IsValid());
if (kClampFromSpace) {
from_space_map_.SetSize(new_capacity);
}
// NOTE: We usually don't use shadow_to_space_map_ and therefore the condition will
// mostly be false.
if (shadow_to_space_map_.IsValid() && shadow_to_space_map_.Size() > new_capacity) {
shadow_to_space_map_.SetSize(new_capacity);
}
clamp_info_map_status_ = ClampInfoStatus::kClampInfoPending;
}
CHECK_EQ(moving_space_begin_, bump_pointer_space_->Begin());
}
void MarkCompact::MaybeClampGcStructures() {
size_t moving_space_size = bump_pointer_space_->Capacity();
DCHECK(thread_running_gc_ != nullptr);
if (UNLIKELY(clamp_info_map_status_ == ClampInfoStatus::kClampInfoPending)) {
CHECK(from_space_map_.IsValid());
if (from_space_map_.Size() > moving_space_size) {
from_space_map_.SetSize(moving_space_size);
}
// Bitmaps and other data structures
live_words_bitmap_->SetBitmapSize(moving_space_size);
size_t set_size = InitializeInfoMap(info_map_.Begin(), moving_space_size);
CHECK_LT(set_size, info_map_.Size());
info_map_.SetSize(set_size);
clamp_info_map_status_ = ClampInfoStatus::kClampInfoFinished;
}
}
void MarkCompact::PrepareCardTableForMarking(bool clear_alloc_space_cards) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
accounting::CardTable* const card_table = heap_->GetCardTable();
// immune_spaces_ is emptied in InitializePhase() before marking starts. This
// function is invoked twice during marking. We only need to populate immune_spaces_
// once per GC cycle. And when it's done (below), all the immune spaces are
// added to it. We can never have partially filled immune_spaces_.
bool update_immune_spaces = immune_spaces_.IsEmpty();
// Mark all of the spaces we never collect as immune.
for (const auto& space : GetHeap()->GetContinuousSpaces()) {
if (space->GetGcRetentionPolicy() == space::kGcRetentionPolicyNeverCollect ||
space->GetGcRetentionPolicy() == space::kGcRetentionPolicyFullCollect) {
CHECK(space->IsZygoteSpace() || space->IsImageSpace());
if (update_immune_spaces) {
immune_spaces_.AddSpace(space);
}
accounting::ModUnionTable* table = heap_->FindModUnionTableFromSpace(space);
if (table != nullptr) {
table->ProcessCards();
} else {
// Keep cards aged if we don't have a mod-union table since we need
// to scan them in future GCs. This case is for app images.
card_table->ModifyCardsAtomic(
space->Begin(),
space->End(),
[](uint8_t card) {
return (card == gc::accounting::CardTable::kCardClean)
? card
: gc::accounting::CardTable::kCardAged;
},
/* card modified visitor */ VoidFunctor());
}
} else if (clear_alloc_space_cards) {
CHECK(!space->IsZygoteSpace());
CHECK(!space->IsImageSpace());
// The card-table corresponding to bump-pointer and non-moving space can
// be cleared, because we are going to traverse all the reachable objects
// in these spaces. This card-table will eventually be used to track
// mutations while concurrent marking is going on.
card_table->ClearCardRange(space->Begin(), space->Limit());
if (space != bump_pointer_space_) {
CHECK_EQ(space, heap_->GetNonMovingSpace());
non_moving_space_ = space;
non_moving_space_bitmap_ = space->GetMarkBitmap();
}
} else {
card_table->ModifyCardsAtomic(
space->Begin(),
space->End(),
[](uint8_t card) {
return (card == gc::accounting::CardTable::kCardDirty) ?
gc::accounting::CardTable::kCardAged :
gc::accounting::CardTable::kCardClean;
},
/* card modified visitor */ VoidFunctor());
}
}
}
void MarkCompact::MarkZygoteLargeObjects() {
Thread* self = thread_running_gc_;
DCHECK_EQ(self, Thread::Current());
space::LargeObjectSpace* const los = heap_->GetLargeObjectsSpace();
if (los != nullptr) {
// Pick the current live bitmap (mark bitmap if swapped).
accounting::LargeObjectBitmap* const live_bitmap = los->GetLiveBitmap();
accounting::LargeObjectBitmap* const mark_bitmap = los->GetMarkBitmap();
// Walk through all of the objects and explicitly mark the zygote ones so they don't get swept.
std::pair<uint8_t*, uint8_t*> range = los->GetBeginEndAtomic();
live_bitmap->VisitMarkedRange(reinterpret_cast<uintptr_t>(range.first),
reinterpret_cast<uintptr_t>(range.second),
[mark_bitmap, los, self](mirror::Object* obj)
REQUIRES(Locks::heap_bitmap_lock_)
REQUIRES_SHARED(Locks::mutator_lock_) {
if (los->IsZygoteLargeObject(self, obj)) {
mark_bitmap->Set(obj);
}
});
}
}
void MarkCompact::InitializePhase() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
mark_stack_ = heap_->GetMarkStack();
CHECK(mark_stack_->IsEmpty());
immune_spaces_.Reset();
moving_first_objs_count_ = 0;
non_moving_first_objs_count_ = 0;
black_page_count_ = 0;
bytes_scanned_ = 0;
freed_objects_ = 0;
// The first buffer is used by gc-thread.
compaction_buffer_counter_.store(1, std::memory_order_relaxed);
from_space_slide_diff_ = from_space_begin_ - bump_pointer_space_->Begin();
black_allocations_begin_ = bump_pointer_space_->Limit();
CHECK_EQ(moving_space_begin_, bump_pointer_space_->Begin());
moving_space_end_ = bump_pointer_space_->Limit();
walk_super_class_cache_ = nullptr;
// TODO: Would it suffice to read it once in the constructor, which is called
// in zygote process?
pointer_size_ = Runtime::Current()->GetClassLinker()->GetImagePointerSize();
}
class MarkCompact::ThreadFlipVisitor : public Closure {
public:
explicit ThreadFlipVisitor(MarkCompact* collector) : collector_(collector) {}
void Run(Thread* thread) override REQUIRES_SHARED(Locks::mutator_lock_) {
// Note: self is not necessarily equal to thread since thread may be suspended.
Thread* self = Thread::Current();
CHECK(thread == self || thread->GetState() != ThreadState::kRunnable)
<< thread->GetState() << " thread " << thread << " self " << self;
thread->VisitRoots(collector_, kVisitRootFlagAllRoots);
// Interpreter cache is thread-local so it needs to be swept either in a
// flip, or a stop-the-world pause.
CHECK(collector_->compacting_);
thread->SweepInterpreterCache(collector_);
thread->AdjustTlab(collector_->black_objs_slide_diff_);
}
private:
MarkCompact* const collector_;
};
class MarkCompact::FlipCallback : public Closure {
public:
explicit FlipCallback(MarkCompact* collector) : collector_(collector) {}
void Run([[maybe_unused]] Thread* thread) override REQUIRES(Locks::mutator_lock_) {
collector_->CompactionPause();
}
private:
MarkCompact* const collector_;
};
void MarkCompact::RunPhases() {
Thread* self = Thread::Current();
thread_running_gc_ = self;
Runtime* runtime = Runtime::Current();
InitializePhase();
GetHeap()->PreGcVerification(this);
{
ReaderMutexLock mu(self, *Locks::mutator_lock_);
MarkingPhase();
}
{
// Marking pause
ScopedPause pause(this);
MarkingPause();
if (kIsDebugBuild) {
bump_pointer_space_->AssertAllThreadLocalBuffersAreRevoked();
}
}
{
ReaderMutexLock mu(self, *Locks::mutator_lock_);
ReclaimPhase();
PrepareForCompaction();
}
if (uffd_ != kFallbackMode && !use_uffd_sigbus_) {
heap_->GetThreadPool()->WaitForWorkersToBeCreated();
}
{
// Compaction pause
ThreadFlipVisitor visitor(this);
FlipCallback callback(this);
runtime->GetThreadList()->FlipThreadRoots(
&visitor, &callback, this, GetHeap()->GetGcPauseListener());
}
if (IsValidFd(uffd_)) {
ReaderMutexLock mu(self, *Locks::mutator_lock_);
CompactionPhase();
}
FinishPhase();
thread_running_gc_ = nullptr;
}
void MarkCompact::InitMovingSpaceFirstObjects(const size_t vec_len) {
// Find the first live word first.
size_t to_space_page_idx = 0;
uint32_t offset_in_chunk_word;
uint32_t offset;
mirror::Object* obj;
const uintptr_t heap_begin = moving_space_bitmap_->HeapBegin();
size_t chunk_idx;
// Find the first live word in the space
for (chunk_idx = 0; chunk_info_vec_[chunk_idx] == 0; chunk_idx++) {
if (chunk_idx > vec_len) {
// We don't have any live data on the moving-space.
return;
}
}
// Use live-words bitmap to find the first word
offset_in_chunk_word = live_words_bitmap_->FindNthLiveWordOffset(chunk_idx, /*n*/ 0);
offset = chunk_idx * kBitsPerVectorWord + offset_in_chunk_word;
DCHECK(live_words_bitmap_->Test(offset)) << "offset=" << offset
<< " chunk_idx=" << chunk_idx
<< " N=0"
<< " offset_in_word=" << offset_in_chunk_word
<< " word=" << std::hex
<< live_words_bitmap_->GetWord(chunk_idx);
// The first object doesn't require using FindPrecedingObject().
obj = reinterpret_cast<mirror::Object*>(heap_begin + offset * kAlignment);
// TODO: add a check to validate the object.
pre_compact_offset_moving_space_[to_space_page_idx] = offset;
first_objs_moving_space_[to_space_page_idx].Assign(obj);
to_space_page_idx++;
uint32_t page_live_bytes = 0;
while (true) {
for (; page_live_bytes <= gPageSize; chunk_idx++) {
if (chunk_idx > vec_len) {
moving_first_objs_count_ = to_space_page_idx;
return;
}
page_live_bytes += chunk_info_vec_[chunk_idx];
}
chunk_idx--;
page_live_bytes -= gPageSize;
DCHECK_LE(page_live_bytes, kOffsetChunkSize);
DCHECK_LE(page_live_bytes, chunk_info_vec_[chunk_idx])
<< " chunk_idx=" << chunk_idx
<< " to_space_page_idx=" << to_space_page_idx
<< " vec_len=" << vec_len;
DCHECK(IsAligned<kAlignment>(chunk_info_vec_[chunk_idx] - page_live_bytes));
offset_in_chunk_word =
live_words_bitmap_->FindNthLiveWordOffset(
chunk_idx, (chunk_info_vec_[chunk_idx] - page_live_bytes) / kAlignment);
offset = chunk_idx * kBitsPerVectorWord + offset_in_chunk_word;
DCHECK(live_words_bitmap_->Test(offset))
<< "offset=" << offset
<< " chunk_idx=" << chunk_idx
<< " N=" << ((chunk_info_vec_[chunk_idx] - page_live_bytes) / kAlignment)
<< " offset_in_word=" << offset_in_chunk_word
<< " word=" << std::hex << live_words_bitmap_->GetWord(chunk_idx);
// TODO: Can we optimize this for large objects? If we are continuing a
// large object that spans multiple pages, then we may be able to do without
// calling FindPrecedingObject().
//
// Find the object which encapsulates offset in it, which could be
// starting at offset itself.
obj = moving_space_bitmap_->FindPrecedingObject(heap_begin + offset * kAlignment);
// TODO: add a check to validate the object.
pre_compact_offset_moving_space_[to_space_page_idx] = offset;
first_objs_moving_space_[to_space_page_idx].Assign(obj);
to_space_page_idx++;
chunk_idx++;
}
}
void MarkCompact::InitNonMovingSpaceFirstObjects() {
accounting::ContinuousSpaceBitmap* bitmap = non_moving_space_->GetLiveBitmap();
uintptr_t begin = reinterpret_cast<uintptr_t>(non_moving_space_->Begin());
const uintptr_t end = reinterpret_cast<uintptr_t>(non_moving_space_->End());
mirror::Object* prev_obj;
size_t page_idx;
{
// Find first live object
mirror::Object* obj = nullptr;
bitmap->VisitMarkedRange</*kVisitOnce*/ true>(begin,
end,
[&obj] (mirror::Object* o) {
obj = o;
});
if (obj == nullptr) {
// There are no live objects in the non-moving space
return;
}
page_idx = (reinterpret_cast<uintptr_t>(obj) - begin) / gPageSize;
first_objs_non_moving_space_[page_idx++].Assign(obj);
prev_obj = obj;
}
// TODO: check obj is valid
uintptr_t prev_obj_end = reinterpret_cast<uintptr_t>(prev_obj)
+ RoundUp(prev_obj->SizeOf<kDefaultVerifyFlags>(), kAlignment);
// For every page find the object starting from which we need to call
// VisitReferences. It could either be an object that started on some
// preceding page, or some object starting within this page.
begin = RoundDown(reinterpret_cast<uintptr_t>(prev_obj) + gPageSize, gPageSize);
while (begin < end) {
// Utilize, if any, large object that started in some preceding page, but
// overlaps with this page as well.
if (prev_obj != nullptr && prev_obj_end > begin) {
DCHECK_LT(prev_obj, reinterpret_cast<mirror::Object*>(begin));
first_objs_non_moving_space_[page_idx].Assign(prev_obj);
mirror::Class* klass = prev_obj->GetClass<kVerifyNone, kWithoutReadBarrier>();
if (HasAddress(klass)) {
LOG(WARNING) << "found inter-page object " << prev_obj
<< " in non-moving space with klass " << klass
<< " in moving space";
}
} else {
prev_obj_end = 0;
// It's sufficient to only search for previous object in the preceding page.
// If no live object started in that page and some object had started in
// the page preceding to that page, which was big enough to overlap with
// the current page, then we wouldn't be in the else part.
prev_obj = bitmap->FindPrecedingObject(begin, begin - gPageSize);
if (prev_obj != nullptr) {
prev_obj_end = reinterpret_cast<uintptr_t>(prev_obj)
+ RoundUp(prev_obj->SizeOf<kDefaultVerifyFlags>(), kAlignment);
}
if (prev_obj_end > begin) {
mirror::Class* klass = prev_obj->GetClass<kVerifyNone, kWithoutReadBarrier>();
if (HasAddress(klass)) {
LOG(WARNING) << "found inter-page object " << prev_obj
<< " in non-moving space with klass " << klass
<< " in moving space";
}
first_objs_non_moving_space_[page_idx].Assign(prev_obj);
} else {
// Find the first live object in this page
bitmap->VisitMarkedRange</*kVisitOnce*/ true>(
begin,
begin + gPageSize,
[this, page_idx] (mirror::Object* obj) {
first_objs_non_moving_space_[page_idx].Assign(obj);
});
}
// An empty entry indicates that the page has no live objects and hence
// can be skipped.
}
begin += gPageSize;
page_idx++;
}
non_moving_first_objs_count_ = page_idx;
}
bool MarkCompact::CanCompactMovingSpaceWithMinorFault() {
size_t min_size = (moving_first_objs_count_ + black_page_count_) * gPageSize;
return minor_fault_initialized_ && shadow_to_space_map_.IsValid() &&
shadow_to_space_map_.Size() >= min_size;
}
class MarkCompact::ConcurrentCompactionGcTask : public SelfDeletingTask {
public:
explicit ConcurrentCompactionGcTask(MarkCompact* collector, size_t idx)
: collector_(collector), index_(idx) {}
void Run([[maybe_unused]] Thread* self) override REQUIRES_SHARED(Locks::mutator_lock_) {
if (collector_->CanCompactMovingSpaceWithMinorFault()) {
collector_->ConcurrentCompaction<MarkCompact::kMinorFaultMode>(/*buf=*/nullptr);
} else {
// The passed page/buf to ConcurrentCompaction is used by the thread as a
// gPageSize buffer for compacting and updating objects into and then
// passing the buf to uffd ioctls.
uint8_t* buf = collector_->compaction_buffers_map_.Begin() + index_ * gPageSize;
collector_->ConcurrentCompaction<MarkCompact::kCopyMode>(buf);
}
}
private:
MarkCompact* const collector_;
size_t index_;
};
void MarkCompact::PrepareForCompaction() {
uint8_t* space_begin = bump_pointer_space_->Begin();
size_t vector_len = (black_allocations_begin_ - space_begin) / kOffsetChunkSize;
DCHECK_LE(vector_len, vector_length_);
for (size_t i = 0; i < vector_len; i++) {
DCHECK_LE(chunk_info_vec_[i], kOffsetChunkSize);
DCHECK_EQ(chunk_info_vec_[i], live_words_bitmap_->LiveBytesInBitmapWord(i));
}
InitMovingSpaceFirstObjects(vector_len);
InitNonMovingSpaceFirstObjects();
// TODO: We can do a lot of neat tricks with this offset vector to tune the
// compaction as we wish. Originally, the compaction algorithm slides all
// live objects towards the beginning of the heap. This is nice because it
// keeps the spatial locality of objects intact.
// However, sometimes it's desired to compact objects in certain portions
// of the heap. For instance, it is expected that, over time,
// objects towards the beginning of the heap are long lived and are always
// densely packed. In this case, it makes sense to only update references in
// there and not try to compact it.
// Furthermore, we might have some large objects and may not want to move such
// objects.
// We can adjust, without too much effort, the values in the chunk_info_vec_ such
// that the objects in the dense beginning area aren't moved. OTOH, large
// objects, which could be anywhere in the heap, could also be kept from
// moving by using a similar trick. The only issue is that by doing this we will
// leave an unused hole in the middle of the heap which can't be used for
// allocations until we do a *full* compaction.
//
// At this point every element in the chunk_info_vec_ contains the live-bytes
// of the corresponding chunk. For old-to-new address computation we need
// every element to reflect total live-bytes till the corresponding chunk.
// Live-bytes count is required to compute post_compact_end_ below.
uint32_t total;
// Update the vector one past the heap usage as it is required for black
// allocated objects' post-compact address computation.
if (vector_len < vector_length_) {
vector_len++;
total = 0;
} else {
// Fetch the value stored in the last element before it gets overwritten by
// std::exclusive_scan().
total = chunk_info_vec_[vector_len - 1];
}
std::exclusive_scan(chunk_info_vec_, chunk_info_vec_ + vector_len, chunk_info_vec_, 0);
total += chunk_info_vec_[vector_len - 1];
for (size_t i = vector_len; i < vector_length_; i++) {
DCHECK_EQ(chunk_info_vec_[i], 0u);
}
post_compact_end_ = AlignUp(space_begin + total, gPageSize);
CHECK_EQ(post_compact_end_, space_begin + moving_first_objs_count_ * gPageSize);
black_objs_slide_diff_ = black_allocations_begin_ - post_compact_end_;
// We shouldn't be consuming more space after compaction than pre-compaction.
CHECK_GE(black_objs_slide_diff_, 0);
// How do we handle compaction of heap portion used for allocations after the
// marking-pause?
// All allocations after the marking-pause are considered black (reachable)
// for this GC cycle. However, they need not be allocated contiguously as
// different mutators use TLABs. So we will compact the heap till the point
// where allocations took place before the marking-pause. And everything after
// that will be slid with TLAB holes, and then TLAB info in TLS will be
// appropriately updated in the pre-compaction pause.
// The chunk-info vector entries for the post marking-pause allocations will be
// also updated in the pre-compaction pause.
bool is_zygote = Runtime::Current()->IsZygote();
if (!uffd_initialized_ && CreateUserfaultfd(/*post_fork*/false)) {
if (!use_uffd_sigbus_) {
// Register the buffer that we use for terminating concurrent compaction
struct uffdio_register uffd_register;
uffd_register.range.start = reinterpret_cast<uintptr_t>(conc_compaction_termination_page_);
uffd_register.range.len = gPageSize;
uffd_register.mode = UFFDIO_REGISTER_MODE_MISSING;
CHECK_EQ(ioctl(uffd_, UFFDIO_REGISTER, &uffd_register), 0)
<< "ioctl_userfaultfd: register compaction termination page: " << strerror(errno);
}
if (!uffd_minor_fault_supported_ && shadow_to_space_map_.IsValid()) {
// A valid shadow-map for moving space is only possible if we
// were able to map it in the constructor. That also means that its size
// matches the moving-space.
CHECK_EQ(shadow_to_space_map_.Size(), bump_pointer_space_->Capacity());
// Release the shadow map for moving-space if we don't support minor-fault
// as it's not required.
shadow_to_space_map_.Reset();
}
}
// For zygote we create the thread pool each time before starting compaction,
// and get rid of it when finished. This is expected to happen rarely as
// zygote spends most of the time in native fork loop.
if (uffd_ != kFallbackMode) {
if (!use_uffd_sigbus_) {
ThreadPool* pool = heap_->GetThreadPool();
if (UNLIKELY(pool == nullptr)) {
// On devices with 2 cores, GetParallelGCThreadCount() will return 1,
// which is desired number of workers on such devices.
heap_->CreateThreadPool(std::min(heap_->GetParallelGCThreadCount(), kMaxNumUffdWorkers));
pool = heap_->GetThreadPool();
}
size_t num_threads = pool->GetThreadCount();
thread_pool_counter_ = num_threads;
for (size_t i = 0; i < num_threads; i++) {
pool->AddTask(thread_running_gc_, new ConcurrentCompactionGcTask(this, i + 1));
}
CHECK_EQ(pool->GetTaskCount(thread_running_gc_), num_threads);
}
/*
* Possible scenarios for mappings:
* A) All zygote GCs (or if minor-fault feature isn't available): uses
* uffd's copy mode
* 1) For moving-space ('to' space is same as the moving-space):
* a) Private-anonymous mappings for 'to' and 'from' space are created in
* the constructor.
* b) In the compaction pause, we mremap(dontunmap) from 'to' space to
* 'from' space. This results in moving all pages to 'from' space and
* emptying the 'to' space, thereby preparing it for userfaultfd
* registration.
*
* 2) For linear-alloc space:
* a) Private-anonymous mappings for the linear-alloc and its 'shadow'
* are created by the arena-pool.
* b) In the compaction pause, we mremap(dontumap) with similar effect as
* (A.1.b) above.
*
* B) First GC after zygote: uses uffd's copy-mode
* 1) For moving-space:
* a) If the mmap for shadow-map has been successful in the constructor,
* then we remap it (mmap with MAP_FIXED) to get a shared-anonymous
* mapping.
* b) Else, we create two memfd and ftruncate them to the moving-space
* size.
* c) Same as (A.1.b)
* d) If (B.1.a), then mremap(dontunmap) from shadow-map to
* 'to' space. This will make both of them map to the same pages
* e) If (B.1.b), then mmap with the first memfd in shared mode on the
* 'to' space.
* f) At the end of compaction, we will have moved the moving-space
* objects to a MAP_SHARED mapping, readying it for minor-fault from next
* GC cycle.
*
* 2) For linear-alloc space:
* a) Same as (A.2.b)
* b) mmap a shared-anonymous mapping onto the linear-alloc space.
* c) Same as (B.1.f)
*
* C) All subsequent GCs: preferable minor-fault mode. But may also require
* using copy-mode.
* 1) For moving-space:
* a) If the shadow-map is created and no memfd was used, then that means
* we are using shared-anonymous. Therefore, mmap a shared-anonymous on
* the shadow-space.
* b) If the shadow-map is not mapped yet, then mmap one with a size
* big enough to hold the compacted moving space. This may fail, in which
* case we will use uffd's copy-mode.
* c) If (b) is successful, then mmap the free memfd onto shadow-map.
* d) Same as (A.1.b)
* e) In compaction pause, if the shadow-map was not created, then use
* copy-mode.
* f) Else, if the created map is smaller than the required-size, then
* use mremap (without dontunmap) to expand the size. If failed, then use
* copy-mode.
* g) Otherwise, same as (B.1.d) and use minor-fault mode.
*
* 2) For linear-alloc space:
* a) Same as (A.2.b)
* b) Use minor-fault mode
*/
auto mmap_shadow_map = [this](int flags, int fd) {
void* ret = mmap(shadow_to_space_map_.Begin(),
shadow_to_space_map_.Size(),
PROT_READ | PROT_WRITE,
flags,
fd,
/*offset=*/0);
DCHECK_NE(ret, MAP_FAILED) << "mmap for moving-space shadow failed:" << strerror(errno);
};
// Setup all the virtual memory ranges required for concurrent compaction.
if (minor_fault_initialized_) {
DCHECK(!is_zygote);
if (UNLIKELY(!shadow_to_space_map_.IsValid())) {
// This case happens only once on the first GC in minor-fault mode, if
// we were unable to reserve shadow-map for moving-space in the
// beginning.
DCHECK_GE(moving_to_space_fd_, 0);
// Take extra 4MB to reduce the likelihood of requiring resizing this
// map in the pause due to black allocations.
size_t reqd_size = std::min(moving_first_objs_count_ * gPageSize + 4 * MB,
bump_pointer_space_->Capacity());
// We cannot support memory-tool with shadow-map (as it requires
// appending a redzone) in this case because the mapping may have to be expanded
// using mremap (in KernelPreparation()), which would ignore the redzone.
// MemMap::MapFile() appends a redzone, but MemMap::MapAnonymous() doesn't.
std::string err_msg;
shadow_to_space_map_ = MemMap::MapAnonymous("moving-space-shadow",
reqd_size,
PROT_NONE,
/*low_4gb=*/kObjPtrPoisoning,
&err_msg);
if (shadow_to_space_map_.IsValid()) {
CHECK(!kMemoryToolAddsRedzones || shadow_to_space_map_.GetRedzoneSize() == 0u);
// We want to use MemMap to get low-4GB mapping, if required, but then also
// want to have its ownership as we may grow it (in
// KernelPreparation()). If the ownership is not taken and we try to
// resize MemMap, then it unmaps the virtual range.
MemMap temp = shadow_to_space_map_.TakeReservedMemory(shadow_to_space_map_.Size(),
/*reuse*/ true);
std::swap(temp, shadow_to_space_map_);
DCHECK(!temp.IsValid());
} else {
LOG(WARNING) << "Failed to create moving space's shadow map of " << PrettySize(reqd_size)
<< " size. " << err_msg;
}
}
if (LIKELY(shadow_to_space_map_.IsValid())) {
int fd = moving_to_space_fd_;
int mmap_flags = MAP_SHARED | MAP_FIXED;
if (fd == kFdUnused) {
// Unused moving-to-space fd means we are using anonymous shared
// mapping.
DCHECK_EQ(shadow_to_space_map_.Size(), bump_pointer_space_->Capacity());
mmap_flags |= MAP_ANONYMOUS;
fd = -1;
}
// If the map is smaller than required, then we'll do mremap in the
// compaction pause to increase the size.
mmap_shadow_map(mmap_flags, fd);
}
for (auto& data : linear_alloc_spaces_data_) {
DCHECK_EQ(mprotect(data.shadow_.Begin(), data.shadow_.Size(), PROT_READ | PROT_WRITE), 0)
<< "mprotect failed: " << strerror(errno);
}
} else if (!is_zygote && uffd_minor_fault_supported_) {
// First GC after zygote-fork. We will still use uffd's copy mode but will
// use it to move objects to MAP_SHARED (to prepare for subsequent GCs, which
// will use uffd's minor-fault feature).
if (shadow_to_space_map_.IsValid() &&
shadow_to_space_map_.Size() == bump_pointer_space_->Capacity()) {
mmap_shadow_map(MAP_SHARED | MAP_FIXED | MAP_ANONYMOUS, /*fd=*/-1);
} else {
size_t size = bump_pointer_space_->Capacity();
DCHECK_EQ(moving_to_space_fd_, kFdUnused);
DCHECK_EQ(moving_from_space_fd_, kFdUnused);
const char* name = bump_pointer_space_->GetName();
moving_to_space_fd_ = memfd_create(name, MFD_CLOEXEC);
CHECK_NE(moving_to_space_fd_, -1)
<< "memfd_create: failed for " << name << ": " << strerror(errno);
moving_from_space_fd_ = memfd_create(name, MFD_CLOEXEC);
CHECK_NE(moving_from_space_fd_, -1)
<< "memfd_create: failed for " << name << ": " << strerror(errno);
// memfds are considered as files from resource limits point of view.
// And the moving space could be several hundred MBs. So increase the
// limit, if it's lower than moving-space size.
bool rlimit_changed = false;
rlimit rlim_read;
CHECK_EQ(getrlimit(RLIMIT_FSIZE, &rlim_read), 0) << "getrlimit failed: " << strerror(errno);
if (rlim_read.rlim_cur < size) {
rlimit_changed = true;
rlimit rlim = rlim_read;
rlim.rlim_cur = size;
CHECK_EQ(setrlimit(RLIMIT_FSIZE, &rlim), 0) << "setrlimit failed: " << strerror(errno);
}
// moving-space will map this fd so that we compact objects into it.
int ret = ftruncate(moving_to_space_fd_, size);
CHECK_EQ(ret, 0) << "ftruncate failed for moving-space:" << strerror(errno);
ret = ftruncate(moving_from_space_fd_, size);
CHECK_EQ(ret, 0) << "ftruncate failed for moving-space:" << strerror(errno);
if (rlimit_changed) {
// reset the rlimit to the original limits.
CHECK_EQ(setrlimit(RLIMIT_FSIZE, &rlim_read), 0)
<< "setrlimit failed: " << strerror(errno);
}
}
}
}
}
class MarkCompact::VerifyRootMarkedVisitor : public SingleRootVisitor {
public:
explicit VerifyRootMarkedVisitor(MarkCompact* collector) : collector_(collector) { }
void VisitRoot(mirror::Object* root, const RootInfo& info) override
REQUIRES_SHARED(Locks::mutator_lock_, Locks::heap_bitmap_lock_) {
CHECK(collector_->IsMarked(root) != nullptr) << info.ToString();
}
private:
MarkCompact* const collector_;
};
void MarkCompact::ReMarkRoots(Runtime* runtime) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
DCHECK_EQ(thread_running_gc_, Thread::Current());
Locks::mutator_lock_->AssertExclusiveHeld(thread_running_gc_);
MarkNonThreadRoots(runtime);
MarkConcurrentRoots(static_cast<VisitRootFlags>(kVisitRootFlagNewRoots
| kVisitRootFlagStopLoggingNewRoots
| kVisitRootFlagClearRootLog),
runtime);
if (kVerifyRootsMarked) {
TimingLogger::ScopedTiming t2("(Paused)VerifyRoots", GetTimings());
VerifyRootMarkedVisitor visitor(this);
runtime->VisitRoots(&visitor);
}
}
void MarkCompact::MarkingPause() {
TimingLogger::ScopedTiming t("(Paused)MarkingPause", GetTimings());
Runtime* runtime = Runtime::Current();
Locks::mutator_lock_->AssertExclusiveHeld(thread_running_gc_);
{
// Handle the dirty objects as we are a concurrent GC
WriterMutexLock mu(thread_running_gc_, *Locks::heap_bitmap_lock_);
{
MutexLock mu2(thread_running_gc_, *Locks::runtime_shutdown_lock_);
MutexLock mu3(thread_running_gc_, *Locks::thread_list_lock_);
std::list<Thread*> thread_list = runtime->GetThreadList()->GetList();
for (Thread* thread : thread_list) {
thread->VisitRoots(this, static_cast<VisitRootFlags>(0));
DCHECK_EQ(thread->GetThreadLocalGcBuffer(), nullptr);
// Need to revoke all the thread-local allocation stacks since we will
// swap the allocation stacks (below) and don't want anybody to allocate
// into the live stack.
thread->RevokeThreadLocalAllocationStack();
bump_pointer_space_->RevokeThreadLocalBuffers(thread);
}
}
// Fetch only the accumulated objects-allocated count as it is guaranteed to
// be up-to-date after the TLAB revocation above.
freed_objects_ += bump_pointer_space_->GetAccumulatedObjectsAllocated();
// Capture 'end' of moving-space at this point. Every allocation beyond this
// point will be considered as black.
// Align-up to page boundary so that black allocations happen from next page
// onwards. Also, it ensures that 'end' is aligned for card-table's
// ClearCardRange().
black_allocations_begin_ = bump_pointer_space_->AlignEnd(thread_running_gc_, gPageSize, heap_);
DCHECK_ALIGNED_PARAM(black_allocations_begin_, gPageSize);
// Re-mark root set. Doesn't include thread-roots as they are already marked
// above.
ReMarkRoots(runtime);
// Scan dirty objects.
RecursiveMarkDirtyObjects(/*paused*/ true, accounting::CardTable::kCardDirty);
{
TimingLogger::ScopedTiming t2("SwapStacks", GetTimings());
heap_->SwapStacks();
live_stack_freeze_size_ = heap_->GetLiveStack()->Size();
}
}
// TODO: For PreSweepingGcVerification(), find correct strategy to visit/walk
// objects in bump-pointer space when we have a mark-bitmap to indicate live
// objects. At the same time we also need to be able to visit black allocations,
// even though they are not marked in the bitmap. Without both of these we fail
// pre-sweeping verification. As well as we leave windows open wherein a
// VisitObjects/Walk on the space would either miss some objects or visit
// unreachable ones. These windows are when we are switching from shared
// mutator-lock to exclusive and vice-versa starting from here till compaction pause.
// heap_->PreSweepingGcVerification(this);
// Disallow new system weaks to prevent a race which occurs when someone adds
// a new system weak before we sweep them. Since this new system weak may not
// be marked, the GC may incorrectly sweep it. This also fixes a race where
// interning may attempt to return a strong reference to a string that is
// about to be swept.
runtime->DisallowNewSystemWeaks();
// Enable the reference processing slow path, needs to be done with mutators
// paused since there is no lock in the GetReferent fast path.
heap_->GetReferenceProcessor()->EnableSlowPath();
}
void MarkCompact::SweepSystemWeaks(Thread* self, Runtime* runtime, const bool paused) {
TimingLogger::ScopedTiming t(paused ? "(Paused)SweepSystemWeaks" : "SweepSystemWeaks",
GetTimings());
ReaderMutexLock mu(self, *Locks::heap_bitmap_lock_);
runtime->SweepSystemWeaks(this);
}
void MarkCompact::ProcessReferences(Thread* self) {
WriterMutexLock mu(self, *Locks::heap_bitmap_lock_);
GetHeap()->GetReferenceProcessor()->ProcessReferences(self, GetTimings());
}
void MarkCompact::Sweep(bool swap_bitmaps) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
// Ensure that nobody inserted objects in the live stack after we swapped the
// stacks.
CHECK_GE(live_stack_freeze_size_, GetHeap()->GetLiveStack()->Size());
{
TimingLogger::ScopedTiming t2("MarkAllocStackAsLive", GetTimings());
// Mark everything allocated since the last GC as live so that we can sweep
// concurrently, knowing that new allocations won't be marked as live.
accounting::ObjectStack* live_stack = heap_->GetLiveStack();
heap_->MarkAllocStackAsLive(live_stack);
live_stack->Reset();
DCHECK(mark_stack_->IsEmpty());
}
for (const auto& space : GetHeap()->GetContinuousSpaces()) {
if (space->IsContinuousMemMapAllocSpace() && space != bump_pointer_space_ &&
!immune_spaces_.ContainsSpace(space)) {
space::ContinuousMemMapAllocSpace* alloc_space = space->AsContinuousMemMapAllocSpace();
DCHECK(!alloc_space->IsZygoteSpace());
TimingLogger::ScopedTiming split("SweepMallocSpace", GetTimings());
RecordFree(alloc_space->Sweep(swap_bitmaps));
}
}
SweepLargeObjects(swap_bitmaps);
}
void MarkCompact::SweepLargeObjects(bool swap_bitmaps) {
space::LargeObjectSpace* los = heap_->GetLargeObjectsSpace();
if (los != nullptr) {
TimingLogger::ScopedTiming split(__FUNCTION__, GetTimings());
RecordFreeLOS(los->Sweep(swap_bitmaps));
}
}
void MarkCompact::ReclaimPhase() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
DCHECK(thread_running_gc_ == Thread::Current());
Runtime* const runtime = Runtime::Current();
// Process the references concurrently.
ProcessReferences(thread_running_gc_);
// TODO: Try to merge this system-weak sweeping with the one while updating
// references during the compaction pause.
SweepSystemWeaks(thread_running_gc_, runtime, /*paused*/ false);
runtime->AllowNewSystemWeaks();
// Clean up class loaders after system weaks are swept since that is how we know if class
// unloading occurred.
runtime->GetClassLinker()->CleanupClassLoaders();
{
WriterMutexLock mu(thread_running_gc_, *Locks::heap_bitmap_lock_);
// Reclaim unmarked objects.
Sweep(false);
// Swap the live and mark bitmaps for each space which we modified space. This is an
// optimization that enables us to not clear live bits inside of the sweep. Only swaps unbound
// bitmaps.
SwapBitmaps();
// Unbind the live and mark bitmaps.
GetHeap()->UnBindBitmaps();
}
}
// We want to avoid checking for every reference if it's within the page or
// not. This can be done if we know where in the page the holder object lies.
// If it doesn't overlap either boundaries then we can skip the checks.
template <bool kCheckBegin, bool kCheckEnd>
class MarkCompact::RefsUpdateVisitor {
public:
explicit RefsUpdateVisitor(MarkCompact* collector,
mirror::Object* obj,
uint8_t* begin,
uint8_t* end)
: collector_(collector),
moving_space_begin_(collector->moving_space_begin_),
moving_space_end_(collector->moving_space_end_),
obj_(obj),
begin_(begin),
end_(end) {
DCHECK(!kCheckBegin || begin != nullptr);
DCHECK(!kCheckEnd || end != nullptr);
}
void operator()([[maybe_unused]] mirror::Object* old,
MemberOffset offset,
[[maybe_unused]] bool is_static) const ALWAYS_INLINE
REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES_SHARED(Locks::heap_bitmap_lock_) {
bool update = true;
if (kCheckBegin || kCheckEnd) {
uint8_t* ref = reinterpret_cast<uint8_t*>(obj_) + offset.Int32Value();
update = (!kCheckBegin || ref >= begin_) && (!kCheckEnd || ref < end_);
}
if (update) {
collector_->UpdateRef(obj_, offset, moving_space_begin_, moving_space_end_);
}
}
// For object arrays we don't need to check boundaries here as it's done in
// VisitReferenes().
// TODO: Optimize reference updating using SIMD instructions. Object arrays
// are perfect as all references are tightly packed.
void operator()([[maybe_unused]] mirror::Object* old,
MemberOffset offset,
[[maybe_unused]] bool is_static,
[[maybe_unused]] bool is_obj_array) const ALWAYS_INLINE
REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES_SHARED(Locks::heap_bitmap_lock_) {
collector_->UpdateRef(obj_, offset, moving_space_begin_, moving_space_end_);
}
void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
ALWAYS_INLINE
REQUIRES_SHARED(Locks::mutator_lock_) {
if (!root->IsNull()) {
VisitRoot(root);
}
}
void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
ALWAYS_INLINE
REQUIRES_SHARED(Locks::mutator_lock_) {
collector_->UpdateRoot(root, moving_space_begin_, moving_space_end_);
}
private:
MarkCompact* const collector_;
uint8_t* const moving_space_begin_;
uint8_t* const moving_space_end_;
mirror::Object* const obj_;
uint8_t* const begin_;
uint8_t* const end_;
};
bool MarkCompact::IsValidObject(mirror::Object* obj) const {
mirror::Class* klass = obj->GetClass<kVerifyNone, kWithoutReadBarrier>();
if (!heap_->GetVerification()->IsValidHeapObjectAddress(klass)) {
return false;
}
return heap_->GetVerification()->IsValidClassUnchecked<kWithFromSpaceBarrier>(
obj->GetClass<kVerifyNone, kWithFromSpaceBarrier>());
}
template <typename Callback>
void MarkCompact::VerifyObject(mirror::Object* ref, Callback& callback) const {
if (kIsDebugBuild) {
mirror::Class* klass = ref->GetClass<kVerifyNone, kWithFromSpaceBarrier>();
mirror::Class* pre_compact_klass = ref->GetClass<kVerifyNone, kWithoutReadBarrier>();
mirror::Class* klass_klass = klass->GetClass<kVerifyNone, kWithFromSpaceBarrier>();
mirror::Class* klass_klass_klass = klass_klass->GetClass<kVerifyNone, kWithFromSpaceBarrier>();
if (HasAddress(pre_compact_klass) &&
reinterpret_cast<uint8_t*>(pre_compact_klass) < black_allocations_begin_) {
CHECK(moving_space_bitmap_->Test(pre_compact_klass))
<< "ref=" << ref
<< " post_compact_end=" << static_cast<void*>(post_compact_end_)
<< " pre_compact_klass=" << pre_compact_klass
<< " black_allocations_begin=" << static_cast<void*>(black_allocations_begin_);
CHECK(live_words_bitmap_->Test(pre_compact_klass));
}
if (!IsValidObject(ref)) {
std::ostringstream oss;
oss << "Invalid object: "
<< "ref=" << ref
<< " klass=" << klass
<< " klass_klass=" << klass_klass
<< " klass_klass_klass=" << klass_klass_klass
<< " pre_compact_klass=" << pre_compact_klass
<< " from_space_begin=" << static_cast<void*>(from_space_begin_)
<< " pre_compact_begin=" << static_cast<void*>(bump_pointer_space_->Begin())
<< " post_compact_end=" << static_cast<void*>(post_compact_end_)
<< " black_allocations_begin=" << static_cast<void*>(black_allocations_begin_);
// Call callback before dumping larger data like RAM and space dumps.
callback(oss);
oss << " \nobject="
<< heap_->GetVerification()->DumpRAMAroundAddress(reinterpret_cast<uintptr_t>(ref), 128)
<< " \nklass(from)="
<< heap_->GetVerification()->DumpRAMAroundAddress(reinterpret_cast<uintptr_t>(klass), 128)
<< "spaces:\n";
heap_->DumpSpaces(oss);
LOG(FATAL) << oss.str();
}
}
}
void MarkCompact::CompactPage(mirror::Object* obj,
uint32_t offset,
uint8_t* addr,
bool needs_memset_zero) {
DCHECK(moving_space_bitmap_->Test(obj)
&& live_words_bitmap_->Test(obj));
DCHECK(live_words_bitmap_->Test(offset)) << "obj=" << obj
<< " offset=" << offset
<< " addr=" << static_cast<void*>(addr)
<< " black_allocs_begin="
<< static_cast<void*>(black_allocations_begin_)
<< " post_compact_addr="
<< static_cast<void*>(post_compact_end_);
uint8_t* const start_addr = addr;
// How many distinct live-strides do we have.
size_t stride_count = 0;
uint8_t* last_stride = addr;
uint32_t last_stride_begin = 0;
auto verify_obj_callback = [&] (std::ostream& os) {
os << " stride_count=" << stride_count
<< " last_stride=" << static_cast<void*>(last_stride)
<< " offset=" << offset
<< " start_addr=" << static_cast<void*>(start_addr);
};
obj = GetFromSpaceAddr(obj);
live_words_bitmap_->VisitLiveStrides(
offset,
black_allocations_begin_,
gPageSize,
[&addr, &last_stride, &stride_count, &last_stride_begin, verify_obj_callback, this](
uint32_t stride_begin, size_t stride_size, [[maybe_unused]] bool is_last)
REQUIRES_SHARED(Locks::mutator_lock_) {
const size_t stride_in_bytes = stride_size * kAlignment;
DCHECK_LE(stride_in_bytes, gPageSize);
last_stride_begin = stride_begin;
DCHECK(IsAligned<kAlignment>(addr));
memcpy(addr, from_space_begin_ + stride_begin * kAlignment, stride_in_bytes);
if (kIsDebugBuild) {
uint8_t* space_begin = bump_pointer_space_->Begin();
// We can interpret the first word of the stride as an
// obj only from second stride onwards, as the first
// stride's first-object may have started on previous
// page. The only exception is the first page of the
// moving space.
if (stride_count > 0 || stride_begin * kAlignment < gPageSize) {
mirror::Object* o =
reinterpret_cast<mirror::Object*>(space_begin + stride_begin * kAlignment);
CHECK(live_words_bitmap_->Test(o)) << "ref=" << o;
CHECK(moving_space_bitmap_->Test(o))
<< "ref=" << o << " bitmap: " << moving_space_bitmap_->DumpMemAround(o);
VerifyObject(reinterpret_cast<mirror::Object*>(addr), verify_obj_callback);
}
}
last_stride = addr;
addr += stride_in_bytes;
stride_count++;
});
DCHECK_LT(last_stride, start_addr + gPageSize);
DCHECK_GT(stride_count, 0u);
size_t obj_size = 0;
uint32_t offset_within_obj = offset * kAlignment
- (reinterpret_cast<uint8_t*>(obj) - from_space_begin_);
// First object
if (offset_within_obj > 0) {
mirror::Object* to_ref = reinterpret_cast<mirror::Object*>(start_addr - offset_within_obj);
if (stride_count > 1) {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/false> visitor(this,
to_ref,
start_addr,
nullptr);
obj_size = obj->VisitRefsForCompaction</*kFetchObjSize*/true, /*kVisitNativeRoots*/false>(
visitor, MemberOffset(offset_within_obj), MemberOffset(-1));
} else {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/true> visitor(this,
to_ref,
start_addr,
start_addr + gPageSize);
obj_size = obj->VisitRefsForCompaction</*kFetchObjSize*/true, /*kVisitNativeRoots*/false>(
visitor, MemberOffset(offset_within_obj), MemberOffset(offset_within_obj
+ gPageSize));
}
obj_size = RoundUp(obj_size, kAlignment);
DCHECK_GT(obj_size, offset_within_obj)
<< "obj:" << obj
<< " class:"
<< obj->GetClass<kDefaultVerifyFlags, kWithFromSpaceBarrier>()
<< " to_addr:" << to_ref
<< " black-allocation-begin:" << reinterpret_cast<void*>(black_allocations_begin_)
<< " post-compact-end:" << reinterpret_cast<void*>(post_compact_end_)
<< " offset:" << offset * kAlignment
<< " class-after-obj-iter:"
<< (class_after_obj_iter_ != class_after_obj_ordered_map_.rend() ?
class_after_obj_iter_->first.AsMirrorPtr() : nullptr)
<< " last-reclaimed-page:" << reinterpret_cast<void*>(last_reclaimed_page_)
<< " last-checked-reclaim-page-idx:" << last_checked_reclaim_page_idx_
<< " offset-of-last-idx:"
<< pre_compact_offset_moving_space_[last_checked_reclaim_page_idx_] * kAlignment
<< " first-obj-of-last-idx:"
<< first_objs_moving_space_[last_checked_reclaim_page_idx_].AsMirrorPtr();
obj_size -= offset_within_obj;
// If there is only one stride, then adjust last_stride_begin to the
// end of the first object.
if (stride_count == 1) {
last_stride_begin += obj_size / kAlignment;
}
}
// Except for the last page being compacted, the pages will have addr ==
// start_addr + gPageSize.
uint8_t* const end_addr = addr;
addr = start_addr;
size_t bytes_done = obj_size;
// All strides except the last one can be updated without any boundary
// checks.
DCHECK_LE(addr, last_stride);
size_t bytes_to_visit = last_stride - addr;
DCHECK_LE(bytes_to_visit, gPageSize);
while (bytes_to_visit > bytes_done) {
mirror::Object* ref = reinterpret_cast<mirror::Object*>(addr + bytes_done);
VerifyObject(ref, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/false>
visitor(this, ref, nullptr, nullptr);
obj_size = ref->VisitRefsForCompaction(visitor, MemberOffset(0), MemberOffset(-1));
obj_size = RoundUp(obj_size, kAlignment);
bytes_done += obj_size;
}
// Last stride may have multiple objects in it and we don't know where the
// last object which crosses the page boundary starts, therefore check
// page-end in all of these objects. Also, we need to call
// VisitRefsForCompaction() with from-space object as we fetch object size,
// which in case of klass requires 'class_size_'.
uint8_t* from_addr = from_space_begin_ + last_stride_begin * kAlignment;
bytes_to_visit = end_addr - addr;
DCHECK_LE(bytes_to_visit, gPageSize);
while (bytes_to_visit > bytes_done) {
mirror::Object* ref = reinterpret_cast<mirror::Object*>(addr + bytes_done);
obj = reinterpret_cast<mirror::Object*>(from_addr);
VerifyObject(ref, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/true>
visitor(this, ref, nullptr, start_addr + gPageSize);
obj_size = obj->VisitRefsForCompaction(visitor,
MemberOffset(0),
MemberOffset(end_addr - (addr + bytes_done)));
obj_size = RoundUp(obj_size, kAlignment);
DCHECK_GT(obj_size, 0u)
<< "from_addr:" << obj
<< " from-space-class:"
<< obj->GetClass<kDefaultVerifyFlags, kWithFromSpaceBarrier>()
<< " to_addr:" << ref
<< " black-allocation-begin:" << reinterpret_cast<void*>(black_allocations_begin_)
<< " post-compact-end:" << reinterpret_cast<void*>(post_compact_end_)
<< " offset:" << offset * kAlignment
<< " bytes_done:" << bytes_done
<< " class-after-obj-iter:"
<< (class_after_obj_iter_ != class_after_obj_ordered_map_.rend() ?
class_after_obj_iter_->first.AsMirrorPtr() : nullptr)
<< " last-reclaimed-page:" << reinterpret_cast<void*>(last_reclaimed_page_)
<< " last-checked-reclaim-page-idx:" << last_checked_reclaim_page_idx_
<< " offset-of-last-idx:"
<< pre_compact_offset_moving_space_[last_checked_reclaim_page_idx_] * kAlignment
<< " first-obj-of-last-idx:"
<< first_objs_moving_space_[last_checked_reclaim_page_idx_].AsMirrorPtr();
from_addr += obj_size;
bytes_done += obj_size;
}
// The last page that we compact may have some bytes left untouched in the
// end, we should zero them as the kernel copies at page granularity.
if (needs_memset_zero && UNLIKELY(bytes_done < gPageSize)) {
std::memset(addr + bytes_done, 0x0, gPageSize - bytes_done);
}
}
// We store the starting point (pre_compact_page - first_obj) and first-chunk's
// size. If more TLAB(s) started in this page, then those chunks are identified
// using mark bitmap. All this info is prepared in UpdateMovingSpaceBlackAllocations().
// If we find a set bit in the bitmap, then we copy the remaining page and then
// use the bitmap to visit each object for updating references.
void MarkCompact::SlideBlackPage(mirror::Object* first_obj,
mirror::Object* next_page_first_obj,
uint32_t first_chunk_size,
uint8_t* const pre_compact_page,
uint8_t* dest,
bool needs_memset_zero) {
DCHECK(IsAlignedParam(pre_compact_page, gPageSize));
size_t bytes_copied;
uint8_t* src_addr = reinterpret_cast<uint8_t*>(GetFromSpaceAddr(first_obj));
uint8_t* pre_compact_addr = reinterpret_cast<uint8_t*>(first_obj);
uint8_t* const pre_compact_page_end = pre_compact_page + gPageSize;
uint8_t* const dest_page_end = dest + gPageSize;
auto verify_obj_callback = [&] (std::ostream& os) {
os << " first_obj=" << first_obj
<< " next_page_first_obj=" << next_page_first_obj
<< " first_chunk_sie=" << first_chunk_size
<< " dest=" << static_cast<void*>(dest)
<< " pre_compact_page="
<< static_cast<void* const>(pre_compact_page);
};
// We have empty portion at the beginning of the page. Zero it.
if (pre_compact_addr > pre_compact_page) {
bytes_copied = pre_compact_addr - pre_compact_page;
DCHECK_LT(bytes_copied, gPageSize);
if (needs_memset_zero) {
std::memset(dest, 0x0, bytes_copied);
}
dest += bytes_copied;
} else {
bytes_copied = 0;
size_t offset = pre_compact_page - pre_compact_addr;
pre_compact_addr = pre_compact_page;
src_addr += offset;
DCHECK(IsAlignedParam(src_addr, gPageSize));
}
// Copy the first chunk of live words
std::memcpy(dest, src_addr, first_chunk_size);
// Update references in the first chunk. Use object size to find next object.
{
size_t bytes_to_visit = first_chunk_size;
size_t obj_size;
// The first object started in some previous page. So we need to check the
// beginning.
DCHECK_LE(reinterpret_cast<uint8_t*>(first_obj), pre_compact_addr);
size_t offset = pre_compact_addr - reinterpret_cast<uint8_t*>(first_obj);
if (bytes_copied == 0 && offset > 0) {
mirror::Object* to_obj = reinterpret_cast<mirror::Object*>(dest - offset);
mirror::Object* from_obj = reinterpret_cast<mirror::Object*>(src_addr - offset);
// If the next page's first-obj is in this page or nullptr, then we don't
// need to check end boundary
if (next_page_first_obj == nullptr
|| (first_obj != next_page_first_obj
&& reinterpret_cast<uint8_t*>(next_page_first_obj) <= pre_compact_page_end)) {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/false> visitor(this,
to_obj,
dest,
nullptr);
obj_size = from_obj->VisitRefsForCompaction<
/*kFetchObjSize*/true, /*kVisitNativeRoots*/false>(visitor,
MemberOffset(offset),
MemberOffset(-1));
} else {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/true> visitor(this,
to_obj,
dest,
dest_page_end);
obj_size = from_obj->VisitRefsForCompaction<
/*kFetchObjSize*/true, /*kVisitNativeRoots*/false>(visitor,
MemberOffset(offset),
MemberOffset(offset
+ gPageSize));
if (first_obj == next_page_first_obj) {
// First object is the only object on this page. So there's nothing else left to do.
return;
}
}
obj_size = RoundUp(obj_size, kAlignment);
obj_size -= offset;
dest += obj_size;
bytes_to_visit -= obj_size;
}
bytes_copied += first_chunk_size;
// If the last object in this page is next_page_first_obj, then we need to check end boundary
bool check_last_obj = false;
if (next_page_first_obj != nullptr
&& reinterpret_cast<uint8_t*>(next_page_first_obj) < pre_compact_page_end
&& bytes_copied == gPageSize) {
size_t diff = pre_compact_page_end - reinterpret_cast<uint8_t*>(next_page_first_obj);
DCHECK_LE(diff, gPageSize);
DCHECK_LE(diff, bytes_to_visit);
bytes_to_visit -= diff;
check_last_obj = true;
}
while (bytes_to_visit > 0) {
mirror::Object* dest_obj = reinterpret_cast<mirror::Object*>(dest);
VerifyObject(dest_obj, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/false> visitor(this,
dest_obj,
nullptr,
nullptr);
obj_size = dest_obj->VisitRefsForCompaction(visitor, MemberOffset(0), MemberOffset(-1));
obj_size = RoundUp(obj_size, kAlignment);
bytes_to_visit -= obj_size;
dest += obj_size;
}
DCHECK_EQ(bytes_to_visit, 0u);
if (check_last_obj) {
mirror::Object* dest_obj = reinterpret_cast<mirror::Object*>(dest);
VerifyObject(dest_obj, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/true> visitor(this,
dest_obj,
nullptr,
dest_page_end);
mirror::Object* obj = GetFromSpaceAddr(next_page_first_obj);
obj->VisitRefsForCompaction</*kFetchObjSize*/false>(visitor,
MemberOffset(0),
MemberOffset(dest_page_end - dest));
return;
}
}
// Probably a TLAB finished on this page and/or a new TLAB started as well.
if (bytes_copied < gPageSize) {
src_addr += first_chunk_size;
pre_compact_addr += first_chunk_size;
// Use mark-bitmap to identify where objects are. First call
// VisitMarkedRange for only the first marked bit. If found, zero all bytes
// until that object and then call memcpy on the rest of the page.
// Then call VisitMarkedRange for all marked bits *after* the one found in
// this invocation. This time to visit references.
uintptr_t start_visit = reinterpret_cast<uintptr_t>(pre_compact_addr);
uintptr_t page_end = reinterpret_cast<uintptr_t>(pre_compact_page_end);
mirror::Object* found_obj = nullptr;
moving_space_bitmap_->VisitMarkedRange</*kVisitOnce*/true>(start_visit,
page_end,
[&found_obj](mirror::Object* obj) {
found_obj = obj;
});
size_t remaining_bytes = gPageSize - bytes_copied;
if (found_obj == nullptr) {
if (needs_memset_zero) {
// No more black objects in this page. Zero the remaining bytes and return.
std::memset(dest, 0x0, remaining_bytes);
}
return;
}
// Copy everything in this page, which includes any zeroed regions
// in-between.
std::memcpy(dest, src_addr, remaining_bytes);
DCHECK_LT(reinterpret_cast<uintptr_t>(found_obj), page_end);
moving_space_bitmap_->VisitMarkedRange(
reinterpret_cast<uintptr_t>(found_obj) + mirror::kObjectHeaderSize,
page_end,
[&found_obj, pre_compact_addr, dest, this, verify_obj_callback] (mirror::Object* obj)
REQUIRES_SHARED(Locks::mutator_lock_) {
ptrdiff_t diff = reinterpret_cast<uint8_t*>(found_obj) - pre_compact_addr;
mirror::Object* ref = reinterpret_cast<mirror::Object*>(dest + diff);
VerifyObject(ref, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/false>
visitor(this, ref, nullptr, nullptr);
ref->VisitRefsForCompaction</*kFetchObjSize*/false>(visitor,
MemberOffset(0),
MemberOffset(-1));
// Remember for next round.
found_obj = obj;
});
// found_obj may have been updated in VisitMarkedRange. Visit the last found
// object.
DCHECK_GT(reinterpret_cast<uint8_t*>(found_obj), pre_compact_addr);
DCHECK_LT(reinterpret_cast<uintptr_t>(found_obj), page_end);
ptrdiff_t diff = reinterpret_cast<uint8_t*>(found_obj) - pre_compact_addr;
mirror::Object* dest_obj = reinterpret_cast<mirror::Object*>(dest + diff);
VerifyObject(dest_obj, verify_obj_callback);
RefsUpdateVisitor</*kCheckBegin*/ false, /*kCheckEnd*/ true> visitor(
this, dest_obj, nullptr, dest_page_end);
// Last object could overlap with next page. And if it happens to be a
// class, then we may access something (like static-fields' offsets) which
// is on the next page. Therefore, use from-space's reference.
mirror::Object* obj = GetFromSpaceAddr(found_obj);
obj->VisitRefsForCompaction</*kFetchObjSize*/ false>(
visitor, MemberOffset(0), MemberOffset(page_end - reinterpret_cast<uintptr_t>(found_obj)));
}
}
template <bool kFirstPageMapping>
void MarkCompact::MapProcessedPages(uint8_t* to_space_start,
Atomic<PageState>* state_arr,
size_t arr_idx,
size_t arr_len) {
DCHECK(minor_fault_initialized_);
DCHECK_LT(arr_idx, arr_len);
DCHECK_ALIGNED_PARAM(to_space_start, gPageSize);
// Claim all the contiguous pages, which are ready to be mapped, and then do
// so in a single ioctl. This helps avoid the overhead of invoking syscall
// several times and also maps the already-processed pages, avoiding
// unnecessary faults on them.
size_t length = kFirstPageMapping ? gPageSize : 0;
if (kFirstPageMapping) {
arr_idx++;
}
// We need to guarantee that we don't end up sucsessfully marking a later
// page 'mapping' and then fail to mark an earlier page. To guarantee that
// we use acq_rel order.
for (; arr_idx < arr_len; arr_idx++, length += gPageSize) {
PageState expected_state = PageState::kProcessed;
if (!state_arr[arr_idx].compare_exchange_strong(
expected_state, PageState::kProcessedAndMapping, std::memory_order_acq_rel)) {
break;
}
}
if (length > 0) {
// Note: We need the first page to be attempted (to be mapped) by the ioctl
// as this function is called due to some mutator thread waiting on the
// 'to_space_start' page. Therefore, the ioctl must always be called
// with 'to_space_start' as the 'start' address because it can bail out in
// the middle (not attempting to map the subsequent pages) if it finds any
// page either already mapped in between, or missing on the shadow-map.
struct uffdio_continue uffd_continue;
uffd_continue.range.start = reinterpret_cast<uintptr_t>(to_space_start);
uffd_continue.range.len = length;
uffd_continue.mode = 0;
int ret = ioctl(uffd_, UFFDIO_CONTINUE, &uffd_continue);
if (UNLIKELY(ret == -1 && errno == EAGAIN)) {
// This can happen only in linear-alloc.
DCHECK(linear_alloc_spaces_data_.end() !=
std::find_if(linear_alloc_spaces_data_.begin(),
linear_alloc_spaces_data_.end(),
[to_space_start](const LinearAllocSpaceData& data) {
return data.begin_ <= to_space_start && to_space_start < data.end_;
}));
// This could happen if userfaultfd couldn't find any pages mapped in the
// shadow map. For instance, if there are certain (contiguous) pages on
// linear-alloc which are allocated and have first-object set-up but have
// not been accessed yet.
// Bail out by setting the remaining pages' state back to kProcessed and
// then waking up any waiting threads.
DCHECK_GE(uffd_continue.mapped, 0);
DCHECK_ALIGNED_PARAM(uffd_continue.mapped, gPageSize);
DCHECK_LT(uffd_continue.mapped, static_cast<ssize_t>(length));
if (kFirstPageMapping) {
// In this case the first page must be mapped.
DCHECK_GE(uffd_continue.mapped, static_cast<ssize_t>(gPageSize));
}
// Nobody would modify these pages' state simultaneously so only atomic
// store is sufficient. Use 'release' order to ensure that all states are
// modified sequentially.
for (size_t remaining_len = length - uffd_continue.mapped; remaining_len > 0;
remaining_len -= gPageSize) {
arr_idx--;
DCHECK_EQ(state_arr[arr_idx].load(std::memory_order_relaxed),
PageState::kProcessedAndMapping);
state_arr[arr_idx].store(PageState::kProcessed, std::memory_order_release);
}
uffd_continue.range.start =
reinterpret_cast<uintptr_t>(to_space_start) + uffd_continue.mapped;
uffd_continue.range.len = length - uffd_continue.mapped;
ret = ioctl(uffd_, UFFDIO_WAKE, &uffd_continue.range);
CHECK_EQ(ret, 0) << "ioctl_userfaultfd: wake failed: " << strerror(errno);
} else {
// We may receive ENOENT if gc-thread unregisters the
// range behind our back, which is fine because that
// happens only when it knows compaction is done.
CHECK(ret == 0 || !kFirstPageMapping || errno == ENOENT)
<< "ioctl_userfaultfd: continue failed: " << strerror(errno);
if (ret == 0) {
DCHECK_EQ(uffd_continue.mapped, static_cast<ssize_t>(length));
}
}
if (use_uffd_sigbus_) {
// Nobody else would modify these pages' state simultaneously so atomic
// store is sufficient.
for (; uffd_continue.mapped > 0; uffd_continue.mapped -= gPageSize) {
arr_idx--;
DCHECK_EQ(state_arr[arr_idx].load(std::memory_order_relaxed),
PageState::kProcessedAndMapping);
state_arr[arr_idx].store(PageState::kProcessedAndMapped, std::memory_order_release);
}
}
}
}
void MarkCompact::ZeropageIoctl(void* addr, bool tolerate_eexist, bool tolerate_enoent) {
struct uffdio_zeropage uffd_zeropage;
DCHECK(IsAlignedParam(addr, gPageSize));
uffd_zeropage.range.start = reinterpret_cast<uintptr_t>(addr);
uffd_zeropage.range.len = gPageSize;
uffd_zeropage.mode = 0;
int ret = ioctl(uffd_, UFFDIO_ZEROPAGE, &uffd_zeropage);
if (LIKELY(ret == 0)) {
DCHECK_EQ(uffd_zeropage.zeropage, static_cast<ssize_t>(gPageSize));
} else {
CHECK((tolerate_enoent && errno == ENOENT) || (tolerate_eexist && errno == EEXIST))
<< "ioctl_userfaultfd: zeropage failed: " << strerror(errno) << ". addr:" << addr;
}
}
void MarkCompact::CopyIoctl(void* dst, void* buffer) {
struct uffdio_copy uffd_copy;
uffd_copy.src = reinterpret_cast<uintptr_t>(buffer);
uffd_copy.dst = reinterpret_cast<uintptr_t>(dst);
uffd_copy.len = gPageSize;
uffd_copy.mode = 0;
CHECK_EQ(ioctl(uffd_, UFFDIO_COPY, &uffd_copy), 0)
<< "ioctl_userfaultfd: copy failed: " << strerror(errno) << ". src:" << buffer
<< " dst:" << dst;
DCHECK_EQ(uffd_copy.copy, static_cast<ssize_t>(gPageSize));
}
template <int kMode, typename CompactionFn>
void MarkCompact::DoPageCompactionWithStateChange(size_t page_idx,
size_t status_arr_len,
uint8_t* to_space_page,
uint8_t* page,
CompactionFn func) {
PageState expected_state = PageState::kUnprocessed;
PageState desired_state =
kMode == kCopyMode ? PageState::kProcessingAndMapping : PageState::kProcessing;
// In the concurrent case (kMode != kFallbackMode) we need to ensure that the update
// to moving_spaces_status_[page_idx] is released before the contents of the page are
// made accessible to other threads.
//
// We need acquire ordering here to ensure that when the CAS fails, another thread
// has completed processing the page, which is guaranteed by the release below.
if (kMode == kFallbackMode || moving_pages_status_[page_idx].compare_exchange_strong(
expected_state, desired_state, std::memory_order_acquire)) {
func();
if (kMode == kCopyMode) {
CopyIoctl(to_space_page, page);
if (use_uffd_sigbus_) {
// Store is sufficient as no other thread would modify the status at this point.
moving_pages_status_[page_idx].store(PageState::kProcessedAndMapped,
std::memory_order_release);
}
} else if (kMode == kMinorFaultMode) {
expected_state = PageState::kProcessing;
desired_state = PageState::kProcessed;
// the CAS needs to be with release order to ensure that stores to the
// page makes it to memory *before* other threads observe that it's
// ready to be mapped.
if (!moving_pages_status_[page_idx].compare_exchange_strong(
expected_state, desired_state, std::memory_order_release)) {
// Some mutator has requested to map the page after processing it.
DCHECK_EQ(expected_state, PageState::kProcessingAndMapping);
MapProcessedPages</*kFirstPageMapping=*/true>(
to_space_page, moving_pages_status_, page_idx, status_arr_len);
}
}
} else {
DCHECK_GT(expected_state, PageState::kProcessed);
}
}
void MarkCompact::FreeFromSpacePages(size_t cur_page_idx, int mode) {
// Thanks to sliding compaction, bump-pointer allocations, and reverse
// compaction (see CompactMovingSpace) the logic here is pretty simple: find
// the to-space page up to which compaction has finished, all the from-space
// pages corresponding to this onwards can be freed. There are some corner
// cases to be taken care of, which are described below.
size_t idx = last_checked_reclaim_page_idx_;
// Find the to-space page up to which the corresponding from-space pages can be
// freed.
for (; idx > cur_page_idx; idx--) {
PageState state = moving_pages_status_[idx - 1].load(std::memory_order_acquire);
if (state == PageState::kMutatorProcessing) {
// Some mutator is working on the page.
break;
}
DCHECK(state >= PageState::kProcessed ||
(state == PageState::kUnprocessed &&
(mode == kFallbackMode || idx > moving_first_objs_count_)));
}
DCHECK_LE(idx, last_checked_reclaim_page_idx_);
if (idx == last_checked_reclaim_page_idx_) {
// Nothing to do.
return;
}
uint8_t* reclaim_begin;
uint8_t* idx_addr;
// Calculate the first from-space page to be freed using 'idx'. If the
// first-object of the idx'th to-space page started before the corresponding
// from-space page, which is almost always the case in the compaction portion
// of the moving-space, then it indicates that the subsequent pages that are
// yet to be compacted will need the from-space pages. Therefore, find the page
// (from the already compacted pages) whose first-object is different from
// ours. All the from-space pages starting from that one are safe to be
// removed. Please note that this iteration is not expected to be long in
// normal cases as objects are smaller than page size.
if (idx >= moving_first_objs_count_) {
// black-allocated portion of the moving-space
idx_addr = black_allocations_begin_ + (idx - moving_first_objs_count_) * gPageSize;
reclaim_begin = idx_addr;
mirror::Object* first_obj = first_objs_moving_space_[idx].AsMirrorPtr();
if (first_obj != nullptr && reinterpret_cast<uint8_t*>(first_obj) < reclaim_begin) {
size_t idx_len = moving_first_objs_count_ + black_page_count_;
for (size_t i = idx + 1; i < idx_len; i++) {
mirror::Object* obj = first_objs_moving_space_[i].AsMirrorPtr();
// A null first-object indicates that the corresponding to-space page is
// not used yet. So we can compute its from-space page and use that.
if (obj != first_obj) {
reclaim_begin = obj != nullptr
? AlignUp(reinterpret_cast<uint8_t*>(obj), gPageSize)
: (black_allocations_begin_ + (i - moving_first_objs_count_) * gPageSize);
break;
}
}
}
} else {
DCHECK_GE(pre_compact_offset_moving_space_[idx], 0u);
idx_addr = bump_pointer_space_->Begin() + pre_compact_offset_moving_space_[idx] * kAlignment;
reclaim_begin = idx_addr;
DCHECK_LE(reclaim_begin, black_allocations_begin_);
mirror::Object* first_obj = first_objs_moving_space_[idx].AsMirrorPtr();
if (reinterpret_cast<uint8_t*>(first_obj) < reclaim_begin) {
DCHECK_LT(idx, moving_first_objs_count_);
mirror::Object* obj = first_obj;
for (size_t i = idx + 1; i < moving_first_objs_count_; i++) {
obj = first_objs_moving_space_[i].AsMirrorPtr();
if (first_obj != obj) {
DCHECK_LT(first_obj, obj);
DCHECK_LT(reclaim_begin, reinterpret_cast<uint8_t*>(obj));
reclaim_begin = reinterpret_cast<uint8_t*>(obj);
break;
}
}
if (obj == first_obj) {
reclaim_begin = black_allocations_begin_;
}
}
reclaim_begin = AlignUp(reclaim_begin, gPageSize);
}
DCHECK_NE(reclaim_begin, nullptr);
DCHECK_ALIGNED_PARAM(reclaim_begin, gPageSize);
DCHECK_ALIGNED_PARAM(last_reclaimed_page_, gPageSize);
// Check if the 'class_after_obj_map_' map allows pages to be freed.
for (; class_after_obj_iter_ != class_after_obj_ordered_map_.rend(); class_after_obj_iter_++) {
mirror::Object* klass = class_after_obj_iter_->first.AsMirrorPtr();
mirror::Class* from_klass = static_cast<mirror::Class*>(GetFromSpaceAddr(klass));
// Check with class' end to ensure that, if required, the entire class survives.
uint8_t* klass_end = reinterpret_cast<uint8_t*>(klass) + from_klass->SizeOf<kVerifyNone>();
DCHECK_LE(klass_end, last_reclaimed_page_);
if (reinterpret_cast<uint8_t*>(klass_end) >= reclaim_begin) {
// Found a class which is in the reclaim range.
uint8_t* obj_addr = reinterpret_cast<uint8_t*>(class_after_obj_iter_->second.AsMirrorPtr());
// NOTE: Don't assert that obj is of 'klass' type as klass could instead
// be its super-class.
if (obj_addr < idx_addr) {
// Its lowest-address object is not compacted yet. Reclaim starting from
// the end of this class.
reclaim_begin = AlignUp(klass_end, gPageSize);
} else {
// Continue consuming pairs wherein the lowest address object has already
// been compacted.
continue;
}
}
// All the remaining class (and thereby corresponding object) addresses are
// lower than the reclaim range.
break;
}
ssize_t size = last_reclaimed_page_ - reclaim_begin;
if (size >= kMinFromSpaceMadviseSize) {
int behavior = minor_fault_initialized_ ? MADV_REMOVE : MADV_DONTNEED;
CHECK_EQ(madvise(reclaim_begin + from_space_slide_diff_, size, behavior), 0)
<< "madvise of from-space failed: " << strerror(errno);
last_reclaimed_page_ = reclaim_begin;
}
last_checked_reclaim_page_idx_ = idx;
}
void MarkCompact::UpdateClassAfterObjMap() {
CHECK(class_after_obj_ordered_map_.empty());
for (const auto& pair : class_after_obj_hash_map_) {
auto super_class_iter = super_class_after_class_hash_map_.find(pair.first);
ObjReference key = super_class_iter != super_class_after_class_hash_map_.end()
? super_class_iter->second
: pair.first;
if (std::less<mirror::Object*>{}(pair.second.AsMirrorPtr(), key.AsMirrorPtr()) &&
HasAddress(key.AsMirrorPtr())) {
auto [ret_iter, success] = class_after_obj_ordered_map_.try_emplace(key, pair.second);
// It could fail only if the class 'key' has objects of its own, which are lower in
// address order, as well of some of its derived class. In this case
// choose the lowest address object.
if (!success &&
std::less<mirror::Object*>{}(pair.second.AsMirrorPtr(), ret_iter->second.AsMirrorPtr())) {
ret_iter->second = pair.second;
}
}
}
class_after_obj_hash_map_.clear();
super_class_after_class_hash_map_.clear();
}
template <int kMode>
void MarkCompact::CompactMovingSpace(uint8_t* page) {
// For every page we have a starting object, which may have started in some
// preceding page, and an offset within that object from where we must start
// copying.
// Consult the live-words bitmap to copy all contiguously live words at a
// time. These words may constitute multiple objects. To avoid the need for
// consulting mark-bitmap to find where does the next live object start, we
// use the object-size returned by VisitRefsForCompaction.
//
// We do the compaction in reverse direction so that the pages containing
// TLAB and latest allocations are processed first.
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
size_t page_status_arr_len = moving_first_objs_count_ + black_page_count_;
size_t idx = page_status_arr_len;
uint8_t* to_space_end = bump_pointer_space_->Begin() + page_status_arr_len * gPageSize;
uint8_t* shadow_space_end = nullptr;
if (kMode == kMinorFaultMode) {
shadow_space_end = shadow_to_space_map_.Begin() + page_status_arr_len * gPageSize;
}
uint8_t* pre_compact_page = black_allocations_begin_ + (black_page_count_ * gPageSize);
DCHECK(IsAlignedParam(pre_compact_page, gPageSize));
UpdateClassAfterObjMap();
// These variables are maintained by FreeFromSpacePages().
last_reclaimed_page_ = pre_compact_page;
last_checked_reclaim_page_idx_ = idx;
class_after_obj_iter_ = class_after_obj_ordered_map_.rbegin();
// Allocated-black pages
mirror::Object* next_page_first_obj = nullptr;
while (idx > moving_first_objs_count_) {
idx--;
pre_compact_page -= gPageSize;
to_space_end -= gPageSize;
if (kMode == kMinorFaultMode) {
shadow_space_end -= gPageSize;
page = shadow_space_end;
} else if (kMode == kFallbackMode) {
page = to_space_end;
}
mirror::Object* first_obj = first_objs_moving_space_[idx].AsMirrorPtr();
uint32_t first_chunk_size = black_alloc_pages_first_chunk_size_[idx];
if (first_obj != nullptr) {
DoPageCompactionWithStateChange<kMode>(idx,
page_status_arr_len,
to_space_end,
page,
[&]() REQUIRES_SHARED(Locks::mutator_lock_) {
SlideBlackPage(first_obj,
next_page_first_obj,
first_chunk_size,
pre_compact_page,
page,
kMode == kCopyMode);
});
// We are sliding here, so no point attempting to madvise for every
// page. Wait for enough pages to be done.
if (idx % (kMinFromSpaceMadviseSize / gPageSize) == 0) {
FreeFromSpacePages(idx, kMode);
}
}
next_page_first_obj = first_obj;
}
DCHECK_EQ(pre_compact_page, black_allocations_begin_);
while (idx > 0) {
idx--;
to_space_end -= gPageSize;
if (kMode == kMinorFaultMode) {
shadow_space_end -= gPageSize;
page = shadow_space_end;
} else if (kMode == kFallbackMode) {
page = to_space_end;
}
mirror::Object* first_obj = first_objs_moving_space_[idx].AsMirrorPtr();
DoPageCompactionWithStateChange<kMode>(
idx, page_status_arr_len, to_space_end, page, [&]() REQUIRES_SHARED(Locks::mutator_lock_) {
CompactPage(first_obj, pre_compact_offset_moving_space_[idx], page, kMode == kCopyMode);
});
FreeFromSpacePages(idx, kMode);
}
DCHECK_EQ(to_space_end, bump_pointer_space_->Begin());
}
void MarkCompact::UpdateNonMovingPage(mirror::Object* first, uint8_t* page) {
DCHECK_LT(reinterpret_cast<uint8_t*>(first), page + gPageSize);
// For every object found in the page, visit the previous object. This ensures
// that we can visit without checking page-end boundary.
// Call VisitRefsForCompaction with from-space read-barrier as the klass object and
// super-class loads require it.
// TODO: Set kVisitNativeRoots to false once we implement concurrent
// compaction
mirror::Object* curr_obj = first;
non_moving_space_bitmap_->VisitMarkedRange(
reinterpret_cast<uintptr_t>(first) + mirror::kObjectHeaderSize,
reinterpret_cast<uintptr_t>(page + gPageSize),
[&](mirror::Object* next_obj) {
// TODO: Once non-moving space update becomes concurrent, we'll
// require fetching the from-space address of 'curr_obj' and then call
// visitor on that.
if (reinterpret_cast<uint8_t*>(curr_obj) < page) {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/false>
visitor(this, curr_obj, page, page + gPageSize);
MemberOffset begin_offset(page - reinterpret_cast<uint8_t*>(curr_obj));
// Native roots shouldn't be visited as they are done when this
// object's beginning was visited in the preceding page.
curr_obj->VisitRefsForCompaction</*kFetchObjSize*/false, /*kVisitNativeRoots*/false>(
visitor, begin_offset, MemberOffset(-1));
} else {
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/false>
visitor(this, curr_obj, page, page + gPageSize);
curr_obj->VisitRefsForCompaction</*kFetchObjSize*/false>(visitor,
MemberOffset(0),
MemberOffset(-1));
}
curr_obj = next_obj;
});
MemberOffset end_offset(page + gPageSize - reinterpret_cast<uint8_t*>(curr_obj));
if (reinterpret_cast<uint8_t*>(curr_obj) < page) {
RefsUpdateVisitor</*kCheckBegin*/true, /*kCheckEnd*/true>
visitor(this, curr_obj, page, page + gPageSize);
curr_obj->VisitRefsForCompaction</*kFetchObjSize*/false, /*kVisitNativeRoots*/false>(
visitor, MemberOffset(page - reinterpret_cast<uint8_t*>(curr_obj)), end_offset);
} else {
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/true>
visitor(this, curr_obj, page, page + gPageSize);
curr_obj->VisitRefsForCompaction</*kFetchObjSize*/false>(visitor, MemberOffset(0), end_offset);
}
}
void MarkCompact::UpdateNonMovingSpace() {
TimingLogger::ScopedTiming t("(Paused)UpdateNonMovingSpace", GetTimings());
// Iterating in reverse ensures that the class pointer in objects which span
// across more than one page gets updated in the end. This is necessary for
// VisitRefsForCompaction() to work correctly.
// TODO: If and when we make non-moving space update concurrent, implement a
// mechanism to remember class pointers for such objects off-heap and pass it
// to VisitRefsForCompaction().
uint8_t* page = non_moving_space_->Begin() + non_moving_first_objs_count_ * gPageSize;
for (ssize_t i = non_moving_first_objs_count_ - 1; i >= 0; i--) {
mirror::Object* obj = first_objs_non_moving_space_[i].AsMirrorPtr();
page -= gPageSize;
// null means there are no objects on the page to update references.
if (obj != nullptr) {
UpdateNonMovingPage(obj, page);
}
}
}
void MarkCompact::UpdateMovingSpaceBlackAllocations() {
// For sliding black pages, we need the first-object, which overlaps with the
// first byte of the page. Additionally, we compute the size of first chunk of
// black objects. This will suffice for most black pages. Unlike, compaction
// pages, here we don't need to pre-compute the offset within first-obj from
// where sliding has to start. That can be calculated using the pre-compact
// address of the page. Therefore, to save space, we store the first chunk's
// size in black_alloc_pages_first_chunk_size_ array.
// For the pages which may have holes after the first chunk, which could happen
// if a new TLAB starts in the middle of the page, we mark the objects in
// the mark-bitmap. So, if the first-chunk size is smaller than gPageSize,
// then we use the mark-bitmap for the remainder of the page.
uint8_t* const begin = bump_pointer_space_->Begin();
uint8_t* black_allocs = black_allocations_begin_;
DCHECK_LE(begin, black_allocs);
size_t consumed_blocks_count = 0;
size_t first_block_size;
// Needed only for debug at the end of the function. Hopefully compiler will
// eliminate it otherwise.
size_t num_blocks = 0;
// Get the list of all blocks allocated in the bump-pointer space.
std::vector<size_t>* block_sizes = bump_pointer_space_->GetBlockSizes(thread_running_gc_,
&first_block_size);
DCHECK_LE(first_block_size, (size_t)(black_allocs - begin));
if (block_sizes != nullptr) {
size_t black_page_idx = moving_first_objs_count_;
uint8_t* block_end = begin + first_block_size;
uint32_t remaining_chunk_size = 0;
uint32_t first_chunk_size = 0;
mirror::Object* first_obj = nullptr;
num_blocks = block_sizes->size();
for (size_t block_size : *block_sizes) {
block_end += block_size;
// Skip the blocks that are prior to the black allocations. These will be
// merged with the main-block later.
if (black_allocs >= block_end) {
consumed_blocks_count++;
continue;
}
mirror::Object* obj = reinterpret_cast<mirror::Object*>(black_allocs);
bool set_mark_bit = remaining_chunk_size > 0;
// We don't know how many objects are allocated in the current block. When we hit
// a null assume it's the end. This works as every block is expected to
// have objects allocated linearly using bump-pointer.
// BumpPointerSpace::Walk() also works similarly.
while (black_allocs < block_end
&& obj->GetClass<kDefaultVerifyFlags, kWithoutReadBarrier>() != nullptr) {
// Try to keep instructions which access class instance together to
// avoid reloading the pointer from object.
size_t obj_size = obj->SizeOf();
bytes_scanned_ += obj_size;
obj_size = RoundUp(obj_size, kAlignment);
UpdateClassAfterObjectMap(obj);
if (first_obj == nullptr) {
first_obj = obj;
}
// We only need the mark-bitmap in the pages wherein a new TLAB starts in
// the middle of the page.
if (set_mark_bit) {
moving_space_bitmap_->Set(obj);
}
// Handle objects which cross page boundary, including objects larger
// than page size.
if (remaining_chunk_size + obj_size >= gPageSize) {
set_mark_bit = false;
first_chunk_size += gPageSize - remaining_chunk_size;
remaining_chunk_size += obj_size;
// We should not store first-object and remaining_chunk_size if there were
// unused bytes before this TLAB, in which case we must have already
// stored the values (below).
if (black_alloc_pages_first_chunk_size_[black_page_idx] == 0) {
black_alloc_pages_first_chunk_size_[black_page_idx] = first_chunk_size;
first_objs_moving_space_[black_page_idx].Assign(first_obj);
}
black_page_idx++;
remaining_chunk_size -= gPageSize;
// Consume an object larger than page size.
while (remaining_chunk_size >= gPageSize) {
black_alloc_pages_first_chunk_size_[black_page_idx] = gPageSize;
first_objs_moving_space_[black_page_idx].Assign(obj);
black_page_idx++;
remaining_chunk_size -= gPageSize;
}
first_obj = remaining_chunk_size > 0 ? obj : nullptr;
first_chunk_size = remaining_chunk_size;
} else {
DCHECK_LE(first_chunk_size, remaining_chunk_size);
first_chunk_size += obj_size;
remaining_chunk_size += obj_size;
}
black_allocs += obj_size;
obj = reinterpret_cast<mirror::Object*>(black_allocs);
}
DCHECK_LE(black_allocs, block_end);
DCHECK_LT(remaining_chunk_size, gPageSize);
// consume the unallocated portion of the block
if (black_allocs < block_end) {
// first-chunk of the current page ends here. Store it.
if (first_chunk_size > 0 && black_alloc_pages_first_chunk_size_[black_page_idx] == 0) {
black_alloc_pages_first_chunk_size_[black_page_idx] = first_chunk_size;
first_objs_moving_space_[black_page_idx].Assign(first_obj);
}
first_chunk_size = 0;
first_obj = nullptr;
size_t page_remaining = gPageSize - remaining_chunk_size;
size_t block_remaining = block_end - black_allocs;
if (page_remaining <= block_remaining) {
block_remaining -= page_remaining;
// current page and the subsequent empty pages in the block
black_page_idx += 1 + block_remaining / gPageSize;
remaining_chunk_size = block_remaining % gPageSize;
} else {
remaining_chunk_size += block_remaining;
}
black_allocs = block_end;
}
}
if (black_page_idx < bump_pointer_space_->Size() / gPageSize) {
// Store the leftover first-chunk, if any, and update page index.
if (black_alloc_pages_first_chunk_size_[black_page_idx] > 0) {
black_page_idx++;
} else if (first_chunk_size > 0) {
black_alloc_pages_first_chunk_size_[black_page_idx] = first_chunk_size;
first_objs_moving_space_[black_page_idx].Assign(first_obj);
black_page_idx++;
}
}
black_page_count_ = black_page_idx - moving_first_objs_count_;
delete block_sizes;
}
// Update bump-pointer space by consuming all the pre-black blocks into the
// main one.
bump_pointer_space_->SetBlockSizes(thread_running_gc_,
post_compact_end_ - begin,
consumed_blocks_count);
if (kIsDebugBuild) {
size_t moving_space_size = bump_pointer_space_->Size();
size_t los_size = 0;
if (heap_->GetLargeObjectsSpace()) {
los_size = heap_->GetLargeObjectsSpace()->GetBytesAllocated();
}
// The moving-space size is already updated to post-compact size in SetBlockSizes above.
// Also, bytes-allocated has already been adjusted with large-object space' freed-bytes
// in Sweep(), but not with moving-space freed-bytes.
CHECK_GE(heap_->GetBytesAllocated() - black_objs_slide_diff_, moving_space_size + los_size)
<< " moving-space size:" << moving_space_size
<< " moving-space bytes-freed:" << black_objs_slide_diff_
<< " large-object-space size:" << los_size
<< " large-object-space bytes-freed:" << GetCurrentIteration()->GetFreedLargeObjectBytes()
<< " num-tlabs-merged:" << consumed_blocks_count
<< " main-block-size:" << (post_compact_end_ - begin)
<< " total-tlabs-moving-space:" << num_blocks;
}
}
void MarkCompact::UpdateNonMovingSpaceBlackAllocations() {
accounting::ObjectStack* stack = heap_->GetAllocationStack();
const StackReference<mirror::Object>* limit = stack->End();
uint8_t* const space_begin = non_moving_space_->Begin();
for (StackReference<mirror::Object>* it = stack->Begin(); it != limit; ++it) {
mirror::Object* obj = it->AsMirrorPtr();
if (obj != nullptr && non_moving_space_bitmap_->HasAddress(obj)) {
non_moving_space_bitmap_->Set(obj);
// Clear so that we don't try to set the bit again in the next GC-cycle.
it->Clear();
size_t idx = (reinterpret_cast<uint8_t*>(obj) - space_begin) / gPageSize;
uint8_t* page_begin = AlignDown(reinterpret_cast<uint8_t*>(obj), gPageSize);
mirror::Object* first_obj = first_objs_non_moving_space_[idx].AsMirrorPtr();
if (first_obj == nullptr
|| (obj < first_obj && reinterpret_cast<uint8_t*>(first_obj) > page_begin)) {
first_objs_non_moving_space_[idx].Assign(obj);
}
mirror::Object* next_page_first_obj = first_objs_non_moving_space_[++idx].AsMirrorPtr();
uint8_t* next_page_begin = page_begin + gPageSize;
if (next_page_first_obj == nullptr
|| reinterpret_cast<uint8_t*>(next_page_first_obj) > next_page_begin) {
size_t obj_size = RoundUp(obj->SizeOf<kDefaultVerifyFlags>(), kAlignment);
uint8_t* obj_end = reinterpret_cast<uint8_t*>(obj) + obj_size;
while (next_page_begin < obj_end) {
first_objs_non_moving_space_[idx++].Assign(obj);
next_page_begin += gPageSize;
}
}
// update first_objs count in case we went past non_moving_first_objs_count_
non_moving_first_objs_count_ = std::max(non_moving_first_objs_count_, idx);
}
}
}
class MarkCompact::ImmuneSpaceUpdateObjVisitor {
public:
explicit ImmuneSpaceUpdateObjVisitor(MarkCompact* collector) : collector_(collector) {}
void operator()(mirror::Object* obj) const ALWAYS_INLINE REQUIRES(Locks::mutator_lock_) {
RefsUpdateVisitor</*kCheckBegin*/false, /*kCheckEnd*/false> visitor(collector_,
obj,
/*begin_*/nullptr,
/*end_*/nullptr);
obj->VisitRefsForCompaction</*kFetchObjSize*/ false>(
visitor, MemberOffset(0), MemberOffset(-1));
}
static void Callback(mirror::Object* obj, void* arg) REQUIRES(Locks::mutator_lock_) {
reinterpret_cast<ImmuneSpaceUpdateObjVisitor*>(arg)->operator()(obj);
}
private:
MarkCompact* const collector_;
};
class MarkCompact::ClassLoaderRootsUpdater : public ClassLoaderVisitor {
public:
explicit ClassLoaderRootsUpdater(MarkCompact* collector)
: collector_(collector),
moving_space_begin_(collector->moving_space_begin_),
moving_space_end_(collector->moving_space_end_) {}
void Visit(ObjPtr<mirror::ClassLoader> class_loader) override
REQUIRES_SHARED(Locks::classlinker_classes_lock_, Locks::mutator_lock_) {
ClassTable* const class_table = class_loader->GetClassTable();
if (class_table != nullptr) {
// Classes are updated concurrently.
class_table->VisitRoots(*this, /*skip_classes=*/true);
}
}
void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const ALWAYS_INLINE
REQUIRES(Locks::heap_bitmap_lock_) REQUIRES_SHARED(Locks::mutator_lock_) {
if (!root->IsNull()) {
VisitRoot(root);
}
}
void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const ALWAYS_INLINE
REQUIRES(Locks::heap_bitmap_lock_) REQUIRES_SHARED(Locks::mutator_lock_) {
collector_->UpdateRoot(
root, moving_space_begin_, moving_space_end_, RootInfo(RootType::kRootVMInternal));
}
private:
MarkCompact* collector_;
uint8_t* const moving_space_begin_;
uint8_t* const moving_space_end_;
};
class MarkCompact::LinearAllocPageUpdater {
public:
explicit LinearAllocPageUpdater(MarkCompact* collector)
: collector_(collector),
moving_space_begin_(collector->moving_space_begin_),
moving_space_end_(collector->moving_space_end_),
last_page_touched_(false) {}
// Update a page in multi-object arena.
void MultiObjectArena(uint8_t* page_begin, uint8_t* first_obj)
REQUIRES_SHARED(Locks::mutator_lock_) {
DCHECK(first_obj != nullptr);
DCHECK_ALIGNED_PARAM(page_begin, gPageSize);
uint8_t* page_end = page_begin + gPageSize;
uint32_t obj_size;
for (uint8_t* byte = first_obj; byte < page_end;) {
TrackingHeader* header = reinterpret_cast<TrackingHeader*>(byte);
obj_size = header->GetSize();
if (UNLIKELY(obj_size == 0)) {
// No more objects in this page to visit.
last_page_touched_ = byte >= page_begin;
return;
}
uint8_t* obj = byte + sizeof(TrackingHeader);
uint8_t* obj_end = byte + obj_size;
if (header->Is16Aligned()) {
obj = AlignUp(obj, 16);
}
uint8_t* begin_boundary = std::max(obj, page_begin);
uint8_t* end_boundary = std::min(obj_end, page_end);
if (begin_boundary < end_boundary) {
VisitObject(header->GetKind(), obj, begin_boundary, end_boundary);
}
if (ArenaAllocator::IsRunningOnMemoryTool()) {
obj_size += ArenaAllocator::kMemoryToolRedZoneBytes;
}
byte += RoundUp(obj_size, LinearAlloc::kAlignment);
}
last_page_touched_ = true;
}
// This version is only used for cases where the entire page is filled with
// GC-roots. For example, class-table and intern-table.
void SingleObjectArena(uint8_t* page_begin, size_t page_size)
REQUIRES_SHARED(Locks::mutator_lock_) {
static_assert(sizeof(uint32_t) == sizeof(GcRoot<mirror::Object>));
DCHECK_ALIGNED(page_begin, kAlignment);
// Least significant bits are used by class-table.
static constexpr uint32_t kMask = kObjectAlignment - 1;
size_t num_roots = page_size / sizeof(GcRoot<mirror::Object>);
uint32_t* root_ptr = reinterpret_cast<uint32_t*>(page_begin);
for (size_t i = 0; i < num_roots; root_ptr++, i++) {
uint32_t word = *root_ptr;
if (word != 0) {
uint32_t lsbs = word & kMask;
word &= ~kMask;
VisitRootIfNonNull(reinterpret_cast<mirror::CompressedReference<mirror::Object>*>(&word));
*root_ptr = word | lsbs;
last_page_touched_ = true;
}
}
}
bool WasLastPageTouched() const { return last_page_touched_; }
void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
if (!root->IsNull()) {
VisitRoot(root);
}
}
void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
ALWAYS_INLINE REQUIRES_SHARED(Locks::mutator_lock_) {
mirror::Object* old_ref = root->AsMirrorPtr();
DCHECK_NE(old_ref, nullptr);
if (MarkCompact::HasAddress(old_ref, moving_space_begin_, moving_space_end_)) {
mirror::Object* new_ref = old_ref;
if (reinterpret_cast<uint8_t*>(old_ref) >= collector_->black_allocations_begin_) {
new_ref = collector_->PostCompactBlackObjAddr(old_ref);
} else if (collector_->live_words_bitmap_->Test(old_ref)) {
DCHECK(collector_->moving_space_bitmap_->Test(old_ref))
<< "ref:" << old_ref << " root:" << root;
new_ref = collector_->PostCompactOldObjAddr(old_ref);
}
if (old_ref != new_ref) {
root->Assign(new_ref);
}
}
}
private:
void VisitObject(LinearAllocKind kind,
void* obj,
uint8_t* start_boundary,
uint8_t* end_boundary) const ALWAYS_INLINE
REQUIRES_SHARED(Locks::mutator_lock_) {
switch (kind) {
case LinearAllocKind::kNoGCRoots:
break;
case LinearAllocKind::kGCRootArray:
{
GcRoot<mirror::Object>* root = reinterpret_cast<GcRoot<mirror::Object>*>(start_boundary);
GcRoot<mirror::Object>* last = reinterpret_cast<GcRoot<mirror::Object>*>(end_boundary);
for (; root < last; root++) {
VisitRootIfNonNull(root->AddressWithoutBarrier());
}
}
break;
case LinearAllocKind::kArtMethodArray:
{
LengthPrefixedArray<ArtMethod>* array = static_cast<LengthPrefixedArray<ArtMethod>*>(obj);
// Old methods are clobbered in debug builds. Check size to confirm if the array
// has any GC roots to visit. See ClassLinker::LinkMethodsHelper::ClobberOldMethods()
if (array->size() > 0) {
if (collector_->pointer_size_ == PointerSize::k64) {
ArtMethod::VisitArrayRoots<PointerSize::k64>(
*this, start_boundary, end_boundary, array);
} else {
DCHECK_EQ(collector_->pointer_size_, PointerSize::k32);
ArtMethod::VisitArrayRoots<PointerSize::k32>(
*this, start_boundary, end_boundary, array);
}
}
}
break;
case LinearAllocKind::kArtMethod:
ArtMethod::VisitRoots(*this, start_boundary, end_boundary, static_cast<ArtMethod*>(obj));
break;
case LinearAllocKind::kArtFieldArray:
ArtField::VisitArrayRoots(*this,
start_boundary,
end_boundary,
static_cast<LengthPrefixedArray<ArtField>*>(obj));
break;
case LinearAllocKind::kDexCacheArray:
{
mirror::DexCachePair<mirror::Object>* first =
reinterpret_cast<mirror::DexCachePair<mirror::Object>*>(start_boundary);
mirror::DexCachePair<mirror::Object>* last =
reinterpret_cast<mirror::DexCachePair<mirror::Object>*>(end_boundary);
mirror::DexCache::VisitDexCachePairRoots(*this, first, last);
}
}
}
MarkCompact* const collector_;
// Cache to speed up checking if GC-root is in moving space or not.
uint8_t* const moving_space_begin_;
uint8_t* const moving_space_end_;
// Whether the last page was touched or not.
bool last_page_touched_ = false;
};
void MarkCompact::UpdateClassTableClasses(Runtime* runtime, bool immune_class_table_only) {
// If the process is debuggable then redefinition is allowed, which may mean
// pre-zygote-fork class-tables may have pointer to class in moving-space.
// So visit classes from class-sets that are not in linear-alloc arena-pool.
if (UNLIKELY(runtime->IsJavaDebuggableAtInit())) {
ClassLinker* linker = runtime->GetClassLinker();
ClassLoaderRootsUpdater updater(this);
GcVisitedArenaPool* pool = static_cast<GcVisitedArenaPool*>(runtime->GetLinearAllocArenaPool());
auto cond = [this, pool, immune_class_table_only](ClassTable::ClassSet& set) -> bool {
if (!set.empty()) {
return immune_class_table_only ?
immune_spaces_.ContainsObject(reinterpret_cast<mirror::Object*>(&*set.begin())) :
!pool->Contains(reinterpret_cast<void*>(&*set.begin()));
}
return false;
};
linker->VisitClassTables([cond, &updater](ClassTable* table)
REQUIRES_SHARED(Locks::mutator_lock_) {
table->VisitClassesIfConditionMet(cond, updater);
});
ReaderMutexLock rmu(thread_running_gc_, *Locks::classlinker_classes_lock_);
linker->GetBootClassTable()->VisitClassesIfConditionMet(cond, updater);
}
}
void MarkCompact::CompactionPause() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
Runtime* runtime = Runtime::Current();
non_moving_space_bitmap_ = non_moving_space_->GetLiveBitmap();
if (kIsDebugBuild) {
DCHECK_EQ(thread_running_gc_, Thread::Current());
stack_low_addr_ = thread_running_gc_->GetStackEnd();
stack_high_addr_ =
reinterpret_cast<char*>(stack_low_addr_) + thread_running_gc_->GetStackSize();
}
{
TimingLogger::ScopedTiming t2("(Paused)UpdateCompactionDataStructures", GetTimings());
ReaderMutexLock rmu(thread_running_gc_, *Locks::heap_bitmap_lock_);
// Refresh data-structures to catch-up on allocations that may have
// happened since marking-phase pause.
// There could be several TLABs that got allocated since marking pause. We
// don't want to compact them and instead update the TLAB info in TLS and
// let mutators continue to use the TLABs.
// We need to set all the bits in live-words bitmap corresponding to allocated
// objects. Also, we need to find the objects that are overlapping with
// page-begin boundaries. Unlike objects allocated before
// black_allocations_begin_, which can be identified via mark-bitmap, we can get
// this info only via walking the space past black_allocations_begin_, which
// involves fetching object size.
// TODO: We can reduce the time spent on this in a pause by performing one
// round of this concurrently prior to the pause.
UpdateMovingSpaceBlackAllocations();
// TODO: If we want to avoid this allocation in a pause then we will have to
// allocate an array for the entire moving-space size, which can be made
// part of info_map_.
moving_pages_status_ = new Atomic<PageState>[moving_first_objs_count_ + black_page_count_];
if (kIsDebugBuild) {
size_t len = moving_first_objs_count_ + black_page_count_;
for (size_t i = 0; i < len; i++) {
CHECK_EQ(moving_pages_status_[i].load(std::memory_order_relaxed),
PageState::kUnprocessed);
}
}
// Iterate over the allocation_stack_, for every object in the non-moving
// space:
// 1. Mark the object in live bitmap
// 2. Erase the object from allocation stack
// 3. In the corresponding page, if the first-object vector needs updating
// then do so.
UpdateNonMovingSpaceBlackAllocations();
// This store is visible to mutator (or uffd worker threads) as the mutator
// lock's unlock guarantees that.
compacting_ = true;
// Start updating roots and system weaks now.
heap_->GetReferenceProcessor()->UpdateRoots(this);
}
{
// TODO: Immune space updation has to happen either before or after
// remapping pre-compact pages to from-space. And depending on when it's
// done, we have to invoke VisitRefsForCompaction() with or without
// read-barrier.
TimingLogger::ScopedTiming t2("(Paused)UpdateImmuneSpaces", GetTimings());
accounting::CardTable* const card_table = heap_->GetCardTable();
for (auto& space : immune_spaces_.GetSpaces()) {
DCHECK(space->IsImageSpace() || space->IsZygoteSpace());
accounting::ContinuousSpaceBitmap* live_bitmap = space->GetLiveBitmap();
accounting::ModUnionTable* table = heap_->FindModUnionTableFromSpace(space);
// Having zygote-space indicates that the first zygote fork has taken
// place and that the classes/dex-caches in immune-spaces may have allocations
// (ArtMethod/ArtField arrays, dex-cache array, etc.) in the
// non-userfaultfd visited private-anonymous mappings. Visit them here.
ImmuneSpaceUpdateObjVisitor visitor(this);
if (table != nullptr) {
table->ProcessCards();
table->VisitObjects(ImmuneSpaceUpdateObjVisitor::Callback, &visitor);
} else {
WriterMutexLock wmu(thread_running_gc_, *Locks::heap_bitmap_lock_);
card_table->Scan<false>(
live_bitmap,
space->Begin(),
space->Limit(),
visitor,
accounting::CardTable::kCardDirty - 1);
}
}
}
{
TimingLogger::ScopedTiming t2("(Paused)UpdateRoots", GetTimings());
runtime->VisitConcurrentRoots(this, kVisitRootFlagAllRoots);
runtime->VisitNonThreadRoots(this);
{
ClassLinker* linker = runtime->GetClassLinker();
ClassLoaderRootsUpdater updater(this);
ReaderMutexLock rmu(thread_running_gc_, *Locks::classlinker_classes_lock_);
linker->VisitClassLoaders(&updater);
linker->GetBootClassTable()->VisitRoots(updater, /*skip_classes=*/true);
}
SweepSystemWeaks(thread_running_gc_, runtime, /*paused=*/true);
bool has_zygote_space = heap_->HasZygoteSpace();
GcVisitedArenaPool* arena_pool =
static_cast<GcVisitedArenaPool*>(runtime->GetLinearAllocArenaPool());
// Update immune/pre-zygote class-tables in case class redefinition took
// place. pre-zygote class-tables that are not in immune spaces are updated
// below if we are in fallback-mode or if there is no zygote space. So in
// that case only visit class-tables that are there in immune-spaces.
UpdateClassTableClasses(runtime, uffd_ == kFallbackMode || !has_zygote_space);
// Acquire arena-pool's lock, which should be released after the pool is
// userfaultfd registered. This is to ensure that no new arenas are
// allocated and used in between. Since they will not be captured in
// linear_alloc_arenas_ below, we will miss updating their pages. The same
// reason also applies to new allocations within the existing arena which
// may change last_byte.
// Since we are in a STW pause, this shouldn't happen anyways, but holding
// the lock confirms it.
// TODO (b/305779657): Replace with ExclusiveTryLock() and assert that it
// doesn't fail once it is available for ReaderWriterMutex.
WriterMutexLock pool_wmu(thread_running_gc_, arena_pool->GetLock());
// TODO: Find out why it's not sufficient to visit native roots of immune
// spaces, and why all the pre-zygote fork arenas have to be linearly updated.
// Is it possible that some native root starts getting pointed to by some object
// in moving space after fork? Or are we missing a write-barrier somewhere
// when a native root is updated?
auto arena_visitor = [this](uint8_t* page_begin, uint8_t* first_obj, size_t page_size)
REQUIRES_SHARED(Locks::mutator_lock_) {
LinearAllocPageUpdater updater(this);
if (first_obj != nullptr) {
updater.MultiObjectArena(page_begin, first_obj);
} else {
updater.SingleObjectArena(page_begin, page_size);
}
};
if (uffd_ == kFallbackMode || (!has_zygote_space && runtime->IsZygote())) {
// Besides fallback-mode, visit linear-alloc space in the pause for zygote
// processes prior to first fork (that's when zygote space gets created).
if (kIsDebugBuild && IsValidFd(uffd_)) {
// All arenas allocated so far are expected to be pre-zygote fork.
arena_pool->ForEachAllocatedArena(
[](const TrackedArena& arena)
REQUIRES_SHARED(Locks::mutator_lock_) { CHECK(arena.IsPreZygoteForkArena()); });
}
arena_pool->VisitRoots(arena_visitor);
} else {
// Inform the arena-pool that compaction is going on. So the TrackedArena
// objects corresponding to the arenas that are freed shouldn't be deleted
// immediately. We will do that in FinishPhase(). This is to avoid ABA
// problem.
arena_pool->DeferArenaFreeing();
arena_pool->ForEachAllocatedArena(
[this, arena_visitor, has_zygote_space](const TrackedArena& arena)
REQUIRES_SHARED(Locks::mutator_lock_) {
// The pre-zygote fork arenas are not visited concurrently in the
// zygote children processes. The native roots of the dirty objects
// are visited during immune space visit below.
if (!arena.IsPreZygoteForkArena()) {
uint8_t* last_byte = arena.GetLastUsedByte();
auto ret = linear_alloc_arenas_.insert({&arena, last_byte});
CHECK(ret.second);
} else if (!arena.IsSingleObjectArena() || !has_zygote_space) {
// Pre-zygote class-table and intern-table don't need to be updated.
// TODO: Explore the possibility of using /proc/self/pagemap to
// fetch which pages in these arenas are private-dirty and then only
// visit those pages. To optimize it further, we can keep all
// pre-zygote arenas in a single memory range so that just one read
// from pagemap is sufficient.
arena.VisitRoots(arena_visitor);
}
});
}
if (use_uffd_sigbus_) {
// Release order wrt to mutator threads' SIGBUS handler load.
sigbus_in_progress_count_.store(0, std::memory_order_release);
}
KernelPreparation();
}
UpdateNonMovingSpace();
// fallback mode
if (uffd_ == kFallbackMode) {
CompactMovingSpace<kFallbackMode>(nullptr);
int32_t freed_bytes = black_objs_slide_diff_;
bump_pointer_space_->RecordFree(freed_objects_, freed_bytes);
RecordFree(ObjectBytePair(freed_objects_, freed_bytes));
} else {
DCHECK_EQ(compaction_in_progress_count_.load(std::memory_order_relaxed), 0u);
DCHECK_EQ(compaction_buffer_counter_.load(std::memory_order_relaxed), 1);
if (!use_uffd_sigbus_) {
// We must start worker threads before resuming mutators to avoid deadlocks.
heap_->GetThreadPool()->StartWorkers(thread_running_gc_);
}
}
stack_low_addr_ = nullptr;
}
void MarkCompact::KernelPrepareRangeForUffd(uint8_t* to_addr,
uint8_t* from_addr,
size_t map_size,
int fd,
uint8_t* shadow_addr) {
int mremap_flags = MREMAP_MAYMOVE | MREMAP_FIXED;
if (gHaveMremapDontunmap) {
mremap_flags |= MREMAP_DONTUNMAP;
}
void* ret = mremap(to_addr, map_size, map_size, mremap_flags, from_addr);
CHECK_EQ(ret, static_cast<void*>(from_addr))
<< "mremap to move pages failed: " << strerror(errno)
<< ". space-addr=" << reinterpret_cast<void*>(to_addr) << " size=" << PrettySize(map_size);
if (shadow_addr != nullptr) {
DCHECK_EQ(fd, kFdUnused);
DCHECK(gHaveMremapDontunmap);
ret = mremap(shadow_addr, map_size, map_size, mremap_flags, to_addr);
CHECK_EQ(ret, static_cast<void*>(to_addr))
<< "mremap from shadow to to-space map failed: " << strerror(errno);
} else if (!gHaveMremapDontunmap || fd > kFdUnused) {
// Without MREMAP_DONTUNMAP the source mapping is unmapped by mremap. So mmap
// the moving space again.
int mmap_flags = MAP_FIXED;
if (fd == kFdUnused) {
// Use MAP_FIXED_NOREPLACE so that if someone else reserves 'to_addr'
// mapping in meantime, which can happen when MREMAP_DONTUNMAP isn't
// available, to avoid unmapping someone else' mapping and then causing
// crashes elsewhere.
mmap_flags = MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED_NOREPLACE;
// On some platforms MAP_ANONYMOUS expects fd to be -1.
fd = -1;
} else if (IsValidFd(fd)) {
mmap_flags |= MAP_SHARED;
} else {
DCHECK_EQ(fd, kFdSharedAnon);
mmap_flags |= MAP_SHARED | MAP_ANONYMOUS;
}
ret = mmap(to_addr, map_size, PROT_READ | PROT_WRITE, mmap_flags, fd, 0);
CHECK_EQ(ret, static_cast<void*>(to_addr))
<< "mmap for moving space failed: " << strerror(errno);
}
}
void MarkCompact::KernelPreparation() {
TimingLogger::ScopedTiming t("(Paused)KernelPreparation", GetTimings());
uint8_t* moving_space_begin = bump_pointer_space_->Begin();
size_t moving_space_size = bump_pointer_space_->Capacity();
int mode = kCopyMode;
size_t moving_space_register_sz;
if (minor_fault_initialized_) {
moving_space_register_sz = (moving_first_objs_count_ + black_page_count_) * gPageSize;
if (shadow_to_space_map_.IsValid()) {
size_t shadow_size = shadow_to_space_map_.Size();
void* addr = shadow_to_space_map_.Begin();
if (shadow_size < moving_space_register_sz) {
addr = mremap(addr,
shadow_size,
moving_space_register_sz,
// Don't allow moving with obj-ptr poisoning as the
// mapping needs to be in <4GB address space.
kObjPtrPoisoning ? 0 : MREMAP_MAYMOVE,
/*new_address=*/nullptr);
if (addr != MAP_FAILED) {
// Succeeded in expanding the mapping. Update the MemMap entry for shadow map.
MemMap temp = MemMap::MapPlaceholder(
"moving-space-shadow", static_cast<uint8_t*>(addr), moving_space_register_sz);
std::swap(shadow_to_space_map_, temp);
}
}
if (addr != MAP_FAILED) {
mode = kMinorFaultMode;
} else {
// We are not going to use shadow map. So protect it to catch any
// potential bugs.
DCHECK_EQ(mprotect(shadow_to_space_map_.Begin(), shadow_to_space_map_.Size(), PROT_NONE), 0)
<< "mprotect failed: " << strerror(errno);
}
}
} else {
moving_space_register_sz = moving_space_size;
}
bool map_shared =
minor_fault_initialized_ || (!Runtime::Current()->IsZygote() && uffd_minor_fault_supported_);
uint8_t* shadow_addr = nullptr;
if (moving_to_space_fd_ == kFdUnused && map_shared) {
DCHECK(gHaveMremapDontunmap);
DCHECK(shadow_to_space_map_.IsValid());
DCHECK_EQ(shadow_to_space_map_.Size(), moving_space_size);
shadow_addr = shadow_to_space_map_.Begin();
}
KernelPrepareRangeForUffd(moving_space_begin,
from_space_begin_,
moving_space_size,
moving_to_space_fd_,
shadow_addr);
if (IsValidFd(uffd_)) {
// Register the moving space with userfaultfd.
RegisterUffd(moving_space_begin, moving_space_register_sz, mode);
// Prepare linear-alloc for concurrent compaction.
for (auto& data : linear_alloc_spaces_data_) {
bool mmap_again = map_shared && !data.already_shared_;
DCHECK_EQ(static_cast<ssize_t>(data.shadow_.Size()), data.end_ - data.begin_);
// There could be threads running in suspended mode when the compaction
// pause is being executed. In order to make the userfaultfd setup atomic,
// the registration has to be done *before* moving the pages to shadow map.
if (!mmap_again) {
// See the comment in the constructor as to why it's conditionally done.
RegisterUffd(data.begin_,
data.shadow_.Size(),
minor_fault_initialized_ ? kMinorFaultMode : kCopyMode);
}
KernelPrepareRangeForUffd(data.begin_,
data.shadow_.Begin(),
data.shadow_.Size(),
mmap_again ? kFdSharedAnon : kFdUnused);
if (mmap_again) {
data.already_shared_ = true;
RegisterUffd(data.begin_,
data.shadow_.Size(),
minor_fault_initialized_ ? kMinorFaultMode : kCopyMode);
}
}
}
if (map_shared) {
// Start mapping linear-alloc MAP_SHARED only after the compaction pause of
// the first GC in non-zygote processes. This is the GC which sets up
// mappings for using minor-fault in future. Up to this point we run
// userfaultfd in copy-mode, which requires the mappings (of linear-alloc)
// to be MAP_PRIVATE.
map_linear_alloc_shared_ = true;
}
}
template <int kMode>
void MarkCompact::ConcurrentCompaction(uint8_t* buf) {
DCHECK_NE(kMode, kFallbackMode);
DCHECK(kMode != kCopyMode || buf != nullptr);
size_t nr_moving_space_used_pages = moving_first_objs_count_ + black_page_count_;
while (true) {
struct uffd_msg msg;
ssize_t nread = read(uffd_, &msg, sizeof(msg));
CHECK_GT(nread, 0);
CHECK_EQ(msg.event, UFFD_EVENT_PAGEFAULT);
DCHECK_EQ(nread, static_cast<ssize_t>(sizeof(msg)));
uint8_t* fault_addr = reinterpret_cast<uint8_t*>(msg.arg.pagefault.address);
if (fault_addr == conc_compaction_termination_page_) {
// The counter doesn't need to be updated atomically as only one thread
// would wake up against the gc-thread's load to this fault_addr. In fact,
// the other threads would wake up serially because every exiting thread
// will wake up gc-thread, which would retry load but again would find the
// page missing. Also, the value will be flushed to caches due to the ioctl
// syscall below.
uint8_t ret = thread_pool_counter_--;
// If 'gKernelHasFaultRetry == true' then only the last thread should map the
// zeropage so that the gc-thread can proceed. Otherwise, each thread does
// it and the gc-thread will repeat this fault until thread_pool_counter == 0.
if (!gKernelHasFaultRetry || ret == 1) {
ZeropageIoctl(fault_addr, /*tolerate_eexist=*/false, /*tolerate_enoent=*/false);
} else {
struct uffdio_range uffd_range;
uffd_range.start = msg.arg.pagefault.address;
uffd_range.len = gPageSize;
CHECK_EQ(ioctl(uffd_, UFFDIO_WAKE, &uffd_range), 0)
<< "ioctl_userfaultfd: wake failed for concurrent-compaction termination page: "
<< strerror(errno);
}
break;
}
uint8_t* fault_page = AlignDown(fault_addr, gPageSize);
if (HasAddress(reinterpret_cast<mirror::Object*>(fault_addr))) {
ConcurrentlyProcessMovingPage<kMode>(fault_page, buf, nr_moving_space_used_pages);
} else if (minor_fault_initialized_) {
ConcurrentlyProcessLinearAllocPage<kMinorFaultMode>(
fault_page, (msg.arg.pagefault.flags & UFFD_PAGEFAULT_FLAG_MINOR) != 0);
} else {
ConcurrentlyProcessLinearAllocPage<kCopyMode>(
fault_page, (msg.arg.pagefault.flags & UFFD_PAGEFAULT_FLAG_MINOR) != 0);
}
}
}
bool MarkCompact::SigbusHandler(siginfo_t* info) {
class ScopedInProgressCount {
public:
explicit ScopedInProgressCount(MarkCompact* collector) : collector_(collector) {
// Increment the count only if compaction is not done yet.
SigbusCounterType prev =
collector_->sigbus_in_progress_count_.load(std::memory_order_relaxed);
while ((prev & kSigbusCounterCompactionDoneMask) == 0) {
if (collector_->sigbus_in_progress_count_.compare_exchange_strong(
prev, prev + 1, std::memory_order_acquire)) {
DCHECK_LT(prev, kSigbusCounterCompactionDoneMask - 1);
compaction_done_ = false;
return;
}
}
compaction_done_ = true;
}
bool IsCompactionDone() const {
return compaction_done_;
}
~ScopedInProgressCount() {
if (!IsCompactionDone()) {
collector_->sigbus_in_progress_count_.fetch_sub(1, std::memory_order_release);
}
}
private:
MarkCompact* const collector_;
bool compaction_done_;
};
DCHECK(use_uffd_sigbus_);
if (info->si_code != BUS_ADRERR) {
// Userfaultfd raises SIGBUS with BUS_ADRERR. All other causes can't be
// handled here.
return false;
}
ScopedInProgressCount spc(this);
uint8_t* fault_page = AlignDown(reinterpret_cast<uint8_t*>(info->si_addr), gPageSize);
if (!spc.IsCompactionDone()) {
if (HasAddress(reinterpret_cast<mirror::Object*>(fault_page))) {
Thread* self = Thread::Current();
Locks::mutator_lock_->AssertSharedHeld(self);
size_t nr_moving_space_used_pages = moving_first_objs_count_ + black_page_count_;
if (minor_fault_initialized_) {
ConcurrentlyProcessMovingPage<kMinorFaultMode>(
fault_page, nullptr, nr_moving_space_used_pages);
} else {
ConcurrentlyProcessMovingPage<kCopyMode>(
fault_page, self->GetThreadLocalGcBuffer(), nr_moving_space_used_pages);
}
return true;
} else {
// Find the linear-alloc space containing fault-addr
for (auto& data : linear_alloc_spaces_data_) {
if (data.begin_ <= fault_page && data.end_ > fault_page) {
if (minor_fault_initialized_) {
ConcurrentlyProcessLinearAllocPage<kMinorFaultMode>(fault_page, false);
} else {
ConcurrentlyProcessLinearAllocPage<kCopyMode>(fault_page, false);
}
return true;
}
}
// Fault address doesn't belong to either moving-space or linear-alloc.
return false;
}
} else {
// We may spuriously get SIGBUS fault, which was initiated before the
// compaction was finished, but ends up here. In that case, if the fault
// address is valid then consider it handled.
return HasAddress(reinterpret_cast<mirror::Object*>(fault_page)) ||
linear_alloc_spaces_data_.end() !=
std::find_if(linear_alloc_spaces_data_.begin(),
linear_alloc_spaces_data_.end(),
[fault_page](const LinearAllocSpaceData& data) {
return data.begin_ <= fault_page && data.end_ > fault_page;
});
}
}
static void BackOff(uint32_t i) {
static constexpr uint32_t kYieldMax = 5;
// TODO: Consider adding x86 PAUSE and/or ARM YIELD here.
if (i <= kYieldMax) {
sched_yield();
} else {
// nanosleep is not in the async-signal-safe list, but bionic implements it
// with a pure system call, so it should be fine.
NanoSleep(10000ull * (i - kYieldMax));
}
}
template <int kMode>
void MarkCompact::ConcurrentlyProcessMovingPage(uint8_t* fault_page,
uint8_t* buf,
size_t nr_moving_space_used_pages) {
class ScopedInProgressCount {
public:
explicit ScopedInProgressCount(MarkCompact* collector) : collector_(collector) {
collector_->compaction_in_progress_count_.fetch_add(1, std::memory_order_relaxed);
}
~ScopedInProgressCount() {
collector_->compaction_in_progress_count_.fetch_sub(1, std::memory_order_relaxed);
}
private:
MarkCompact* collector_;
};
uint8_t* unused_space_begin =
bump_pointer_space_->Begin() + nr_moving_space_used_pages * gPageSize;
DCHECK(IsAlignedParam(unused_space_begin, gPageSize));
DCHECK(kMode == kCopyMode || fault_page < unused_space_begin);
if (kMode == kCopyMode && fault_page >= unused_space_begin) {
// There is a race which allows more than one thread to install a
// zero-page. But we can tolerate that. So absorb the EEXIST returned by
// the ioctl and move on.
ZeropageIoctl(fault_page, /*tolerate_eexist=*/true, /*tolerate_enoent=*/true);
return;
}
size_t page_idx = (fault_page - bump_pointer_space_->Begin()) / gPageSize;
DCHECK_LT(page_idx, moving_first_objs_count_ + black_page_count_);
mirror::Object* first_obj = first_objs_moving_space_[page_idx].AsMirrorPtr();
if (first_obj == nullptr) {
// We should never have a case where two workers are trying to install a
// zeropage in this range as we synchronize using moving_pages_status_[page_idx].
PageState expected_state = PageState::kUnprocessed;
if (moving_pages_status_[page_idx].compare_exchange_strong(
expected_state, PageState::kProcessedAndMapping, std::memory_order_relaxed)) {
// Note: ioctl acts as an acquire fence.
ZeropageIoctl(fault_page, /*tolerate_eexist=*/false, /*tolerate_enoent=*/true);
} else {
DCHECK_EQ(expected_state, PageState::kProcessedAndMapping);
}
return;
}
PageState state = moving_pages_status_[page_idx].load(
use_uffd_sigbus_ ? std::memory_order_acquire : std::memory_order_relaxed);
uint32_t backoff_count = 0;
while (true) {
switch (state) {
case PageState::kUnprocessed: {
// The increment to the in-progress counter must be done before updating
// the page's state. Otherwise, we will end up leaving a window wherein
// the GC-thread could observe that no worker is working on compaction
// and could end up unregistering the moving space from userfaultfd.
ScopedInProgressCount spc(this);
// Acquire order to ensure we don't start writing to shadow map, which is
// shared, before the CAS is successful. Release order to ensure that the
// increment to moving_compactions_in_progress above is not re-ordered
// after the CAS.
if (moving_pages_status_[page_idx].compare_exchange_strong(
state, PageState::kMutatorProcessing, std::memory_order_acq_rel)) {
if (kMode == kMinorFaultMode) {
DCHECK_EQ(buf, nullptr);
buf = shadow_to_space_map_.Begin() + page_idx * gPageSize;
} else if (UNLIKELY(buf == nullptr)) {
DCHECK_EQ(kMode, kCopyMode);
uint16_t idx = compaction_buffer_counter_.fetch_add(1, std::memory_order_relaxed);
// The buffer-map is one page bigger as the first buffer is used by GC-thread.
CHECK_LE(idx, kMutatorCompactionBufferCount);
buf = compaction_buffers_map_.Begin() + idx * gPageSize;
DCHECK(compaction_buffers_map_.HasAddress(buf));
Thread::Current()->SetThreadLocalGcBuffer(buf);
}
if (fault_page < post_compact_end_) {
// The page has to be compacted.
CompactPage(
first_obj, pre_compact_offset_moving_space_[page_idx], buf, kMode == kCopyMode);
} else {
DCHECK_NE(first_obj, nullptr);
DCHECK_GT(pre_compact_offset_moving_space_[page_idx], 0u);
uint8_t* pre_compact_page = black_allocations_begin_ + (fault_page - post_compact_end_);
uint32_t first_chunk_size = black_alloc_pages_first_chunk_size_[page_idx];
mirror::Object* next_page_first_obj = nullptr;
if (page_idx + 1 < moving_first_objs_count_ + black_page_count_) {
next_page_first_obj = first_objs_moving_space_[page_idx + 1].AsMirrorPtr();
}
DCHECK(IsAlignedParam(pre_compact_page, gPageSize));
SlideBlackPage(first_obj,
next_page_first_obj,
first_chunk_size,
pre_compact_page,
buf,
kMode == kCopyMode);
}
// Nobody else would simultaneously modify this page's state so an
// atomic store is sufficient. Use 'release' order to guarantee that
// loads/stores to the page are finished before this store.
moving_pages_status_[page_idx].store(PageState::kProcessedAndMapping,
std::memory_order_release);
if (kMode == kCopyMode) {
CopyIoctl(fault_page, buf);
if (use_uffd_sigbus_) {
// Store is sufficient as no other thread modifies the status at this stage.
moving_pages_status_[page_idx].store(PageState::kProcessedAndMapped,
std::memory_order_release);
}
return;
} else {
break;
}
}
}
continue;
case PageState::kProcessing:
DCHECK_EQ(kMode, kMinorFaultMode);
if (moving_pages_status_[page_idx].compare_exchange_strong(
state, PageState::kProcessingAndMapping, std::memory_order_relaxed) &&
!use_uffd_sigbus_) {
// Somebody else took or will take care of finishing the compaction and
// then mapping the page.
return;
}
continue;
case PageState::kProcessed:
// The page is processed but not mapped. We should map it.
break;
case PageState::kProcessingAndMapping:
case PageState::kMutatorProcessing:
case PageState::kProcessedAndMapping:
if (use_uffd_sigbus_) {
// Wait for the page to be mapped before returning.
BackOff(backoff_count++);
state = moving_pages_status_[page_idx].load(std::memory_order_acquire);
continue;
}
return;
case PageState::kProcessedAndMapped:
// Somebody else took care of the page.
return;
}
break;
}
DCHECK_EQ(kMode, kMinorFaultMode);
if (state == PageState::kUnprocessed) {
MapProcessedPages</*kFirstPageMapping=*/true>(
fault_page, moving_pages_status_, page_idx, nr_moving_space_used_pages);
} else {
DCHECK_EQ(state, PageState::kProcessed);
MapProcessedPages</*kFirstPageMapping=*/false>(
fault_page, moving_pages_status_, page_idx, nr_moving_space_used_pages);
}
}
void MarkCompact::MapUpdatedLinearAllocPage(uint8_t* page,
uint8_t* shadow_page,
Atomic<PageState>& state,
bool page_touched) {
DCHECK(!minor_fault_initialized_);
if (page_touched) {
CopyIoctl(page, shadow_page);
} else {
// If the page wasn't touched, then it means it is empty and
// is most likely not present on the shadow-side. Furthermore,
// since the shadow is also userfaultfd registered doing copy
// ioctl fail as the copy-from-user in the kernel will cause
// userfault. Instead, just map a zeropage, which is not only
// correct but also efficient as it avoids unnecessary memcpy
// in the kernel.
ZeropageIoctl(page, /*tolerate_eexist=*/false, /*tolerate_enoent=*/false);
}
if (use_uffd_sigbus_) {
// Store is sufficient as no other thread can modify the
// status of this page at this point.
state.store(PageState::kProcessedAndMapped, std::memory_order_release);
}
}
template <int kMode>
void MarkCompact::ConcurrentlyProcessLinearAllocPage(uint8_t* fault_page, bool is_minor_fault) {
DCHECK(!is_minor_fault || kMode == kMinorFaultMode);
auto arena_iter = linear_alloc_arenas_.end();
{
TrackedArena temp_arena(fault_page);
arena_iter = linear_alloc_arenas_.upper_bound(&temp_arena);
arena_iter = arena_iter != linear_alloc_arenas_.begin() ? std::prev(arena_iter)
: linear_alloc_arenas_.end();
}
// Unlike ProcessLinearAlloc(), we don't need to hold arena-pool's lock here
// because a thread trying to access the page and as a result causing this
// userfault confirms that nobody can delete the corresponding arena and
// release its pages.
// NOTE: We may have some memory range be recycled several times during a
// compaction cycle, thereby potentially causing userfault on the same page
// several times. That's not a problem as all of them (except for possibly the
// first one) would require us mapping a zero-page, which we do without updating
// the 'state_arr'.
if (arena_iter == linear_alloc_arenas_.end() ||
arena_iter->first->IsWaitingForDeletion() ||
arena_iter->second <= fault_page) {
// Fault page isn't in any of the arenas that existed before we started
// compaction. So map zeropage and return.
ZeropageIoctl(fault_page, /*tolerate_eexist=*/true, /*tolerate_enoent=*/false);
} else {
// Find the linear-alloc space containing fault-page
LinearAllocSpaceData* space_data = nullptr;
for (auto& data : linear_alloc_spaces_data_) {
if (data.begin_ <= fault_page && fault_page < data.end_) {
space_data = &data;
break;
}
}
DCHECK_NE(space_data, nullptr);
ptrdiff_t diff = space_data->shadow_.Begin() - space_data->begin_;
size_t page_idx = (fault_page - space_data->begin_) / gPageSize;
Atomic<PageState>* state_arr =
reinterpret_cast<Atomic<PageState>*>(space_data->page_status_map_.Begin());
PageState state = state_arr[page_idx].load(use_uffd_sigbus_ ? std::memory_order_acquire :
std::memory_order_relaxed);
uint32_t backoff_count = 0;
while (true) {
switch (state) {
case PageState::kUnprocessed: {
// Acquire order to ensure we don't start writing to shadow map, which is
// shared, before the CAS is successful.
if (state_arr[page_idx].compare_exchange_strong(
state, PageState::kProcessingAndMapping, std::memory_order_acquire)) {
if (kMode == kCopyMode || is_minor_fault) {
LinearAllocPageUpdater updater(this);
uint8_t* first_obj = arena_iter->first->GetFirstObject(fault_page);
// null first_obj indicates that it's a page from arena for
// intern-table/class-table. So first object isn't required.
if (first_obj != nullptr) {
updater.MultiObjectArena(fault_page + diff, first_obj + diff);
} else {
updater.SingleObjectArena(fault_page + diff, gPageSize);
}
if (kMode == kCopyMode) {
MapUpdatedLinearAllocPage(fault_page,
fault_page + diff,
state_arr[page_idx],
updater.WasLastPageTouched());
return;
}
} else {
// Don't touch the page in this case (there is no reason to do so
// anyways) as it would mean reading from first_obj, which could be on
// another missing page and hence may cause this thread to block, leading
// to deadlocks.
// Force read the page if it is missing so that a zeropage gets mapped on
// the shadow map and then CONTINUE ioctl will map it on linear-alloc.
ForceRead(fault_page + diff);
}
MapProcessedPages</*kFirstPageMapping=*/true>(
fault_page, state_arr, page_idx, space_data->page_status_map_.Size());
return;
}
}
continue;
case PageState::kProcessing:
DCHECK_EQ(kMode, kMinorFaultMode);
if (state_arr[page_idx].compare_exchange_strong(
state, PageState::kProcessingAndMapping, std::memory_order_relaxed) &&
!use_uffd_sigbus_) {
// Somebody else took or will take care of finishing the updates and
// then mapping the page.
return;
}
continue;
case PageState::kProcessed:
// The page is processed but not mapped. We should map it.
break;
case PageState::kMutatorProcessing:
UNREACHABLE();
case PageState::kProcessingAndMapping:
case PageState::kProcessedAndMapping:
if (use_uffd_sigbus_) {
// Wait for the page to be mapped before returning.
BackOff(backoff_count++);
state = state_arr[page_idx].load(std::memory_order_acquire);
continue;
}
return;
case PageState::kProcessedAndMapped:
// Somebody else took care of the page.
return;
}
break;
}
DCHECK_EQ(kMode, kMinorFaultMode);
DCHECK_EQ(state, PageState::kProcessed);
if (!is_minor_fault) {
// Force read the page if it is missing so that a zeropage gets mapped on
// the shadow map and then CONTINUE ioctl will map it on linear-alloc.
ForceRead(fault_page + diff);
}
MapProcessedPages</*kFirstPageMapping=*/false>(
fault_page, state_arr, page_idx, space_data->page_status_map_.Size());
}
}
void MarkCompact::ProcessLinearAlloc() {
GcVisitedArenaPool* arena_pool =
static_cast<GcVisitedArenaPool*>(Runtime::Current()->GetLinearAllocArenaPool());
DCHECK_EQ(thread_running_gc_, Thread::Current());
for (auto& pair : linear_alloc_arenas_) {
const TrackedArena* arena = pair.first;
size_t arena_size;
uint8_t* arena_begin;
ptrdiff_t diff;
bool others_processing;
{
// Acquire arena-pool's lock (in shared-mode) so that the arena being updated
// does not get deleted at the same time. If this critical section is too
// long and impacts mutator response time, then we get rid of this lock by
// holding onto memory ranges of all deleted (since compaction pause)
// arenas until completion finishes.
ReaderMutexLock rmu(thread_running_gc_, arena_pool->GetLock());
// If any arenas were freed since compaction pause then skip them from
// visiting.
if (arena->IsWaitingForDeletion()) {
continue;
}
uint8_t* last_byte = pair.second;
DCHECK_ALIGNED_PARAM(last_byte, gPageSize);
others_processing = false;
arena_begin = arena->Begin();
arena_size = arena->Size();
// Find the linear-alloc space containing the arena
LinearAllocSpaceData* space_data = nullptr;
for (auto& data : linear_alloc_spaces_data_) {
if (data.begin_ <= arena_begin && arena_begin < data.end_) {
space_data = &data;
break;
}
}
CHECK_NE(space_data, nullptr);
diff = space_data->shadow_.Begin() - space_data->begin_;
auto visitor = [space_data, last_byte, diff, this, &others_processing](
uint8_t* page_begin,
uint8_t* first_obj,
size_t page_size) REQUIRES_SHARED(Locks::mutator_lock_) {
// No need to process pages past last_byte as they already have updated
// gc-roots, if any.
if (page_begin >= last_byte) {
return;
}
LinearAllocPageUpdater updater(this);
size_t page_idx = (page_begin - space_data->begin_) / gPageSize;
DCHECK_LT(page_idx, space_data->page_status_map_.Size());
Atomic<PageState>* state_arr =
reinterpret_cast<Atomic<PageState>*>(space_data->page_status_map_.Begin());
PageState expected_state = PageState::kUnprocessed;
PageState desired_state =
minor_fault_initialized_ ? PageState::kProcessing : PageState::kProcessingAndMapping;
// Acquire order to ensure that we don't start accessing the shadow page,
// which is shared with other threads, prior to CAS. Also, for same
// reason, we used 'release' order for changing the state to 'processed'.
if (state_arr[page_idx].compare_exchange_strong(
expected_state, desired_state, std::memory_order_acquire)) {
// null first_obj indicates that it's a page from arena for
// intern-table/class-table. So first object isn't required.
if (first_obj != nullptr) {
updater.MultiObjectArena(page_begin + diff, first_obj + diff);
} else {
DCHECK_EQ(page_size, gPageSize);
updater.SingleObjectArena(page_begin + diff, page_size);
}
expected_state = PageState::kProcessing;
if (!minor_fault_initialized_) {
MapUpdatedLinearAllocPage(
page_begin, page_begin + diff, state_arr[page_idx], updater.WasLastPageTouched());
} else if (!state_arr[page_idx].compare_exchange_strong(
expected_state, PageState::kProcessed, std::memory_order_release)) {
DCHECK_EQ(expected_state, PageState::kProcessingAndMapping);
// Force read in case the page was missing and updater didn't touch it
// as there was nothing to do. This will ensure that a zeropage is
// faulted on the shadow map.
ForceRead(page_begin + diff);
MapProcessedPages</*kFirstPageMapping=*/true>(
page_begin, state_arr, page_idx, space_data->page_status_map_.Size());
}
} else {
others_processing = true;
}
};
arena->VisitRoots(visitor);
}
// If we are not in minor-fault mode and if no other thread was found to be
// processing any pages in this arena, then we can madvise the shadow size.
// Otherwise, we will double the memory use for linear-alloc.
if (!minor_fault_initialized_ && !others_processing) {
ZeroAndReleaseMemory(arena_begin + diff, arena_size);
}
}
}
void MarkCompact::RegisterUffd(void* addr, size_t size, int mode) {
DCHECK(IsValidFd(uffd_));
struct uffdio_register uffd_register;
uffd_register.range.start = reinterpret_cast<uintptr_t>(addr);
uffd_register.range.len = size;
uffd_register.mode = UFFDIO_REGISTER_MODE_MISSING;
if (mode == kMinorFaultMode) {
uffd_register.mode |= UFFDIO_REGISTER_MODE_MINOR;
}
CHECK_EQ(ioctl(uffd_, UFFDIO_REGISTER, &uffd_register), 0)
<< "ioctl_userfaultfd: register failed: " << strerror(errno)
<< ". start:" << static_cast<void*>(addr) << " len:" << PrettySize(size);
}
void MarkCompact::UnregisterUffd(uint8_t* start, size_t len) {
DCHECK(IsValidFd(uffd_));
struct uffdio_range range;
range.start = reinterpret_cast<uintptr_t>(start);
range.len = len;
CHECK_EQ(ioctl(uffd_, UFFDIO_UNREGISTER, &range), 0)
<< "ioctl_userfaultfd: unregister failed: " << strerror(errno)
<< ". addr:" << static_cast<void*>(start) << " len:" << PrettySize(len);
// Due to an oversight in the kernel implementation of 'unregister', the
// waiting threads are woken up only for copy uffds. Therefore, for now, we
// have to explicitly wake up the threads in minor-fault case.
// TODO: The fix in the kernel is being worked on. Once the kernel version
// containing the fix is known, make it conditional on that as well.
if (minor_fault_initialized_) {
CHECK_EQ(ioctl(uffd_, UFFDIO_WAKE, &range), 0)
<< "ioctl_userfaultfd: wake failed: " << strerror(errno)
<< ". addr:" << static_cast<void*>(start) << " len:" << PrettySize(len);
}
}
void MarkCompact::CompactionPhase() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
{
int32_t freed_bytes = black_objs_slide_diff_;
bump_pointer_space_->RecordFree(freed_objects_, freed_bytes);
RecordFree(ObjectBytePair(freed_objects_, freed_bytes));
}
size_t moving_space_size = bump_pointer_space_->Capacity();
size_t used_size = (moving_first_objs_count_ + black_page_count_) * gPageSize;
if (CanCompactMovingSpaceWithMinorFault()) {
CompactMovingSpace<kMinorFaultMode>(/*page=*/nullptr);
} else {
if (used_size < moving_space_size) {
// mremap clears 'anon_vma' field of anonymous mappings. If we
// uffd-register only the used portion of the space, then the vma gets
// split (between used and unused portions) and as soon as pages are
// mapped to the vmas, they get different `anon_vma` assigned, which
// ensures that the two vmas cannot merged after we uffd-unregister the
// used portion. OTOH, registering the entire space avoids the split, but
// unnecessarily causes userfaults on allocations.
// By mapping a zero-page (below) we let the kernel assign an 'anon_vma'
// *before* the vma-split caused by uffd-unregister of the unused portion
// This ensures that when we unregister the used portion after compaction,
// the two split vmas merge. This is necessary for the mremap of the
// next GC cycle to not fail due to having more than one vmas in the source
// range.
uint8_t* unused_first_page = bump_pointer_space_->Begin() + used_size;
// It's ok if somebody else already mapped the page.
ZeropageIoctl(unused_first_page, /*tolerate_eexist*/ true, /*tolerate_enoent*/ false);
UnregisterUffd(unused_first_page, moving_space_size - used_size);
}
CompactMovingSpace<kCopyMode>(compaction_buffers_map_.Begin());
}
// Make sure no mutator is reading from the from-space before unregistering
// userfaultfd from moving-space and then zapping from-space. The mutator
// and GC may race to set a page state to processing or further along. The two
// attempts are ordered. If the collector wins, then the mutator will see that
// and not access the from-space page. If the muator wins, then the
// compaction_in_progress_count_ increment by the mutator happens-before the test
// here, and we will not see a zero value until the mutator has completed.
for (uint32_t i = 0; compaction_in_progress_count_.load(std::memory_order_acquire) > 0; i++) {
BackOff(i);
}
if (used_size > 0) {
UnregisterUffd(bump_pointer_space_->Begin(), used_size);
}
// Release all of the memory taken by moving-space's from-map
if (minor_fault_initialized_) {
if (IsValidFd(moving_from_space_fd_)) {
// A strange behavior is observed wherein between GC cycles the from-space'
// first page is accessed. But the memfd that is mapped on from-space, is
// used on to-space in next GC cycle, causing issues with userfaultfd as the
// page isn't missing. A possible reason for this could be prefetches. The
// mprotect ensures that such accesses don't succeed.
int ret = mprotect(from_space_begin_, moving_space_size, PROT_NONE);
CHECK_EQ(ret, 0) << "mprotect(PROT_NONE) for from-space failed: " << strerror(errno);
// madvise(MADV_REMOVE) needs PROT_WRITE. Use fallocate() instead, which
// does the same thing.
ret = fallocate(moving_from_space_fd_,
FALLOC_FL_PUNCH_HOLE | FALLOC_FL_KEEP_SIZE,
/*offset=*/0,
moving_space_size);
CHECK_EQ(ret, 0) << "fallocate for from-space failed: " << strerror(errno);
} else {
// We don't have a valid fd, so use madvise(MADV_REMOVE) instead. mprotect
// is not required in this case as we create fresh
// MAP_SHARED+MAP_ANONYMOUS mapping in each GC cycle.
int ret = madvise(from_space_begin_, moving_space_size, MADV_REMOVE);
CHECK_EQ(ret, 0) << "madvise(MADV_REMOVE) failed for from-space map:" << strerror(errno);
}
} else {
from_space_map_.MadviseDontNeedAndZero();
}
// mprotect(PROT_NONE) all maps except to-space in debug-mode to catch any unexpected accesses.
if (shadow_to_space_map_.IsValid()) {
DCHECK_EQ(mprotect(shadow_to_space_map_.Begin(), shadow_to_space_map_.Size(), PROT_NONE), 0)
<< "mprotect(PROT_NONE) for shadow-map failed:" << strerror(errno);
}
if (!IsValidFd(moving_from_space_fd_)) {
// The other case is already mprotected above.
DCHECK_EQ(mprotect(from_space_begin_, moving_space_size, PROT_NONE), 0)
<< "mprotect(PROT_NONE) for from-space failed: " << strerror(errno);
}
ProcessLinearAlloc();
if (use_uffd_sigbus_) {
// Set compaction-done bit so that no new mutator threads start compaction
// process in the SIGBUS handler.
SigbusCounterType count = sigbus_in_progress_count_.fetch_or(kSigbusCounterCompactionDoneMask,
std::memory_order_acq_rel);
// Wait for SIGBUS handlers already in play.
for (uint32_t i = 0; count > 0; i++) {
BackOff(i);
count = sigbus_in_progress_count_.load(std::memory_order_acquire);
count &= ~kSigbusCounterCompactionDoneMask;
}
} else {
DCHECK(IsAlignedParam(conc_compaction_termination_page_, gPageSize));
// We will only iterate once if gKernelHasFaultRetry is true.
do {
// madvise the page so that we can get userfaults on it.
ZeroAndReleaseMemory(conc_compaction_termination_page_, gPageSize);
// The following load triggers 'special' userfaults. When received by the
// thread-pool workers, they will exit out of the compaction task. This fault
// happens because we madvised the page.
ForceRead(conc_compaction_termination_page_);
} while (thread_pool_counter_ > 0);
}
// Unregister linear-alloc spaces
for (auto& data : linear_alloc_spaces_data_) {
DCHECK_EQ(data.end_ - data.begin_, static_cast<ssize_t>(data.shadow_.Size()));
UnregisterUffd(data.begin_, data.shadow_.Size());
// madvise linear-allocs's page-status array
data.page_status_map_.MadviseDontNeedAndZero();
// Madvise the entire linear-alloc space's shadow. In copy-mode it gets rid
// of the pages which are still mapped. In minor-fault mode this unmaps all
// pages, which is good in reducing the mremap (done in STW pause) time in
// next GC cycle.
data.shadow_.MadviseDontNeedAndZero();
if (minor_fault_initialized_) {
DCHECK_EQ(mprotect(data.shadow_.Begin(), data.shadow_.Size(), PROT_NONE), 0)
<< "mprotect failed: " << strerror(errno);
}
}
if (!use_uffd_sigbus_) {
heap_->GetThreadPool()->StopWorkers(thread_running_gc_);
}
}
template <size_t kBufferSize>
class MarkCompact::ThreadRootsVisitor : public RootVisitor {
public:
explicit ThreadRootsVisitor(MarkCompact* mark_compact, Thread* const self)
: mark_compact_(mark_compact), self_(self) {}
~ThreadRootsVisitor() {
Flush();
}
void VisitRoots(mirror::Object*** roots,
size_t count,
[[maybe_unused]] const RootInfo& info) override
REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_) {
for (size_t i = 0; i < count; i++) {
mirror::Object* obj = *roots[i];
if (mark_compact_->MarkObjectNonNullNoPush</*kParallel*/true>(obj)) {
Push(obj);
}
}
}
void VisitRoots(mirror::CompressedReference<mirror::Object>** roots,
size_t count,
[[maybe_unused]] const RootInfo& info) override
REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(Locks::heap_bitmap_lock_) {
for (size_t i = 0; i < count; i++) {
mirror::Object* obj = roots[i]->AsMirrorPtr();
if (mark_compact_->MarkObjectNonNullNoPush</*kParallel*/true>(obj)) {
Push(obj);
}
}
}
private:
void Flush() REQUIRES_SHARED(Locks::mutator_lock_)
REQUIRES(Locks::heap_bitmap_lock_) {
StackReference<mirror::Object>* start;
StackReference<mirror::Object>* end;
{
MutexLock mu(self_, mark_compact_->lock_);
// Loop here because even after expanding once it may not be sufficient to
// accommodate all references. It's almost impossible, but there is no harm
// in implementing it this way.
while (!mark_compact_->mark_stack_->BumpBack(idx_, &start, &end)) {
mark_compact_->ExpandMarkStack();
}
}
while (idx_ > 0) {
*start++ = roots_[--idx_];
}
DCHECK_EQ(start, end);
}
void Push(mirror::Object* obj) REQUIRES_SHARED(Locks::mutator_lock_)
REQUIRES(Locks::heap_bitmap_lock_) {
if (UNLIKELY(idx_ >= kBufferSize)) {
Flush();
}
roots_[idx_++].Assign(obj);
}
StackReference<mirror::Object> roots_[kBufferSize];
size_t idx_ = 0;
MarkCompact* const mark_compact_;
Thread* const self_;
};
class MarkCompact::CheckpointMarkThreadRoots : public Closure {
public:
explicit CheckpointMarkThreadRoots(MarkCompact* mark_compact) : mark_compact_(mark_compact) {}
void Run(Thread* thread) override NO_THREAD_SAFETY_ANALYSIS {
ScopedTrace trace("Marking thread roots");
// Note: self is not necessarily equal to thread since thread may be
// suspended.
Thread* const self = Thread::Current();
CHECK(thread == self
|| thread->IsSuspended()
|| thread->GetState() == ThreadState::kWaitingPerformingGc)
<< thread->GetState() << " thread " << thread << " self " << self;
{
ThreadRootsVisitor</*kBufferSize*/ 20> visitor(mark_compact_, self);
thread->VisitRoots(&visitor, kVisitRootFlagAllRoots);
}
// Clear page-buffer to prepare for compaction phase.
thread->SetThreadLocalGcBuffer(nullptr);
// If thread is a running mutator, then act on behalf of the garbage
// collector. See the code in ThreadList::RunCheckpoint.
mark_compact_->GetBarrier().Pass(self);
}
private:
MarkCompact* const mark_compact_;
};
void MarkCompact::MarkRootsCheckpoint(Thread* self, Runtime* runtime) {
// We revote TLABs later during paused round of marking.
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
CheckpointMarkThreadRoots check_point(this);
ThreadList* thread_list = runtime->GetThreadList();
gc_barrier_.Init(self, 0);
// Request the check point is run on all threads returning a count of the threads that must
// run through the barrier including self.
size_t barrier_count = thread_list->RunCheckpoint(&check_point);
// Release locks then wait for all mutator threads to pass the barrier.
// If there are no threads to wait which implys that all the checkpoint functions are finished,
// then no need to release locks.
if (barrier_count == 0) {
return;
}
Locks::heap_bitmap_lock_->ExclusiveUnlock(self);
Locks::mutator_lock_->SharedUnlock(self);
{
ScopedThreadStateChange tsc(self, ThreadState::kWaitingForCheckPointsToRun);
gc_barrier_.Increment(self, barrier_count);
}
Locks::mutator_lock_->SharedLock(self);
Locks::heap_bitmap_lock_->ExclusiveLock(self);
}
void MarkCompact::MarkNonThreadRoots(Runtime* runtime) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
runtime->VisitNonThreadRoots(this);
}
void MarkCompact::MarkConcurrentRoots(VisitRootFlags flags, Runtime* runtime) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
runtime->VisitConcurrentRoots(this, flags);
}
void MarkCompact::RevokeAllThreadLocalBuffers() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
bump_pointer_space_->RevokeAllThreadLocalBuffers();
}
class MarkCompact::ScanObjectVisitor {
public:
explicit ScanObjectVisitor(MarkCompact* const mark_compact) ALWAYS_INLINE
: mark_compact_(mark_compact) {}
void operator()(ObjPtr<mirror::Object> obj) const
ALWAYS_INLINE
REQUIRES(Locks::heap_bitmap_lock_)
REQUIRES_SHARED(Locks::mutator_lock_) {
mark_compact_->ScanObject</*kUpdateLiveWords*/ false>(obj.Ptr());
}
private:
MarkCompact* const mark_compact_;
};
void MarkCompact::UpdateAndMarkModUnion() {
accounting::CardTable* const card_table = heap_->GetCardTable();
for (const auto& space : immune_spaces_.GetSpaces()) {
const char* name = space->IsZygoteSpace()
? "UpdateAndMarkZygoteModUnionTable"
: "UpdateAndMarkImageModUnionTable";
DCHECK(space->IsZygoteSpace() || space->IsImageSpace()) << *space;
TimingLogger::ScopedTiming t(name, GetTimings());
accounting::ModUnionTable* table = heap_->FindModUnionTableFromSpace(space);
if (table != nullptr) {
// UpdateAndMarkReferences() doesn't visit Reference-type objects. But
// that's fine because these objects are immutable enough (referent can
// only be cleared) and hence the only referents they can have are intra-space.
table->UpdateAndMarkReferences(this);
} else {
// No mod-union table, scan all dirty/aged cards in the corresponding
// card-table. This can only occur for app images.
card_table->Scan</*kClearCard*/ false>(space->GetMarkBitmap(),
space->Begin(),
space->End(),
ScanObjectVisitor(this),
gc::accounting::CardTable::kCardAged);
}
}
}
void MarkCompact::MarkReachableObjects() {
UpdateAndMarkModUnion();
// Recursively mark all the non-image bits set in the mark bitmap.
ProcessMarkStack();
}
void MarkCompact::ScanDirtyObjects(bool paused, uint8_t minimum_age) {
accounting::CardTable* card_table = heap_->GetCardTable();
for (const auto& space : heap_->GetContinuousSpaces()) {
const char* name = nullptr;
switch (space->GetGcRetentionPolicy()) {
case space::kGcRetentionPolicyNeverCollect:
name = paused ? "(Paused)ScanGrayImmuneSpaceObjects" : "ScanGrayImmuneSpaceObjects";
break;
case space::kGcRetentionPolicyFullCollect:
name = paused ? "(Paused)ScanGrayZygoteSpaceObjects" : "ScanGrayZygoteSpaceObjects";
break;
case space::kGcRetentionPolicyAlwaysCollect:
name = paused ? "(Paused)ScanGrayAllocSpaceObjects" : "ScanGrayAllocSpaceObjects";
break;
default:
LOG(FATAL) << "Unreachable";
UNREACHABLE();
}
TimingLogger::ScopedTiming t(name, GetTimings());
card_table->Scan</*kClearCard*/ false>(
space->GetMarkBitmap(), space->Begin(), space->End(), ScanObjectVisitor(this), minimum_age);
}
}
void MarkCompact::RecursiveMarkDirtyObjects(bool paused, uint8_t minimum_age) {
ScanDirtyObjects(paused, minimum_age);
ProcessMarkStack();
}
void MarkCompact::MarkRoots(VisitRootFlags flags) {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
Runtime* runtime = Runtime::Current();
// Make sure that the checkpoint which collects the stack roots is the first
// one capturning GC-roots. As this one is supposed to find the address
// everything allocated after that (during this marking phase) will be
// considered 'marked'.
MarkRootsCheckpoint(thread_running_gc_, runtime);
MarkNonThreadRoots(runtime);
MarkConcurrentRoots(flags, runtime);
}
void MarkCompact::PreCleanCards() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
CHECK(!Locks::mutator_lock_->IsExclusiveHeld(thread_running_gc_));
// Age the card-table before thread stack scanning checkpoint in MarkRoots()
// as it ensures that there are no in-progress write barriers which started
// prior to aging the card-table.
PrepareCardTableForMarking(/*clear_alloc_space_cards*/ false);
MarkRoots(static_cast<VisitRootFlags>(kVisitRootFlagClearRootLog | kVisitRootFlagNewRoots));
RecursiveMarkDirtyObjects(/*paused*/ false, accounting::CardTable::kCardDirty - 1);
}
// In a concurrent marking algorithm, if we are not using a write/read barrier, as
// in this case, then we need a stop-the-world (STW) round in the end to mark
// objects which were written into concurrently while concurrent marking was
// performed.
// In order to minimize the pause time, we could take one of the two approaches:
// 1. Keep repeating concurrent marking of dirty cards until the time spent goes
// below a threshold.
// 2. Do two rounds concurrently and then attempt a paused one. If we figure
// that it's taking too long, then resume mutators and retry.
//
// Given the non-trivial fixed overhead of running a round (card table and root
// scan), it might be better to go with approach 2.
void MarkCompact::MarkingPhase() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
DCHECK_EQ(thread_running_gc_, Thread::Current());
WriterMutexLock mu(thread_running_gc_, *Locks::heap_bitmap_lock_);
MaybeClampGcStructures();
PrepareCardTableForMarking(/*clear_alloc_space_cards*/ true);
MarkZygoteLargeObjects();
MarkRoots(
static_cast<VisitRootFlags>(kVisitRootFlagAllRoots | kVisitRootFlagStartLoggingNewRoots));
MarkReachableObjects();
// Pre-clean dirtied cards to reduce pauses.
PreCleanCards();
// Setup reference processing and forward soft references once before enabling
// slow path (in MarkingPause)
ReferenceProcessor* rp = GetHeap()->GetReferenceProcessor();
bool clear_soft_references = GetCurrentIteration()->GetClearSoftReferences();
rp->Setup(thread_running_gc_, this, /*concurrent=*/ true, clear_soft_references);
if (!clear_soft_references) {
// Forward as many SoftReferences as possible before inhibiting reference access.
rp->ForwardSoftReferences(GetTimings());
}
}
class MarkCompact::RefFieldsVisitor {
public:
ALWAYS_INLINE explicit RefFieldsVisitor(MarkCompact* const mark_compact)
: mark_compact_(mark_compact) {}
ALWAYS_INLINE void operator()(mirror::Object* obj,
MemberOffset offset,
[[maybe_unused]] bool is_static) const
REQUIRES(Locks::heap_bitmap_lock_) REQUIRES_SHARED(Locks::mutator_lock_) {
if (kCheckLocks) {
Locks::mutator_lock_->AssertSharedHeld(Thread::Current());
Locks::heap_bitmap_lock_->AssertExclusiveHeld(Thread::Current());
}
mark_compact_->MarkObject(obj->GetFieldObject<mirror::Object>(offset), obj, offset);
}
void operator()(ObjPtr<mirror::Class> klass, ObjPtr<mirror::Reference> ref) const ALWAYS_INLINE
REQUIRES(Locks::heap_bitmap_lock_) REQUIRES_SHARED(Locks::mutator_lock_) {
mark_compact_->DelayReferenceReferent(klass, ref);
}
void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const ALWAYS_INLINE
REQUIRES(Locks::heap_bitmap_lock_) REQUIRES_SHARED(Locks::mutator_lock_) {
if (!root->IsNull()) {
VisitRoot(root);
}
}
void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
REQUIRES(Locks::heap_bitmap_lock_)
REQUIRES_SHARED(Locks::mutator_lock_) {
if (kCheckLocks) {
Locks::mutator_lock_->AssertSharedHeld(Thread::Current());
Locks::heap_bitmap_lock_->AssertExclusiveHeld(Thread::Current());
}
mark_compact_->MarkObject(root->AsMirrorPtr());
}
private:
MarkCompact* const mark_compact_;
};
template <size_t kAlignment>
size_t MarkCompact::LiveWordsBitmap<kAlignment>::LiveBytesInBitmapWord(size_t chunk_idx) const {
const size_t index = chunk_idx * kBitmapWordsPerVectorWord;
size_t words = 0;
for (uint32_t i = 0; i < kBitmapWordsPerVectorWord; i++) {
words += POPCOUNT(Bitmap::Begin()[index + i]);
}
return words * kAlignment;
}
void MarkCompact::UpdateLivenessInfo(mirror::Object* obj, size_t obj_size) {
DCHECK(obj != nullptr);
DCHECK_EQ(obj_size, obj->SizeOf<kDefaultVerifyFlags>());
uintptr_t obj_begin = reinterpret_cast<uintptr_t>(obj);
UpdateClassAfterObjectMap(obj);
size_t size = RoundUp(obj_size, kAlignment);
uintptr_t bit_index = live_words_bitmap_->SetLiveWords(obj_begin, size);
size_t chunk_idx = (obj_begin - live_words_bitmap_->Begin()) / kOffsetChunkSize;
// Compute the bit-index within the chunk-info vector word.
bit_index %= kBitsPerVectorWord;
size_t first_chunk_portion = std::min(size, (kBitsPerVectorWord - bit_index) * kAlignment);
chunk_info_vec_[chunk_idx++] += first_chunk_portion;
DCHECK_LE(first_chunk_portion, size);
for (size -= first_chunk_portion; size > kOffsetChunkSize; size -= kOffsetChunkSize) {
DCHECK_EQ(chunk_info_vec_[chunk_idx], 0u);
chunk_info_vec_[chunk_idx++] = kOffsetChunkSize;
}
chunk_info_vec_[chunk_idx] += size;
freed_objects_--;
}
template <bool kUpdateLiveWords>
void MarkCompact::ScanObject(mirror::Object* obj) {
// The size of `obj` is used both here (to update `bytes_scanned_`) and in
// `UpdateLivenessInfo`. As fetching this value can be expensive, do it once
// here and pass that information to `UpdateLivenessInfo`.
size_t obj_size = obj->SizeOf<kDefaultVerifyFlags>();
bytes_scanned_ += obj_size;
RefFieldsVisitor visitor(this);
DCHECK(IsMarked(obj)) << "Scanning marked object " << obj << "\n" << heap_->DumpSpaces();
if (kUpdateLiveWords && HasAddress(obj)) {
UpdateLivenessInfo(obj, obj_size);
}
obj->VisitReferences(visitor, visitor);
}
// Scan anything that's on the mark stack.
void MarkCompact::ProcessMarkStack() {
TimingLogger::ScopedTiming t(__FUNCTION__, GetTimings());
// TODO: try prefetch like in CMS
while (!mark_stack_->IsEmpty()) {
mirror::Object* obj = mark_stack_->PopBack();
DCHECK(obj != nullptr);
ScanObject</*kUpdateLiveWords*/ true>(obj);
}
}
void MarkCompact::ExpandMarkStack() {
const size_t new_size = mark_stack_->Capacity() * 2;
std::vector<StackReference<mirror::Object>> temp(mark_stack_->Begin(),
mark_stack_->End());
mark_stack_->Resize(new_size);
for (auto& ref : temp) {
mark_stack_->PushBack(ref.AsMirrorPtr());
}
DCHECK(!mark_stack_->IsFull());
}
inline void MarkCompact::PushOnMarkStack(mirror::Object* obj) {
if (UNLIKELY(mark_stack_->IsFull())) {
ExpandMarkStack();
}
mark_stack_->PushBack(obj);
}
inline void MarkCompact::MarkObjectNonNull(mirror::Object* obj,
mirror::Object* holder,
MemberOffset offset) {
DCHECK(obj != nullptr);
if (MarkObjectNonNullNoPush</*kParallel*/false>(obj, holder, offset)) {
PushOnMarkStack(obj);
}
}
template <bool kParallel>
inline bool MarkCompact::MarkObjectNonNullNoPush(mirror::Object* obj,
mirror::Object* holder,
MemberOffset offset) {
// We expect most of the referenes to be in bump-pointer space, so try that
// first to keep the cost of this function minimal.
if (LIKELY(HasAddress(obj))) {
return kParallel ? !moving_space_bitmap_->AtomicTestAndSet(obj)
: !moving_space_bitmap_->Set(obj);
} else if (non_moving_space_bitmap_->HasAddress(obj)) {
return kParallel ? !non_moving_space_bitmap_->AtomicTestAndSet(obj)
: !non_moving_space_bitmap_->Set(obj);
} else if (immune_spaces_.ContainsObject(obj)) {
DCHECK(IsMarked(obj) != nullptr);
return false;
} else {
// Must be a large-object space, otherwise it's a case of heap corruption.
if (!IsAlignedParam(obj, space::LargeObjectSpace::ObjectAlignment())) {
// Objects in large-object space are aligned to the large-object alignment.
// So if we have an object which doesn't belong to any space and is not
// page-aligned as well, then it's memory corruption.
// TODO: implement protect/unprotect in bump-pointer space.
heap_->GetVerification()->LogHeapCorruption(holder, offset, obj, /*fatal*/ true);
}
DCHECK_NE(heap_->GetLargeObjectsSpace(), nullptr)
<< "ref=" << obj
<< " doesn't belong to any of the spaces and large object space doesn't exist";
accounting::LargeObjectBitmap* los_bitmap = heap_->GetLargeObjectsSpace()->GetMarkBitmap();
DCHECK(los_bitmap->HasAddress(obj));
if (kParallel) {
los_bitmap->AtomicTestAndSet(obj);
} else {
los_bitmap->Set(obj);
}
// We only have primitive arrays in large object space. So there is no
// reason to push into mark-stack.
DCHECK(obj->IsString() || (obj->IsArrayInstance() && !obj->IsObjectArray()));
return false;
}
}
inline void MarkCompact::MarkObject(mirror::Object* obj,
mirror::Object* holder,
MemberOffset offset) {
if (obj != nullptr) {
MarkObjectNonNull(obj, holder, offset);
}
}
mirror::Object* MarkCompact::MarkObject(mirror::Object* obj) {
MarkObject(obj, nullptr, MemberOffset(0));
return obj;
}
void MarkCompact::MarkHeapReference(mirror::HeapReference<mirror::Object>* obj,
[[maybe_unused]] bool do_atomic_update) {
MarkObject(obj->AsMirrorPtr(), nullptr, MemberOffset(0));
}
void MarkCompact::VisitRoots(mirror::Object*** roots,
size_t count,
const RootInfo& info) {
if (compacting_) {
uint8_t* moving_space_begin = moving_space_begin_;
uint8_t* moving_space_end = moving_space_end_;
for (size_t i = 0; i < count; ++i) {
UpdateRoot(roots[i], moving_space_begin, moving_space_end, info);
}
} else {
for (size_t i = 0; i < count; ++i) {
MarkObjectNonNull(*roots[i]);
}
}
}
void MarkCompact::VisitRoots(mirror::CompressedReference<mirror::Object>** roots,
size_t count,
const RootInfo& info) {
// TODO: do we need to check if the root is null or not?
if (compacting_) {
uint8_t* moving_space_begin = moving_space_begin_;
uint8_t* moving_space_end = moving_space_end_;
for (size_t i = 0; i < count; ++i) {
UpdateRoot(roots[i], moving_space_begin, moving_space_end, info);
}
} else {
for (size_t i = 0; i < count; ++i) {
MarkObjectNonNull(roots[i]->AsMirrorPtr());
}
}
}
mirror::Object* MarkCompact::IsMarked(mirror::Object* obj) {
if (HasAddress(obj)) {
const bool is_black = reinterpret_cast<uint8_t*>(obj) >= black_allocations_begin_;
if (compacting_) {
if (is_black) {
return PostCompactBlackObjAddr(obj);
} else if (live_words_bitmap_->Test(obj)) {
return PostCompactOldObjAddr(obj);
} else {
return nullptr;
}
}
return (is_black || moving_space_bitmap_->Test(obj)) ? obj : nullptr;
} else if (non_moving_space_bitmap_->HasAddress(obj)) {
return non_moving_space_bitmap_->Test(obj) ? obj : nullptr;
} else if (immune_spaces_.ContainsObject(obj)) {
return obj;
} else {
DCHECK(heap_->GetLargeObjectsSpace())
<< "ref=" << obj
<< " doesn't belong to any of the spaces and large object space doesn't exist";
accounting::LargeObjectBitmap* los_bitmap = heap_->GetLargeObjectsSpace()->GetMarkBitmap();
if (los_bitmap->HasAddress(obj)) {
DCHECK(IsAlignedParam(obj, space::LargeObjectSpace::ObjectAlignment()));
return los_bitmap->Test(obj) ? obj : nullptr;
} else {
// The given obj is not in any of the known spaces, so return null. This could
// happen for instance in interpreter caches wherein a concurrent updation
// to the cache could result in obj being a non-reference. This is
// tolerable because SweepInterpreterCaches only updates if the given
// object has moved, which can't be the case for the non-reference.
return nullptr;
}
}
}
bool MarkCompact::IsNullOrMarkedHeapReference(mirror::HeapReference<mirror::Object>* obj,
[[maybe_unused]] bool do_atomic_update) {
mirror::Object* ref = obj->AsMirrorPtr();
if (ref == nullptr) {
return true;
}
return IsMarked(ref);
}
// Process the 'referent' field in a java.lang.ref.Reference. If the referent
// has not yet been marked, put it on the appropriate list in the heap for later
// processing.
void MarkCompact::DelayReferenceReferent(ObjPtr<mirror::Class> klass,
ObjPtr<mirror::Reference> ref) {
heap_->GetReferenceProcessor()->DelayReferenceReferent(klass, ref, this);
}
void MarkCompact::FinishPhase() {
GetCurrentIteration()->SetScannedBytes(bytes_scanned_);
bool is_zygote = Runtime::Current()->IsZygote();
compacting_ = false;
minor_fault_initialized_ = !is_zygote && uffd_minor_fault_supported_;
// Madvise compaction buffers. When using threaded implementation, skip the first page,
// which is used by the gc-thread for the next iteration. Otherwise, we get into a
// deadlock due to userfault on it in the next iteration. This page is not consuming any
// physical memory because we already madvised it above and then we triggered a read
// userfault, which maps a special zero-page.
if (use_uffd_sigbus_ || !minor_fault_initialized_ || !shadow_to_space_map_.IsValid() ||
shadow_to_space_map_.Size() < (moving_first_objs_count_ + black_page_count_) * gPageSize) {
size_t adjustment = use_uffd_sigbus_ ? 0 : gPageSize;
ZeroAndReleaseMemory(compaction_buffers_map_.Begin() + adjustment,
compaction_buffers_map_.Size() - adjustment);
} else if (shadow_to_space_map_.Size() == bump_pointer_space_->Capacity()) {
// Now that we are going to use minor-faults from next GC cycle, we can
// unmap the buffers used by worker threads.
compaction_buffers_map_.SetSize(gPageSize);
}
info_map_.MadviseDontNeedAndZero();
live_words_bitmap_->ClearBitmap();
// TODO: We can clear this bitmap right before compaction pause. But in that
// case we need to ensure that we don't assert on this bitmap afterwards.
// Also, we would still need to clear it here again as we may have to use the
// bitmap for black-allocations (see UpdateMovingSpaceBlackAllocations()).
moving_space_bitmap_->Clear();
if (UNLIKELY(is_zygote && IsValidFd(uffd_))) {
heap_->DeleteThreadPool();
// This unregisters all ranges as a side-effect.
close(uffd_);
uffd_ = kFdUnused;
uffd_initialized_ = false;
}
CHECK(mark_stack_->IsEmpty()); // Ensure that the mark stack is empty.
mark_stack_->Reset();
DCHECK_EQ(thread_running_gc_, Thread::Current());
if (kIsDebugBuild) {
MutexLock mu(thread_running_gc_, lock_);
if (updated_roots_.get() != nullptr) {
updated_roots_->clear();
}
}
class_after_obj_ordered_map_.clear();
delete[] moving_pages_status_;
linear_alloc_arenas_.clear();
{
ReaderMutexLock mu(thread_running_gc_, *Locks::mutator_lock_);
WriterMutexLock mu2(thread_running_gc_, *Locks::heap_bitmap_lock_);
heap_->ClearMarkedObjects();
}
std::swap(moving_to_space_fd_, moving_from_space_fd_);
if (IsValidFd(moving_to_space_fd_)) {
// Confirm that the memfd to be used on to-space in next GC cycle is empty.
struct stat buf;
DCHECK_EQ(fstat(moving_to_space_fd_, &buf), 0) << "fstat failed: " << strerror(errno);
DCHECK_EQ(buf.st_blocks, 0u);
}
GcVisitedArenaPool* arena_pool =
static_cast<GcVisitedArenaPool*>(Runtime::Current()->GetLinearAllocArenaPool());
arena_pool->DeleteUnusedArenas();
}
} // namespace collector
} // namespace gc
} // namespace art