普通文本  |  201行  |  8.68 KB

/*
 * 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