167 lines
6.0 KiB
C++
167 lines
6.0 KiB
C++
/*
|
|
* 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 "art_dex_file_loader.h"
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <memory>
|
|
|
|
#include "android-base/stringprintf.h"
|
|
#include "base/file_magic.h"
|
|
#include "base/file_utils.h"
|
|
#include "base/logging.h"
|
|
#include "base/mem_map.h"
|
|
#include "base/mman.h" // For the PROT_* and MAP_* constants.
|
|
#include "base/stl_util.h"
|
|
#include "base/systrace.h"
|
|
#include "base/unix_file/fd_file.h"
|
|
#include "base/zip_archive.h"
|
|
#include "dex/compact_dex_file.h"
|
|
#include "dex/dex_file.h"
|
|
#include "dex/dex_file_verifier.h"
|
|
#include "dex/standard_dex_file.h"
|
|
|
|
namespace art {
|
|
|
|
using android::base::StringPrintf;
|
|
|
|
bool ArtDexFileLoader::GetMultiDexChecksums(const char* filename,
|
|
std::vector<uint32_t>* checksums,
|
|
std::vector<std::string>* dex_locations,
|
|
std::string* error_msg,
|
|
int zip_fd,
|
|
bool* zip_file_only_contains_uncompressed_dex) {
|
|
CHECK(checksums != nullptr);
|
|
uint32_t magic;
|
|
|
|
File fd;
|
|
if (zip_fd != -1) {
|
|
if (ReadMagicAndReset(zip_fd, &magic, error_msg)) {
|
|
fd = File(DupCloexec(zip_fd), /* check_usage= */ false);
|
|
}
|
|
} else {
|
|
fd = OpenAndReadMagic(filename, &magic, error_msg);
|
|
}
|
|
if (fd.Fd() == -1) {
|
|
DCHECK(!error_msg->empty());
|
|
return false;
|
|
}
|
|
if (IsZipMagic(magic)) {
|
|
std::unique_ptr<ZipArchive> zip_archive(
|
|
ZipArchive::OpenFromFd(fd.Release(), filename, error_msg));
|
|
if (zip_archive.get() == nullptr) {
|
|
*error_msg = StringPrintf("Failed to open zip archive '%s' (error msg: %s)", filename,
|
|
error_msg->c_str());
|
|
return false;
|
|
}
|
|
|
|
if (zip_file_only_contains_uncompressed_dex != nullptr) {
|
|
// Start by assuming everything is uncompressed.
|
|
*zip_file_only_contains_uncompressed_dex = true;
|
|
}
|
|
|
|
uint32_t idx = 0;
|
|
std::string zip_entry_name = GetMultiDexClassesDexName(idx);
|
|
std::unique_ptr<ZipEntry> zip_entry(zip_archive->Find(zip_entry_name.c_str(), error_msg));
|
|
if (zip_entry.get() == nullptr) {
|
|
// A zip file with no dex code should be accepted. It's likely a config split APK, which we
|
|
// are currently passing from higher levels.
|
|
VLOG(dex) << StringPrintf("Zip archive '%s' doesn't contain %s (error msg: %s)",
|
|
filename,
|
|
zip_entry_name.c_str(),
|
|
error_msg->c_str());
|
|
return true;
|
|
}
|
|
|
|
do {
|
|
if (zip_file_only_contains_uncompressed_dex != nullptr) {
|
|
if (!(zip_entry->IsUncompressed() && zip_entry->IsAlignedTo(alignof(DexFile::Header)))) {
|
|
*zip_file_only_contains_uncompressed_dex = false;
|
|
}
|
|
}
|
|
checksums->push_back(zip_entry->GetCrc32());
|
|
dex_locations->push_back(GetMultiDexLocation(idx, filename));
|
|
zip_entry_name = GetMultiDexClassesDexName(++idx);
|
|
zip_entry.reset(zip_archive->Find(zip_entry_name.c_str(), error_msg));
|
|
} while (zip_entry.get() != nullptr);
|
|
return true;
|
|
}
|
|
if (IsMagicValid(magic)) {
|
|
ArtDexFileLoader loader(fd.Release(), filename);
|
|
std::vector<std::unique_ptr<const DexFile>> dex_files;
|
|
if (!loader.Open(/* verify= */ false,
|
|
/* verify_checksum= */ false,
|
|
error_msg,
|
|
&dex_files)) {
|
|
return false;
|
|
}
|
|
for (auto& dex_file : dex_files) {
|
|
checksums->push_back(dex_file->GetHeader().checksum_);
|
|
dex_locations->push_back(dex_file->GetLocation());
|
|
}
|
|
return true;
|
|
}
|
|
*error_msg = StringPrintf("Expected valid zip or dex file: '%s'", filename);
|
|
return false;
|
|
}
|
|
|
|
std::unique_ptr<const DexFile> ArtDexFileLoader::Open(
|
|
const uint8_t* base,
|
|
size_t size,
|
|
const std::string& location,
|
|
uint32_t location_checksum,
|
|
const OatDexFile* oat_dex_file,
|
|
bool verify,
|
|
bool verify_checksum,
|
|
std::string* error_msg,
|
|
std::unique_ptr<DexFileContainer> container) const {
|
|
return OpenCommon(base,
|
|
size,
|
|
/*data_base=*/nullptr,
|
|
/*data_size=*/0,
|
|
location,
|
|
location_checksum,
|
|
oat_dex_file,
|
|
verify,
|
|
verify_checksum,
|
|
error_msg,
|
|
std::move(container),
|
|
/*verify_result=*/nullptr);
|
|
}
|
|
|
|
std::unique_ptr<const DexFile> ArtDexFileLoader::Open(const std::string& location,
|
|
uint32_t location_checksum,
|
|
MemMap&& mem_map,
|
|
bool verify,
|
|
bool verify_checksum,
|
|
std::string* error_msg) const {
|
|
ArtDexFileLoader loader(std::move(mem_map), location);
|
|
return loader.Open(location_checksum, verify, verify_checksum, error_msg);
|
|
}
|
|
|
|
bool ArtDexFileLoader::Open(const char* filename,
|
|
const std::string& location,
|
|
bool verify,
|
|
bool verify_checksum,
|
|
std::string* error_msg,
|
|
std::vector<std::unique_ptr<const DexFile>>* dex_files) const {
|
|
ArtDexFileLoader loader(filename, location);
|
|
return loader.Open(verify, verify_checksum, error_msg, dex_files);
|
|
}
|
|
|
|
} // namespace art
|