/*
|
* Copyright (C) 2015 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 "oat_file.h"
|
|
#include <string>
|
|
#include <gtest/gtest.h>
|
|
#include "common_runtime_test.h"
|
#include "dexopt_test.h"
|
#include "scoped_thread_state_change-inl.h"
|
#include "vdex_file.h"
|
|
namespace art {
|
|
class OatFileTest : public DexoptTest {
|
};
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_NullAbsLocation) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation(nullptr,
|
"/data/app/foo/base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_NullAbsLocation_Multidex) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation(nullptr,
|
"/data/app/foo/base.apk!classes2.dex",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk!classes2.dex", dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk!classes2.dex", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelLocationAbsolute) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("base.apk",
|
"/system/framework/base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ(kIsTargetBuild ? "/system/framework/base.apk" : "base.apk", dex_file_name);
|
ASSERT_EQ("/system/framework/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_BothAbsoluteLocations) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/base.apk",
|
"/system/framework/base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ(kIsTargetBuild ? "/system/framework/base.apk" : "/data/app/foo/base.apk",
|
dex_file_name);
|
ASSERT_EQ("/system/framework/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelSuffixOfAbsLocation1) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/base.apk",
|
"base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelSuffixOfAbsLocation2) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/base.apk",
|
"foo/base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelSuffixOfAbsLocation_Multidex) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/base.apk",
|
"base.apk!classes11.dex",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk!classes11.dex", dex_file_name);
|
ASSERT_EQ("/data/app/foo/base.apk!classes11.dex", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelNotSuffixOfAbsLocation1) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/sludge.apk",
|
"base.apk!classes2.dex",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ(kIsTargetBuild ? "base.apk!classes2.dex" : "/data/app/foo/sludge.apk!classes2.dex",
|
dex_file_name);
|
ASSERT_EQ("base.apk!classes2.dex", dex_location);
|
}
|
|
TEST_F(OatFileTest, ResolveRelativeEncodedDexLocation_RelNotSuffixOfAbsLocation2) {
|
std::string dex_location;
|
std::string dex_file_name;
|
OatFile::ResolveRelativeEncodedDexLocation("/data/app/foo/sludge.apk",
|
"o/base.apk",
|
&dex_location,
|
&dex_file_name);
|
ASSERT_EQ(kIsTargetBuild ? "o/base.apk" : "/data/app/foo/sludge.apk", dex_file_name);
|
ASSERT_EQ("o/base.apk", dex_location);
|
}
|
|
TEST_F(OatFileTest, LoadOat) {
|
std::string dex_location = GetScratchDir() + "/LoadOat.jar";
|
|
Copy(GetDexSrc1(), dex_location);
|
GenerateOatForTest(dex_location.c_str(), CompilerFilter::kSpeed);
|
|
std::string oat_location;
|
std::string error_msg;
|
ASSERT_TRUE(OatFileAssistant::DexLocationToOatFilename(
|
dex_location, kRuntimeISA, &oat_location, &error_msg)) << error_msg;
|
std::unique_ptr<OatFile> odex_file(OatFile::Open(/*zip_fd=*/ -1,
|
oat_location.c_str(),
|
oat_location.c_str(),
|
/*executable=*/ false,
|
/*low_4gb=*/ false,
|
dex_location.c_str(),
|
/*reservation=*/ nullptr,
|
&error_msg));
|
ASSERT_TRUE(odex_file.get() != nullptr);
|
|
// Check that the vdex file was loaded in the reserved space of odex file.
|
EXPECT_EQ(odex_file->GetVdexFile()->Begin(), odex_file->VdexBegin());
|
}
|
|
TEST_F(OatFileTest, ChangingMultiDexUncompressed) {
|
std::string dex_location = GetScratchDir() + "/MultiDexUncompressed.jar";
|
|
Copy(GetTestDexFileName("MultiDexUncompressed"), dex_location);
|
GenerateOatForTest(dex_location.c_str(), CompilerFilter::kQuicken);
|
|
std::string oat_location;
|
std::string error_msg;
|
ASSERT_TRUE(OatFileAssistant::DexLocationToOatFilename(
|
dex_location, kRuntimeISA, &oat_location, &error_msg)) << error_msg;
|
|
// Ensure we can load that file. Just a precondition.
|
{
|
std::unique_ptr<OatFile> odex_file(OatFile::Open(/*zip_fd=*/ -1,
|
oat_location.c_str(),
|
oat_location.c_str(),
|
/*executable=*/ false,
|
/*low_4gb=*/ false,
|
dex_location.c_str(),
|
/*reservation=*/ nullptr,
|
&error_msg));
|
ASSERT_TRUE(odex_file != nullptr);
|
ASSERT_EQ(2u, odex_file->GetOatDexFiles().size());
|
}
|
|
// Now replace the source.
|
Copy(GetTestDexFileName("MainUncompressed"), dex_location);
|
|
// And try to load again.
|
std::unique_ptr<OatFile> odex_file(OatFile::Open(/*zip_fd=*/ -1,
|
oat_location,
|
oat_location,
|
/*executable=*/ false,
|
/*low_4gb=*/ false,
|
dex_location.c_str(),
|
/*reservation=*/ nullptr,
|
&error_msg));
|
EXPECT_TRUE(odex_file == nullptr);
|
EXPECT_NE(std::string::npos, error_msg.find("expected 2 uncompressed dex files, but found 1"))
|
<< error_msg;
|
}
|
|
} // namespace art
|