• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
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 "dex_to_dex_decompiler.h"
18 
19 #include "class_linker.h"
20 #include "common_compiler_test.h"
21 #include "compiled_method-inl.h"
22 #include "compiler_callbacks.h"
23 #include "dex/dex_file.h"
24 #include "driver/compiler_driver.h"
25 #include "driver/compiler_options.h"
26 #include "handle_scope-inl.h"
27 #include "mirror/class_loader.h"
28 #include "runtime.h"
29 #include "scoped_thread_state_change-inl.h"
30 #include "thread.h"
31 #include "verifier/method_verifier-inl.h"
32 #include "verifier/verifier_deps.h"
33 
34 namespace art {
35 
36 class DexToDexDecompilerTest : public CommonCompilerTest {
37  public:
CompileAll(jobject class_loader)38   void CompileAll(jobject class_loader) REQUIRES(!Locks::mutator_lock_) {
39     TimingLogger timings("CompilerDriverTest::CompileAll", false, false);
40     TimingLogger::ScopedTiming t(__FUNCTION__, &timings);
41     compiler_options_->boot_image_ = false;
42     compiler_options_->SetCompilerFilter(CompilerFilter::kQuicken);
43     // Create the main VerifierDeps, here instead of in the compiler since we want to aggregate
44     // the results for all the dex files, not just the results for the current dex file.
45     Runtime::Current()->GetCompilerCallbacks()->SetVerifierDeps(
46         new verifier::VerifierDeps(GetDexFiles(class_loader)));
47     compiler_driver_->SetDexFilesForOatFile(GetDexFiles(class_loader));
48     compiler_driver_->CompileAll(class_loader, GetDexFiles(class_loader), &timings);
49   }
50 
RunTest(const char * dex_name)51   void RunTest(const char* dex_name) {
52     Thread* self = Thread::Current();
53     // First load the original dex file.
54     jobject original_class_loader;
55     {
56       ScopedObjectAccess soa(self);
57       original_class_loader = LoadDex(dex_name);
58     }
59     const DexFile* original_dex_file = GetDexFiles(original_class_loader)[0];
60 
61     // Load the dex file again and make it writable to quicken them.
62     jobject class_loader;
63     const DexFile* updated_dex_file = nullptr;
64     {
65       ScopedObjectAccess soa(self);
66       class_loader = LoadDex(dex_name);
67       updated_dex_file = GetDexFiles(class_loader)[0];
68       Runtime::Current()->GetClassLinker()->RegisterDexFile(
69           *updated_dex_file, soa.Decode<mirror::ClassLoader>(class_loader).Ptr());
70     }
71     // The dex files should be identical.
72     int cmp = memcmp(original_dex_file->Begin(),
73                      updated_dex_file->Begin(),
74                      updated_dex_file->Size());
75     ASSERT_EQ(0, cmp);
76 
77     updated_dex_file->EnableWrite();
78     CompileAll(class_loader);
79     // The dex files should be different after quickening.
80     cmp = memcmp(original_dex_file->Begin(), updated_dex_file->Begin(), updated_dex_file->Size());
81     ASSERT_NE(0, cmp);
82 
83     // Unquicken the dex file.
84     for (uint32_t i = 0; i < updated_dex_file->NumClassDefs(); ++i) {
85       const DexFile::ClassDef& class_def = updated_dex_file->GetClassDef(i);
86       const uint8_t* class_data = updated_dex_file->GetClassData(class_def);
87       if (class_data == nullptr) {
88         continue;
89       }
90       ClassDataItemIterator it(*updated_dex_file, class_data);
91       it.SkipAllFields();
92 
93       // Unquicken each method.
94       while (it.HasNextMethod()) {
95         uint32_t method_idx = it.GetMemberIndex();
96         CompiledMethod* compiled_method =
97             compiler_driver_->GetCompiledMethod(MethodReference(updated_dex_file, method_idx));
98         ArrayRef<const uint8_t> table;
99         if (compiled_method != nullptr) {
100           table = compiled_method->GetVmapTable();
101         }
102         optimizer::ArtDecompileDEX(*updated_dex_file,
103                                    *it.GetMethodCodeItem(),
104                                    table,
105                                    /* decompile_return_instruction */ true);
106         it.Next();
107       }
108       DCHECK(!it.HasNext());
109     }
110 
111     // Make sure after unquickening we go back to the same contents as the original dex file.
112     cmp = memcmp(original_dex_file->Begin(), updated_dex_file->Begin(), updated_dex_file->Size());
113     ASSERT_EQ(0, cmp);
114   }
115 };
116 
TEST_F(DexToDexDecompilerTest,VerifierDeps)117 TEST_F(DexToDexDecompilerTest, VerifierDeps) {
118   RunTest("VerifierDeps");
119 }
120 
TEST_F(DexToDexDecompilerTest,DexToDexDecompiler)121 TEST_F(DexToDexDecompilerTest, DexToDexDecompiler) {
122   RunTest("DexToDexDecompiler");
123 }
124 
125 }  // namespace art
126