/*
* 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 "compact_dex_file.h"
#include "base/leb128.h"
#include "code_item_accessors-inl.h"
#include "dex_file-inl.h"
namespace art {
constexpr uint8_t CompactDexFile::kDexMagic[kDexMagicSize];
constexpr uint8_t CompactDexFile::kDexMagicVersion[];
void CompactDexFile::WriteMagic(uint8_t* magic) {
std::copy_n(kDexMagic, kDexMagicSize, magic);
}
void CompactDexFile::WriteCurrentVersion(uint8_t* magic) {
std::copy_n(kDexMagicVersion, kDexVersionLen, magic + kDexMagicSize);
}
bool CompactDexFile::IsMagicValid(const uint8_t* magic) {
return (memcmp(magic, kDexMagic, sizeof(kDexMagic)) == 0);
}
bool CompactDexFile::IsVersionValid(const uint8_t* magic) {
const uint8_t* version = &magic[sizeof(kDexMagic)];
return memcmp(version, kDexMagicVersion, kDexVersionLen) == 0;
}
bool CompactDexFile::IsMagicValid() const {
return IsMagicValid(header_->magic_);
}
bool CompactDexFile::IsVersionValid() const {
return IsVersionValid(header_->magic_);
}
bool CompactDexFile::SupportsDefaultMethods() const {
return (GetHeader().GetFeatureFlags() &
static_cast<uint32_t>(FeatureFlags::kDefaultMethods)) != 0;
}
uint32_t CompactDexFile::GetCodeItemSize(const dex::CodeItem& item) const {
DCHECK(IsInDataSection(&item));
return reinterpret_cast<uintptr_t>(CodeItemDataAccessor(*this, &item).CodeItemDataEnd()) -
reinterpret_cast<uintptr_t>(&item);
}
uint32_t CompactDexFile::CalculateChecksum(const uint8_t* base_begin,
size_t base_size,
const uint8_t* data_begin,
size_t data_size) {
Header temp_header(*Header::At(base_begin));
// Zero out fields that are not included in the sum.
temp_header.checksum_ = 0u;
temp_header.data_off_ = 0u;
temp_header.data_size_ = 0u;
uint32_t checksum = ChecksumMemoryRange(reinterpret_cast<const uint8_t*>(&temp_header),
sizeof(temp_header));
// Exclude the header since we already computed it's checksum.
checksum = (checksum * 31) ^ ChecksumMemoryRange(base_begin + sizeof(temp_header),
base_size - sizeof(temp_header));
checksum = (checksum * 31) ^ ChecksumMemoryRange(data_begin, data_size);
return checksum;
}
uint32_t CompactDexFile::CalculateChecksum() const {
return CalculateChecksum(Begin(), Size(), DataBegin(), DataSize());
}
CompactDexFile::CompactDexFile(const uint8_t* base,
size_t size,
const uint8_t* data_begin,
size_t data_size,
const std::string& location,
uint32_t location_checksum,
const OatDexFile* oat_dex_file,
std::unique_ptr<DexFileContainer> container)
: DexFile(base,
size,
data_begin,
data_size,
location,
location_checksum,
oat_dex_file,
std::move(container),
/*is_compact_dex=*/ true),
debug_info_offsets_(DataBegin() + GetHeader().debug_info_offsets_pos_,
GetHeader().debug_info_base_,
GetHeader().debug_info_offsets_table_offset_) {}
} // namespace art