// Copyright 2011 Google Inc. All Rights Reserved. #include #include #include #include #include #include "UniquePtr.h" #include "base64.h" #include "class_linker.h" #include "class_loader.h" #include "compiler.h" #include "constants.h" #include "dex_file.h" #include "file.h" #include "gtest/gtest.h" #include "heap.h" #include "oat_file.h" #include "os.h" #include "runtime.h" #include "stl_util.h" #include "stringprintf.h" #include "thread.h" #include "unicode/uclean.h" #include "unicode/uvernum.h" namespace art { static inline const DexFile* OpenDexFileBase64(const char* base64, const std::string& location) { // decode base64 CHECK(base64 != NULL); size_t length; byte* dex_bytes = DecodeBase64(base64, &length); CHECK(dex_bytes != NULL); // write to provided file UniquePtr file(OS::OpenFile(location.c_str(), true)); CHECK(file.get() != NULL); if (!file->WriteFully(dex_bytes, length)) { PLOG(FATAL) << "Failed to write base64 as dex file"; } file.reset(); // read dex file const DexFile* dex_file = DexFile::Open(location, ""); CHECK(dex_file != NULL); return dex_file; } class ScratchFile { public: ScratchFile() { filename_ = getenv("ANDROID_DATA"); filename_ += "/TmpFile-XXXXXX"; fd_ = mkstemp(&filename_[0]); CHECK_NE(-1, fd_); } ~ScratchFile() { int unlink_result = unlink(filename_.c_str()); CHECK_EQ(0, unlink_result); int close_result = close(fd_); CHECK_EQ(0, close_result); } const char* GetFilename() const { return filename_.c_str(); } int GetFd() const { return fd_; } private: std::string filename_; int fd_; }; class CommonTest : public testing::Test { public: static void MakeExecutable(const ByteArray* code_array) { CHECK(code_array != NULL); MakeExecutable(code_array->GetData(), code_array->GetLength()); } static void MakeExecutable(const std::vector& code) { CHECK_NE(code.size(), 0U); MakeExecutable(&code[0], code.size()); } void MakeExecutable(Method* method) { CHECK(method != NULL); const CompiledInvokeStub* compiled_invoke_stub = compiler_->GetCompiledInvokeStub(method); CHECK(compiled_invoke_stub != NULL) << PrettyMethod(method); const std::vector& invoke_stub = compiled_invoke_stub->GetCode(); MakeExecutable(invoke_stub); const Method::InvokeStub* method_invoke_stub = reinterpret_cast(&invoke_stub[0]); LOG(INFO) << "MakeExecutable " << PrettyMethod(method) << " invoke_stub=" << reinterpret_cast(method_invoke_stub); if (!method->IsAbstract()) { const CompiledMethod* compiled_method = compiler_->GetCompiledMethod(method); CHECK(compiled_method != NULL) << PrettyMethod(method); const std::vector& code = compiled_method->GetCode(); MakeExecutable(code); const void* method_code = CompiledMethod::CodePointer(&code[0], compiled_method->GetInstructionSet()); LOG(INFO) << "MakeExecutable " << PrettyMethod(method) << " code=" << method_code; OatFile::OatMethod oat_method(method_code, compiled_method->GetFrameSizeInBytes(), compiled_method->GetReturnPcOffsetInBytes(), compiled_method->GetCoreSpillMask(), compiled_method->GetFpSpillMask(), &compiled_method->GetMappingTable()[0], &compiled_method->GetVmapTable()[0], method_invoke_stub); oat_method.LinkMethod(method); } else { MakeExecutable(runtime_->GetAbstractMethodErrorStubArray()); const void* method_code = runtime_->GetAbstractMethodErrorStubArray()->GetData(); LOG(INFO) << "MakeExecutable " << PrettyMethod(method) << " code=" << method_code; OatFile::OatMethod oat_method(method_code, kStackAlignment, 0, 0, 0, NULL, NULL, method_invoke_stub); oat_method.LinkMethod(method); } } static void MakeExecutable(const void* code_start, size_t code_length) { CHECK(code_start != NULL); CHECK_NE(code_length, 0U); uintptr_t data = reinterpret_cast(code_start); uintptr_t base = RoundDown(data, kPageSize); uintptr_t limit = RoundUp(data + code_length, kPageSize); uintptr_t len = limit - base; int result = mprotect(reinterpret_cast(base), len, PROT_READ | PROT_WRITE | PROT_EXEC); CHECK_EQ(result, 0); } protected: virtual void SetUp() { is_host_ = getenv("ANDROID_BUILD_TOP") != NULL; if (is_host_) { // $ANDROID_ROOT is set on the device, but not on the host. // We need to set this so that icu4c can find its locale data. std::string root; root += getenv("ANDROID_BUILD_TOP"); root += "/out/host/linux-x86"; setenv("ANDROID_ROOT", root.c_str(), 1); } // On target, Cannot use /mnt/sdcard because it is mounted noexec, so use subdir of art-cache android_data_ = (is_host_ ? "/tmp/art-data-XXXXXX" : "/data/art-cache/art-data-XXXXXX"); if (mkdtemp(&android_data_[0]) == NULL) { PLOG(FATAL) << "mkdtemp(\"" << &android_data_[0] << "\") failed"; } setenv("ANDROID_DATA", android_data_.c_str(), 1); art_cache_.append(android_data_.c_str()); art_cache_.append("/art-cache"); int mkdir_result = mkdir(art_cache_.c_str(), 0700); ASSERT_EQ(mkdir_result, 0); java_lang_dex_file_.reset(GetLibCoreDex()); std::string boot_class_path; boot_class_path += "-Xbootclasspath:"; boot_class_path += GetLibCoreDexFileName(); Runtime::Options options; options.push_back(std::make_pair(boot_class_path.c_str(), reinterpret_cast(NULL))); options.push_back(std::make_pair("-Xcheck:jni", reinterpret_cast(NULL))); options.push_back(std::make_pair("-Xms64m", reinterpret_cast(NULL))); options.push_back(std::make_pair("-Xmx64m", reinterpret_cast(NULL))); runtime_.reset(Runtime::Create(options, false)); ASSERT_TRUE(runtime_.get() != NULL); class_linker_ = runtime_->GetClassLinker(); InstructionSet instruction_set = kNone; #if defined(__i386__) instruction_set = kX86; #elif defined(__arm__) instruction_set = kThumb2; #endif runtime_->SetJniStubArray(JniCompiler::CreateJniStub(instruction_set)); runtime_->SetAbstractMethodErrorStubArray(Compiler::CreateAbstractMethodErrorStub(instruction_set)); for (int i=0; i < Runtime::kLastTrampolineMethodType; i++) { Runtime::TrampolineType type = Runtime::TrampolineType(i); if (!runtime_->HasResolutionStubArray(type)) { runtime_->SetResolutionStubArray( Compiler::CreateResolutionStub(instruction_set, type), type); } } for (int i=0; i < Runtime::kLastCalleeSaveType; i++) { Runtime::CalleeSaveType type = Runtime::CalleeSaveType(i); if (!runtime_->HasCalleeSaveMethod(type)) { runtime_->SetCalleeSaveMethod( runtime_->CreateCalleeSaveMethod(instruction_set, type), type); } } compiler_.reset(new Compiler(instruction_set)); Heap::VerifyHeap(); // Check for heap corruption before the test } virtual void TearDown() { const char* android_data = getenv("ANDROID_DATA"); ASSERT_TRUE(android_data != NULL); DIR* dir = opendir(art_cache_.c_str()); ASSERT_TRUE(dir != NULL); while (true) { struct dirent entry; struct dirent* entry_ptr; int readdir_result = readdir_r(dir, &entry, &entry_ptr); ASSERT_EQ(0, readdir_result); if (entry_ptr == NULL) { break; } if ((strcmp(entry_ptr->d_name, ".") == 0) || (strcmp(entry_ptr->d_name, "..") == 0)) { continue; } std::string filename(art_cache_); filename.push_back('/'); filename.append(entry_ptr->d_name); int unlink_result = unlink(filename.c_str()); ASSERT_EQ(0, unlink_result); } closedir(dir); int rmdir_cache_result = rmdir(art_cache_.c_str()); ASSERT_EQ(0, rmdir_cache_result); int rmdir_data_result = rmdir(android_data_.c_str()); ASSERT_EQ(0, rmdir_data_result); // icu4c has a fixed 10-element array "gCommonICUDataArray". // If we run > 10 tests, we fill that array and u_setCommonData fails. // There's a function to clear the array, but it's not public... typedef void (*IcuCleanupFn)(); void* sym = dlsym(RTLD_DEFAULT, "u_cleanup_" U_ICU_VERSION_SHORT); CHECK(sym != NULL); IcuCleanupFn icu_cleanup_fn = reinterpret_cast(sym); (*icu_cleanup_fn)(); compiler_.reset(); Heap::VerifyHeap(); // Check for heap corruption after the test } std::string GetLibCoreDexFileName() { if (is_host_) { const char* host_dir = getenv("ANDROID_HOST_OUT"); CHECK(host_dir != NULL); return StringPrintf("%s/framework/core-hostdex.jar", host_dir); } return std::string("/system/framework/core.jar"); } const DexFile* GetLibCoreDex() { std::string libcore_dex_file_name = GetLibCoreDexFileName(); return DexFile::Open(libcore_dex_file_name, ""); } uint32_t FindTypeIdxByDescriptor(const DexFile& dex_file, const StringPiece& descriptor) { for (size_t i = 0; i < dex_file.NumTypeIds(); i++) { const DexFile::TypeId& type_id = dex_file.GetTypeId(i); if (descriptor == dex_file.GetTypeDescriptor(type_id)) { return i; } } CHECK(false) << "Failed to find type index for " << descriptor; return 0; } uint32_t FindFieldIdxByDescriptorAndName(const DexFile& dex_file, const StringPiece& class_descriptor, const StringPiece& field_name) { for (size_t i = 0; i < dex_file.NumFieldIds(); i++) { const DexFile::FieldId& field_id = dex_file.GetFieldId(i); if (class_descriptor == dex_file.GetFieldClassDescriptor(field_id) && field_name == dex_file.GetFieldName(field_id)) { return i; } } CHECK(false) << "Failed to find field index for " << class_descriptor << " " << field_name; return 0; } const ClassLoader* AllocPathClassLoader(const DexFile* dex_file) { CHECK(dex_file != NULL); class_linker_->RegisterDexFile(*dex_file); std::vector dex_files; dex_files.push_back(dex_file); return PathClassLoader::Alloc(dex_files); } const DexFile* OpenTestDexFile(const char* name) { CHECK(name != NULL); std::string filename; if (is_host_) { // on the host, just read target dex file filename += getenv("ANDROID_PRODUCT_OUT"); } filename += "/system/framework/art-test-dex-"; filename += name; filename += ".jar"; const DexFile* dex_file = DexFile::Open(filename, ""); CHECK(dex_file != NULL) << "Failed to open " << filename; return dex_file; } const ClassLoader* LoadDex(const char* dex_name) { const DexFile* dex_file = OpenTestDexFile(dex_name); CHECK(dex_file != NULL); loaded_dex_files_.push_back(dex_file); class_linker_->RegisterDexFile(*dex_file); std::vector class_path; class_path.push_back(dex_file); const ClassLoader* class_loader = PathClassLoader::Alloc(class_path); CHECK(class_loader != NULL); Thread::Current()->SetClassLoaderOverride(class_loader); return class_loader; } void CompileMethod(Method* method) { CHECK(method != NULL); compiler_->CompileOne(method); MakeExecutable(method); MakeExecutable(runtime_->GetJniStubArray()); } void CompileDirectMethod(const ClassLoader* class_loader, const char* class_name, const char* method_name, const char* signature) { std::string class_descriptor = DotToDescriptor(class_name); Class* klass = class_linker_->FindClass(class_descriptor, class_loader); CHECK(klass != NULL) << "Class not found " << class_name; Method* method = klass->FindDirectMethod(method_name, signature); CHECK(method != NULL) << "Direct method not found: " << class_name << "." << method_name << signature; CompileMethod(method); } void CompileVirtualMethod(const ClassLoader* class_loader, const char* class_name, const char* method_name, const char* signature) { std::string class_descriptor = DotToDescriptor(class_name); Class* klass = class_linker_->FindClass(class_descriptor, class_loader); CHECK(klass != NULL) << "Class not found " << class_name; Method* method = klass->FindVirtualMethod(method_name, signature); CHECK(method != NULL) << "Virtual method not found: " << class_name << "." << method_name << signature; CompileMethod(method); } bool is_host_; std::string android_data_; std::string art_cache_; UniquePtr java_lang_dex_file_; std::vector boot_class_path_; UniquePtr runtime_; // Owned by the runtime ClassLinker* class_linker_; UniquePtr compiler_; private: std::vector loaded_dex_files_; }; } // namespace art namespace std { // TODO: isn't gtest supposed to be able to print STL types for itself? template std::ostream& operator<<(std::ostream& os, const std::vector& rhs) { os << ::art::ToString(rhs); return os; } } // namespace std