/*
* Copyright (C) 2017 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 "dex_file_layout.h"
#include "base/bit_utils.h"
#include "base/mman.h"
#include "dex_file.h"
namespace art {
int DexLayoutSection::MadviseLargestPageAlignedRegion(const uint8_t* begin,
const uint8_t* end,
int advice) {
#ifdef _WIN32
UNUSED(begin);
UNUSED(end);
UNUSED(advice);
PLOG(WARNING) << "madvise is unsupported on Windows.";
#else
DCHECK_LE(begin, end);
begin = AlignUp(begin, kPageSize);
end = AlignDown(end, kPageSize);
if (begin < end) {
// TODO: remove the direct dependency on madvise here.
int result = madvise(const_cast<uint8_t*>(begin), end - begin, advice);
if (result != 0) {
PLOG(WARNING) << "madvise failed " << result;
}
return result;
}
#endif
return 0;
}
void DexLayoutSection::Subsection::Madvise(const DexFile* dex_file, int advice) const {
DCHECK(dex_file != nullptr);
DCHECK_LT(start_offset_, dex_file->Size());
DCHECK_LE(end_offset_, dex_file->Size());
MadviseLargestPageAlignedRegion(dex_file->Begin() + start_offset_,
dex_file->Begin() + end_offset_,
advice);
}
void DexLayoutSections::Madvise(const DexFile* dex_file, MadviseState state) const {
#ifdef _WIN32
UNUSED(dex_file);
UNUSED(state);
PLOG(WARNING) << "madvise is unsupported on Windows.";
#else
// The dex file is already defaulted to random access everywhere.
for (const DexLayoutSection& section : sections_) {
switch (state) {
case MadviseState::kMadviseStateAtLoad: {
section.parts_[static_cast<size_t>(LayoutType::kLayoutTypeStartupOnly)].Madvise(
dex_file,
MADV_WILLNEED);
section.parts_[static_cast<size_t>(LayoutType::kLayoutTypeHot)].Madvise(
dex_file,
MADV_WILLNEED);
break;
}
case MadviseState::kMadviseStateFinishedLaunch: {
section.parts_[static_cast<size_t>(LayoutType::kLayoutTypeStartupOnly)].Madvise(
dex_file,
MADV_DONTNEED);
break;
}
case MadviseState::kMadviseStateFinishedTrim: {
section.parts_[static_cast<size_t>(LayoutType::kLayoutTypeSometimesUsed)].Madvise(
dex_file,
MADV_DONTNEED);
section.parts_[static_cast<size_t>(LayoutType::kLayoutTypeUsedOnce)].Madvise(
dex_file,
MADV_DONTNEED);
break;
}
}
}
#endif
}
std::ostream& operator<<(std::ostream& os, const DexLayoutSection& section) {
for (size_t i = 0; i < static_cast<size_t>(LayoutType::kLayoutTypeCount); ++i) {
const DexLayoutSection::Subsection& part = section.parts_[i];
os << static_cast<LayoutType>(i) << "("
<< part.start_offset_ << "-" << part.end_offset_ << ") ";
}
return os;
}
std::ostream& operator<<(std::ostream& os, const DexLayoutSections& sections) {
for (size_t i = 0; i < static_cast<size_t>(DexLayoutSections::SectionType::kSectionCount); ++i) {
os << static_cast<DexLayoutSections::SectionType>(i) << ":" << sections.sections_[i] << "\n";
}
return os;
}
} // namespace art