blob: f446c0919bf5d0987acc5039c0999e2755c6f3c1 [file] [log] [blame]
Calin Juravle2e2db782016-02-23 12:00:03 +00001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <gtest/gtest.h>
Alex Lightcff30a02021-03-25 17:33:32 -070018#include <sstream>
19#include <string>
Calin Juravle2e2db782016-02-23 12:00:03 +000020
Calin Juravle0f7f4fc2020-05-11 20:16:50 -070021#include "android-base/file.h"
Calin Juravlee10c1e22018-01-26 20:10:15 -080022#include "android-base/strings.h"
Calin Juravlee0ac1152017-02-13 19:03:47 -080023#include "art_method-inl.h"
Alex Lighta2f13192021-02-03 18:19:03 -080024#include "base/globals.h"
Calin Juravle2e2db782016-02-23 12:00:03 +000025#include "base/unix_file/fd_file.h"
David Sehrc431b9d2018-03-02 12:01:51 -080026#include "base/utils.h"
Calin Juravle2e2db782016-02-23 12:00:03 +000027#include "common_runtime_test.h"
David Sehrb2ec9f52018-02-21 13:20:31 -080028#include "dex/descriptors_names.h"
Alex Lightcff30a02021-03-25 17:33:32 -070029#include "dex/dex_file_structs.h"
Alex Lighta2f13192021-02-03 18:19:03 -080030#include "dex/dex_instruction-inl.h"
31#include "dex/dex_instruction_iterator.h"
Vladimir Markoa8bba7d2018-05-30 15:18:48 +010032#include "dex/type_reference.h"
David Sehr97c381e2017-02-01 15:09:58 -080033#include "exec_utils.h"
Calin Juravlecc3171a2017-05-19 16:47:53 -070034#include "linear_alloc.h"
Calin Juravlee0ac1152017-02-13 19:03:47 -080035#include "mirror/class-inl.h"
Mathieu Chartierd808e8b2017-03-21 13:37:41 -070036#include "obj_ptr-inl.h"
David Sehr82d046e2018-04-23 08:14:19 -070037#include "profile/profile_compilation_info.h"
Vladimir Markof4d05ff2021-04-14 09:39:04 +000038#include "profile/profile_test_helper.h"
Calin Juravlee0ac1152017-02-13 19:03:47 -080039#include "profile_assistant.h"
40#include "scoped_thread_state_change-inl.h"
Calin Juravle2e2db782016-02-23 12:00:03 +000041
42namespace art {
43
Vladimir Markoa8bba7d2018-05-30 15:18:48 +010044using TypeReferenceSet = std::set<TypeReference, TypeReferenceValueComparator>;
Calin Juravleee9cb412018-02-13 20:32:35 -080045
Calin Juravle1061c7d2019-09-23 21:00:29 -040046// TODO(calin): These tests share a lot with the ProfileCompilationInfo tests.
47// we should introduce a better abstraction to extract the common parts.
Vladimir Markof4d05ff2021-04-14 09:39:04 +000048class ProfileAssistantTest : public CommonRuntimeTest, public ProfileTestHelper {
Calin Juravlecc3171a2017-05-19 16:47:53 -070049 public:
Roland Levillainbbc6e7e2018-08-24 16:58:47 +010050 void PostRuntimeCreate() override {
Vladimir Marko69d310e2017-10-09 14:12:23 +010051 allocator_.reset(new ArenaAllocator(Runtime::Current()->GetArenaPool()));
Calin Juravle1061c7d2019-09-23 21:00:29 -040052
Vladimir Marko95f4c662021-05-11 12:25:20 +010053 dex1 = BuildDex("location1", /*checksum=*/ 1, "LUnique1;", /*num_method_ids=*/ 10001);
54 dex2 = BuildDex("location2", /*checksum=*/ 2, "LUnique2;", /*num_method_ids=*/ 10002);
55 dex3 = BuildDex("location3", /*checksum=*/ 3, "LUnique3;", /*num_method_ids=*/ 10003);
56 dex4 = BuildDex("location4", /*checksum=*/ 4, "LUnique4;", /*num_method_ids=*/ 10004);
Calin Juravle1061c7d2019-09-23 21:00:29 -040057
Vladimir Markof4d05ff2021-04-14 09:39:04 +000058 dex1_checksum_missmatch =
Vladimir Marko95f4c662021-05-11 12:25:20 +010059 BuildDex("location1", /*checksum=*/ 12, "LUnique1;", /*num_method_ids=*/ 10001);
Calin Juravlecc3171a2017-05-19 16:47:53 -070060 }
61
Calin Juravle2e2db782016-02-23 12:00:03 +000062 protected:
Calin Juravle1061c7d2019-09-23 21:00:29 -040063 void SetupProfile(const DexFile* dex_file1,
64 const DexFile* dex_file2,
Calin Juravle2e2db782016-02-23 12:00:03 +000065 uint16_t number_of_methods,
Calin Juravlec824b512016-03-29 20:33:33 +010066 uint16_t number_of_classes,
Calin Juravle2e2db782016-02-23 12:00:03 +000067 const ScratchFile& profile,
68 ProfileCompilationInfo* info,
Calin Juravlecea9e9d2017-03-23 19:04:59 -070069 uint16_t start_method_index = 0,
70 bool reverse_dex_write_order = false) {
Calin Juravle2e2db782016-02-23 12:00:03 +000071 for (uint16_t i = start_method_index; i < start_method_index + number_of_methods; i++) {
Calin Juravlecea9e9d2017-03-23 19:04:59 -070072 // reverse_dex_write_order controls the order in which the dex files will be added to
73 // the profile and thus written to disk.
Alex Lighta2f13192021-02-03 18:19:03 -080074 std::vector<ProfileInlineCache> inline_caches =
75 GetTestInlineCaches(dex_file1, dex_file2, dex3);
76 Hotness::Flag flags =
77 static_cast<Hotness::Flag>(Hotness::kFlagHot | Hotness::kFlagPostStartup);
Calin Juravlecea9e9d2017-03-23 19:04:59 -070078 if (reverse_dex_write_order) {
Calin Juravle1061c7d2019-09-23 21:00:29 -040079 ASSERT_TRUE(AddMethod(info, dex_file2, i, inline_caches, flags));
80 ASSERT_TRUE(AddMethod(info, dex_file1, i, inline_caches, flags));
Calin Juravlecea9e9d2017-03-23 19:04:59 -070081 } else {
Calin Juravle1061c7d2019-09-23 21:00:29 -040082 ASSERT_TRUE(AddMethod(info, dex_file1, i, inline_caches, flags));
83 ASSERT_TRUE(AddMethod(info, dex_file2, i, inline_caches, flags));
Calin Juravlecea9e9d2017-03-23 19:04:59 -070084 }
Calin Juravle2e2db782016-02-23 12:00:03 +000085 }
Calin Juravlec824b512016-03-29 20:33:33 +010086 for (uint16_t i = 0; i < number_of_classes; i++) {
Calin Juravle45cdd052019-09-23 23:03:18 -040087 ASSERT_TRUE(AddClass(info, dex_file1, dex::TypeIndex(i)));
Calin Juravlec824b512016-03-29 20:33:33 +010088 }
89
Calin Juravle2e2db782016-02-23 12:00:03 +000090 ASSERT_TRUE(info->Save(GetFd(profile)));
91 ASSERT_EQ(0, profile.GetFile()->Flush());
Calin Juravle2e2db782016-02-23 12:00:03 +000092 }
93
Calin Juravle1061c7d2019-09-23 21:00:29 -040094 void SetupBasicProfile(const DexFile* dex,
Stephen Hines48ba1972018-09-24 13:35:54 -070095 const std::vector<uint32_t>& hot_methods,
96 const std::vector<uint32_t>& startup_methods,
97 const std::vector<uint32_t>& post_startup_methods,
Mathieu Chartier28b5c582017-06-06 14:12:50 -070098 const ScratchFile& profile,
99 ProfileCompilationInfo* info) {
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700100 for (uint32_t idx : hot_methods) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400101 AddMethod(info, dex, idx, Hotness::kFlagHot);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700102 }
103 for (uint32_t idx : startup_methods) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400104 AddMethod(info, dex, idx, Hotness::kFlagStartup);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700105 }
106 for (uint32_t idx : post_startup_methods) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400107 AddMethod(info, dex, idx, Hotness::kFlagPostStartup);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700108 }
109 ASSERT_TRUE(info->Save(GetFd(profile)));
110 ASSERT_EQ(0, profile.GetFile()->Flush());
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700111 }
112
Calin Juravle1061c7d2019-09-23 21:00:29 -0400113 // The dex1_substitute can be used to replace the default dex1 file.
114 std::vector<ProfileInlineCache> GetTestInlineCaches(
115 const DexFile* dex_file1, const DexFile* dex_file2, const DexFile* dex_file3) {
116 std::vector<ProfileInlineCache> inline_caches;
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700117 // Monomorphic
118 for (uint16_t dex_pc = 0; dex_pc < 11; dex_pc++) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400119 std::vector<TypeReference> types = {TypeReference(dex_file1, dex::TypeIndex(0))};
120 inline_caches.push_back(ProfileInlineCache(dex_pc, /* missing_types*/ false, types));
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700121 }
122 // Polymorphic
123 for (uint16_t dex_pc = 11; dex_pc < 22; dex_pc++) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400124 std::vector<TypeReference> types = {
125 TypeReference(dex_file1, dex::TypeIndex(0)),
126 TypeReference(dex_file2, dex::TypeIndex(1)),
127 TypeReference(dex_file3, dex::TypeIndex(2))};
128 inline_caches.push_back(ProfileInlineCache(dex_pc, /* missing_types*/ false, types));
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700129 }
130 // Megamorphic
131 for (uint16_t dex_pc = 22; dex_pc < 33; dex_pc++) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400132 // we need 5 types to make the cache megamorphic
133 std::vector<TypeReference> types = {
134 TypeReference(dex_file1, dex::TypeIndex(0)),
135 TypeReference(dex_file1, dex::TypeIndex(1)),
136 TypeReference(dex_file1, dex::TypeIndex(2)),
137 TypeReference(dex_file1, dex::TypeIndex(3)),
138 TypeReference(dex_file1, dex::TypeIndex(4))};
139 inline_caches.push_back(ProfileInlineCache(dex_pc, /* missing_types*/ false, types));
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700140 }
141 // Missing types
142 for (uint16_t dex_pc = 33; dex_pc < 44; dex_pc++) {
Calin Juravle1061c7d2019-09-23 21:00:29 -0400143 std::vector<TypeReference> types;
144 inline_caches.push_back(ProfileInlineCache(dex_pc, /* missing_types*/ true, types));
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700145 }
146
Calin Juravle1061c7d2019-09-23 21:00:29 -0400147 return inline_caches;
Calin Juravlecea9e9d2017-03-23 19:04:59 -0700148 }
149
Calin Juravle2e2db782016-02-23 12:00:03 +0000150 int GetFd(const ScratchFile& file) const {
151 return static_cast<int>(file.GetFd());
152 }
153
154 void CheckProfileInfo(ScratchFile& file, const ProfileCompilationInfo& info) {
155 ProfileCompilationInfo file_info;
Calin Juravle2e2db782016-02-23 12:00:03 +0000156 ASSERT_TRUE(file_info.Load(GetFd(file)));
157 ASSERT_TRUE(file_info.Equals(info));
158 }
159
Calin Juravle7bcdb532016-06-07 16:14:47 +0100160 std::string GetProfmanCmd() {
Martin Stjernholme58624f2019-09-20 15:53:40 +0100161 std::string file_path = GetArtBinDir() + "/profman";
Calin Juravle2e2db782016-02-23 12:00:03 +0000162 if (kIsDebugBuild) {
163 file_path += "d";
164 }
Roland Levillainfb6a5c02019-03-29 20:20:16 +0000165 EXPECT_TRUE(OS::FileExists(file_path.c_str())) << file_path << " should be a valid file path";
Calin Juravle7bcdb532016-06-07 16:14:47 +0100166 return file_path;
167 }
Mathieu Chartierf70fe3d2017-06-21 15:24:02 -0700168
Calin Juravle7bcdb532016-06-07 16:14:47 +0100169 // Runs test with given arguments.
Calin Juravle0e82e0f2019-11-11 18:34:10 -0800170 int ProcessProfiles(
171 const std::vector<int>& profiles_fd,
172 int reference_profile_fd,
Calin Juravlec0200a92019-11-14 09:01:11 -0800173 const std::vector<const std::string>& extra_args = std::vector<const std::string>()) {
Calin Juravle7bcdb532016-06-07 16:14:47 +0100174 std::string profman_cmd = GetProfmanCmd();
Calin Juravle2e2db782016-02-23 12:00:03 +0000175 std::vector<std::string> argv_str;
Calin Juravle7bcdb532016-06-07 16:14:47 +0100176 argv_str.push_back(profman_cmd);
Calin Juravle2e2db782016-02-23 12:00:03 +0000177 for (size_t k = 0; k < profiles_fd.size(); k++) {
178 argv_str.push_back("--profile-file-fd=" + std::to_string(profiles_fd[k]));
179 }
180 argv_str.push_back("--reference-profile-file-fd=" + std::to_string(reference_profile_fd));
Calin Juravle0e82e0f2019-11-11 18:34:10 -0800181 argv_str.insert(argv_str.end(), extra_args.begin(), extra_args.end());
Calin Juravle2e2db782016-02-23 12:00:03 +0000182
183 std::string error;
184 return ExecAndReturnCode(argv_str, &error);
185 }
Calin Juravle7bcdb532016-06-07 16:14:47 +0100186
187 bool GenerateTestProfile(const std::string& filename) {
188 std::string profman_cmd = GetProfmanCmd();
189 std::vector<std::string> argv_str;
190 argv_str.push_back(profman_cmd);
191 argv_str.push_back("--generate-test-profile=" + filename);
192 std::string error;
Calin Juravled4e69922020-05-18 18:32:22 -0700193 return ExecAndReturnCode(argv_str, &error);
Calin Juravle7bcdb532016-06-07 16:14:47 +0100194 }
David Sehr7c80f2d2017-02-07 16:47:58 -0800195
Jeff Haof0a31f82017-03-27 15:50:37 -0700196 bool GenerateTestProfileWithInputDex(const std::string& filename) {
197 std::string profman_cmd = GetProfmanCmd();
198 std::vector<std::string> argv_str;
199 argv_str.push_back(profman_cmd);
200 argv_str.push_back("--generate-test-profile=" + filename);
201 argv_str.push_back("--generate-test-profile-seed=0");
202 argv_str.push_back("--apk=" + GetLibCoreDexFileNames()[0]);
203 argv_str.push_back("--dex-location=" + GetLibCoreDexFileNames()[0]);
204 std::string error;
205 return ExecAndReturnCode(argv_str, &error);
206 }
207
Andreas Gampe641a4732017-08-24 13:21:35 -0700208 bool CreateProfile(const std::string& profile_file_contents,
Calin Juravlee0ac1152017-02-13 19:03:47 -0800209 const std::string& filename,
Vladimir Markoadfa1ad2021-04-23 15:02:49 +0100210 const std::string& dex_location,
211 bool for_boot_image = false) {
David Sehr7c80f2d2017-02-07 16:47:58 -0800212 ScratchFile class_names_file;
213 File* file = class_names_file.GetFile();
Calin Juravlee0ac1152017-02-13 19:03:47 -0800214 EXPECT_TRUE(file->WriteFully(profile_file_contents.c_str(), profile_file_contents.length()));
David Sehr7c80f2d2017-02-07 16:47:58 -0800215 EXPECT_EQ(0, file->Flush());
David Sehr7c80f2d2017-02-07 16:47:58 -0800216 std::string profman_cmd = GetProfmanCmd();
217 std::vector<std::string> argv_str;
218 argv_str.push_back(profman_cmd);
Vladimir Markoadfa1ad2021-04-23 15:02:49 +0100219 argv_str.push_back(for_boot_image ? "--output-profile-type=boot" : "--output-profile-type=app");
David Sehr7c80f2d2017-02-07 16:47:58 -0800220 argv_str.push_back("--create-profile-from=" + class_names_file.GetFilename());
221 argv_str.push_back("--reference-profile-file=" + filename);
Calin Juravlee0ac1152017-02-13 19:03:47 -0800222 argv_str.push_back("--apk=" + dex_location);
223 argv_str.push_back("--dex-location=" + dex_location);
David Sehr7c80f2d2017-02-07 16:47:58 -0800224 std::string error;
Alex Lighta2f13192021-02-03 18:19:03 -0800225 EXPECT_EQ(ExecAndReturnCode(argv_str, &error), 0) << error;
David Sehr7c80f2d2017-02-07 16:47:58 -0800226 return true;
227 }
228
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700229 bool RunProfman(const std::string& filename,
230 std::vector<std::string>& extra_args,
Alex Lighta2f13192021-02-03 18:19:03 -0800231 std::string* output,
232 std::string_view target_apk) {
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700233 ScratchFile output_file;
David Sehr7c80f2d2017-02-07 16:47:58 -0800234 std::string profman_cmd = GetProfmanCmd();
235 std::vector<std::string> argv_str;
236 argv_str.push_back(profman_cmd);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700237 argv_str.insert(argv_str.end(), extra_args.begin(), extra_args.end());
David Sehr7c80f2d2017-02-07 16:47:58 -0800238 argv_str.push_back("--profile-file=" + filename);
Alex Lighta2f13192021-02-03 18:19:03 -0800239 argv_str.push_back(std::string("--apk=").append(target_apk));
240 argv_str.push_back(std::string("--dex-location=").append(target_apk));
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700241 argv_str.push_back("--dump-output-to-fd=" + std::to_string(GetFd(output_file)));
David Sehr7c80f2d2017-02-07 16:47:58 -0800242 std::string error;
Alex Lighta2f13192021-02-03 18:19:03 -0800243 EXPECT_EQ(ExecAndReturnCode(argv_str, &error), 0) << error;
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700244 File* file = output_file.GetFile();
David Sehr7c80f2d2017-02-07 16:47:58 -0800245 EXPECT_EQ(0, file->Flush());
David Sehr7c80f2d2017-02-07 16:47:58 -0800246 int64_t length = file->GetLength();
247 std::unique_ptr<char[]> buf(new char[length]);
248 EXPECT_EQ(file->Read(buf.get(), length, 0), length);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700249 *output = std::string(buf.get(), length);
David Sehr7c80f2d2017-02-07 16:47:58 -0800250 return true;
251 }
252
Alex Lighta2f13192021-02-03 18:19:03 -0800253 bool DumpClassesAndMethods(const std::string& filename,
254 std::string* file_contents,
255 std::optional<const std::string_view> target = std::nullopt) {
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700256 std::vector<std::string> extra_args;
257 extra_args.push_back("--dump-classes-and-methods");
Alex Lighta2f13192021-02-03 18:19:03 -0800258 return RunProfman(
259 filename, extra_args, file_contents, target.value_or(GetLibCoreDexFileNames()[0]));
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700260 }
261
262 bool DumpOnly(const std::string& filename, std::string* file_contents) {
263 std::vector<std::string> extra_args;
264 extra_args.push_back("--dump-only");
Alex Lighta2f13192021-02-03 18:19:03 -0800265 return RunProfman(filename, extra_args, file_contents, GetLibCoreDexFileNames()[0]);
Mathieu Chartier28b5c582017-06-06 14:12:50 -0700266 }
267
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700268 bool CreateAndDump(const std::string& input_file_contents,
Alex Lighta2f13192021-02-03 18:19:03 -0800269 std::string* output_file_contents,
270 std::optional<const std::string> target = std::nullopt) {
David Sehr7c80f2d2017-02-07 16:47:58 -0800271 ScratchFile profile_file;
Calin Juravlee0ac1152017-02-13 19:03:47 -0800272 EXPECT_TRUE(CreateProfile(input_file_contents,
273 profile_file.GetFilename(),
Alex Lighta2f13192021-02-03 18:19:03 -0800274 target.value_or(GetLibCoreDexFileNames()[0])));
Alex Lighta2f13192021-02-03 18:19:03 -0800275 EXPECT_TRUE(DumpClassesAndMethods(profile_file.GetFilename(), output_file_contents, target));
David Sehr7c80f2d2017-02-07 16:47:58 -0800276 return true;
277 }
Calin Juravlee0ac1152017-02-13 19:03:47 -0800278
Vladimir Markoa8bba7d2018-05-30 15:18:48 +0100279 ObjPtr<mirror::Class> GetClass(ScopedObjectAccess& soa,
280 jobject class_loader,
281 const std::string& clazz) REQUIRES_SHARED(Locks::mutator_lock_) {
Calin Juravlee0ac1152017-02-13 19:03:47 -0800282 ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
Vladimir Markoa8bba7d2018-05-30 15:18:48 +0100283 StackHandleScope<1> hs(soa.Self());
284 Handle<mirror::ClassLoader> h_loader(hs.NewHandle(
285 ObjPtr<mirror::ClassLoader>::DownCast(soa.Self()->DecodeJObject(class_loader))));
286 return class_linker->FindClass(soa.Self(), clazz.c_str(), h_loader);
Calin Juravlee0ac1152017-02-13 19:03:47 -0800287 }
288
289 ArtMethod* GetVirtualMethod(jobject class_loader,
290 const std::string& clazz,
291 const std::string& name) {
Vladimir Markoa8bba7d2018-05-30 15:18:48 +0100292 ScopedObjectAccess soa(Thread::Current());
293 ObjPtr<mirror::Class> klass = GetClass(soa, class_loader, clazz);
Calin Juravlee0ac1152017-02-13 19:03:47 -0800294 ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
295 const auto pointer_size = class_linker->GetImagePointerSize();
296 ArtMethod* method = nullptr;
Calin Juravlee0ac1152017-02-13 19:03:47 -0800297 for (auto& m : klass->GetVirtualMethods(pointer_size)) {
298 if (name == m.GetName()) {
299 EXPECT_TRUE(method == nullptr);
300 method = &m;
301 }
302 }
303 return method;
304 }
305
Vladimir Markoa8bba7d2018-05-30 15:18:48 +0100306 static TypeReference MakeTypeReference(ObjPtr<mirror::Class> klass)
307 REQUIRES_SHARED(Locks::mutator_lock_) {
308 return TypeReference(&klass->GetDexFile(), klass->GetDexTypeIndex());
309 }
310
Alex Lighta2f13192021-02-03 18:19:03 -0800311 // Find the first dex-pc in the given method after 'start_pc' (if given) which
312 // contains a call to any method of 'klass'. If 'start_pc' is not given we
313 // will search from the first dex-pc.
314 uint16_t GetDexPcOfCallTo(ArtMethod* method,
315 Handle<mirror::Class> klass,
316 std::optional<uint32_t> start_pc = std::nullopt)
317 REQUIRES_SHARED(Locks::mutator_lock_) {
318 const DexFile* dex_file = method->GetDexFile();
319 for (const DexInstructionPcPair& inst :
320 CodeItemInstructionAccessor(*dex_file, method->GetCodeItem())) {
321 if (start_pc && inst.DexPc() <= *start_pc) {
322 continue;
323 } else if (inst->IsInvoke()) {
324 const dex::MethodId& method_id = dex_file->GetMethodId(inst->VRegB());
325 std::string_view desc(
326 dex_file->GetTypeDescriptor(dex_file->GetTypeId(method_id.class_idx_)));
327 std::string scratch;
328 if (desc == klass->GetDescriptor(&scratch)) {
329 return inst.DexPc();
330 }
331 }
332 }
333 EXPECT_TRUE(false) << "Unable to find dex-pc in " << method->PrettyMethod() << " for call to "
334 << klass->PrettyClass()
335 << " after dexpc: " << (start_pc ? static_cast<int64_t>(*start_pc) : -1);
336 return -1;
337 }
338
339 void AssertInlineCaches(ArtMethod* method,
340 uint16_t dex_pc,
Vladimir Markoc63d9672021-03-31 15:50:39 +0100341 const TypeReferenceSet& expected_classes,
Alex Lighta2f13192021-02-03 18:19:03 -0800342 const ProfileCompilationInfo& info,
343 bool is_megamorphic,
344 bool is_missing_types)
345 REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000346 ProfileCompilationInfo::MethodHotness hotness =
347 info.GetMethodHotness(MethodReference(method->GetDexFile(), method->GetDexMethodIndex()));
348 ASSERT_TRUE(hotness.IsHot());
349 const ProfileCompilationInfo::InlineCacheMap* inline_caches = hotness.GetInlineCacheMap();
350 ASSERT_TRUE(inline_caches->find(dex_pc) != inline_caches->end());
Vladimir Markoc63d9672021-03-31 15:50:39 +0100351 AssertInlineCaches(expected_classes,
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000352 info,
Vladimir Markoc63d9672021-03-31 15:50:39 +0100353 method,
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000354 inline_caches->find(dex_pc)->second,
Alex Lighta2f13192021-02-03 18:19:03 -0800355 is_megamorphic,
356 is_missing_types);
357 }
Calin Juravlee0ac1152017-02-13 19:03:47 -0800358 void AssertInlineCaches(ArtMethod* method,
Vladimir Markoc63d9672021-03-31 15:50:39 +0100359 const TypeReferenceSet& expected_classes,
Calin Juravlee0ac1152017-02-13 19:03:47 -0800360 const ProfileCompilationInfo& info,
Calin Juravle589e71e2017-03-03 16:05:05 -0800361 bool is_megamorphic,
362 bool is_missing_types)
Calin Juravlee0ac1152017-02-13 19:03:47 -0800363 REQUIRES_SHARED(Locks::mutator_lock_) {
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000364 ProfileCompilationInfo::MethodHotness hotness =
365 info.GetMethodHotness(MethodReference(method->GetDexFile(), method->GetDexMethodIndex()));
366 ASSERT_TRUE(hotness.IsHot());
367 const ProfileCompilationInfo::InlineCacheMap* inline_caches = hotness.GetInlineCacheMap();
368 ASSERT_EQ(inline_caches->size(), 1u);
Vladimir Markoc63d9672021-03-31 15:50:39 +0100369 AssertInlineCaches(expected_classes,
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000370 info,
Vladimir Markoc63d9672021-03-31 15:50:39 +0100371 method,
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000372 inline_caches->begin()->second,
Alex Lighta2f13192021-02-03 18:19:03 -0800373 is_megamorphic,
374 is_missing_types);
375 }
Calin Juravlee0ac1152017-02-13 19:03:47 -0800376
Alex Lighta2f13192021-02-03 18:19:03 -0800377 void AssertInlineCaches(const TypeReferenceSet& expected_clases,
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000378 const ProfileCompilationInfo& info,
Vladimir Markoc63d9672021-03-31 15:50:39 +0100379 ArtMethod* method,
Alex Lighta2f13192021-02-03 18:19:03 -0800380 const ProfileCompilationInfo::DexPcData& dex_pc_data,
381 bool is_megamorphic,
382 bool is_missing_types)
383 REQUIRES_SHARED(Locks::mutator_lock_) {
Calin Juravle589e71e2017-03-03 16:05:05 -0800384 ASSERT_EQ(dex_pc_data.is_megamorphic, is_megamorphic);
385 ASSERT_EQ(dex_pc_data.is_missing_types, is_missing_types);
Calin Juravlee0ac1152017-02-13 19:03:47 -0800386 ASSERT_EQ(expected_clases.size(), dex_pc_data.classes.size());
Vladimir Markoc63d9672021-03-31 15:50:39 +0100387 const DexFile* dex_file = method->GetDexFile();
Calin Juravlee0ac1152017-02-13 19:03:47 -0800388 size_t found = 0;
Vladimir Markoa8bba7d2018-05-30 15:18:48 +0100389 for (const TypeReference& type_ref : expected_clases) {
Vladimir Markoc63d9672021-03-31 15:50:39 +0100390 if (type_ref.dex_file == dex_file) {
391 CHECK_LT(type_ref.TypeIndex().index_, dex_file->NumTypeIds());
392 for (dex::TypeIndex type_index : dex_pc_data.classes) {
393 ASSERT_TRUE(type_index.IsValid());
394 if (type_ref.TypeIndex() == type_index) {
395 ++found;
396 }
397 }
398 } else {
399 // Match by descriptor.
400 const char* expected_descriptor = type_ref.dex_file->StringByTypeIdx(type_ref.TypeIndex());
401 for (dex::TypeIndex type_index : dex_pc_data.classes) {
402 ASSERT_TRUE(type_index.IsValid());
403 const char* descriptor = info.GetTypeDescriptor(dex_file, type_index);
404 if (strcmp(expected_descriptor, descriptor) == 0) {
405 ++found;
406 }
Calin Juravlee0ac1152017-02-13 19:03:47 -0800407 }
408 }
409 }
410
411 ASSERT_EQ(expected_clases.size(), found);
412 }
Calin Juravlecc3171a2017-05-19 16:47:53 -0700413
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700414 int CheckCompilationMethodPercentChange(uint16_t methods_in_cur_profile,
yawanng7c618802020-11-05 20:02:27 +0000415 uint16_t methods_in_ref_profile,
416 const std::vector<const std::string>& extra_args =
417 std::vector<const std::string>()) {
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700418 ScratchFile profile;
419 ScratchFile reference_profile;
420 std::vector<int> profile_fds({ GetFd(profile)});
421 int reference_profile_fd = GetFd(reference_profile);
422 std::vector<uint32_t> hot_methods_cur;
423 std::vector<uint32_t> hot_methods_ref;
424 std::vector<uint32_t> empty_vector;
425 for (size_t i = 0; i < methods_in_cur_profile; ++i) {
426 hot_methods_cur.push_back(i);
427 }
428 for (size_t i = 0; i < methods_in_ref_profile; ++i) {
429 hot_methods_ref.push_back(i);
430 }
431 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400432 SetupBasicProfile(dex1, hot_methods_cur, empty_vector, empty_vector,
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700433 profile, &info1);
434 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400435 SetupBasicProfile(dex1, hot_methods_ref, empty_vector, empty_vector,
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700436 reference_profile, &info2);
yawanng7c618802020-11-05 20:02:27 +0000437 return ProcessProfiles(profile_fds, reference_profile_fd, extra_args);
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700438 }
439
440 int CheckCompilationClassPercentChange(uint16_t classes_in_cur_profile,
yawanng7c618802020-11-05 20:02:27 +0000441 uint16_t classes_in_ref_profile,
442 const std::vector<const std::string>& extra_args =
443 std::vector<const std::string>()) {
Vladimir Markoc63d9672021-03-31 15:50:39 +0100444 uint16_t max_classes = std::max(classes_in_cur_profile, classes_in_ref_profile);
445 const DexFile* dex1_x = BuildDex(
446 "location1_x", /*checksum=*/ 0x101, "LUnique1_x;", /*num_method_ids=*/ 0, max_classes);
447 const DexFile* dex2_x = BuildDex(
448 "location2_x", /*checksum=*/ 0x102, "LUnique2_x;", /*num_method_ids=*/ 0, max_classes);
449
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700450 ScratchFile profile;
451 ScratchFile reference_profile;
452
453 std::vector<int> profile_fds({ GetFd(profile)});
454 int reference_profile_fd = GetFd(reference_profile);
455
456 ProfileCompilationInfo info1;
Vladimir Markoc63d9672021-03-31 15:50:39 +0100457 SetupProfile(dex1_x, dex2_x, 0, classes_in_cur_profile, profile, &info1);
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700458 ProfileCompilationInfo info2;
Vladimir Markoc63d9672021-03-31 15:50:39 +0100459 SetupProfile(dex1_x, dex2_x, 0, classes_in_ref_profile, reference_profile, &info2);
yawanng7c618802020-11-05 20:02:27 +0000460 return ProcessProfiles(profile_fds, reference_profile_fd, extra_args);
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700461 }
462
Vladimir Marko69d310e2017-10-09 14:12:23 +0100463 std::unique_ptr<ArenaAllocator> allocator_;
Calin Juravlee6f87cc2017-05-24 17:41:05 -0700464
Calin Juravle1061c7d2019-09-23 21:00:29 -0400465 const DexFile* dex1;
466 const DexFile* dex2;
467 const DexFile* dex3;
468 const DexFile* dex4;
469 const DexFile* dex1_checksum_missmatch;
Calin Juravle2e2db782016-02-23 12:00:03 +0000470};
471
472TEST_F(ProfileAssistantTest, AdviseCompilationEmptyReferences) {
473 ScratchFile profile1;
474 ScratchFile profile2;
475 ScratchFile reference_profile;
476
477 std::vector<int> profile_fds({
478 GetFd(profile1),
479 GetFd(profile2)});
480 int reference_profile_fd = GetFd(reference_profile);
481
482 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
483 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400484 SetupProfile(dex1, dex2, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1);
Calin Juravle2e2db782016-02-23 12:00:03 +0000485 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400486 SetupProfile(dex3, dex4, kNumberOfMethodsToEnableCompilation, 0, profile2, &info2);
Calin Juravle2e2db782016-02-23 12:00:03 +0000487
488 // We should advise compilation.
489 ASSERT_EQ(ProfileAssistant::kCompile,
490 ProcessProfiles(profile_fds, reference_profile_fd));
491 // The resulting compilation info must be equal to the merge of the inputs.
492 ProfileCompilationInfo result;
Calin Juravle2e2db782016-02-23 12:00:03 +0000493 ASSERT_TRUE(result.Load(reference_profile_fd));
494
495 ProfileCompilationInfo expected;
Calin Juravle67265462016-03-18 16:23:40 +0000496 ASSERT_TRUE(expected.MergeWith(info1));
497 ASSERT_TRUE(expected.MergeWith(info2));
Calin Juravle2e2db782016-02-23 12:00:03 +0000498 ASSERT_TRUE(expected.Equals(result));
499
500 // The information from profiles must remain the same.
501 CheckProfileInfo(profile1, info1);
502 CheckProfileInfo(profile2, info2);
503}
504
Calin Juravlec824b512016-03-29 20:33:33 +0100505// TODO(calin): Add more tests for classes.
506TEST_F(ProfileAssistantTest, AdviseCompilationEmptyReferencesBecauseOfClasses) {
Vladimir Markoc63d9672021-03-31 15:50:39 +0100507 const uint16_t kNumberOfClassesToEnableCompilation = 100;
508 const DexFile* dex1_100 = BuildDex("location1_100",
509 /*checksum=*/ 101,
510 "LUnique1_100;",
511 /*num_method_ids=*/ 0,
512 /*num_type_ids=*/ 100);
513 const DexFile* dex2_100 = BuildDex("location2_100",
514 /*checksum=*/ 102,
515 "LUnique2_100;",
516 /*num_method_ids=*/ 0,
517 /*num_type_ids=*/ 100);
518
Calin Juravlec824b512016-03-29 20:33:33 +0100519 ScratchFile profile1;
520 ScratchFile reference_profile;
521
522 std::vector<int> profile_fds({
523 GetFd(profile1)});
524 int reference_profile_fd = GetFd(reference_profile);
525
Calin Juravlec824b512016-03-29 20:33:33 +0100526 ProfileCompilationInfo info1;
Vladimir Markoc63d9672021-03-31 15:50:39 +0100527 SetupProfile(dex1_100, dex2_100, 0, kNumberOfClassesToEnableCompilation, profile1, &info1);
Calin Juravlec824b512016-03-29 20:33:33 +0100528
529 // We should advise compilation.
530 ASSERT_EQ(ProfileAssistant::kCompile,
531 ProcessProfiles(profile_fds, reference_profile_fd));
532 // The resulting compilation info must be equal to the merge of the inputs.
533 ProfileCompilationInfo result;
Calin Juravlec824b512016-03-29 20:33:33 +0100534 ASSERT_TRUE(result.Load(reference_profile_fd));
535
536 ProfileCompilationInfo expected;
537 ASSERT_TRUE(expected.MergeWith(info1));
538 ASSERT_TRUE(expected.Equals(result));
539
540 // The information from profiles must remain the same.
541 CheckProfileInfo(profile1, info1);
542}
543
Calin Juravle2e2db782016-02-23 12:00:03 +0000544TEST_F(ProfileAssistantTest, AdviseCompilationNonEmptyReferences) {
545 ScratchFile profile1;
546 ScratchFile profile2;
547 ScratchFile reference_profile;
548
549 std::vector<int> profile_fds({
550 GetFd(profile1),
551 GetFd(profile2)});
552 int reference_profile_fd = GetFd(reference_profile);
553
554 // The new profile info will contain the methods with indices 0-100.
555 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
556 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400557 SetupProfile(dex1, dex2, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1);
Calin Juravle2e2db782016-02-23 12:00:03 +0000558 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400559 SetupProfile(dex3, dex4, kNumberOfMethodsToEnableCompilation, 0, profile2, &info2);
Calin Juravle2e2db782016-02-23 12:00:03 +0000560
561
562 // The reference profile info will contain the methods with indices 50-150.
563 const uint16_t kNumberOfMethodsAlreadyCompiled = 100;
564 ProfileCompilationInfo reference_info;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400565 SetupProfile(dex1, dex2, kNumberOfMethodsAlreadyCompiled, 0, reference_profile,
Calin Juravle2e2db782016-02-23 12:00:03 +0000566 &reference_info, kNumberOfMethodsToEnableCompilation / 2);
567
568 // We should advise compilation.
569 ASSERT_EQ(ProfileAssistant::kCompile,
570 ProcessProfiles(profile_fds, reference_profile_fd));
571
572 // The resulting compilation info must be equal to the merge of the inputs
573 ProfileCompilationInfo result;
Calin Juravle2e2db782016-02-23 12:00:03 +0000574 ASSERT_TRUE(result.Load(reference_profile_fd));
575
576 ProfileCompilationInfo expected;
Calin Juravle67265462016-03-18 16:23:40 +0000577 ASSERT_TRUE(expected.MergeWith(info1));
578 ASSERT_TRUE(expected.MergeWith(info2));
579 ASSERT_TRUE(expected.MergeWith(reference_info));
Calin Juravle2e2db782016-02-23 12:00:03 +0000580 ASSERT_TRUE(expected.Equals(result));
581
582 // The information from profiles must remain the same.
583 CheckProfileInfo(profile1, info1);
584 CheckProfileInfo(profile2, info2);
585}
586
Calin Juravle17374092021-06-08 13:45:09 -0700587TEST_F(ProfileAssistantTest, DoNotAdviseCompilationEmptyProfile) {
588 ScratchFile profile1;
589 ScratchFile profile2;
590 ScratchFile reference_profile;
591
592 std::vector<int> profile_fds({
593 GetFd(profile1),
594 GetFd(profile2)});
595 int reference_profile_fd = GetFd(reference_profile);
596
597 ProfileCompilationInfo info1;
598 SetupProfile(dex1, dex2, /*number_of_methods=*/ 0, /*number_of_classes*/ 0, profile1, &info1);
599 ProfileCompilationInfo info2;
600 SetupProfile(dex3, dex4, /*number_of_methods=*/ 0, /*number_of_classes*/ 0, profile2, &info2);
601
602 // We should not advise compilation.
603 ASSERT_EQ(ProfileAssistant::kSkipCompilationEmptyProfiles,
604 ProcessProfiles(profile_fds, reference_profile_fd));
605
606 // The information from profiles must remain the same.
607 ProfileCompilationInfo file_info1;
608 ASSERT_TRUE(file_info1.Load(GetFd(profile1)));
609 ASSERT_TRUE(file_info1.Equals(info1));
610
611 ProfileCompilationInfo file_info2;
612 ASSERT_TRUE(file_info2.Load(GetFd(profile2)));
613 ASSERT_TRUE(file_info2.Equals(info2));
614
615 // Reference profile files must remain empty.
616 ASSERT_EQ(0, reference_profile.GetFile()->GetLength());
617
618 // The information from profiles must remain the same.
619 CheckProfileInfo(profile1, info1);
620 CheckProfileInfo(profile2, info2);
621}
622
Calin Juravle2e2db782016-02-23 12:00:03 +0000623TEST_F(ProfileAssistantTest, DoNotAdviseCompilation) {
624 ScratchFile profile1;
625 ScratchFile profile2;
626 ScratchFile reference_profile;
627
628 std::vector<int> profile_fds({
629 GetFd(profile1),
630 GetFd(profile2)});
631 int reference_profile_fd = GetFd(reference_profile);
632
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700633 const uint16_t kNumberOfMethodsToSkipCompilation = 24; // Threshold is 100.
Calin Juravle2e2db782016-02-23 12:00:03 +0000634 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400635 SetupProfile(dex1, dex2, kNumberOfMethodsToSkipCompilation, 0, profile1, &info1);
Calin Juravle2e2db782016-02-23 12:00:03 +0000636 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400637 SetupProfile(dex3, dex4, kNumberOfMethodsToSkipCompilation, 0, profile2, &info2);
Calin Juravle2e2db782016-02-23 12:00:03 +0000638
639 // We should not advise compilation.
Calin Juravle17374092021-06-08 13:45:09 -0700640 ASSERT_EQ(ProfileAssistant::kSkipCompilationSmallDelta,
Calin Juravle2e2db782016-02-23 12:00:03 +0000641 ProcessProfiles(profile_fds, reference_profile_fd));
642
643 // The information from profiles must remain the same.
644 ProfileCompilationInfo file_info1;
Calin Juravle2e2db782016-02-23 12:00:03 +0000645 ASSERT_TRUE(file_info1.Load(GetFd(profile1)));
646 ASSERT_TRUE(file_info1.Equals(info1));
647
648 ProfileCompilationInfo file_info2;
Calin Juravle2e2db782016-02-23 12:00:03 +0000649 ASSERT_TRUE(file_info2.Load(GetFd(profile2)));
650 ASSERT_TRUE(file_info2.Equals(info2));
651
652 // Reference profile files must remain empty.
653 ASSERT_EQ(0, reference_profile.GetFile()->GetLength());
654
655 // The information from profiles must remain the same.
656 CheckProfileInfo(profile1, info1);
657 CheckProfileInfo(profile2, info2);
658}
659
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700660TEST_F(ProfileAssistantTest, DoNotAdviseCompilationMethodPercentage) {
661 const uint16_t kNumberOfMethodsInRefProfile = 6000;
662 const uint16_t kNumberOfMethodsInCurProfile = 6100; // Threshold is 2%.
yawanngb209a042020-11-11 20:17:56 +0000663 std::vector<const std::string> extra_args({"--min-new-methods-percent-change=2"});
yawanng7c618802020-11-05 20:02:27 +0000664
665 // We should not advise compilation.
Calin Juravle17374092021-06-08 13:45:09 -0700666 ASSERT_EQ(ProfileAssistant::kSkipCompilationSmallDelta,
yawanng7c618802020-11-05 20:02:27 +0000667 CheckCompilationMethodPercentChange(kNumberOfMethodsInCurProfile,
668 kNumberOfMethodsInRefProfile,
669 extra_args));
670}
671
yawanngb209a042020-11-11 20:17:56 +0000672TEST_F(ProfileAssistantTest, ShouldAdviseCompilationMethodPercentage) {
673 const uint16_t kNumberOfMethodsInRefProfile = 6000;
674 const uint16_t kNumberOfMethodsInCurProfile = 6200; // Threshold is 2%.
675 std::vector<const std::string> extra_args({"--min-new-methods-percent-change=2"});
676
677 // We should advise compilation.
678 ASSERT_EQ(ProfileAssistant::kCompile,
679 CheckCompilationMethodPercentChange(kNumberOfMethodsInCurProfile,
680 kNumberOfMethodsInRefProfile,
681 extra_args));
682}
683
684TEST_F(ProfileAssistantTest, DoNotAdviseCompilationMethodPercentageWithNewMin) {
685 const uint16_t kNumberOfMethodsInRefProfile = 6000;
686 const uint16_t kNumberOfMethodsInCurProfile = 6200; // Threshold is 20%.
687
688 // We should not advise compilation.
Calin Juravle17374092021-06-08 13:45:09 -0700689 ASSERT_EQ(ProfileAssistant::kSkipCompilationSmallDelta,
yawanngb209a042020-11-11 20:17:56 +0000690 CheckCompilationMethodPercentChange(kNumberOfMethodsInCurProfile,
691 kNumberOfMethodsInRefProfile));
692}
693
Vladimir Markoc63d9672021-03-31 15:50:39 +0100694TEST_F(ProfileAssistantTest, DoNotAdviseCompilationClassPercentage) {
Shubham Ajmera9ab6e1d2017-09-25 18:40:54 -0700695 const uint16_t kNumberOfClassesInRefProfile = 6000;
696 const uint16_t kNumberOfClassesInCurProfile = 6110; // Threshold is 2%.
yawanngb209a042020-11-11 20:17:56 +0000697 std::vector<const std::string> extra_args({"--min-new-classes-percent-change=2"});
yawanng7c618802020-11-05 20:02:27 +0000698
699 // We should not advise compilation.
Calin Juravle17374092021-06-08 13:45:09 -0700700 ASSERT_EQ(ProfileAssistant::kSkipCompilationSmallDelta,
yawanng7c618802020-11-05 20:02:27 +0000701 CheckCompilationClassPercentChange(kNumberOfClassesInCurProfile,
702 kNumberOfClassesInRefProfile,
703 extra_args));
704}
705
yawanngb209a042020-11-11 20:17:56 +0000706TEST_F(ProfileAssistantTest, ShouldAdviseCompilationClassPercentage) {
707 const uint16_t kNumberOfClassesInRefProfile = 6000;
708 const uint16_t kNumberOfClassesInCurProfile = 6120; // Threshold is 2%.
709 std::vector<const std::string> extra_args({"--min-new-classes-percent-change=2"});
710
711 // We should advise compilation.
712 ASSERT_EQ(ProfileAssistant::kCompile,
713 CheckCompilationClassPercentChange(kNumberOfClassesInCurProfile,
714 kNumberOfClassesInRefProfile,
715 extra_args));
716}
717
718TEST_F(ProfileAssistantTest, DoNotAdviseCompilationClassPercentageWithNewMin) {
719 const uint16_t kNumberOfClassesInRefProfile = 6000;
720 const uint16_t kNumberOfClassesInCurProfile = 6200; // Threshold is 20%.
721
722 // We should not advise compilation.
Calin Juravle17374092021-06-08 13:45:09 -0700723 ASSERT_EQ(ProfileAssistant::kSkipCompilationSmallDelta,
yawanngb209a042020-11-11 20:17:56 +0000724 CheckCompilationClassPercentChange(kNumberOfClassesInCurProfile,
725 kNumberOfClassesInRefProfile));
726}
727
Calin Juravle2e2db782016-02-23 12:00:03 +0000728TEST_F(ProfileAssistantTest, FailProcessingBecauseOfProfiles) {
729 ScratchFile profile1;
730 ScratchFile profile2;
731 ScratchFile reference_profile;
732
733 std::vector<int> profile_fds({
734 GetFd(profile1),
735 GetFd(profile2)});
736 int reference_profile_fd = GetFd(reference_profile);
737
738 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
739 // Assign different hashes for the same dex file. This will make merging of information to fail.
740 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400741 SetupProfile(dex1, dex2, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1);
Calin Juravle2e2db782016-02-23 12:00:03 +0000742 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400743 SetupProfile(
744 dex1_checksum_missmatch, dex2, kNumberOfMethodsToEnableCompilation, 0, profile2, &info2);
Calin Juravle2e2db782016-02-23 12:00:03 +0000745
746 // We should fail processing.
747 ASSERT_EQ(ProfileAssistant::kErrorBadProfiles,
748 ProcessProfiles(profile_fds, reference_profile_fd));
749
750 // The information from profiles must remain the same.
751 CheckProfileInfo(profile1, info1);
752 CheckProfileInfo(profile2, info2);
753
754 // Reference profile files must still remain empty.
755 ASSERT_EQ(0, reference_profile.GetFile()->GetLength());
756}
757
758TEST_F(ProfileAssistantTest, FailProcessingBecauseOfReferenceProfiles) {
759 ScratchFile profile1;
760 ScratchFile reference_profile;
761
762 std::vector<int> profile_fds({
763 GetFd(profile1)});
764 int reference_profile_fd = GetFd(reference_profile);
765
766 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
767 // Assign different hashes for the same dex file. This will make merging of information to fail.
768 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -0400769 SetupProfile(dex1, dex2, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1);
Calin Juravle2e2db782016-02-23 12:00:03 +0000770 ProfileCompilationInfo reference_info;
Alex Lighta2f13192021-02-03 18:19:03 -0800771 SetupProfile(dex1_checksum_missmatch,
772 dex2,
773 kNumberOfMethodsToEnableCompilation,
774 0,
775 reference_profile,
776 &reference_info);
Calin Juravle2e2db782016-02-23 12:00:03 +0000777
778 // We should not advise compilation.
Calin Juravle2e2db782016-02-23 12:00:03 +0000779 ASSERT_EQ(ProfileAssistant::kErrorBadProfiles,
780 ProcessProfiles(profile_fds, reference_profile_fd));
781
782 // The information from profiles must remain the same.
783 CheckProfileInfo(profile1, info1);
784}
785
Calin Juravle7bcdb532016-06-07 16:14:47 +0100786TEST_F(ProfileAssistantTest, TestProfileGeneration) {
787 ScratchFile profile;
788 // Generate a test profile.
789 GenerateTestProfile(profile.GetFilename());
790
791 // Verify that the generated profile is valid and can be loaded.
Calin Juravle7bcdb532016-06-07 16:14:47 +0100792 ProfileCompilationInfo info;
793 ASSERT_TRUE(info.Load(GetFd(profile)));
794}
795
Jeff Haof0a31f82017-03-27 15:50:37 -0700796TEST_F(ProfileAssistantTest, TestProfileGenerationWithIndexDex) {
797 ScratchFile profile;
798 // Generate a test profile passing in a dex file as reference.
799 GenerateTestProfileWithInputDex(profile.GetFilename());
800
801 // Verify that the generated profile is valid and can be loaded.
Jeff Haof0a31f82017-03-27 15:50:37 -0700802 ProfileCompilationInfo info;
803 ASSERT_TRUE(info.Load(GetFd(profile)));
804}
805
David Sehr7c80f2d2017-02-07 16:47:58 -0800806TEST_F(ProfileAssistantTest, TestProfileCreationAllMatch) {
807 // Class names put here need to be in sorted order.
808 std::vector<std::string> class_names = {
Mathieu Chartierea650f32017-05-24 12:04:13 -0700809 "HLjava/lang/Object;-><init>()V",
Calin Juravlee0ac1152017-02-13 19:03:47 -0800810 "Ljava/lang/Comparable;",
811 "Ljava/lang/Math;",
Mathieu Chartier34067262017-04-06 13:55:46 -0700812 "Ljava/lang/Object;",
Mathieu Chartierea650f32017-05-24 12:04:13 -0700813 "SPLjava/lang/Comparable;->compareTo(Ljava/lang/Object;)I",
Vladimir Markoc63d9672021-03-31 15:50:39 +0100814 "[[[[[[[[I", // No `TypeId`s in core-oj with this many array dimensions,
815 "[[[[[[[[Ljava/lang/Object;", // "extra descriptors" shall be used for these array classes.
David Sehr7c80f2d2017-02-07 16:47:58 -0800816 };
Mathieu Chartier34067262017-04-06 13:55:46 -0700817 std::string file_contents;
David Sehr7c80f2d2017-02-07 16:47:58 -0800818 for (std::string& class_name : class_names) {
Mathieu Chartier34067262017-04-06 13:55:46 -0700819 file_contents += class_name + std::string("\n");
David Sehr7c80f2d2017-02-07 16:47:58 -0800820 }
821 std::string output_file_contents;
Mathieu Chartier34067262017-04-06 13:55:46 -0700822 ASSERT_TRUE(CreateAndDump(file_contents, &output_file_contents));
823 ASSERT_EQ(output_file_contents, file_contents);
David Sehr7c80f2d2017-02-07 16:47:58 -0800824}
825
Nicolas Geoffrayf5e26f82019-08-13 15:21:05 +0100826TEST_F(ProfileAssistantTest, TestArrayClass) {
827 std::vector<std::string> class_names = {
828 "[Ljava/lang/Comparable;",
829 };
830 std::string file_contents;
831 for (std::string& class_name : class_names) {
832 file_contents += class_name + std::string("\n");
833 }
834 std::string output_file_contents;
835 ASSERT_TRUE(CreateAndDump(file_contents, &output_file_contents));
836 ASSERT_EQ(output_file_contents, file_contents);
837}
838
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700839TEST_F(ProfileAssistantTest, TestProfileCreationGenerateMethods) {
840 // Class names put here need to be in sorted order.
841 std::vector<std::string> class_names = {
Calin Juravlea6c9b782019-09-16 18:57:26 -0700842 "HLjava/lang/Math;->*",
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700843 };
844 std::string input_file_contents;
845 std::string expected_contents;
846 for (std::string& class_name : class_names) {
847 input_file_contents += class_name + std::string("\n");
848 expected_contents += DescriptorToDot(class_name.c_str()) +
849 std::string("\n");
850 }
851 std::string output_file_contents;
852 ScratchFile profile_file;
853 EXPECT_TRUE(CreateProfile(input_file_contents,
854 profile_file.GetFilename(),
Calin Juravle1bfe4bd2018-04-26 16:00:11 -0700855 GetLibCoreDexFileNames()[0]));
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700856 ProfileCompilationInfo info;
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700857 ASSERT_TRUE(info.Load(GetFd(profile_file)));
858 // Verify that the profile has matching methods.
859 ScopedObjectAccess soa(Thread::Current());
Vladimir Marko95f4c662021-05-11 12:25:20 +0100860 ObjPtr<mirror::Class> klass = GetClass(soa, /*class_loader=*/ nullptr, "Ljava/lang/Math;");
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700861 ASSERT_TRUE(klass != nullptr);
862 size_t method_count = 0;
863 for (ArtMethod& method : klass->GetMethods(kRuntimePointerSize)) {
864 if (!method.IsCopied() && method.GetCodeItem() != nullptr) {
865 ++method_count;
Vladimir Markoa64c1ad2021-03-08 14:27:05 +0000866 ProfileCompilationInfo::MethodHotness hotness =
867 info.GetMethodHotness(MethodReference(method.GetDexFile(), method.GetDexMethodIndex()));
868 ASSERT_TRUE(hotness.IsHot()) << method.PrettyMethod();
Mathieu Chartierd808e8b2017-03-21 13:37:41 -0700869 }
870 }
871 EXPECT_GT(method_count, 0u);
872}
873
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700874static std::string JoinProfileLines(const std::vector<std::string>& lines) {
875 std::string result = android::base::Join(lines, '\n');
876 return result + '\n';
877}
878
Mathieu Chartier2f794552017-06-19 10:58:08 -0700879TEST_F(ProfileAssistantTest, TestBootImageProfile) {
880 const std::string core_dex = GetLibCoreDexFileNames()[0];
881
882 std::vector<ScratchFile> profiles;
883
884 // In image with enough clean occurrences.
885 const std::string kCleanClass = "Ljava/lang/CharSequence;";
886 // In image with enough dirty occurrences.
887 const std::string kDirtyClass = "Ljava/lang/Object;";
888 // Not in image becauseof not enough occurrences.
889 const std::string kUncommonCleanClass = "Ljava/lang/Process;";
890 const std::string kUncommonDirtyClass = "Ljava/lang/Package;";
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700891 // Method that is common and hot. Should end up in profile.
892 const std::string kCommonHotMethod = "Ljava/lang/Comparable;->compareTo(Ljava/lang/Object;)I";
893 // Uncommon method, should not end up in profile
894 const std::string kUncommonMethod = "Ljava/util/HashMap;-><init>()V";
895 // Method that gets marked as hot since it's in multiple profile and marked as startup.
896 const std::string kStartupMethodForUpgrade = "Ljava/util/ArrayList;->clear()V";
897 // Startup method used by a special package which will get a different threshold;
898 const std::string kSpecialPackageStartupMethod =
899 "Ljava/lang/Object;->toString()Ljava/lang/String;";
900 // Method used by a special package which will get a different threshold;
901 const std::string kUncommonSpecialPackageMethod = "Ljava/lang/Object;->hashCode()I";
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100902 // Denylisted class
903 const std::string kPreloadedDenylistedClass = "Ljava/lang/Thread;";
Mathieu Chartier2f794552017-06-19 10:58:08 -0700904
905 // Thresholds for this test.
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700906 static const size_t kDirtyThreshold = 100;
907 static const size_t kCleanThreshold = 50;
908 static const size_t kPreloadedThreshold = 100;
909 static const size_t kMethodThreshold = 75;
910 static const size_t kSpecialThreshold = 50;
911 const std::string kSpecialPackage = "dex4";
Mathieu Chartier2f794552017-06-19 10:58:08 -0700912
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700913 // Create boot profile content, attributing the classes and methods to different dex files.
914 std::vector<std::string> input_data = {
915 "{dex1}" + kCleanClass,
916 "{dex1}" + kDirtyClass,
917 "{dex1}" + kUncommonCleanClass,
918 "{dex1}H" + kCommonHotMethod,
919 "{dex1}P" + kStartupMethodForUpgrade,
920 "{dex1}" + kUncommonDirtyClass,
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100921 "{dex1}" + kPreloadedDenylistedClass,
Mathieu Chartier2f794552017-06-19 10:58:08 -0700922
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700923 "{dex2}" + kCleanClass,
924 "{dex2}" + kDirtyClass,
925 "{dex2}P" + kCommonHotMethod,
926 "{dex2}P" + kStartupMethodForUpgrade,
927 "{dex2}" + kUncommonDirtyClass,
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100928 "{dex2}" + kPreloadedDenylistedClass,
Mathieu Chartier2f794552017-06-19 10:58:08 -0700929
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700930 "{dex3}P" + kUncommonMethod,
931 "{dex3}PS" + kStartupMethodForUpgrade,
932 "{dex3}S" + kCommonHotMethod,
933 "{dex3}S" + kSpecialPackageStartupMethod,
934 "{dex3}" + kDirtyClass,
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100935 "{dex3}" + kPreloadedDenylistedClass,
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700936
937 "{dex4}" + kDirtyClass,
938 "{dex4}P" + kCommonHotMethod,
939 "{dex4}S" + kSpecialPackageStartupMethod,
Calin Juravle48030c42020-06-17 19:16:01 -0700940 "{dex4}P" + kUncommonSpecialPackageMethod,
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100941 "{dex4}" + kPreloadedDenylistedClass,
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700942 };
943 std::string input_file_contents = JoinProfileLines(input_data);
944
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100945 ScratchFile preloaded_class_denylist;
946 std::string denylist_content = DescriptorToDot(kPreloadedDenylistedClass.c_str());
947 EXPECT_TRUE(preloaded_class_denylist.GetFile()->WriteFully(
948 denylist_content.c_str(), denylist_content.length()));
Calin Juravle48030c42020-06-17 19:16:01 -0700949
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100950 EXPECT_EQ(0, preloaded_class_denylist.GetFile()->Flush());
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700951 // Expected data
952 std::vector<std::string> expected_data = {
953 kCleanClass,
954 kDirtyClass,
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100955 kPreloadedDenylistedClass,
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700956 "HSP" + kCommonHotMethod,
957 "HS" + kSpecialPackageStartupMethod,
958 "HSP" + kStartupMethodForUpgrade
959 };
960 std::string expected_profile_content = JoinProfileLines(expected_data);
961
962 std::vector<std::string> expected_preloaded_data = {
963 DescriptorToDot(kDirtyClass.c_str())
964 };
965 std::string expected_preloaded_content = JoinProfileLines(expected_preloaded_data);
966
967 ScratchFile profile;
Vladimir Markoadfa1ad2021-04-23 15:02:49 +0100968 EXPECT_TRUE(CreateProfile(input_file_contents,
969 profile.GetFilename(),
970 core_dex,
971 /*for_boot_image=*/ true));
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700972
Vladimir Markoadfa1ad2021-04-23 15:02:49 +0100973 ProfileCompilationInfo bootProfile(/*for_boot_image=*/ true);
974 bootProfile.Load(profile.GetFilename(), /*clear_if_invalid=*/ true);
Mathieu Chartier2f794552017-06-19 10:58:08 -0700975
976 // Generate the boot profile.
977 ScratchFile out_profile;
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700978 ScratchFile out_preloaded_classes;
Mathieu Chartier2f794552017-06-19 10:58:08 -0700979 std::vector<std::string> args;
980 args.push_back(GetProfmanCmd());
981 args.push_back("--generate-boot-image-profile");
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700982 args.push_back("--class-threshold=" + std::to_string(kDirtyThreshold));
983 args.push_back("--clean-class-threshold=" + std::to_string(kCleanThreshold));
984 args.push_back("--method-threshold=" + std::to_string(kMethodThreshold));
985 args.push_back("--preloaded-class-threshold=" + std::to_string(kPreloadedThreshold));
986 args.push_back(
987 "--special-package=" + kSpecialPackage + ":" + std::to_string(kSpecialThreshold));
988 args.push_back("--profile-file=" + profile.GetFilename());
989 args.push_back("--out-profile-path=" + out_profile.GetFilename());
990 args.push_back("--out-preloaded-classes-path=" + out_preloaded_classes.GetFilename());
Mathieu Chartier2f794552017-06-19 10:58:08 -0700991 args.push_back("--apk=" + core_dex);
992 args.push_back("--dex-location=" + core_dex);
Orion Hodson3e8caeb2020-08-07 15:41:24 +0100993 args.push_back("--preloaded-classes-denylist=" + preloaded_class_denylist.GetFilename());
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700994
Mathieu Chartier2f794552017-06-19 10:58:08 -0700995 std::string error;
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700996 ASSERT_EQ(ExecAndReturnCode(args, &error), 0) << error;
Mathieu Chartier2f794552017-06-19 10:58:08 -0700997
998 // Verify the boot profile contents.
Calin Juravle0f7f4fc2020-05-11 20:16:50 -0700999 std::string output_profile_contents;
1000 ASSERT_TRUE(android::base::ReadFileToString(
1001 out_profile.GetFilename(), &output_profile_contents));
1002 ASSERT_EQ(output_profile_contents, expected_profile_content);
1003
Calin Juravle0e02d612020-06-18 18:05:29 -07001004 // Verify the preloaded classes content.
Calin Juravle0f7f4fc2020-05-11 20:16:50 -07001005 std::string output_preloaded_contents;
1006 ASSERT_TRUE(android::base::ReadFileToString(
1007 out_preloaded_classes.GetFilename(), &output_preloaded_contents));
1008 ASSERT_EQ(output_preloaded_contents, expected_preloaded_content);
Mathieu Chartier2f794552017-06-19 10:58:08 -07001009}
1010
Calin Juravle0e02d612020-06-18 18:05:29 -07001011TEST_F(ProfileAssistantTest, TestBootImageProfileWith2RawProfiles) {
1012 const std::string core_dex = GetLibCoreDexFileNames()[0];
1013
1014 std::vector<ScratchFile> profiles;
1015
1016 const std::string kCommonClassUsedByDex1 = "Ljava/lang/CharSequence;";
1017 const std::string kCommonClassUsedByDex1Dex2 = "Ljava/lang/Object;";
1018 const std::string kUncommonClass = "Ljava/lang/Process;";
1019 const std::string kCommonHotMethodUsedByDex1 =
1020 "Ljava/lang/Comparable;->compareTo(Ljava/lang/Object;)I";
1021 const std::string kCommonHotMethodUsedByDex1Dex2 = "Ljava/lang/Object;->hashCode()I";
1022 const std::string kUncommonHotMethod = "Ljava/util/HashMap;-><init>()V";
1023
1024
1025 // Thresholds for this test.
1026 static const size_t kDirtyThreshold = 100;
1027 static const size_t kCleanThreshold = 100;
1028 static const size_t kMethodThreshold = 100;
1029
1030 // Create boot profile content, attributing the classes and methods to different dex files.
1031 std::vector<std::string> input_data1 = {
1032 "{dex1}" + kCommonClassUsedByDex1,
1033 "{dex1}" + kCommonClassUsedByDex1Dex2,
1034 "{dex1}" + kUncommonClass,
1035 "{dex1}H" + kCommonHotMethodUsedByDex1Dex2,
1036 "{dex1}" + kCommonHotMethodUsedByDex1,
1037 };
1038 std::vector<std::string> input_data2 = {
1039 "{dex1}" + kCommonClassUsedByDex1,
1040 "{dex2}" + kCommonClassUsedByDex1Dex2,
1041 "{dex1}H" + kCommonHotMethodUsedByDex1,
1042 "{dex2}" + kCommonHotMethodUsedByDex1Dex2,
1043 "{dex1}" + kUncommonHotMethod,
1044 };
1045 std::string input_file_contents1 = JoinProfileLines(input_data1);
1046 std::string input_file_contents2 = JoinProfileLines(input_data2);
1047
1048 // Expected data
1049 std::vector<std::string> expected_data = {
1050 kCommonClassUsedByDex1,
1051 kCommonClassUsedByDex1Dex2,
1052 "H" + kCommonHotMethodUsedByDex1,
1053 "H" + kCommonHotMethodUsedByDex1Dex2
1054 };
1055 std::string expected_profile_content = JoinProfileLines(expected_data);
1056
1057 ScratchFile profile1;
1058 ScratchFile profile2;
Vladimir Markoadfa1ad2021-04-23 15:02:49 +01001059 EXPECT_TRUE(CreateProfile(input_file_contents1,
1060 profile1.GetFilename(),
1061 core_dex,
1062 /*for_boot_image=*/ true));
1063 EXPECT_TRUE(CreateProfile(input_file_contents2,
1064 profile2.GetFilename(),
1065 core_dex,
1066 /*for_boot_image=*/ true));
Calin Juravle0e02d612020-06-18 18:05:29 -07001067
1068 ProfileCompilationInfo boot_profile1;
1069 ProfileCompilationInfo boot_profile2;
Vladimir Marko95f4c662021-05-11 12:25:20 +01001070 boot_profile1.Load(profile1.GetFilename(), /*for_boot_image=*/ true);
1071 boot_profile2.Load(profile2.GetFilename(), /*for_boot_image=*/ true);
Calin Juravle0e02d612020-06-18 18:05:29 -07001072
1073 // Generate the boot profile.
1074 ScratchFile out_profile;
1075 ScratchFile out_preloaded_classes;
Calin Juravle0e02d612020-06-18 18:05:29 -07001076 std::vector<std::string> args;
1077 args.push_back(GetProfmanCmd());
1078 args.push_back("--generate-boot-image-profile");
1079 args.push_back("--class-threshold=" + std::to_string(kDirtyThreshold));
1080 args.push_back("--clean-class-threshold=" + std::to_string(kCleanThreshold));
1081 args.push_back("--method-threshold=" + std::to_string(kMethodThreshold));
1082 args.push_back("--profile-file=" + profile1.GetFilename());
1083 args.push_back("--profile-file=" + profile2.GetFilename());
1084 args.push_back("--out-profile-path=" + out_profile.GetFilename());
1085 args.push_back("--out-preloaded-classes-path=" + out_preloaded_classes.GetFilename());
1086 args.push_back("--apk=" + core_dex);
1087 args.push_back("--dex-location=" + core_dex);
1088
1089 std::string error;
1090 ASSERT_EQ(ExecAndReturnCode(args, &error), 0) << error;
Calin Juravle0e02d612020-06-18 18:05:29 -07001091
1092 // Verify the boot profile contents.
1093 std::string output_profile_contents;
1094 ASSERT_TRUE(android::base::ReadFileToString(
1095 out_profile.GetFilename(), &output_profile_contents));
1096 ASSERT_EQ(output_profile_contents, expected_profile_content);
1097}
1098
David Sehr7c80f2d2017-02-07 16:47:58 -08001099TEST_F(ProfileAssistantTest, TestProfileCreationOneNotMatched) {
1100 // Class names put here need to be in sorted order.
1101 std::vector<std::string> class_names = {
Calin Juravlee0ac1152017-02-13 19:03:47 -08001102 "Ldoesnt/match/this/one;",
1103 "Ljava/lang/Comparable;",
1104 "Ljava/lang/Object;"
David Sehr7c80f2d2017-02-07 16:47:58 -08001105 };
1106 std::string input_file_contents;
1107 for (std::string& class_name : class_names) {
1108 input_file_contents += class_name + std::string("\n");
1109 }
1110 std::string output_file_contents;
1111 ASSERT_TRUE(CreateAndDump(input_file_contents, &output_file_contents));
1112 std::string expected_contents =
Mathieu Chartier34067262017-04-06 13:55:46 -07001113 class_names[1] + std::string("\n") +
1114 class_names[2] + std::string("\n");
David Sehr7c80f2d2017-02-07 16:47:58 -08001115 ASSERT_EQ(output_file_contents, expected_contents);
1116}
1117
1118TEST_F(ProfileAssistantTest, TestProfileCreationNoneMatched) {
1119 // Class names put here need to be in sorted order.
1120 std::vector<std::string> class_names = {
Calin Juravlee0ac1152017-02-13 19:03:47 -08001121 "Ldoesnt/match/this/one;",
1122 "Ldoesnt/match/this/one/either;",
1123 "Lnor/this/one;"
David Sehr7c80f2d2017-02-07 16:47:58 -08001124 };
1125 std::string input_file_contents;
1126 for (std::string& class_name : class_names) {
1127 input_file_contents += class_name + std::string("\n");
1128 }
1129 std::string output_file_contents;
1130 ASSERT_TRUE(CreateAndDump(input_file_contents, &output_file_contents));
1131 std::string expected_contents("");
1132 ASSERT_EQ(output_file_contents, expected_contents);
1133}
1134
Alex Lighta2f13192021-02-03 18:19:03 -08001135// Test that we can dump profiles in a way they can be re-constituted.
1136// Test goes 'txt -> prof -> txt -> prof' and then compares the two profs.
1137TEST_F(ProfileAssistantTest, TestProfileRoundTrip) {
Calin Juravlee0ac1152017-02-13 19:03:47 -08001138 // Create the profile content.
Alex Lighta2f13192021-02-03 18:19:03 -08001139 std::vector<std::string_view> methods = {
Calin Juravlea6c9b782019-09-16 18:57:26 -07001140 "HLTestInline;->inlineMonomorphic(LSuper;)I+LSubA;",
1141 "HLTestInline;->inlinePolymorphic(LSuper;)I+LSubA;,LSubB;,LSubC;",
1142 "HLTestInline;->inlineMegamorphic(LSuper;)I+LSubA;,LSubB;,LSubC;,LSubD;,LSubE;",
1143 "HLTestInline;->inlineMissingTypes(LSuper;)I+missing_types",
Alex Lighta2f13192021-02-03 18:19:03 -08001144 "HLTestInline;->noInlineCache(LSuper;)I",
Alex Light0b58ec52021-03-02 14:11:59 -08001145 "HLTestInline;->inlineMultiMonomorphic(LSuper;LSecret;)I+]LSuper;LSubA;]LSecret;LSubB;",
1146 "HLTestInline;->inlineMultiPolymorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
1147 "HLTestInline;->inlineMultiMegamorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;,LSubD;,LSubE;]LSecret;megamorphic_types",
1148 "HLTestInline;->inlineMultiMissingTypes(LSuper;LSecret;)I+]LSuper;missing_types]LSecret;missing_types",
1149 "HLTestInline;->inlineTriplePolymorphic(LSuper;LSecret;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
Alex Lighta2f13192021-02-03 18:19:03 -08001150 "HLTestInline;->noInlineCacheMulti(LSuper;LSecret;)I",
Calin Juravlee0ac1152017-02-13 19:03:47 -08001151 };
Alex Lighta2f13192021-02-03 18:19:03 -08001152 std::ostringstream input_file_contents;
1153 for (const std::string_view& m : methods) {
1154 input_file_contents << m << "\n";
Calin Juravlee0ac1152017-02-13 19:03:47 -08001155 }
1156
1157 // Create the profile and save it to disk.
1158 ScratchFile profile_file;
Alex Lighta2f13192021-02-03 18:19:03 -08001159 ASSERT_TRUE(CreateProfile(input_file_contents.str(),
1160 profile_file.GetFilename(),
1161 GetTestDexFileName("ProfileTestMultiDex")));
Alex Lighta2f13192021-02-03 18:19:03 -08001162
1163 // Dump the file back into text.
1164 std::string text_two;
1165 ASSERT_TRUE(DumpClassesAndMethods(
1166 profile_file.GetFilename(), &text_two, GetTestDexFileName("ProfileTestMultiDex")));
1167
1168 // Create another profile and save it to the disk as well.
1169 ScratchFile profile_two;
1170 ASSERT_TRUE(CreateProfile(
1171 text_two, profile_two.GetFilename(), GetTestDexFileName("ProfileTestMultiDex")));
Alex Lighta2f13192021-02-03 18:19:03 -08001172
1173 // These two profiles should be bit-identical.
1174 // TODO We could compare the 'text_two' to the methods but since the order is
1175 // arbitrary for many parts and there are multiple 'correct' dumps we'd need
1176 // to basically parse everything and this is simply easier.
1177 std::string error;
1178 std::vector<std::string> args { kIsTargetBuild ? "/system/bin/cmp" : "/usr/bin/cmp",
1179 "-s",
1180 profile_file.GetFilename(),
1181 profile_two.GetFilename() };
1182 ASSERT_EQ(ExecAndReturnCode(args, &error), 0) << error << " from " << text_two;
1183}
1184
Alex Lightd83edf32021-02-22 15:07:18 -08001185
1186// Test that we can dump profiles in a way they can be re-constituted and
1187// annotations don't interfere. Test goes 'txt -> ProfileWithAnnotations -> txt
1188// -> prof' and then compares that to one that is 'txt ->
1189// prof_without_annotations'.
1190TEST_F(ProfileAssistantTest, TestProfileRoundTripWithAnnotations) {
1191 // Create the profile content.
1192 std::vector<std::string_view> methods = {
1193 "HLTestInline;->inlineMonomorphic(LSuper;)I+LSubA;",
1194 "HLTestInline;->inlinePolymorphic(LSuper;)I+LSubA;,LSubB;,LSubC;",
1195 "HLTestInline;->inlineMegamorphic(LSuper;)I+LSubA;,LSubB;,LSubC;,LSubD;,LSubE;",
1196 "HLTestInline;->inlineMissingTypes(LSuper;)I+missing_types",
1197 "HLTestInline;->noInlineCache(LSuper;)I",
Alex Light0b58ec52021-03-02 14:11:59 -08001198 "HLTestInline;->inlineMultiMonomorphic(LSuper;LSecret;)I+]LSuper;LSubA;]LSecret;LSubB;",
1199 "HLTestInline;->inlineMultiPolymorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
1200 "HLTestInline;->inlineMultiMegamorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;,LSubD;,LSubE;]LSecret;megamorphic_types",
1201 "HLTestInline;->inlineMultiMissingTypes(LSuper;LSecret;)I+]LSuper;missing_types]LSecret;missing_types",
1202 "HLTestInline;->inlineTriplePolymorphic(LSuper;LSecret;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
Alex Lightd83edf32021-02-22 15:07:18 -08001203 "HLTestInline;->noInlineCacheMulti(LSuper;LSecret;)I",
1204 };
1205 std::ostringstream no_annotation_input_file_contents;
1206 std::ostringstream with_annotation_input_file_contents;
1207 for (const std::string_view& m : methods) {
1208 no_annotation_input_file_contents << m << "\n";
1209 with_annotation_input_file_contents << "{foobar}" << m << "\n";
1210 }
1211
1212 // Create the profile and save it to disk.
1213 ScratchFile with_annotation_profile_file;
1214 ASSERT_TRUE(CreateProfile(with_annotation_input_file_contents.str(),
1215 with_annotation_profile_file.GetFilename(),
1216 GetTestDexFileName("ProfileTestMultiDex")));
Alex Lightd83edf32021-02-22 15:07:18 -08001217
1218 ScratchFile no_annotation_profile_file;
1219 ASSERT_TRUE(CreateProfile(no_annotation_input_file_contents.str(),
1220 no_annotation_profile_file.GetFilename(),
1221 GetTestDexFileName("ProfileTestMultiDex")));
Alex Lightd83edf32021-02-22 15:07:18 -08001222
1223 // Dump the file back into text.
1224 std::string text_two;
1225 ASSERT_TRUE(DumpClassesAndMethods(with_annotation_profile_file.GetFilename(),
1226 &text_two,
1227 GetTestDexFileName("ProfileTestMultiDex")));
1228
1229 // Create another profile and save it to the disk as well.
1230 ScratchFile profile_two;
1231 ASSERT_TRUE(CreateProfile(
1232 text_two, profile_two.GetFilename(), GetTestDexFileName("ProfileTestMultiDex")));
Alex Lightd83edf32021-02-22 15:07:18 -08001233
1234 // These two profiles should be bit-identical.
1235 // TODO We could compare the 'text_two' to the methods but since the order is
1236 // arbitrary for many parts and there are multiple 'correct' dumps we'd need
1237 // to basically parse everything and this is simply easier.
1238 std::string error;
1239 std::vector<std::string> args { kIsTargetBuild ? "/system/bin/cmp" : "/usr/bin/cmp",
1240 "-s",
1241 no_annotation_profile_file.GetFilename(),
1242 profile_two.GetFilename() };
1243 ASSERT_EQ(ExecAndReturnCode(args, &error), 0) << error << " from " << text_two;
1244}
1245
Alex Lighta2f13192021-02-03 18:19:03 -08001246TEST_F(ProfileAssistantTest, TestProfileCreateInlineCache) {
1247 // Create the profile content.
1248 std::vector<std::string_view> methods = {
1249 "HLTestInline;->inlineMonomorphic(LSuper;)I+LSubA;",
1250 "HLTestInline;->inlinePolymorphic(LSuper;)I+LSubA;,LSubB;,LSubC;",
1251 "HLTestInline;->inlineMegamorphic(LSuper;)I+LSubA;,LSubB;,LSubC;,LSubD;,LSubE;",
1252 "HLTestInline;->inlineMissingTypes(LSuper;)I+missing_types",
1253 "HLTestInline;->noInlineCache(LSuper;)I",
Alex Light0b58ec52021-03-02 14:11:59 -08001254 "HLTestInline;->inlineMultiMonomorphic(LSuper;LSecret;)I+]LSuper;LSubA;]LSecret;LSubB;",
1255 "HLTestInline;->inlineMultiPolymorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
1256 "HLTestInline;->inlineMultiMegamorphic(LSuper;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;,LSubD;,LSubE;]LSecret;LSubA;,LSubB;,LSubC;,LSubD;,LSubE;",
1257 "HLTestInline;->inlineMultiMissingTypes(LSuper;LSecret;)I+]LSuper;missing_types]LSecret;missing_types",
1258 "HLTestInline;->inlineTriplePolymorphic(LSuper;LSecret;LSecret;)I+]LSuper;LSubA;,LSubB;,LSubC;]LSecret;LSubB;,LSubC;",
Alex Lighta2f13192021-02-03 18:19:03 -08001259 "HLTestInline;->noInlineCacheMulti(LSuper;LSecret;)I",
1260 };
1261 std::ostringstream input_file_contents;
1262 for (const std::string_view& m : methods) {
1263 input_file_contents << m << "\n";
1264 }
1265
1266 // Create the profile and save it to disk.
1267 ScratchFile profile_file;
1268 ASSERT_TRUE(CreateProfile(input_file_contents.str(),
Calin Juravlee0ac1152017-02-13 19:03:47 -08001269 profile_file.GetFilename(),
Calin Juravle1bfe4bd2018-04-26 16:00:11 -07001270 GetTestDexFileName("ProfileTestMultiDex")));
Calin Juravlee0ac1152017-02-13 19:03:47 -08001271
1272 // Load the profile from disk.
1273 ProfileCompilationInfo info;
Calin Juravlee0ac1152017-02-13 19:03:47 -08001274 ASSERT_TRUE(info.Load(GetFd(profile_file)));
1275
1276 // Load the dex files and verify that the profile contains the expected methods info.
1277 ScopedObjectAccess soa(Thread::Current());
1278 jobject class_loader = LoadDex("ProfileTestMultiDex");
1279 ASSERT_NE(class_loader, nullptr);
1280
Alex Lighta2f13192021-02-03 18:19:03 -08001281 StackHandleScope<5> hs(soa.Self());
1282 Handle<mirror::Class> super_klass = hs.NewHandle(GetClass(soa, class_loader, "LSuper;"));
1283 Handle<mirror::Class> secret_klass = hs.NewHandle(GetClass(soa, class_loader, "LSecret;"));
Vladimir Markoa8bba7d2018-05-30 15:18:48 +01001284 Handle<mirror::Class> sub_a = hs.NewHandle(GetClass(soa, class_loader, "LSubA;"));
1285 Handle<mirror::Class> sub_b = hs.NewHandle(GetClass(soa, class_loader, "LSubB;"));
1286 Handle<mirror::Class> sub_c = hs.NewHandle(GetClass(soa, class_loader, "LSubC;"));
Calin Juravlee0ac1152017-02-13 19:03:47 -08001287
Alex Lighta2f13192021-02-03 18:19:03 -08001288 ASSERT_TRUE(super_klass != nullptr);
1289 ASSERT_TRUE(secret_klass != nullptr);
Calin Juravlee0ac1152017-02-13 19:03:47 -08001290 ASSERT_TRUE(sub_a != nullptr);
1291 ASSERT_TRUE(sub_b != nullptr);
1292 ASSERT_TRUE(sub_c != nullptr);
1293
1294 {
1295 // Verify that method inlineMonomorphic has the expected inline caches and nothing else.
1296 ArtMethod* inline_monomorphic = GetVirtualMethod(class_loader,
1297 "LTestInline;",
1298 "inlineMonomorphic");
1299 ASSERT_TRUE(inline_monomorphic != nullptr);
Vladimir Markoa8bba7d2018-05-30 15:18:48 +01001300 TypeReferenceSet expected_monomorphic;
1301 expected_monomorphic.insert(MakeTypeReference(sub_a.Get()));
Calin Juravle589e71e2017-03-03 16:05:05 -08001302 AssertInlineCaches(inline_monomorphic,
1303 expected_monomorphic,
1304 info,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001305 /*is_megamorphic=*/false,
1306 /*is_missing_types=*/false);
Calin Juravlee0ac1152017-02-13 19:03:47 -08001307 }
1308
1309 {
1310 // Verify that method inlinePolymorphic has the expected inline caches and nothing else.
1311 ArtMethod* inline_polymorhic = GetVirtualMethod(class_loader,
1312 "LTestInline;",
1313 "inlinePolymorphic");
1314 ASSERT_TRUE(inline_polymorhic != nullptr);
Vladimir Markoa8bba7d2018-05-30 15:18:48 +01001315 TypeReferenceSet expected_polymorphic;
1316 expected_polymorphic.insert(MakeTypeReference(sub_a.Get()));
1317 expected_polymorphic.insert(MakeTypeReference(sub_b.Get()));
1318 expected_polymorphic.insert(MakeTypeReference(sub_c.Get()));
Calin Juravle589e71e2017-03-03 16:05:05 -08001319 AssertInlineCaches(inline_polymorhic,
1320 expected_polymorphic,
1321 info,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001322 /*is_megamorphic=*/false,
1323 /*is_missing_types=*/false);
Calin Juravlee0ac1152017-02-13 19:03:47 -08001324 }
1325
1326 {
1327 // Verify that method inlineMegamorphic has the expected inline caches and nothing else.
1328 ArtMethod* inline_megamorphic = GetVirtualMethod(class_loader,
1329 "LTestInline;",
1330 "inlineMegamorphic");
1331 ASSERT_TRUE(inline_megamorphic != nullptr);
Vladimir Markoa8bba7d2018-05-30 15:18:48 +01001332 TypeReferenceSet expected_megamorphic;
Calin Juravle589e71e2017-03-03 16:05:05 -08001333 AssertInlineCaches(inline_megamorphic,
1334 expected_megamorphic,
1335 info,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001336 /*is_megamorphic=*/true,
1337 /*is_missing_types=*/false);
Calin Juravle589e71e2017-03-03 16:05:05 -08001338 }
1339
1340 {
1341 // Verify that method inlineMegamorphic has the expected inline caches and nothing else.
1342 ArtMethod* inline_missing_types = GetVirtualMethod(class_loader,
1343 "LTestInline;",
1344 "inlineMissingTypes");
1345 ASSERT_TRUE(inline_missing_types != nullptr);
Vladimir Markoa8bba7d2018-05-30 15:18:48 +01001346 TypeReferenceSet expected_missing_Types;
Calin Juravle589e71e2017-03-03 16:05:05 -08001347 AssertInlineCaches(inline_missing_types,
1348 expected_missing_Types,
1349 info,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001350 /*is_megamorphic=*/false,
1351 /*is_missing_types=*/true);
Calin Juravlee0ac1152017-02-13 19:03:47 -08001352 }
1353
1354 {
1355 // Verify that method noInlineCache has no inline caches in the profile.
1356 ArtMethod* no_inline_cache = GetVirtualMethod(class_loader, "LTestInline;", "noInlineCache");
1357 ASSERT_TRUE(no_inline_cache != nullptr);
Vladimir Markoa64c1ad2021-03-08 14:27:05 +00001358 ProfileCompilationInfo::MethodHotness hotness_no_inline_cache = info.GetMethodHotness(
1359 MethodReference(no_inline_cache->GetDexFile(), no_inline_cache->GetDexMethodIndex()));
1360 ASSERT_TRUE(hotness_no_inline_cache.IsHot());
1361 ASSERT_TRUE(hotness_no_inline_cache.GetInlineCacheMap()->empty());
Calin Juravlee0ac1152017-02-13 19:03:47 -08001362 }
Alex Lighta2f13192021-02-03 18:19:03 -08001363
1364 {
1365 // Verify that method inlineMonomorphic has the expected inline caches and nothing else.
1366 ArtMethod* inline_monomorphic = GetVirtualMethod(class_loader,
1367 "LTestInline;",
1368 "inlineMultiMonomorphic");
1369 ASSERT_TRUE(inline_monomorphic != nullptr);
1370 TypeReferenceSet expected_monomorphic_super;
1371 TypeReferenceSet expected_monomorphic_secret;
1372 expected_monomorphic_super.insert(MakeTypeReference(sub_a.Get()));
1373 expected_monomorphic_secret.insert(MakeTypeReference(sub_b.Get()));
1374 AssertInlineCaches(inline_monomorphic,
1375 GetDexPcOfCallTo(inline_monomorphic, super_klass),
1376 expected_monomorphic_super,
1377 info,
1378 /*is_megamorphic=*/false,
1379 /*is_missing_types=*/false);
1380 AssertInlineCaches(inline_monomorphic,
1381 GetDexPcOfCallTo(inline_monomorphic, secret_klass),
1382 expected_monomorphic_secret,
1383 info,
1384 /*is_megamorphic=*/false,
1385 /*is_missing_types=*/false);
1386 }
1387
1388 {
1389 // Verify that method inlinePolymorphic has the expected inline caches and nothing else.
1390 ArtMethod* inline_polymorhic = GetVirtualMethod(class_loader,
1391 "LTestInline;",
1392 "inlineMultiPolymorphic");
1393 ASSERT_TRUE(inline_polymorhic != nullptr);
1394 TypeReferenceSet expected_polymorphic_super;
1395 expected_polymorphic_super.insert(MakeTypeReference(sub_a.Get()));
1396 expected_polymorphic_super.insert(MakeTypeReference(sub_b.Get()));
1397 expected_polymorphic_super.insert(MakeTypeReference(sub_c.Get()));
1398 TypeReferenceSet expected_polymorphic_secret;
1399 expected_polymorphic_secret.insert(MakeTypeReference(sub_b.Get()));
1400 expected_polymorphic_secret.insert(MakeTypeReference(sub_c.Get()));
1401 AssertInlineCaches(inline_polymorhic,
1402 GetDexPcOfCallTo(inline_polymorhic, super_klass),
1403 expected_polymorphic_super,
1404 info,
1405 /*is_megamorphic=*/false,
1406 /*is_missing_types=*/false);
1407 AssertInlineCaches(inline_polymorhic,
1408 GetDexPcOfCallTo(inline_polymorhic, secret_klass),
1409 expected_polymorphic_secret,
1410 info,
1411 /*is_megamorphic=*/false,
1412 /*is_missing_types=*/false);
1413 }
1414
1415 {
1416 // Verify that method inlinePolymorphic has the expected inline caches and nothing else.
1417 ArtMethod* inline_polymorhic = GetVirtualMethod(class_loader,
1418 "LTestInline;",
1419 "inlineTriplePolymorphic");
1420 ASSERT_TRUE(inline_polymorhic != nullptr);
1421 TypeReferenceSet expected_polymorphic_super;
1422 expected_polymorphic_super.insert(MakeTypeReference(sub_a.Get()));
1423 expected_polymorphic_super.insert(MakeTypeReference(sub_b.Get()));
1424 expected_polymorphic_super.insert(MakeTypeReference(sub_c.Get()));
1425 TypeReferenceSet expected_polymorphic_secret;
1426 expected_polymorphic_secret.insert(MakeTypeReference(sub_b.Get()));
1427 expected_polymorphic_secret.insert(MakeTypeReference(sub_c.Get()));
1428 AssertInlineCaches(inline_polymorhic,
1429 GetDexPcOfCallTo(inline_polymorhic, super_klass),
1430 expected_polymorphic_super,
1431 info,
1432 /*is_megamorphic=*/false,
1433 /*is_missing_types=*/false);
1434 uint16_t first_call = GetDexPcOfCallTo(inline_polymorhic, secret_klass);
1435 AssertInlineCaches(inline_polymorhic,
1436 first_call,
1437 expected_polymorphic_secret,
1438 info,
1439 /*is_megamorphic=*/false,
1440 /*is_missing_types=*/false);
1441 uint16_t second_call = GetDexPcOfCallTo(inline_polymorhic, secret_klass, first_call);
1442 ASSERT_LT(first_call, second_call);
1443 AssertInlineCaches(inline_polymorhic,
1444 second_call,
1445 expected_polymorphic_secret,
1446 info,
1447 /*is_megamorphic=*/false,
1448 /*is_missing_types=*/false);
1449 }
1450
1451 {
1452 // Verify that method inlineMegamorphic has the expected inline caches and nothing else.
1453 ArtMethod* inline_megamorphic = GetVirtualMethod(class_loader,
1454 "LTestInline;",
1455 "inlineMultiMegamorphic");
1456 ASSERT_TRUE(inline_megamorphic != nullptr);
1457 TypeReferenceSet expected_megamorphic;
1458 AssertInlineCaches(inline_megamorphic,
1459 GetDexPcOfCallTo(inline_megamorphic, super_klass),
1460 expected_megamorphic,
1461 info,
1462 /*is_megamorphic=*/true,
1463 /*is_missing_types=*/false);
1464 AssertInlineCaches(inline_megamorphic,
1465 GetDexPcOfCallTo(inline_megamorphic, secret_klass),
1466 expected_megamorphic,
1467 info,
1468 /*is_megamorphic=*/true,
1469 /*is_missing_types=*/false);
1470 }
1471
1472 {
1473 // Verify that method inlineMegamorphic has the expected inline caches and nothing else.
1474 ArtMethod* inline_missing_types = GetVirtualMethod(class_loader,
1475 "LTestInline;",
1476 "inlineMultiMissingTypes");
1477 ASSERT_TRUE(inline_missing_types != nullptr);
1478 TypeReferenceSet expected_missing_Types;
1479 AssertInlineCaches(inline_missing_types,
1480 GetDexPcOfCallTo(inline_missing_types, super_klass),
1481 expected_missing_Types,
1482 info,
1483 /*is_megamorphic=*/false,
1484 /*is_missing_types=*/true);
1485 AssertInlineCaches(inline_missing_types,
1486 GetDexPcOfCallTo(inline_missing_types, secret_klass),
1487 expected_missing_Types,
1488 info,
1489 /*is_megamorphic=*/false,
1490 /*is_missing_types=*/true);
1491 }
1492
1493 {
1494 // Verify that method noInlineCacheMulti has no inline caches in the profile.
1495 ArtMethod* no_inline_cache =
1496 GetVirtualMethod(class_loader, "LTestInline;", "noInlineCacheMulti");
1497 ASSERT_TRUE(no_inline_cache != nullptr);
Vladimir Markoa64c1ad2021-03-08 14:27:05 +00001498 ProfileCompilationInfo::MethodHotness hotness_no_inline_cache = info.GetMethodHotness(
1499 MethodReference(no_inline_cache->GetDexFile(), no_inline_cache->GetDexMethodIndex()));
1500 ASSERT_TRUE(hotness_no_inline_cache.IsHot());
1501 ASSERT_TRUE(hotness_no_inline_cache.GetInlineCacheMap()->empty());
Alex Lighta2f13192021-02-03 18:19:03 -08001502 }
Calin Juravlee0ac1152017-02-13 19:03:47 -08001503}
1504
Calin Juravlecea9e9d2017-03-23 19:04:59 -07001505TEST_F(ProfileAssistantTest, MergeProfilesWithDifferentDexOrder) {
1506 ScratchFile profile1;
1507 ScratchFile reference_profile;
1508
1509 std::vector<int> profile_fds({GetFd(profile1)});
1510 int reference_profile_fd = GetFd(reference_profile);
1511
1512 // The new profile info will contain the methods with indices 0-100.
1513 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
1514 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001515 SetupProfile(dex1, dex2, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001516 /*start_method_index=*/0, /*reverse_dex_write_order=*/false);
Calin Juravlecea9e9d2017-03-23 19:04:59 -07001517
1518 // The reference profile info will contain the methods with indices 50-150.
1519 // When setting up the profile reverse the order in which the dex files
1520 // are added to the profile. This will verify that profman merges profiles
1521 // with a different dex order correctly.
1522 const uint16_t kNumberOfMethodsAlreadyCompiled = 100;
1523 ProfileCompilationInfo reference_info;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001524 SetupProfile(dex1, dex2, kNumberOfMethodsAlreadyCompiled, 0, reference_profile,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001525 &reference_info, kNumberOfMethodsToEnableCompilation / 2, /*reverse_dex_write_order=*/true);
Calin Juravlecea9e9d2017-03-23 19:04:59 -07001526
1527 // We should advise compilation.
1528 ASSERT_EQ(ProfileAssistant::kCompile,
1529 ProcessProfiles(profile_fds, reference_profile_fd));
1530
1531 // The resulting compilation info must be equal to the merge of the inputs.
1532 ProfileCompilationInfo result;
Calin Juravlecea9e9d2017-03-23 19:04:59 -07001533 ASSERT_TRUE(result.Load(reference_profile_fd));
1534
1535 ProfileCompilationInfo expected;
1536 ASSERT_TRUE(expected.MergeWith(reference_info));
1537 ASSERT_TRUE(expected.MergeWith(info1));
1538 ASSERT_TRUE(expected.Equals(result));
1539
1540 // The information from profile must remain the same.
1541 CheckProfileInfo(profile1, info1);
1542}
1543
Alex Lightcff30a02021-03-25 17:33:32 -07001544TEST_F(ProfileAssistantTest, TestProfileCreateWithSubtype) {
1545 // Create the profile content.
1546 std::vector<std::string> profile_methods = {
1547 "HLTestInlineSubtype;->inlineMonomorphic(LSuper;)I+]LSuper;LSubA;",
1548 };
1549 std::string input_file_contents;
1550 for (std::string& m : profile_methods) {
1551 input_file_contents += m + std::string("\n");
1552 }
1553
1554 // Create the profile and save it to disk.
1555 ScratchFile profile_file;
1556 std::string dex_filename = GetTestDexFileName("ProfileTestMultiDex");
1557 ASSERT_TRUE(CreateProfile(input_file_contents, profile_file.GetFilename(), dex_filename));
1558
1559 // Load the profile from disk.
1560 ProfileCompilationInfo info;
Alex Lightcff30a02021-03-25 17:33:32 -07001561 ASSERT_TRUE(info.Load(GetFd(profile_file)));
1562 LOG(ERROR) << profile_file.GetFilename();
1563
1564 // Load the dex files and verify that the profile contains the expected
1565 // methods info.
1566 ScopedObjectAccess soa(Thread::Current());
1567 jobject class_loader = LoadDex("ProfileTestMultiDex");
1568 ASSERT_NE(class_loader, nullptr);
1569
1570 // NB This is the supertype of the declared line!
1571 ArtMethod* inline_monomorphic_super =
1572 GetVirtualMethod(class_loader, "LTestInline;", "inlineMonomorphic");
1573 const DexFile* dex_file = inline_monomorphic_super->GetDexFile();
1574
1575 // Verify that the inline cache is present in the superclass
1576 ProfileCompilationInfo::MethodHotness hotness_super = info.GetMethodHotness(
1577 MethodReference(dex_file, inline_monomorphic_super->GetDexMethodIndex()));
1578 ASSERT_TRUE(hotness_super.IsHot());
1579 const ProfileCompilationInfo::InlineCacheMap* inline_caches = hotness_super.GetInlineCacheMap();
1580 ASSERT_EQ(inline_caches->size(), 1u);
1581 const ProfileCompilationInfo::DexPcData& dex_pc_data = inline_caches->begin()->second;
1582 dex::TypeIndex target_type_index(dex_file->GetIndexForTypeId(*dex_file->FindTypeId("LSubA;")));
1583 ASSERT_EQ(1u, dex_pc_data.classes.size());
Vladimir Markoc63d9672021-03-31 15:50:39 +01001584 ASSERT_EQ(target_type_index, *dex_pc_data.classes.begin());
Alex Lightcff30a02021-03-25 17:33:32 -07001585
1586 // Verify that the method is present in subclass but there are no
1587 // inline-caches (since there is no code).
1588 const dex::MethodId& super_method_id =
1589 dex_file->GetMethodId(inline_monomorphic_super->GetDexMethodIndex());
1590 uint32_t sub_method_index = dex_file->GetIndexForMethodId(
1591 *dex_file->FindMethodId(*dex_file->FindTypeId("LTestInlineSubtype;"),
1592 dex_file->GetStringId(super_method_id.name_idx_),
1593 dex_file->GetProtoId(super_method_id.proto_idx_)));
1594 ProfileCompilationInfo::MethodHotness hotness_sub =
1595 info.GetMethodHotness(MethodReference(dex_file, sub_method_index));
1596 ASSERT_TRUE(hotness_sub.IsHot());
1597 ASSERT_EQ(hotness_sub.GetInlineCacheMap()->size(), 0u);
1598}
1599
1600TEST_F(ProfileAssistantTest, TestProfileCreateWithSubtypeAndDump) {
1601 // Create the profile content.
1602 std::vector<std::string> profile_methods = {
1603 "HLTestInlineSubtype;->inlineMonomorphic(LSuper;)I+]LSuper;LSubA;",
1604 };
1605 std::string input_file_contents;
1606 for (std::string& m : profile_methods) {
1607 input_file_contents += m + std::string("\n");
1608 }
1609
1610 // Create the profile and save it to disk.
1611 ScratchFile profile_file;
1612 std::string dex_filename = GetTestDexFileName("ProfileTestMultiDex");
1613 ASSERT_TRUE(CreateProfile(input_file_contents, profile_file.GetFilename(), dex_filename));
1614
1615 std::string dump_ic;
1616 ASSERT_TRUE(DumpClassesAndMethods(
1617 profile_file.GetFilename(), &dump_ic, GetTestDexFileName("ProfileTestMultiDex")));
1618
1619 std::vector<std::string> lines;
1620 std::stringstream dump_stream(dump_ic);
1621 std::string cur;
1622 while (std::getline(dump_stream, cur, '\n')) {
1623 lines.push_back(std::move(cur));
1624 }
1625
1626 EXPECT_EQ(lines.size(), 2u);
1627 EXPECT_TRUE(std::find(lines.cbegin(),
1628 lines.cend(),
1629 "HLTestInline;->inlineMonomorphic(LSuper;)I+]LSuper;LSubA;") !=
1630 lines.cend());
1631 EXPECT_TRUE(std::find(lines.cbegin(),
1632 lines.cend(),
1633 "HLTestInlineSubtype;->inlineMonomorphic(LSuper;)I") != lines.cend());
1634}
1635
Calin Juravle08556882017-05-26 16:40:45 -07001636TEST_F(ProfileAssistantTest, TestProfileCreateWithInvalidData) {
1637 // Create the profile content.
1638 std::vector<std::string> profile_methods = {
Vladimir Markoc63d9672021-03-31 15:50:39 +01001639 "HLTestInline;->inlineMonomorphic(LSuper;)I+invalid_class", // Invalid descriptor for IC.
1640 "HLTestInline;->invalid_method", // Invalid method spec (no signature).
1641 "invalid_class", // Invalid descriptor.
Calin Juravle08556882017-05-26 16:40:45 -07001642 };
1643 std::string input_file_contents;
1644 for (std::string& m : profile_methods) {
1645 input_file_contents += m + std::string("\n");
1646 }
1647
1648 // Create the profile and save it to disk.
1649 ScratchFile profile_file;
1650 std::string dex_filename = GetTestDexFileName("ProfileTestMultiDex");
1651 ASSERT_TRUE(CreateProfile(input_file_contents,
1652 profile_file.GetFilename(),
Calin Juravle1bfe4bd2018-04-26 16:00:11 -07001653 dex_filename));
Calin Juravle08556882017-05-26 16:40:45 -07001654
1655 // Load the profile from disk.
1656 ProfileCompilationInfo info;
Calin Juravle08556882017-05-26 16:40:45 -07001657 ASSERT_TRUE(info.Load(GetFd(profile_file)));
1658
1659 // Load the dex files and verify that the profile contains the expected methods info.
1660 ScopedObjectAccess soa(Thread::Current());
1661 jobject class_loader = LoadDex("ProfileTestMultiDex");
1662 ASSERT_NE(class_loader, nullptr);
1663
1664 ArtMethod* inline_monomorphic = GetVirtualMethod(class_loader,
1665 "LTestInline;",
1666 "inlineMonomorphic");
1667 const DexFile* dex_file = inline_monomorphic->GetDexFile();
1668
Vladimir Markoc63d9672021-03-31 15:50:39 +01001669 // Invalid descriptor in IC results in rejection of the entire line.
Vladimir Markoa64c1ad2021-03-08 14:27:05 +00001670 ProfileCompilationInfo::MethodHotness hotness =
1671 info.GetMethodHotness(MethodReference(dex_file, inline_monomorphic->GetDexMethodIndex()));
Vladimir Markoc63d9672021-03-31 15:50:39 +01001672 ASSERT_FALSE(hotness.IsHot());
Calin Juravle08556882017-05-26 16:40:45 -07001673
Vladimir Markoc63d9672021-03-31 15:50:39 +01001674 // No data was recorded, so the dex file does not appear in the profile.
1675 // TODO: Record all dex files passed to `profman` in the profile. Note that
1676 // this makes sense only if there are no annotations, otherwise we do not
1677 // know what annotation to use with each dex file.
Calin Juravle08556882017-05-26 16:40:45 -07001678 std::set<dex::TypeIndex> classes;
Mathieu Chartierea650f32017-05-24 12:04:13 -07001679 std::set<uint16_t> hot_methods;
1680 std::set<uint16_t> startup_methods;
1681 std::set<uint16_t> post_start_methods;
Vladimir Markoc63d9672021-03-31 15:50:39 +01001682 ASSERT_FALSE(info.GetClassesAndMethods(*dex_file,
1683 &classes,
1684 &hot_methods,
1685 &startup_methods,
1686 &post_start_methods));
Calin Juravle08556882017-05-26 16:40:45 -07001687}
1688
Mathieu Chartier28b5c582017-06-06 14:12:50 -07001689TEST_F(ProfileAssistantTest, DumpOnly) {
1690 ScratchFile profile;
1691
1692 const uint32_t kNumberOfMethods = 64;
1693 std::vector<uint32_t> hot_methods;
1694 std::vector<uint32_t> startup_methods;
1695 std::vector<uint32_t> post_startup_methods;
1696 for (size_t i = 0; i < kNumberOfMethods; ++i) {
1697 if (i % 2 == 0) {
1698 hot_methods.push_back(i);
1699 }
1700 if (i % 3 == 1) {
1701 startup_methods.push_back(i);
1702 }
1703 if (i % 4 == 2) {
1704 post_startup_methods.push_back(i);
1705 }
1706 }
1707 EXPECT_GT(hot_methods.size(), 0u);
1708 EXPECT_GT(startup_methods.size(), 0u);
1709 EXPECT_GT(post_startup_methods.size(), 0u);
1710 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001711 SetupBasicProfile(dex1,
Mathieu Chartier28b5c582017-06-06 14:12:50 -07001712 hot_methods,
1713 startup_methods,
1714 post_startup_methods,
1715 profile,
1716 &info1);
1717 std::string output;
1718 DumpOnly(profile.GetFilename(), &output);
1719 const size_t hot_offset = output.find("hot methods:");
1720 const size_t startup_offset = output.find("startup methods:");
1721 const size_t post_startup_offset = output.find("post startup methods:");
1722 const size_t classes_offset = output.find("classes:");
1723 ASSERT_NE(hot_offset, std::string::npos);
1724 ASSERT_NE(startup_offset, std::string::npos);
1725 ASSERT_NE(post_startup_offset, std::string::npos);
1726 ASSERT_LT(hot_offset, startup_offset);
1727 ASSERT_LT(startup_offset, post_startup_offset);
1728 // Check the actual contents of the dump by looking at the offsets of the methods.
1729 for (uint32_t m : hot_methods) {
1730 const size_t pos = output.find(std::to_string(m) + "[],", hot_offset);
Mathieu Chartierf120ffc2018-04-23 11:27:31 -07001731 ASSERT_NE(pos, std::string::npos) << output;
1732 EXPECT_LT(pos, startup_offset) << output;
Mathieu Chartier28b5c582017-06-06 14:12:50 -07001733 }
1734 for (uint32_t m : startup_methods) {
1735 const size_t pos = output.find(std::to_string(m) + ",", startup_offset);
Mathieu Chartierf120ffc2018-04-23 11:27:31 -07001736 ASSERT_NE(pos, std::string::npos) << output;
1737 EXPECT_LT(pos, post_startup_offset) << output;
Mathieu Chartier28b5c582017-06-06 14:12:50 -07001738 }
1739 for (uint32_t m : post_startup_methods) {
1740 const size_t pos = output.find(std::to_string(m) + ",", post_startup_offset);
Mathieu Chartierf120ffc2018-04-23 11:27:31 -07001741 ASSERT_NE(pos, std::string::npos) << output;
1742 EXPECT_LT(pos, classes_offset) << output;
Mathieu Chartier28b5c582017-06-06 14:12:50 -07001743 }
1744}
1745
Calin Juravlee10c1e22018-01-26 20:10:15 -08001746TEST_F(ProfileAssistantTest, MergeProfilesWithFilter) {
1747 ScratchFile profile1;
1748 ScratchFile profile2;
1749 ScratchFile reference_profile;
1750
1751 std::vector<int> profile_fds({
1752 GetFd(profile1),
1753 GetFd(profile2)});
1754 int reference_profile_fd = GetFd(reference_profile);
1755
1756 // Use a real dex file to generate profile test data.
1757 // The file will be used during merging to filter unwanted data.
1758 std::vector<std::unique_ptr<const DexFile>> dex_files = OpenTestDexFiles("ProfileTestMultiDex");
1759 const DexFile& d1 = *dex_files[0];
1760 const DexFile& d2 = *dex_files[1];
1761 // The new profile info will contain the methods with indices 0-100.
1762 const uint16_t kNumberOfMethodsToEnableCompilation = 100;
1763 ProfileCompilationInfo info1;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001764 SetupProfile(&d1, dex1, kNumberOfMethodsToEnableCompilation, 0, profile1, &info1);
Calin Juravlee10c1e22018-01-26 20:10:15 -08001765 ProfileCompilationInfo info2;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001766 SetupProfile(&d2, dex2, kNumberOfMethodsToEnableCompilation, 0, profile2, &info2);
Calin Juravlee10c1e22018-01-26 20:10:15 -08001767
1768
1769 // The reference profile info will contain the methods with indices 50-150.
1770 const uint16_t kNumberOfMethodsAlreadyCompiled = 100;
1771 ProfileCompilationInfo reference_info;
Calin Juravle1061c7d2019-09-23 21:00:29 -04001772 SetupProfile(&d1, dex1,
Calin Juravlee10c1e22018-01-26 20:10:15 -08001773 kNumberOfMethodsAlreadyCompiled, 0, reference_profile,
1774 &reference_info, kNumberOfMethodsToEnableCompilation / 2);
1775
1776 // Run profman and pass the dex file with --apk-fd.
1777 android::base::unique_fd apk_fd(
Andreas Gampedfcd82c2018-10-16 20:22:37 -07001778 open(GetTestDexFileName("ProfileTestMultiDex").c_str(), O_RDONLY)); // NOLINT
Calin Juravlee10c1e22018-01-26 20:10:15 -08001779 ASSERT_GE(apk_fd.get(), 0);
1780
1781 std::string profman_cmd = GetProfmanCmd();
1782 std::vector<std::string> argv_str;
1783 argv_str.push_back(profman_cmd);
1784 argv_str.push_back("--profile-file-fd=" + std::to_string(profile1.GetFd()));
1785 argv_str.push_back("--profile-file-fd=" + std::to_string(profile2.GetFd()));
1786 argv_str.push_back("--reference-profile-file-fd=" + std::to_string(reference_profile.GetFd()));
1787 argv_str.push_back("--apk-fd=" + std::to_string(apk_fd.get()));
1788 std::string error;
1789
Calin Juravlec0200a92019-11-14 09:01:11 -08001790 EXPECT_EQ(ExecAndReturnCode(argv_str, &error), ProfileAssistant::kCompile) << error;
Calin Juravlee10c1e22018-01-26 20:10:15 -08001791
1792 // Verify that we can load the result.
1793
1794 ProfileCompilationInfo result;
Calin Juravlee10c1e22018-01-26 20:10:15 -08001795 ASSERT_TRUE(result.Load(reference_profile_fd));
1796
Calin Juravlee10c1e22018-01-26 20:10:15 -08001797 // Verify that the result filtered out data not belonging to the dex file.
1798 // This is equivalent to checking that the result is equal to the merging of
1799 // all profiles while filtering out data not belonging to the dex file.
1800
1801 ProfileCompilationInfo::ProfileLoadFilterFn filter_fn =
1802 [&d1, &d2](const std::string& dex_location, uint32_t checksum) -> bool {
Calin Juravle849439a2019-09-16 15:09:16 -07001803 return (dex_location == ProfileCompilationInfo::GetProfileDexFileBaseKey(d1.GetLocation())
Calin Juravlee10c1e22018-01-26 20:10:15 -08001804 && checksum == d1.GetLocationChecksum())
Calin Juravle849439a2019-09-16 15:09:16 -07001805 || (dex_location == ProfileCompilationInfo::GetProfileDexFileBaseKey(d2.GetLocation())
Calin Juravlee10c1e22018-01-26 20:10:15 -08001806 && checksum == d2.GetLocationChecksum());
1807 };
1808
1809 ProfileCompilationInfo info1_filter;
1810 ProfileCompilationInfo info2_filter;
1811 ProfileCompilationInfo expected;
1812
Andreas Gampe9b031f72018-10-04 11:03:34 -07001813 info2_filter.Load(profile1.GetFd(), /*merge_classes=*/ true, filter_fn);
1814 info2_filter.Load(profile2.GetFd(), /*merge_classes=*/ true, filter_fn);
1815 expected.Load(reference_profile.GetFd(), /*merge_classes=*/ true, filter_fn);
Calin Juravlee10c1e22018-01-26 20:10:15 -08001816
1817 ASSERT_TRUE(expected.MergeWith(info1_filter));
1818 ASSERT_TRUE(expected.MergeWith(info2_filter));
1819
1820 ASSERT_TRUE(expected.Equals(result));
1821}
1822
Calin Juravle02c08792018-02-15 19:40:48 -08001823TEST_F(ProfileAssistantTest, CopyAndUpdateProfileKey) {
1824 ScratchFile profile1;
1825 ScratchFile reference_profile;
1826
1827 // Use a real dex file to generate profile test data. During the copy-and-update the
1828 // matching is done based on checksum so we have to match with the real thing.
1829 std::vector<std::unique_ptr<const DexFile>> dex_files = OpenTestDexFiles("ProfileTestMultiDex");
1830 const DexFile& d1 = *dex_files[0];
1831 const DexFile& d2 = *dex_files[1];
1832
1833 ProfileCompilationInfo info1;
1834 uint16_t num_methods_to_add = std::min(d1.NumMethodIds(), d2.NumMethodIds());
Calin Juravle1061c7d2019-09-23 21:00:29 -04001835
Vladimir Markoc63d9672021-03-31 15:50:39 +01001836 const DexFile* dex_to_be_updated1 = BuildDex(
1837 "fake-location1", d1.GetLocationChecksum(), "LC;", d1.NumMethodIds(), d1.NumTypeIds());
1838 const DexFile* dex_to_be_updated2 = BuildDex(
1839 "fake-location2", d2.GetLocationChecksum(), "LC;", d2.NumMethodIds(), d2.NumTypeIds());
Calin Juravle1061c7d2019-09-23 21:00:29 -04001840 SetupProfile(dex_to_be_updated1,
1841 dex_to_be_updated2,
Calin Juravle02c08792018-02-15 19:40:48 -08001842 num_methods_to_add,
Andreas Gampe9b031f72018-10-04 11:03:34 -07001843 /*number_of_classes=*/ 0,
Calin Juravle02c08792018-02-15 19:40:48 -08001844 profile1,
Calin Juravle1061c7d2019-09-23 21:00:29 -04001845 &info1);
Calin Juravle02c08792018-02-15 19:40:48 -08001846
1847 // Run profman and pass the dex file with --apk-fd.
1848 android::base::unique_fd apk_fd(
Andreas Gampedfcd82c2018-10-16 20:22:37 -07001849 open(GetTestDexFileName("ProfileTestMultiDex").c_str(), O_RDONLY)); // NOLINT
Calin Juravle02c08792018-02-15 19:40:48 -08001850 ASSERT_GE(apk_fd.get(), 0);
1851
1852 std::string profman_cmd = GetProfmanCmd();
1853 std::vector<std::string> argv_str;
1854 argv_str.push_back(profman_cmd);
1855 argv_str.push_back("--profile-file-fd=" + std::to_string(profile1.GetFd()));
1856 argv_str.push_back("--reference-profile-file-fd=" + std::to_string(reference_profile.GetFd()));
1857 argv_str.push_back("--apk-fd=" + std::to_string(apk_fd.get()));
1858 argv_str.push_back("--copy-and-update-profile-key");
1859 std::string error;
1860
1861 ASSERT_EQ(ExecAndReturnCode(argv_str, &error), 0) << error;
1862
1863 // Verify that we can load the result.
1864 ProfileCompilationInfo result;
Calin Juravle02c08792018-02-15 19:40:48 -08001865 ASSERT_TRUE(result.Load(reference_profile.GetFd()));
1866
1867 // Verify that the renaming was done.
1868 for (uint16_t i = 0; i < num_methods_to_add; i ++) {
Vladimir Markoa64c1ad2021-03-08 14:27:05 +00001869 ASSERT_TRUE(result.GetMethodHotness(MethodReference(&d1, i)).IsHot()) << i;
1870 ASSERT_TRUE(result.GetMethodHotness(MethodReference(&d2, i)).IsHot()) << i;
Calin Juravle02c08792018-02-15 19:40:48 -08001871
Vladimir Markoa64c1ad2021-03-08 14:27:05 +00001872 ASSERT_FALSE(result.GetMethodHotness(MethodReference(dex_to_be_updated1, i)).IsHot()) << i;
1873 ASSERT_FALSE(result.GetMethodHotness(MethodReference(dex_to_be_updated2, i)).IsHot()) << i;
Calin Juravle02c08792018-02-15 19:40:48 -08001874 }
1875}
1876
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001877TEST_F(ProfileAssistantTest, BootImageMerge) {
1878 ScratchFile profile;
1879 ScratchFile reference_profile;
1880 std::vector<int> profile_fds({GetFd(profile)});
1881 int reference_profile_fd = GetFd(reference_profile);
1882 std::vector<uint32_t> hot_methods_cur;
1883 std::vector<uint32_t> hot_methods_ref;
1884 std::vector<uint32_t> empty_vector;
1885 size_t num_methods = 100;
1886 for (size_t i = 0; i < num_methods; ++i) {
1887 hot_methods_cur.push_back(i);
1888 }
1889 for (size_t i = 0; i < num_methods; ++i) {
1890 hot_methods_ref.push_back(i);
1891 }
Vladimir Markoc63d9672021-03-31 15:50:39 +01001892 ProfileCompilationInfo info1(/*for_boot_image=*/ true);
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001893 SetupBasicProfile(dex1, hot_methods_cur, empty_vector, empty_vector,
1894 profile, &info1);
1895 ProfileCompilationInfo info2(/*for_boot_image=*/true);
1896 SetupBasicProfile(dex1, hot_methods_ref, empty_vector, empty_vector,
1897 reference_profile, &info2);
1898
1899 std::vector<const std::string> extra_args({"--force-merge", "--boot-image-merge"});
1900
1901 int return_code = ProcessProfiles(profile_fds, reference_profile_fd, extra_args);
1902
1903 ASSERT_EQ(return_code, ProfileAssistant::kSuccess);
1904
1905 // Verify the result: it should be equal to info2 since info1 is a regular profile
1906 // and should be ignored.
Vladimir Markoc63d9672021-03-31 15:50:39 +01001907 ProfileCompilationInfo result(/*for_boot_image=*/ true);
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001908 ASSERT_TRUE(result.Load(reference_profile.GetFd()));
1909 ASSERT_TRUE(result.Equals(info2));
1910}
1911
1912// Under default behaviour we should not advice compilation
1913// and the reference profile should not be updated.
1914// However we pass --force-merge to force aggregation and in this case
1915// we should see an update.
1916TEST_F(ProfileAssistantTest, ForceMerge) {
1917 const uint16_t kNumberOfClassesInRefProfile = 6000;
1918 const uint16_t kNumberOfClassesInCurProfile = 6110; // Threshold is 2%.
1919
Vladimir Markoc63d9672021-03-31 15:50:39 +01001920 const DexFile* dex1_7000 = BuildDex("location1_7000",
1921 /*checksum=*/ 7001,
1922 "LUnique1_7000;",
1923 /*num_method_ids=*/ 0,
1924 /*num_type_ids=*/ 7000);
1925 const DexFile* dex2_7000 = BuildDex("location2_7000",
1926 /*checksum=*/ 7002,
1927 "LUnique2_7000;",
1928 /*num_method_ids=*/ 0,
1929 /*num_type_ids=*/ 7000);
1930
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001931 ScratchFile profile;
1932 ScratchFile reference_profile;
1933
1934 std::vector<int> profile_fds({ GetFd(profile)});
1935 int reference_profile_fd = GetFd(reference_profile);
1936
1937 ProfileCompilationInfo info1;
Vladimir Markoc63d9672021-03-31 15:50:39 +01001938 SetupProfile(dex1_7000, dex2_7000, 0, kNumberOfClassesInRefProfile, profile, &info1);
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001939 ProfileCompilationInfo info2;
Vladimir Markoc63d9672021-03-31 15:50:39 +01001940 SetupProfile(dex1_7000, dex2_7000, 0, kNumberOfClassesInCurProfile, reference_profile, &info2);
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001941
1942 std::vector<const std::string> extra_args({"--force-merge"});
1943 int return_code = ProcessProfiles(profile_fds, reference_profile_fd, extra_args);
1944
1945 ASSERT_EQ(return_code, ProfileAssistant::kSuccess);
1946
1947 // Check that the result is the aggregation.
1948 ProfileCompilationInfo result;
Calin Juravle0e82e0f2019-11-11 18:34:10 -08001949 ASSERT_TRUE(result.Load(reference_profile.GetFd()));
1950 ASSERT_TRUE(info1.MergeWith(info2));
1951 ASSERT_TRUE(result.Equals(info1));
1952}
Calin Juravlead88cbe2019-11-11 18:43:15 -08001953
1954// Test that we consider the annations when we merge boot image profiles.
1955TEST_F(ProfileAssistantTest, BootImageMergeWithAnnotations) {
1956 ScratchFile profile;
1957 ScratchFile reference_profile;
1958
1959 std::vector<int> profile_fds({GetFd(profile)});
1960 int reference_profile_fd = GetFd(reference_profile);
1961
1962 // Use a real dex file to generate profile test data so that we can pass descriptors to profman.
1963 std::vector<std::unique_ptr<const DexFile>> dex_files = OpenTestDexFiles("ProfileTestMultiDex");
1964 const DexFile& d1 = *dex_files[0];
1965 const DexFile& d2 = *dex_files[1];
1966 // The new profile info will contain the methods with indices 0-100.
Vladimir Marko95f4c662021-05-11 12:25:20 +01001967 ProfileCompilationInfo info(/*for_boot_image=*/ true);
Calin Juravlead88cbe2019-11-11 18:43:15 -08001968 ProfileCompilationInfo::ProfileSampleAnnotation psa1("package1");
1969 ProfileCompilationInfo::ProfileSampleAnnotation psa2("package2");
1970
1971 AddMethod(&info, &d1, 0, Hotness::kFlagHot, psa1);
1972 AddMethod(&info, &d2, 0, Hotness::kFlagHot, psa2);
1973 info.Save(profile.GetFd());
Calin Juravlead88cbe2019-11-11 18:43:15 -08001974
1975 // Run profman and pass the dex file with --apk-fd.
1976 android::base::unique_fd apk_fd(
1977 open(GetTestDexFileName("ProfileTestMultiDex").c_str(), O_RDONLY)); // NOLINT
1978 ASSERT_GE(apk_fd.get(), 0);
1979
1980 std::string profman_cmd = GetProfmanCmd();
1981 std::vector<std::string> argv_str;
1982 argv_str.push_back(profman_cmd);
1983 argv_str.push_back("--profile-file-fd=" + std::to_string(profile.GetFd()));
1984 argv_str.push_back("--reference-profile-file-fd=" + std::to_string(reference_profile.GetFd()));
1985 argv_str.push_back("--apk-fd=" + std::to_string(apk_fd.get()));
1986 argv_str.push_back("--force-merge");
1987 argv_str.push_back("--boot-image-merge");
1988 std::string error;
1989
1990 EXPECT_EQ(ExecAndReturnCode(argv_str, &error), ProfileAssistant::kSuccess) << error;
1991
1992 // Verify that we can load the result and that it equals to what we saved.
Vladimir Markoc63d9672021-03-31 15:50:39 +01001993 ProfileCompilationInfo result(/*for_boot_image=*/ true);
Calin Juravlead88cbe2019-11-11 18:43:15 -08001994 ASSERT_TRUE(result.Load(reference_profile_fd));
1995 ASSERT_TRUE(info.Equals(result));
1996}
Calin Juravlec0200a92019-11-14 09:01:11 -08001997
1998TEST_F(ProfileAssistantTest, DifferentProfileVersions) {
1999 ScratchFile profile1;
2000 ScratchFile profile2;
2001
Vladimir Marko95f4c662021-05-11 12:25:20 +01002002 ProfileCompilationInfo info1(/*for_boot_image=*/ false);
Calin Juravlec0200a92019-11-14 09:01:11 -08002003 info1.Save(profile1.GetFd());
Calin Juravlec0200a92019-11-14 09:01:11 -08002004
Vladimir Marko95f4c662021-05-11 12:25:20 +01002005 ProfileCompilationInfo info2(/*for_boot_image=*/ true);
Calin Juravlec0200a92019-11-14 09:01:11 -08002006 info2.Save(profile2.GetFd());
Calin Juravlec0200a92019-11-14 09:01:11 -08002007
2008 std::vector<int> profile_fds({ GetFd(profile1)});
2009 int reference_profile_fd = GetFd(profile2);
Vladimir Markoc63d9672021-03-31 15:50:39 +01002010 std::vector<const std::string> boot_image_args({"--boot-image-merge"});
2011 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd, boot_image_args),
Calin Juravlec0200a92019-11-14 09:01:11 -08002012 ProfileAssistant::kErrorDifferentVersions);
Vladimir Markoc63d9672021-03-31 15:50:39 +01002013 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd),
2014 ProfileAssistant::kErrorBadProfiles);
Calin Juravlec0200a92019-11-14 09:01:11 -08002015
2016 // Reverse the order of the profiles to verify we get the same behaviour.
2017 profile_fds[0] = GetFd(profile2);
2018 reference_profile_fd = GetFd(profile1);
Vladimir Markoc63d9672021-03-31 15:50:39 +01002019 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd, boot_image_args),
2020 ProfileAssistant::kErrorBadProfiles);
Calin Juravlec0200a92019-11-14 09:01:11 -08002021 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd),
2022 ProfileAssistant::kErrorDifferentVersions);
2023}
Calin Juravlee571a282019-12-03 18:36:01 -08002024
2025// Under default behaviour we will abort if we cannot load a profile during a merge
2026// operation. However, if we pass --force-merge to force aggregation we should
2027// ignore files we cannot load
2028TEST_F(ProfileAssistantTest, ForceMergeIgnoreProfilesItCannotLoad) {
2029 ScratchFile profile1;
2030 ScratchFile profile2;
2031
2032 // Write corrupt data in the first file.
2033 std::string content = "giberish";
2034 ASSERT_TRUE(profile1.GetFile()->WriteFully(content.c_str(), content.length()));
Calin Juravlee571a282019-12-03 18:36:01 -08002035
Vladimir Marko95f4c662021-05-11 12:25:20 +01002036 ProfileCompilationInfo info2(/*for_boot_image=*/ true);
Calin Juravlee571a282019-12-03 18:36:01 -08002037 info2.Save(profile2.GetFd());
Calin Juravlee571a282019-12-03 18:36:01 -08002038
2039 std::vector<int> profile_fds({ GetFd(profile1)});
2040 int reference_profile_fd = GetFd(profile2);
2041
2042 // With force-merge we should merge successfully.
Vladimir Markoc63d9672021-03-31 15:50:39 +01002043 std::vector<const std::string> extra_args({"--force-merge", "--boot-image-merge"});
Calin Juravlee571a282019-12-03 18:36:01 -08002044 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd, extra_args),
2045 ProfileAssistant::kSuccess);
2046
Vladimir Markoc63d9672021-03-31 15:50:39 +01002047 ProfileCompilationInfo result(/*for_boot_image=*/ true);
Calin Juravlee571a282019-12-03 18:36:01 -08002048 ASSERT_TRUE(result.Load(reference_profile_fd));
2049 ASSERT_TRUE(info2.Equals(result));
2050
2051 // Without force-merge we should fail.
Vladimir Markoc63d9672021-03-31 15:50:39 +01002052 std::vector<const std::string> extra_args2({"--boot-image-merge"});
2053 ASSERT_EQ(ProcessProfiles(profile_fds, reference_profile_fd, extra_args2),
Calin Juravlee571a282019-12-03 18:36:01 -08002054 ProfileAssistant::kErrorBadProfiles);
2055}
Vladimir Markoc63d9672021-03-31 15:50:39 +01002056
Calin Juravle2e2db782016-02-23 12:00:03 +00002057} // namespace art