• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 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 #ifndef ART_RUNTIME_COMMON_TEST_H_
18 #define ART_RUNTIME_COMMON_TEST_H_
19 
20 #include <dirent.h>
21 #include <dlfcn.h>
22 #include <sys/mman.h>
23 #include <sys/stat.h>
24 #include <sys/types.h>
25 
26 #include "../../external/icu4c/common/unicode/uvernum.h"
27 #include "base/macros.h"
28 #include "base/stl_util.h"
29 #include "base/stringprintf.h"
30 #include "base/unix_file/fd_file.h"
31 #include "class_linker.h"
32 #include "compiler/driver/compiler_driver.h"
33 #include "dex_file-inl.h"
34 #include "entrypoints/entrypoint_utils.h"
35 #include "gc/heap.h"
36 #include "gtest/gtest.h"
37 #include "instruction_set.h"
38 #include "interpreter/interpreter.h"
39 #include "mirror/class_loader.h"
40 #include "oat_file.h"
41 #include "object_utils.h"
42 #include "os.h"
43 #include "runtime.h"
44 #include "scoped_thread_state_change.h"
45 #include "ScopedLocalRef.h"
46 #include "thread.h"
47 #include "UniquePtr.h"
48 #include "well_known_classes.h"
49 
50 namespace art {
51 
52 static const byte kBase64Map[256] = {
53   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
54   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
55   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
56   255, 255, 255, 255, 255, 255, 255,  62, 255, 255, 255,  63,
57   52,  53,  54,  55,  56,  57,  58,  59,  60,  61, 255, 255,
58   255, 254, 255, 255, 255,   0,   1,   2,   3,   4,   5,   6,
59     7,   8,   9,  10,  11,  12,  13,  14,  15,  16,  17,  18,  // NOLINT
60    19,  20,  21,  22,  23,  24,  25, 255, 255, 255, 255, 255,  // NOLINT
61   255,  26,  27,  28,  29,  30,  31,  32,  33,  34,  35,  36,
62    37,  38,  39,  40,  41,  42,  43,  44,  45,  46,  47,  48,  // NOLINT
63    49,  50,  51, 255, 255, 255, 255, 255, 255, 255, 255, 255,  // NOLINT
64   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
65   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
66   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
67   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
68   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
69   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
70   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
71   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
72   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
73   255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
74   255, 255, 255, 255
75 };
76 
DecodeBase64(const char * src,size_t * dst_size)77 byte* DecodeBase64(const char* src, size_t* dst_size) {
78   std::vector<byte> tmp;
79   uint32_t t = 0, y = 0;
80   int g = 3;
81   for (size_t i = 0; src[i] != '\0'; ++i) {
82     byte c = kBase64Map[src[i] & 0xFF];
83     if (c == 255) continue;
84     // the final = symbols are read and used to trim the remaining bytes
85     if (c == 254) {
86       c = 0;
87       // prevent g < 0 which would potentially allow an overflow later
88       if (--g < 0) {
89         *dst_size = 0;
90         return NULL;
91       }
92     } else if (g != 3) {
93       // we only allow = to be at the end
94       *dst_size = 0;
95       return NULL;
96     }
97     t = (t << 6) | c;
98     if (++y == 4) {
99       tmp.push_back((t >> 16) & 255);
100       if (g > 1) {
101         tmp.push_back((t >> 8) & 255);
102       }
103       if (g > 2) {
104         tmp.push_back(t & 255);
105       }
106       y = t = 0;
107     }
108   }
109   if (y != 0) {
110     *dst_size = 0;
111     return NULL;
112   }
113   UniquePtr<byte[]> dst(new byte[tmp.size()]);
114   if (dst_size != NULL) {
115     *dst_size = tmp.size();
116   } else {
117     *dst_size = 0;
118   }
119   std::copy(tmp.begin(), tmp.end(), dst.get());
120   return dst.release();
121 }
122 
123 class ScratchFile {
124  public:
ScratchFile()125   ScratchFile() {
126     filename_ = getenv("ANDROID_DATA");
127     filename_ += "/TmpFile-XXXXXX";
128     int fd = mkstemp(&filename_[0]);
129     CHECK_NE(-1, fd);
130     file_.reset(new File(fd, GetFilename()));
131   }
132 
~ScratchFile()133   ~ScratchFile() {
134     int unlink_result = unlink(filename_.c_str());
135     CHECK_EQ(0, unlink_result);
136   }
137 
GetFilename()138   const std::string& GetFilename() const {
139     return filename_;
140   }
141 
GetFile()142   File* GetFile() const {
143     return file_.get();
144   }
145 
GetFd()146   int GetFd() const {
147     return file_->Fd();
148   }
149 
150  private:
151   std::string filename_;
152   UniquePtr<File> file_;
153 };
154 
155 class CommonTest : public testing::Test {
156  public:
MakeExecutable(const mirror::ByteArray * code_array)157   static void MakeExecutable(const mirror::ByteArray* code_array) {
158     CHECK(code_array != NULL);
159     MakeExecutable(code_array->GetData(), code_array->GetLength());
160   }
161 
MakeExecutable(const std::vector<uint8_t> & code)162   static void MakeExecutable(const std::vector<uint8_t>& code) {
163     CHECK_NE(code.size(), 0U);
164     MakeExecutable(&code[0], code.size());
165   }
166 
167   // Create an OatMethod based on pointers (for unit tests)
CreateOatMethod(const void * code,const size_t frame_size_in_bytes,const uint32_t core_spill_mask,const uint32_t fp_spill_mask,const uint8_t * mapping_table,const uint8_t * vmap_table,const uint8_t * gc_map)168   OatFile::OatMethod CreateOatMethod(const void* code,
169                                      const size_t frame_size_in_bytes,
170                                      const uint32_t core_spill_mask,
171                                      const uint32_t fp_spill_mask,
172                                      const uint8_t* mapping_table,
173                                      const uint8_t* vmap_table,
174                                      const uint8_t* gc_map) {
175       return OatFile::OatMethod(NULL,
176                                 reinterpret_cast<uint32_t>(code),
177                                 frame_size_in_bytes,
178                                 core_spill_mask,
179                                 fp_spill_mask,
180                                 reinterpret_cast<uint32_t>(mapping_table),
181                                 reinterpret_cast<uint32_t>(vmap_table),
182                                 reinterpret_cast<uint32_t>(gc_map));
183   }
184 
MakeExecutable(mirror::ArtMethod * method)185   void MakeExecutable(mirror::ArtMethod* method) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
186     CHECK(method != NULL);
187     LOG(INFO) << "MakeExecutable " << PrettyMethod(method);
188 
189     const CompiledMethod* compiled_method = NULL;
190     if (!method->IsAbstract()) {
191       const mirror::DexCache* dex_cache = method->GetDeclaringClass()->GetDexCache();
192       const DexFile& dex_file = *dex_cache->GetDexFile();
193       compiled_method =
194           compiler_driver_->GetCompiledMethod(MethodReference(&dex_file,
195                                                               method->GetDexMethodIndex()));
196     }
197     if (compiled_method != NULL) {
198       const std::vector<uint8_t>& code = compiled_method->GetCode();
199       MakeExecutable(code);
200       const void* method_code = CompiledMethod::CodePointer(&code[0],
201                                                             compiled_method->GetInstructionSet());
202       LOG(INFO) << "MakeExecutable " << PrettyMethod(method) << " code=" << method_code;
203       OatFile::OatMethod oat_method = CreateOatMethod(method_code,
204                                                       compiled_method->GetFrameSizeInBytes(),
205                                                       compiled_method->GetCoreSpillMask(),
206                                                       compiled_method->GetFpSpillMask(),
207                                                       &compiled_method->GetMappingTable()[0],
208                                                       &compiled_method->GetVmapTable()[0],
209                                                       NULL);
210       oat_method.LinkMethod(method);
211       method->SetEntryPointFromInterpreter(artInterpreterToCompiledCodeBridge);
212     } else {
213       const void* method_code;
214       // No code? You must mean to go into the interpreter.
215       method_code = GetCompiledCodeToInterpreterBridge();
216       LOG(INFO) << "MakeExecutable " << PrettyMethod(method) << " code=" << method_code;
217       OatFile::OatMethod oat_method = CreateOatMethod(method_code,
218                                                       kStackAlignment,
219                                                       0,
220                                                       0,
221                                                       NULL,
222                                                       NULL,
223                                                       NULL);
224       oat_method.LinkMethod(method);
225       method->SetEntryPointFromInterpreter(interpreter::artInterpreterToInterpreterBridge);
226     }
227   }
228 
MakeExecutable(const void * code_start,size_t code_length)229   static void MakeExecutable(const void* code_start, size_t code_length) {
230     CHECK(code_start != NULL);
231     CHECK_NE(code_length, 0U);
232     uintptr_t data = reinterpret_cast<uintptr_t>(code_start);
233     uintptr_t base = RoundDown(data, kPageSize);
234     uintptr_t limit = RoundUp(data + code_length, kPageSize);
235     uintptr_t len = limit - base;
236     int result = mprotect(reinterpret_cast<void*>(base), len, PROT_READ | PROT_WRITE | PROT_EXEC);
237     CHECK_EQ(result, 0);
238 
239     // Flush instruction cache
240     // Only uses __builtin___clear_cache if GCC >= 4.3.3
241 #if GCC_VERSION >= 40303
242     __builtin___clear_cache(reinterpret_cast<void*>(base), reinterpret_cast<void*>(base + len));
243 #else
244     LOG(FATAL) << "UNIMPLEMENTED: cache flush";
245 #endif
246   }
247 
SetEnvironmentVariables(std::string & android_data)248   static void SetEnvironmentVariables(std::string& android_data) {
249     if (IsHost()) {
250       // $ANDROID_ROOT is set on the device, but not on the host.
251       // We need to set this so that icu4c can find its locale data.
252       std::string root;
253       root += getenv("ANDROID_BUILD_TOP");
254 #if defined(__linux__)
255       root += "/out/host/linux-x86";
256 #elif defined(__APPLE__)
257       root += "/out/host/darwin-x86";
258 #else
259 #error unsupported OS
260 #endif
261       setenv("ANDROID_ROOT", root.c_str(), 1);
262       setenv("LD_LIBRARY_PATH", ":", 0);  // Required by java.lang.System.<clinit>.
263     }
264 
265     // On target, Cannot use /mnt/sdcard because it is mounted noexec, so use subdir of dalvik-cache
266     android_data = (IsHost() ? "/tmp/art-data-XXXXXX" : "/data/dalvik-cache/art-data-XXXXXX");
267     if (mkdtemp(&android_data[0]) == NULL) {
268       PLOG(FATAL) << "mkdtemp(\"" << &android_data[0] << "\") failed";
269     }
270     setenv("ANDROID_DATA", android_data.c_str(), 1);
271   }
272 
273  protected:
IsHost()274   static bool IsHost() {
275     return (getenv("ANDROID_BUILD_TOP") != NULL);
276   }
277 
SetUp()278   virtual void SetUp() {
279     SetEnvironmentVariables(android_data_);
280     dalvik_cache_.append(android_data_.c_str());
281     dalvik_cache_.append("/dalvik-cache");
282     int mkdir_result = mkdir(dalvik_cache_.c_str(), 0700);
283     ASSERT_EQ(mkdir_result, 0);
284 
285     java_lang_dex_file_ = DexFile::Open(GetLibCoreDexFileName(), GetLibCoreDexFileName());
286     if (java_lang_dex_file_ == NULL) {
287       LOG(FATAL) << "Could not open .dex file '" << GetLibCoreDexFileName() << "'\n";
288     }
289     conscrypt_file_ = DexFile::Open(GetConscryptFileName(), GetConscryptFileName());
290     if (conscrypt_file_  == NULL) {
291       LOG(FATAL) << "Could not open .dex file '" << GetConscryptFileName() << "'\n";
292     }
293     boot_class_path_.push_back(java_lang_dex_file_);
294     boot_class_path_.push_back(conscrypt_file_);
295 
296     std::string min_heap_string(StringPrintf("-Xms%zdm", gc::Heap::kDefaultInitialSize / MB));
297     std::string max_heap_string(StringPrintf("-Xmx%zdm", gc::Heap::kDefaultMaximumSize / MB));
298 
299     Runtime::Options options;
300     options.push_back(std::make_pair("compiler", reinterpret_cast<void*>(NULL)));
301     options.push_back(std::make_pair("bootclasspath", &boot_class_path_));
302     options.push_back(std::make_pair("-Xcheck:jni", reinterpret_cast<void*>(NULL)));
303     options.push_back(std::make_pair(min_heap_string.c_str(), reinterpret_cast<void*>(NULL)));
304     options.push_back(std::make_pair(max_heap_string.c_str(), reinterpret_cast<void*>(NULL)));
305     if (!Runtime::Create(options, false)) {
306       LOG(FATAL) << "Failed to create runtime";
307       return;
308     }
309     runtime_.reset(Runtime::Current());
310     // Runtime::Create acquired the mutator_lock_ that is normally given away when we Runtime::Start,
311     // give it away now and then switch to a more managable ScopedObjectAccess.
312     Thread::Current()->TransitionFromRunnableToSuspended(kNative);
313     {
314       ScopedObjectAccess soa(Thread::Current());
315       ASSERT_TRUE(runtime_.get() != NULL);
316       class_linker_ = runtime_->GetClassLinker();
317 
318       InstructionSet instruction_set = kNone;
319 #if defined(__arm__)
320       instruction_set = kThumb2;
321 #elif defined(__mips__)
322       instruction_set = kMips;
323 #elif defined(__i386__)
324       instruction_set = kX86;
325 #endif
326 
327       // TODO: make selectable
328 #if defined(ART_USE_PORTABLE_COMPILER)
329       CompilerBackend compiler_backend = kPortable;
330 #else
331       CompilerBackend compiler_backend = kQuick;
332 #endif
333 
334       if (!runtime_->HasResolutionMethod()) {
335         runtime_->SetResolutionMethod(runtime_->CreateResolutionMethod());
336       }
337       for (int i = 0; i < Runtime::kLastCalleeSaveType; i++) {
338         Runtime::CalleeSaveType type = Runtime::CalleeSaveType(i);
339         if (!runtime_->HasCalleeSaveMethod(type)) {
340           runtime_->SetCalleeSaveMethod(
341               runtime_->CreateCalleeSaveMethod(instruction_set, type), type);
342         }
343       }
344       class_linker_->FixupDexCaches(runtime_->GetResolutionMethod());
345       compiler_driver_.reset(new CompilerDriver(compiler_backend, instruction_set,
346                                                 true, new CompilerDriver::DescriptorSet,
347                                                 2, true));
348     }
349     // We typically don't generate an image in unit tests, disable this optimization by default.
350     compiler_driver_->SetSupportBootImageFixup(false);
351 
352     // We're back in native, take the opportunity to initialize well known classes.
353     WellKnownClasses::Init(Thread::Current()->GetJniEnv());
354     // Create the heap thread pool so that the GC runs in parallel for tests. Normally, the thread
355     // pool is created by the runtime.
356     runtime_->GetHeap()->CreateThreadPool();
357     runtime_->GetHeap()->VerifyHeap();  // Check for heap corruption before the test
358   }
359 
TearDown()360   virtual void TearDown() {
361     const char* android_data = getenv("ANDROID_DATA");
362     ASSERT_TRUE(android_data != NULL);
363     DIR* dir = opendir(dalvik_cache_.c_str());
364     ASSERT_TRUE(dir != NULL);
365     dirent* e;
366     while ((e = readdir(dir)) != NULL) {
367       if ((strcmp(e->d_name, ".") == 0) || (strcmp(e->d_name, "..") == 0)) {
368         continue;
369       }
370       std::string filename(dalvik_cache_);
371       filename.push_back('/');
372       filename.append(e->d_name);
373       int unlink_result = unlink(filename.c_str());
374       ASSERT_EQ(0, unlink_result);
375     }
376     closedir(dir);
377     int rmdir_cache_result = rmdir(dalvik_cache_.c_str());
378     ASSERT_EQ(0, rmdir_cache_result);
379     int rmdir_data_result = rmdir(android_data_.c_str());
380     ASSERT_EQ(0, rmdir_data_result);
381 
382     // icu4c has a fixed 10-element array "gCommonICUDataArray".
383     // If we run > 10 tests, we fill that array and u_setCommonData fails.
384     // There's a function to clear the array, but it's not public...
385     typedef void (*IcuCleanupFn)();
386     void* sym = dlsym(RTLD_DEFAULT, "u_cleanup_" U_ICU_VERSION_SHORT);
387     CHECK(sym != NULL);
388     IcuCleanupFn icu_cleanup_fn = reinterpret_cast<IcuCleanupFn>(sym);
389     (*icu_cleanup_fn)();
390 
391     compiler_driver_.reset();
392     STLDeleteElements(&opened_dex_files_);
393 
394     Runtime::Current()->GetHeap()->VerifyHeap();  // Check for heap corruption after the test
395   }
396 
GetLibCoreDexFileName()397   std::string GetLibCoreDexFileName() {
398     return GetDexFileName("core-libart");
399   }
400 
GetConscryptFileName()401   std::string GetConscryptFileName() {
402     return GetDexFileName("conscrypt");
403   }
404 
GetDexFileName(const std::string & jar_prefix)405   std::string GetDexFileName(const std::string& jar_prefix) {
406     if (IsHost()) {
407       const char* host_dir = getenv("ANDROID_HOST_OUT");
408       CHECK(host_dir != NULL);
409       return StringPrintf("%s/framework/%s-hostdex.jar", host_dir, jar_prefix.c_str());
410     }
411     return StringPrintf("%s/framework/%s.jar", GetAndroidRoot(), jar_prefix.c_str());
412   }
413 
GetTestAndroidRoot()414   std::string GetTestAndroidRoot() {
415     if (IsHost()) {
416       const char* host_dir = getenv("ANDROID_HOST_OUT");
417       CHECK(host_dir != NULL);
418       return host_dir;
419     }
420     return GetAndroidRoot();
421   }
422 
OpenTestDexFile(const char * name)423   const DexFile* OpenTestDexFile(const char* name) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
424     CHECK(name != NULL);
425     std::string filename;
426     if (IsHost()) {
427       filename += getenv("ANDROID_HOST_OUT");
428       filename += "/framework/";
429     } else {
430       filename += "/data/nativetest/art/";
431     }
432     filename += "art-test-dex-";
433     filename += name;
434     filename += ".jar";
435     const DexFile* dex_file = DexFile::Open(filename, filename);
436     CHECK(dex_file != NULL) << "Failed to open " << filename;
437     CHECK_EQ(PROT_READ, dex_file->GetPermissions());
438     CHECK(dex_file->IsReadOnly());
439     opened_dex_files_.push_back(dex_file);
440     return dex_file;
441   }
442 
LoadDex(const char * dex_name)443   jobject LoadDex(const char* dex_name) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
444     const DexFile* dex_file = OpenTestDexFile(dex_name);
445     CHECK(dex_file != NULL);
446     class_linker_->RegisterDexFile(*dex_file);
447     std::vector<const DexFile*> class_path;
448     class_path.push_back(dex_file);
449     ScopedObjectAccessUnchecked soa(Thread::Current());
450     ScopedLocalRef<jobject> class_loader_local(soa.Env(),
451         soa.Env()->AllocObject(WellKnownClasses::dalvik_system_PathClassLoader));
452     jobject class_loader = soa.Env()->NewGlobalRef(class_loader_local.get());
453     soa.Self()->SetClassLoaderOverride(soa.Decode<mirror::ClassLoader*>(class_loader_local.get()));
454     Runtime::Current()->SetCompileTimeClassPath(class_loader, class_path);
455     return class_loader;
456   }
457 
CompileClass(mirror::ClassLoader * class_loader,const char * class_name)458   void CompileClass(mirror::ClassLoader* class_loader, const char* class_name)
459       SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
460     std::string class_descriptor(DotToDescriptor(class_name));
461     mirror::Class* klass = class_linker_->FindClass(class_descriptor.c_str(), class_loader);
462     CHECK(klass != NULL) << "Class not found " << class_name;
463     for (size_t i = 0; i < klass->NumDirectMethods(); i++) {
464       CompileMethod(klass->GetDirectMethod(i));
465     }
466     for (size_t i = 0; i < klass->NumVirtualMethods(); i++) {
467       CompileMethod(klass->GetVirtualMethod(i));
468     }
469   }
470 
CompileMethod(mirror::ArtMethod * method)471   void CompileMethod(mirror::ArtMethod* method) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
472     CHECK(method != NULL);
473     base::TimingLogger timings("CommonTest::CompileMethod", false, false);
474     timings.StartSplit("CompileOne");
475     compiler_driver_->CompileOne(method, timings);
476     MakeExecutable(method);
477   }
478 
CompileDirectMethod(mirror::ClassLoader * class_loader,const char * class_name,const char * method_name,const char * signature)479   void CompileDirectMethod(mirror::ClassLoader* class_loader,
480                            const char* class_name,
481                            const char* method_name,
482                            const char* signature)
483       SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
484     std::string class_descriptor(DotToDescriptor(class_name));
485     mirror::Class* klass = class_linker_->FindClass(class_descriptor.c_str(), class_loader);
486     CHECK(klass != NULL) << "Class not found " << class_name;
487     mirror::ArtMethod* method = klass->FindDirectMethod(method_name, signature);
488     CHECK(method != NULL) << "Direct method not found: "
489                           << class_name << "." << method_name << signature;
490     CompileMethod(method);
491   }
492 
CompileVirtualMethod(mirror::ClassLoader * class_loader,const char * class_name,const char * method_name,const char * signature)493   void CompileVirtualMethod(mirror::ClassLoader* class_loader,
494                             const char* class_name,
495                             const char* method_name,
496                             const char* signature)
497       SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
498     std::string class_descriptor(DotToDescriptor(class_name));
499     mirror::Class* klass = class_linker_->FindClass(class_descriptor.c_str(), class_loader);
500     CHECK(klass != NULL) << "Class not found " << class_name;
501     mirror::ArtMethod* method = klass->FindVirtualMethod(method_name, signature);
502     CHECK(method != NULL) << "Virtual method not found: "
503                           << class_name << "." << method_name << signature;
504     CompileMethod(method);
505   }
506 
ReserveImageSpace()507   void ReserveImageSpace() {
508     // Reserve where the image will be loaded up front so that other parts of test set up don't
509     // accidentally end up colliding with the fixed memory address when we need to load the image.
510     image_reservation_.reset(MemMap::MapAnonymous("image reservation",
511                                                   reinterpret_cast<byte*>(ART_BASE_ADDRESS),
512                                                   (size_t)100 * 1024 * 1024,  // 100MB
513                                                   PROT_NONE));
514   }
515 
UnreserveImageSpace()516   void UnreserveImageSpace() {
517     image_reservation_.reset();
518   }
519 
520   std::string android_data_;
521   std::string dalvik_cache_;
522   const DexFile* java_lang_dex_file_;  // owned by runtime_
523   const DexFile* conscrypt_file_;  // owned by runtime_
524   std::vector<const DexFile*> boot_class_path_;
525   UniquePtr<Runtime> runtime_;
526   // Owned by the runtime
527   ClassLinker* class_linker_;
528   UniquePtr<CompilerDriver> compiler_driver_;
529 
530  private:
531   std::vector<const DexFile*> opened_dex_files_;
532   UniquePtr<MemMap> image_reservation_;
533 };
534 
535 // Sets a CheckJni abort hook to catch failures. Note that this will cause CheckJNI to carry on
536 // rather than aborting, so be careful!
537 class CheckJniAbortCatcher {
538  public:
CheckJniAbortCatcher()539   CheckJniAbortCatcher() : vm_(Runtime::Current()->GetJavaVM()) {
540     vm_->check_jni_abort_hook = Hook;
541     vm_->check_jni_abort_hook_data = &actual_;
542   }
543 
~CheckJniAbortCatcher()544   ~CheckJniAbortCatcher() {
545     vm_->check_jni_abort_hook = NULL;
546     vm_->check_jni_abort_hook_data = NULL;
547     EXPECT_TRUE(actual_.empty()) << actual_;
548   }
549 
Check(const char * expected_text)550   void Check(const char* expected_text) {
551     EXPECT_TRUE(actual_.find(expected_text) != std::string::npos) << "\n"
552         << "Expected to find: " << expected_text << "\n"
553         << "In the output   : " << actual_;
554     actual_.clear();
555   }
556 
557  private:
Hook(void * data,const std::string & reason)558   static void Hook(void* data, const std::string& reason) {
559     // We use += because when we're hooking the aborts like this, multiple problems can be found.
560     *reinterpret_cast<std::string*>(data) += reason;
561   }
562 
563   JavaVMExt* vm_;
564   std::string actual_;
565 
566   DISALLOW_COPY_AND_ASSIGN(CheckJniAbortCatcher);
567 };
568 
569 // TODO: These tests were disabled for portable when we went to having
570 // MCLinker link LLVM ELF output because we no longer just have code
571 // blobs in memory. We'll need to dlopen to load and relocate
572 // temporary output to resurrect these tests.
573 #if defined(ART_USE_PORTABLE_COMPILER)
574 #define TEST_DISABLED_FOR_PORTABLE() printf("WARNING: TEST DISABLED FOR PORTABLE\n"); return
575 #else
576 #define TEST_DISABLED_FOR_PORTABLE()
577 #endif
578 
579 }  // namespace art
580 
581 namespace std {
582 
583 // TODO: isn't gtest supposed to be able to print STL types for itself?
584 template <typename T>
585 std::ostream& operator<<(std::ostream& os, const std::vector<T>& rhs) {
586   os << ::art::ToString(rhs);
587   return os;
588 }
589 
590 }  // namespace std
591 
592 #endif  // ART_RUNTIME_COMMON_TEST_H_
593