• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 <regex>
18 #include <sstream>
19 #include <string>
20 #include <vector>
21 
22 #include <sys/wait.h>
23 #include <unistd.h>
24 
25 #include <android-base/logging.h>
26 
27 #include "common_runtime_test.h"
28 
29 #include "base/file_utils.h"
30 #include "base/macros.h"
31 #include "base/unix_file/fd_file.h"
32 #include "base/utils.h"
33 #include "dex/art_dex_file_loader.h"
34 #include "dex/dex_file-inl.h"
35 #include "dex/dex_file_loader.h"
36 #include "dex/method_reference.h"
37 #include "jit/profile_compilation_info.h"
38 #include "runtime.h"
39 
40 namespace art {
41 
42 struct ImageSizes {
43   size_t art_size = 0;
44   size_t oat_size = 0;
45   size_t vdex_size = 0;
46 };
47 
operator <<(std::ostream & os,const ImageSizes & sizes)48 std::ostream& operator<<(std::ostream& os, const ImageSizes& sizes) {
49   os << "art=" << sizes.art_size << " oat=" << sizes.oat_size << " vdex=" << sizes.vdex_size;
50   return os;
51 }
52 
53 class Dex2oatImageTest : public CommonRuntimeTest {
54  public:
TearDown()55   virtual void TearDown() OVERRIDE {}
56 
57  protected:
58   // Visitors take method and type references
59   template <typename MethodVisitor, typename ClassVisitor>
VisitLibcoreDexes(const MethodVisitor & method_visitor,const ClassVisitor & class_visitor,size_t method_frequency=1,size_t class_frequency=1)60   void VisitLibcoreDexes(const MethodVisitor& method_visitor,
61                          const ClassVisitor& class_visitor,
62                          size_t method_frequency = 1,
63                          size_t class_frequency = 1) {
64     size_t method_counter = 0;
65     size_t class_counter = 0;
66     for (const std::string& dex : GetLibCoreDexFileNames()) {
67       std::vector<std::unique_ptr<const DexFile>> dex_files;
68       std::string error_msg;
69       const ArtDexFileLoader dex_file_loader;
70       CHECK(dex_file_loader.Open(dex.c_str(),
71                                  dex,
72                                  /*verify*/ true,
73                                  /*verify_checksum*/ false,
74                                  &error_msg,
75                                  &dex_files))
76           << error_msg;
77       for (const std::unique_ptr<const DexFile>& dex_file : dex_files) {
78         for (size_t i = 0; i < dex_file->NumMethodIds(); ++i) {
79           if (++method_counter % method_frequency == 0) {
80             method_visitor(MethodReference(dex_file.get(), i));
81           }
82         }
83         for (size_t i = 0; i < dex_file->NumTypeIds(); ++i) {
84           if (++class_counter % class_frequency == 0) {
85             class_visitor(TypeReference(dex_file.get(), dex::TypeIndex(i)));
86           }
87         }
88       }
89     }
90   }
91 
WriteLine(File * file,std::string line)92   static void WriteLine(File* file, std::string line) {
93     line += '\n';
94     EXPECT_TRUE(file->WriteFully(&line[0], line.length()));
95   }
96 
GenerateClasses(File * out_file,size_t frequency=1)97   void GenerateClasses(File* out_file, size_t frequency = 1) {
98     VisitLibcoreDexes(VoidFunctor(),
99                       [out_file](TypeReference ref) {
100       WriteLine(out_file, ref.dex_file->PrettyType(ref.TypeIndex()));
101     }, frequency, frequency);
102     EXPECT_EQ(out_file->Flush(), 0);
103   }
104 
GenerateMethods(File * out_file,size_t frequency=1)105   void GenerateMethods(File* out_file, size_t frequency = 1) {
106     VisitLibcoreDexes([out_file](MethodReference ref) {
107       WriteLine(out_file, ref.PrettyMethod());
108     }, VoidFunctor(), frequency, frequency);
109     EXPECT_EQ(out_file->Flush(), 0);
110   }
111 
AddRuntimeArg(std::vector<std::string> & args,const std::string & arg)112   void AddRuntimeArg(std::vector<std::string>& args, const std::string& arg) {
113     args.push_back("--runtime-arg");
114     args.push_back(arg);
115   }
116 
CompileImageAndGetSizes(const std::vector<std::string> & extra_args)117   ImageSizes CompileImageAndGetSizes(const std::vector<std::string>& extra_args) {
118     ImageSizes ret;
119     ScratchFile scratch;
120     std::string scratch_dir = scratch.GetFilename();
121     while (!scratch_dir.empty() && scratch_dir.back() != '/') {
122       scratch_dir.pop_back();
123     }
124     CHECK(!scratch_dir.empty()) << "No directory " << scratch.GetFilename();
125     std::string error_msg;
126     if (!CompileBootImage(extra_args, scratch.GetFilename(), &error_msg)) {
127       LOG(ERROR) << "Failed to compile image " << scratch.GetFilename() << error_msg;
128     }
129     std::string art_file = scratch.GetFilename() + ".art";
130     std::string oat_file = scratch.GetFilename() + ".oat";
131     std::string vdex_file = scratch.GetFilename() + ".vdex";
132     int64_t art_size = OS::GetFileSizeBytes(art_file.c_str());
133     int64_t oat_size = OS::GetFileSizeBytes(oat_file.c_str());
134     int64_t vdex_size = OS::GetFileSizeBytes(vdex_file.c_str());
135     CHECK_GT(art_size, 0u) << art_file;
136     CHECK_GT(oat_size, 0u) << oat_file;
137     CHECK_GT(vdex_size, 0u) << vdex_file;
138     ret.art_size = art_size;
139     ret.oat_size = oat_size;
140     ret.vdex_size = vdex_size;
141     scratch.Close();
142     // Clear image files since we compile the image multiple times and don't want to leave any
143     // artifacts behind.
144     ClearDirectory(scratch_dir.c_str(), /*recursive*/ false);
145     return ret;
146   }
147 
CompileBootImage(const std::vector<std::string> & extra_args,const std::string & image_file_name_prefix,std::string * error_msg)148   bool CompileBootImage(const std::vector<std::string>& extra_args,
149                         const std::string& image_file_name_prefix,
150                         std::string* error_msg) {
151     Runtime* const runtime = Runtime::Current();
152     std::vector<std::string> argv;
153     argv.push_back(runtime->GetCompilerExecutable());
154     AddRuntimeArg(argv, "-Xms64m");
155     AddRuntimeArg(argv, "-Xmx64m");
156     std::vector<std::string> dex_files = GetLibCoreDexFileNames();
157     for (const std::string& dex_file : dex_files) {
158       argv.push_back("--dex-file=" + dex_file);
159       argv.push_back("--dex-location=" + dex_file);
160     }
161     if (runtime->IsJavaDebuggable()) {
162       argv.push_back("--debuggable");
163     }
164     runtime->AddCurrentRuntimeFeaturesAsDex2OatArguments(&argv);
165 
166     AddRuntimeArg(argv, "-Xverify:softfail");
167 
168     if (!kIsTargetBuild) {
169       argv.push_back("--host");
170     }
171 
172     argv.push_back("--image=" + image_file_name_prefix + ".art");
173     argv.push_back("--oat-file=" + image_file_name_prefix + ".oat");
174     argv.push_back("--oat-location=" + image_file_name_prefix + ".oat");
175     argv.push_back("--base=0x60000000");
176 
177     std::vector<std::string> compiler_options = runtime->GetCompilerOptions();
178     argv.insert(argv.end(), compiler_options.begin(), compiler_options.end());
179 
180     // We must set --android-root.
181     const char* android_root = getenv("ANDROID_ROOT");
182     CHECK(android_root != nullptr);
183     argv.push_back("--android-root=" + std::string(android_root));
184     argv.insert(argv.end(), extra_args.begin(), extra_args.end());
185 
186     return RunDex2Oat(argv, error_msg);
187   }
188 
RunDex2Oat(const std::vector<std::string> & args,std::string * error_msg)189   int RunDex2Oat(const std::vector<std::string>& args, std::string* error_msg) {
190     int link[2];
191 
192     if (pipe(link) == -1) {
193       return false;
194     }
195 
196     pid_t pid = fork();
197     if (pid == -1) {
198       return false;
199     }
200 
201     if (pid == 0) {
202       // We need dex2oat to actually log things.
203       setenv("ANDROID_LOG_TAGS", "*:f", 1);
204       dup2(link[1], STDERR_FILENO);
205       close(link[0]);
206       close(link[1]);
207       std::vector<const char*> c_args;
208       for (const std::string& str : args) {
209         c_args.push_back(str.c_str());
210       }
211       c_args.push_back(nullptr);
212       execv(c_args[0], const_cast<char* const*>(c_args.data()));
213       exit(1);
214       UNREACHABLE();
215     } else {
216       close(link[1]);
217       char buffer[128];
218       memset(buffer, 0, 128);
219       ssize_t bytes_read = 0;
220 
221       while (TEMP_FAILURE_RETRY(bytes_read = read(link[0], buffer, 128)) > 0) {
222         *error_msg += std::string(buffer, bytes_read);
223       }
224       close(link[0]);
225       int status = -1;
226       if (waitpid(pid, &status, 0) != -1) {
227         return (status == 0);
228       }
229       return false;
230     }
231   }
232 };
233 
TEST_F(Dex2oatImageTest,TestModesAndFilters)234 TEST_F(Dex2oatImageTest, TestModesAndFilters) {
235   if (kIsTargetBuild) {
236     // This test is too slow for target builds.
237     return;
238   }
239   ImageSizes base_sizes = CompileImageAndGetSizes({});
240   ImageSizes image_classes_sizes;
241   ImageSizes compiled_classes_sizes;
242   ImageSizes compiled_all_classes_sizes;
243   ImageSizes compiled_methods_sizes;
244   ImageSizes compiled_all_methods_sizes;
245   ImageSizes profile_sizes;
246   std::cout << "Base compile sizes " << base_sizes << std::endl;
247   // Test image classes
248   {
249     ScratchFile classes;
250     GenerateClasses(classes.GetFile(), /*frequency*/ 1u);
251     image_classes_sizes = CompileImageAndGetSizes(
252         {"--image-classes=" + classes.GetFilename()});
253     classes.Close();
254     std::cout << "Image classes sizes " << image_classes_sizes << std::endl;
255     // Putting all classes as image classes should increase art size
256     EXPECT_GE(image_classes_sizes.art_size, base_sizes.art_size);
257     // Sanity check that dex is the same size.
258     EXPECT_EQ(image_classes_sizes.vdex_size, base_sizes.vdex_size);
259   }
260   // Test compiled classes with all the classes.
261   {
262     ScratchFile classes;
263     // Only compile every even class.
264     GenerateClasses(classes.GetFile(), /*frequency*/ 1u);
265     compiled_all_classes_sizes = CompileImageAndGetSizes(
266         {"--compiled-classes=" + classes.GetFilename()});
267     classes.Close();
268     std::cout << "Compiled all classes sizes " << compiled_all_classes_sizes << std::endl;
269     // Check that oat size is smaller since we didn't compile everything.
270     EXPECT_EQ(compiled_all_classes_sizes.art_size, base_sizes.art_size);
271     // TODO(mathieuc): Find a reliable way to check compiled code.
272     // EXPECT_EQ(compiled_all_classes_sizes.oat_size, base_sizes.oat_size);
273     EXPECT_EQ(compiled_all_classes_sizes.vdex_size, base_sizes.vdex_size);
274   }
275   // Test compiled classes.
276   {
277     ScratchFile classes;
278     // Only compile every even class.
279     GenerateClasses(classes.GetFile(), /*frequency*/ 2u);
280     compiled_classes_sizes = CompileImageAndGetSizes(
281         {"--image-classes=" + classes.GetFilename(),
282          "--compiled-classes=" + classes.GetFilename()});
283     classes.Close();
284     std::cout << "Compiled classes sizes " << compiled_classes_sizes << std::endl;
285     // Check that oat size is smaller since we didn't compile everything.
286     // TODO(mathieuc): Find a reliable way to check compiled code.
287     // EXPECT_LT(compiled_classes_sizes.oat_size, base_sizes.oat_size);
288     // Art file should be smaller than image classes version since we included fewer classes in the
289     // list.
290     EXPECT_LT(compiled_classes_sizes.art_size, image_classes_sizes.art_size);
291   }
292   // Test compiled methods.
293   {
294     ScratchFile methods;
295     // Only compile every even class.
296     GenerateMethods(methods.GetFile(), /*frequency*/ 1u);
297     compiled_all_methods_sizes = CompileImageAndGetSizes(
298         {"--compiled-methods=" + methods.GetFilename()});
299     methods.Close();
300     std::cout << "Compiled all methods sizes " << compiled_all_methods_sizes << std::endl;
301     EXPECT_EQ(compiled_all_classes_sizes.art_size, base_sizes.art_size);
302     // TODO(mathieuc): Find a reliable way to check compiled code. b/63746626
303     // EXPECT_EQ(compiled_all_classes_sizes.oat_size, base_sizes.oat_size);
304     EXPECT_EQ(compiled_all_classes_sizes.vdex_size, base_sizes.vdex_size);
305   }
306   static size_t kMethodFrequency = 3;
307   static size_t kTypeFrequency = 4;
308   // Test compiling fewer methods and classes.
309   {
310     ScratchFile methods;
311     ScratchFile classes;
312     // Only compile every even class.
313     GenerateMethods(methods.GetFile(), kMethodFrequency);
314     GenerateClasses(classes.GetFile(), kTypeFrequency);
315     compiled_methods_sizes = CompileImageAndGetSizes(
316         {"--image-classes=" + classes.GetFilename(),
317          "--compiled-methods=" + methods.GetFilename()});
318     methods.Close();
319     classes.Close();
320     std::cout << "Compiled fewer methods sizes " << compiled_methods_sizes << std::endl;
321   }
322   // Cross verify profile based image against image-classes and compiled-methods to make sure it
323   // matches.
324   {
325     ProfileCompilationInfo profile;
326     VisitLibcoreDexes([&profile](MethodReference ref) {
327       uint32_t flags = ProfileCompilationInfo::MethodHotness::kFlagHot |
328           ProfileCompilationInfo::MethodHotness::kFlagStartup;
329       EXPECT_TRUE(profile.AddMethodIndex(
330           static_cast<ProfileCompilationInfo::MethodHotness::Flag>(flags),
331           ref));
332     }, [&profile](TypeReference ref) {
333       EXPECT_TRUE(profile.AddClassForDex(ref));
334     }, kMethodFrequency, kTypeFrequency);
335     ScratchFile profile_file;
336     profile.Save(profile_file.GetFile()->Fd());
337     EXPECT_EQ(profile_file.GetFile()->Flush(), 0);
338     profile_sizes = CompileImageAndGetSizes(
339         {"--profile-file=" + profile_file.GetFilename(),
340          "--compiler-filter=speed-profile"});
341     profile_file.Close();
342     std::cout << "Profile sizes " << profile_sizes << std::endl;
343     // Since there is some difference between profile vs image + methods due to layout, check that
344     // the range is within expected margins (+-10%).
345     const double kRatio = 0.90;
346     EXPECT_LE(profile_sizes.art_size * kRatio, compiled_methods_sizes.art_size);
347     // TODO(mathieuc): Find a reliable way to check compiled code. b/63746626
348     // EXPECT_LE(profile_sizes.oat_size * kRatio, compiled_methods_sizes.oat_size);
349     EXPECT_LE(profile_sizes.vdex_size * kRatio, compiled_methods_sizes.vdex_size);
350     EXPECT_GE(profile_sizes.art_size / kRatio, compiled_methods_sizes.art_size);
351     // TODO(mathieuc): Find a reliable way to check compiled code. b/63746626
352     // EXPECT_GE(profile_sizes.oat_size / kRatio, compiled_methods_sizes.oat_size);
353     EXPECT_GE(profile_sizes.vdex_size / kRatio, compiled_methods_sizes.vdex_size);
354   }
355   // Test dirty image objects.
356   {
357     ScratchFile classes;
358     GenerateClasses(classes.GetFile(), /*frequency*/ 1u);
359     image_classes_sizes = CompileImageAndGetSizes(
360         {"--dirty-image-objects=" + classes.GetFilename()});
361     classes.Close();
362     std::cout << "Dirty image object sizes " << image_classes_sizes << std::endl;
363   }
364 }
365 
366 }  // namespace art
367