1 /*
2 * Copyright (C) 2017 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 "boot_image_profile.h"
18
19 #include <memory>
20 #include <set>
21
22 #include "android-base/file.h"
23 #include "base/unix_file/fd_file.h"
24 #include "dex/class_accessor-inl.h"
25 #include "dex/descriptors_names.h"
26 #include "dex/dex_file-inl.h"
27 #include "dex/method_reference.h"
28 #include "dex/type_reference.h"
29 #include "profile/profile_compilation_info.h"
30
31 namespace art {
32
33 using Hotness = ProfileCompilationInfo::MethodHotness;
34
35 static const std::string kMethodSep = "->"; // NOLINT [runtime/string] [4]
36 static const std::string kPackageUseDelim = "@"; // NOLINT [runtime/string] [4]
37 static constexpr char kMethodFlagStringHot = 'H';
38 static constexpr char kMethodFlagStringStartup = 'S';
39 static constexpr char kMethodFlagStringPostStartup = 'P';
40
41 // Returns the type descriptor of the given reference.
GetTypeDescriptor(const TypeReference & ref)42 static std::string GetTypeDescriptor(const TypeReference& ref) {
43 const dex::TypeId& type_id = ref.dex_file->GetTypeId(ref.TypeIndex());
44 return ref.dex_file->GetTypeDescriptor(type_id);
45 }
46
47 // Returns the method representation used in the text format of the boot image profile.
BootImageRepresentation(const MethodReference & ref)48 static std::string BootImageRepresentation(const MethodReference& ref) {
49 const DexFile* dex_file = ref.dex_file;
50 const dex::MethodId& id = ref.GetMethodId();
51 std::string signature_string(dex_file->GetMethodSignature(id).ToString());
52 std::string type_string(dex_file->GetTypeDescriptor(dex_file->GetTypeId(id.class_idx_)));
53 std::string method_name(dex_file->GetMethodName(id));
54 return type_string +
55 kMethodSep +
56 method_name +
57 signature_string;
58 }
59
60 // Returns the class representation used in the text format of the boot image profile.
BootImageRepresentation(const TypeReference & ref)61 static std::string BootImageRepresentation(const TypeReference& ref) {
62 return GetTypeDescriptor(ref);
63 }
64
65 // Returns the class representation used in preloaded classes.
PreloadedClassesRepresentation(const TypeReference & ref)66 static std::string PreloadedClassesRepresentation(const TypeReference& ref) {
67 std::string descriptor = GetTypeDescriptor(ref);
68 return DescriptorToDot(descriptor.c_str());
69 }
70
71 // Formats the list of packages from the item metadata as a debug string.
GetPackageUseString(const FlattenProfileData::ItemMetadata & metadata)72 static std::string GetPackageUseString(const FlattenProfileData::ItemMetadata& metadata) {
73 std::string result;
74 for (const auto& it : metadata.GetAnnotations()) {
75 result += it.GetOriginPackageName() + ",";
76 }
77
78 return metadata.GetAnnotations().empty()
79 ? result
80 : result.substr(0, result.size() - 1);
81 }
82
83 // Converts a method representation to its final profile format.
MethodToProfileFormat(const std::string & method,const FlattenProfileData::ItemMetadata & metadata,bool output_package_use)84 static std::string MethodToProfileFormat(
85 const std::string& method,
86 const FlattenProfileData::ItemMetadata& metadata,
87 bool output_package_use) {
88 std::string flags_string;
89 if (metadata.HasFlagSet(Hotness::kFlagHot)) {
90 flags_string += kMethodFlagStringHot;
91 }
92 if (metadata.HasFlagSet(Hotness::kFlagStartup)) {
93 flags_string += kMethodFlagStringStartup;
94 }
95 if (metadata.HasFlagSet(Hotness::kFlagPostStartup)) {
96 flags_string += kMethodFlagStringPostStartup;
97 }
98 std::string extra;
99 if (output_package_use) {
100 extra = kPackageUseDelim + GetPackageUseString(metadata);
101 }
102
103 return flags_string + method + extra;
104 }
105
106 // Converts a class representation to its final profile or preloaded classes format.
ClassToProfileFormat(const std::string & classString,const FlattenProfileData::ItemMetadata & metadata,bool output_package_use)107 static std::string ClassToProfileFormat(
108 const std::string& classString,
109 const FlattenProfileData::ItemMetadata& metadata,
110 bool output_package_use) {
111 std::string extra;
112 if (output_package_use) {
113 extra = kPackageUseDelim + GetPackageUseString(metadata);
114 }
115
116 return classString + extra;
117 }
118
119 // Tries to asses if the given type reference is a clean class.
MaybeIsClassClean(const TypeReference & ref)120 static bool MaybeIsClassClean(const TypeReference& ref) {
121 const dex::ClassDef* class_def = ref.dex_file->FindClassDef(ref.TypeIndex());
122 if (class_def == nullptr) {
123 return false;
124 }
125
126 ClassAccessor accessor(*ref.dex_file, *class_def);
127 for (auto& it : accessor.GetStaticFields()) {
128 if (!it.IsFinal()) {
129 // Not final static field will probably dirty the class.
130 return false;
131 }
132 }
133 for (auto& it : accessor.GetMethods()) {
134 uint32_t flags = it.GetAccessFlags();
135 if ((flags & kAccNative) != 0) {
136 // Native method will get dirtied.
137 return false;
138 }
139 if ((flags & kAccConstructor) != 0 && (flags & kAccStatic) != 0) {
140 // Class initializer, may get dirtied (not sure).
141 return false;
142 }
143 }
144
145 return true;
146 }
147
148 // Returns true iff the item should be included in the profile.
149 // (i.e. it passes the given aggregation thresholds)
IncludeItemInProfile(uint32_t max_aggregation_count,uint32_t item_threshold,const FlattenProfileData::ItemMetadata & metadata,const BootImageOptions & options)150 static bool IncludeItemInProfile(uint32_t max_aggregation_count,
151 uint32_t item_threshold,
152 const FlattenProfileData::ItemMetadata& metadata,
153 const BootImageOptions& options) {
154 CHECK_NE(max_aggregation_count, 0u);
155 float item_percent = metadata.GetAnnotations().size() / static_cast<float>(max_aggregation_count);
156 for (const auto& annotIt : metadata.GetAnnotations()) {
157 const auto&thresholdIt =
158 options.special_packages_thresholds.find(annotIt.GetOriginPackageName());
159 if (thresholdIt != options.special_packages_thresholds.end()) {
160 if (item_percent >= (thresholdIt->second / 100.f)) {
161 return true;
162 }
163 }
164 }
165 return item_percent >= (item_threshold / 100.f);
166 }
167
168 // Returns true iff a method with the given metada should be included in the profile.
IncludeMethodInProfile(uint32_t max_aggregation_count,const FlattenProfileData::ItemMetadata & metadata,const BootImageOptions & options)169 static bool IncludeMethodInProfile(uint32_t max_aggregation_count,
170 const FlattenProfileData::ItemMetadata& metadata,
171 const BootImageOptions& options) {
172 return IncludeItemInProfile(max_aggregation_count, options.method_threshold, metadata, options);
173 }
174
175 // Returns true iff a class with the given metada should be included in the profile.
IncludeClassInProfile(const TypeReference & type_ref,uint32_t max_aggregation_count,const FlattenProfileData::ItemMetadata & metadata,const BootImageOptions & options)176 static bool IncludeClassInProfile(const TypeReference& type_ref,
177 uint32_t max_aggregation_count,
178 const FlattenProfileData::ItemMetadata& metadata,
179 const BootImageOptions& options) {
180 uint32_t threshold = MaybeIsClassClean(type_ref)
181 ? options.image_class_clean_threshold
182 : options.image_class_threshold;
183 return IncludeItemInProfile(max_aggregation_count, threshold, metadata, options);
184 }
185
186 // Returns true iff a class with the given metada should be included in the list of
187 // prelaoded classes.
IncludeInPreloadedClasses(const std::string & class_name,uint32_t max_aggregation_count,const FlattenProfileData::ItemMetadata & metadata,const BootImageOptions & options)188 static bool IncludeInPreloadedClasses(const std::string& class_name,
189 uint32_t max_aggregation_count,
190 const FlattenProfileData::ItemMetadata& metadata,
191 const BootImageOptions& options) {
192 bool denylisted = options.preloaded_classes_denylist.find(class_name) !=
193 options.preloaded_classes_denylist.end();
194 return !denylisted && IncludeItemInProfile(
195 max_aggregation_count, options.preloaded_class_threshold, metadata, options);
196 }
197
GenerateBootImageProfile(const std::vector<std::unique_ptr<const DexFile>> & dex_files,const std::vector<std::string> & profile_files,const BootImageOptions & options,const std::string & boot_profile_out_path,const std::string & preloaded_classes_out_path)198 bool GenerateBootImageProfile(
199 const std::vector<std::unique_ptr<const DexFile>>& dex_files,
200 const std::vector<std::string>& profile_files,
201 const BootImageOptions& options,
202 const std::string& boot_profile_out_path,
203 const std::string& preloaded_classes_out_path) {
204 if (boot_profile_out_path.empty()) {
205 LOG(ERROR) << "No output file specified";
206 return false;
207 }
208
209 bool generate_preloaded_classes = !preloaded_classes_out_path.empty();
210
211 std::unique_ptr<FlattenProfileData> flattend_data(new FlattenProfileData());
212 for (const std::string& profile_file : profile_files) {
213 ProfileCompilationInfo profile(/*for_boot_image=*/ true);
214 if (!profile.Load(profile_file, /*clear_if_invalid=*/ false)) {
215 LOG(ERROR) << "Profile is not a valid: " << profile_file;
216 return false;
217 }
218 std::unique_ptr<FlattenProfileData> currentData = profile.ExtractProfileData(dex_files);
219 flattend_data->MergeData(*currentData);
220 }
221
222 // We want the output sorted by the method/class name.
223 // So we use an intermediate map for that.
224 // There's no attempt to optimize this as it's not part of any critical path,
225 // and mostly executed on hosts.
226 SafeMap<std::string, FlattenProfileData::ItemMetadata> profile_methods;
227 SafeMap<std::string, FlattenProfileData::ItemMetadata> profile_classes;
228 SafeMap<std::string, FlattenProfileData::ItemMetadata> preloaded_classes;
229
230 for (const auto& it : flattend_data->GetMethodData()) {
231 if (IncludeMethodInProfile(flattend_data->GetMaxAggregationForMethods(), it.second, options)) {
232 FlattenProfileData::ItemMetadata metadata(it.second);
233 if (options.upgrade_startup_to_hot
234 && ((metadata.GetFlags() & Hotness::Flag::kFlagStartup) != 0)) {
235 metadata.AddFlag(Hotness::Flag::kFlagHot);
236 }
237 profile_methods.Put(BootImageRepresentation(it.first), metadata);
238 }
239 }
240
241 for (const auto& it : flattend_data->GetClassData()) {
242 const TypeReference& type_ref = it.first;
243 const FlattenProfileData::ItemMetadata& metadata = it.second;
244 if (IncludeClassInProfile(type_ref,
245 flattend_data->GetMaxAggregationForClasses(),
246 metadata,
247 options)) {
248 profile_classes.Put(BootImageRepresentation(it.first), it.second);
249 }
250 std::string preloaded_class_representation = PreloadedClassesRepresentation(it.first);
251 if (generate_preloaded_classes && IncludeInPreloadedClasses(
252 preloaded_class_representation,
253 flattend_data->GetMaxAggregationForClasses(),
254 metadata,
255 options)) {
256 preloaded_classes.Put(preloaded_class_representation, it.second);
257 }
258 }
259
260 // Create the output content
261 std::string profile_content;
262 std::string preloaded_content;
263 for (const auto& it : profile_classes) {
264 profile_content += ClassToProfileFormat(it.first, it.second, options.append_package_use_list)
265 + "\n";
266 }
267 for (const auto& it : profile_methods) {
268 profile_content += MethodToProfileFormat(it.first, it.second, options.append_package_use_list)
269 + "\n";
270 }
271
272 if (generate_preloaded_classes) {
273 for (const auto& it : preloaded_classes) {
274 preloaded_content +=
275 ClassToProfileFormat(it.first, it.second, options.append_package_use_list) + "\n";
276 }
277 }
278
279 return android::base::WriteStringToFile(profile_content, boot_profile_out_path)
280 && (!generate_preloaded_classes
281 || android::base::WriteStringToFile(preloaded_content, preloaded_classes_out_path));
282 }
283
284 } // namespace art
285