blob: 7886e60ed37d35bf11bbbb5c7700eb291dc2bfb6 [file] [log] [blame]
buzbeec143c552011-08-20 17:38:58 -07001// Copyright 2011 Google Inc. All Rights Reserved.
2
Elliott Hughes90a33692011-08-30 13:27:07 -07003#include "compiler.h"
4
5#include <stdint.h>
6#include <stdio.h>
7
8#include "UniquePtr.h"
buzbeec143c552011-08-20 17:38:58 -07009#include "class_linker.h"
10#include "common_test.h"
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070011#include "dex_cache.h"
buzbeec143c552011-08-20 17:38:58 -070012#include "dex_file.h"
13#include "heap.h"
14#include "object.h"
buzbeec143c552011-08-20 17:38:58 -070015
16namespace art {
17
18class CompilerTest : public CommonTest {
Brian Carlstrombffb1552011-08-25 12:23:53 -070019 protected:
Brian Carlstrom8a487412011-08-29 20:08:52 -070020
Elliott Hughes7957d542011-09-09 17:16:01 -070021 void AssertStaticIntMethod(jint expected, const ClassLoader* class_loader,
Elliott Hughes1240dad2011-09-09 16:24:50 -070022 const char* class_name, const char* method, const char* signature,
Elliott Hughes7957d542011-09-09 17:16:01 -070023 ...) {
Shih-wei Liao303b01e2011-09-14 00:46:13 -070024 EnsureCompiled(class_loader, class_name, method, signature, false);
Elliott Hughesd3a72972011-09-07 15:31:59 -070025#if defined(__arm__)
Brian Carlstrombffb1552011-08-25 12:23:53 -070026 va_list args;
Elliott Hughes7957d542011-09-09 17:16:01 -070027 va_start(args, signature);
Elliott Hughes1240dad2011-09-09 16:24:50 -070028 jint result = env_->CallStaticIntMethodV(class_, mid_, args);
Brian Carlstrombffb1552011-08-25 12:23:53 -070029 va_end(args);
Elliott Hughes1240dad2011-09-09 16:24:50 -070030 LOG(INFO) << class_name << "." << method << "(...) result is " << result;
Brian Carlstrombffb1552011-08-25 12:23:53 -070031 EXPECT_EQ(expected, result);
32#endif // __arm__
33 }
Elliott Hughes1240dad2011-09-09 16:24:50 -070034
Elliott Hughes7957d542011-09-09 17:16:01 -070035 void AssertStaticLongMethod(jlong expected, const ClassLoader* class_loader,
Elliott Hughes1240dad2011-09-09 16:24:50 -070036 const char* class_name, const char* method, const char* signature,
Elliott Hughes7957d542011-09-09 17:16:01 -070037 ...) {
Shih-wei Liao303b01e2011-09-14 00:46:13 -070038 EnsureCompiled(class_loader, class_name, method, signature, false);
Elliott Hughesd3a72972011-09-07 15:31:59 -070039#if defined(__arm__)
buzbeebafc3422011-08-25 15:22:55 -070040 va_list args;
Elliott Hughes7957d542011-09-09 17:16:01 -070041 va_start(args, signature);
Elliott Hughes1240dad2011-09-09 16:24:50 -070042 jlong result = env_->CallStaticLongMethodV(class_, mid_, args);
buzbeebafc3422011-08-25 15:22:55 -070043 va_end(args);
Elliott Hughes1240dad2011-09-09 16:24:50 -070044 LOG(INFO) << class_name << "." << method << "(...) result is " << result;
buzbeebafc3422011-08-25 15:22:55 -070045 EXPECT_EQ(expected, result);
46#endif // __arm__
47 }
Elliott Hughes1240dad2011-09-09 16:24:50 -070048
49 void CompileAll(const ClassLoader* class_loader) {
50 compiler_->CompileAll(class_loader);
51 MakeAllExecutable(class_loader);
52 }
53
54 void EnsureCompiled(const ClassLoader* class_loader,
Shih-wei Liao303b01e2011-09-14 00:46:13 -070055 const char* class_name, const char* method, const char* signature, bool is_virtual) {
Elliott Hughes1240dad2011-09-09 16:24:50 -070056 CompileAll(class_loader);
57 env_ = Thread::Current()->GetJniEnv();
58 class_ = env_->FindClass(class_name);
59 CHECK(class_ != NULL) << "Class not found: " << class_name;
Shih-wei Liao303b01e2011-09-14 00:46:13 -070060 if (is_virtual) {
61 mid_ = env_->GetMethodID(class_, method, signature);
62 } else {
63 mid_ = env_->GetStaticMethodID(class_, method, signature);
64 }
Elliott Hughes1240dad2011-09-09 16:24:50 -070065 CHECK(mid_ != NULL) << "Method not found: " << class_name << "." << method << signature;
66 }
67
68 void MakeAllExecutable(const ClassLoader* class_loader) {
69 const std::vector<const DexFile*>& class_path = ClassLoader::GetClassPath(class_loader);
70 for (size_t i = 0; i != class_path.size(); ++i) {
71 const DexFile* dex_file = class_path[i];
72 CHECK(dex_file != NULL);
73 MakeDexFileExecutable(class_loader, *dex_file);
74 }
75 }
76
77 void MakeDexFileExecutable(const ClassLoader* class_loader, const DexFile& dex_file) {
78 ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
79 for (size_t i = 0; i < dex_file.NumClassDefs(); i++) {
80 const DexFile::ClassDef& class_def = dex_file.GetClassDef(i);
81 const char* descriptor = dex_file.GetClassDescriptor(class_def);
82 Class* c = class_linker->FindClass(descriptor, class_loader);
83 CHECK(c != NULL);
84 for (size_t i = 0; i < c->NumDirectMethods(); i++) {
85 MakeMethodExecutable(c->GetDirectMethod(i));
86 }
87 for (size_t i = 0; i < c->NumVirtualMethods(); i++) {
88 MakeMethodExecutable(c->GetVirtualMethod(i));
89 }
90 }
91 }
92
93 void MakeMethodExecutable(Method* m) {
94 if (m->GetCodeArray() != NULL) {
95 MakeExecutable(m->GetCodeArray());
96 } else {
97 LOG(WARNING) << "no code for " << PrettyMethod(m);
98 }
99 if (m->GetInvokeStubArray() != NULL) {
100 MakeExecutable(m->GetInvokeStubArray());
101 } else {
102 LOG(WARNING) << "no invoke stub for " << PrettyMethod(m);
103 }
104 }
105
106 JNIEnv* env_;
107 jclass class_;
108 jmethodID mid_;
buzbeec143c552011-08-20 17:38:58 -0700109};
110
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700111// Disabled due to 10 second runtime on host
Elliott Hughes1240dad2011-09-09 16:24:50 -0700112TEST_F(CompilerTest, DISABLED_LARGE_CompileDexLibCore) {
113 CompileAll(NULL);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700114
115 // All libcore references should resolve
116 const DexFile* dex = java_lang_dex_file_.get();
117 DexCache* dex_cache = class_linker_->FindDexCache(*dex);
118 EXPECT_EQ(dex->NumStringIds(), dex_cache->NumStrings());
119 for (size_t i = 0; i < dex_cache->NumStrings(); i++) {
Elliott Hughescf4c6c42011-09-01 15:16:42 -0700120 const String* string = dex_cache->GetResolvedString(i);
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700121 EXPECT_TRUE(string != NULL) << "string_idx=" << i;
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700122 }
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700123 EXPECT_EQ(dex->NumTypeIds(), dex_cache->NumResolvedTypes());
124 for (size_t i = 0; i < dex_cache->NumResolvedTypes(); i++) {
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700125 Class* type = dex_cache->GetResolvedType(i);
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700126 EXPECT_TRUE(type != NULL) << "type_idx=" << i
127 << " " << dex->GetTypeDescriptor(dex->GetTypeId(i));
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700128 }
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700129 EXPECT_EQ(dex->NumMethodIds(), dex_cache->NumResolvedMethods());
130 for (size_t i = 0; i < dex_cache->NumResolvedMethods(); i++) {
Brian Carlstrom20cfffa2011-08-26 02:31:27 -0700131 Method* method = dex_cache->GetResolvedMethod(i);
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700132 EXPECT_TRUE(method != NULL) << "method_idx=" << i
133 << " " << dex->GetMethodClassDescriptor(dex->GetMethodId(i))
134 << " " << dex->GetMethodName(dex->GetMethodId(i));
Shih-wei Liaoc486c112011-09-13 16:43:52 -0700135 EXPECT_TRUE(method->GetCode() != NULL) << "method_idx=" << i
136 << " "
137 << dex->GetMethodClassDescriptor(dex->GetMethodId(i))
138 << " " << dex->GetMethodName(dex->GetMethodId(i));
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700139 }
Brian Carlstrom1caa2c22011-08-28 13:02:33 -0700140 EXPECT_EQ(dex->NumFieldIds(), dex_cache->NumResolvedFields());
141 for (size_t i = 0; i < dex_cache->NumResolvedFields(); i++) {
Brian Carlstrom20cfffa2011-08-26 02:31:27 -0700142 Field* field = dex_cache->GetResolvedField(i);
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700143 EXPECT_TRUE(field != NULL) << "field_idx=" << i
144 << " " << dex->GetFieldClassDescriptor(dex->GetFieldId(i))
145 << " " << dex->GetFieldName(dex->GetFieldId(i));
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700146 }
147
Brian Carlstrom83db7722011-08-26 17:32:56 -0700148 // TODO check Class::IsVerified for all classes
149
150 // TODO: check that all Method::GetCode() values are non-null
151
Brian Carlstrom9cc262e2011-08-28 12:45:30 -0700152 EXPECT_EQ(dex->NumMethodIds(), dex_cache->NumCodeAndDirectMethods());
153 CodeAndDirectMethods* code_and_direct_methods = dex_cache->GetCodeAndDirectMethods();
154 for (size_t i = 0; i < dex_cache->NumCodeAndDirectMethods(); i++) {
Brian Carlstrom83db7722011-08-26 17:32:56 -0700155 Method* method = dex_cache->GetResolvedMethod(i);
Brian Carlstrom7540ff42011-09-04 16:38:46 -0700156 if (method->IsDirect()) {
157 EXPECT_EQ(method->GetCode(), code_and_direct_methods->GetResolvedCode(i));
158 EXPECT_EQ(method, code_and_direct_methods->GetResolvedMethod(i));
159 } else {
160 EXPECT_EQ(0U, code_and_direct_methods->GetResolvedCode(i));
161 EXPECT_TRUE(code_and_direct_methods->GetResolvedMethod(i) == NULL);
162 }
Brian Carlstrom83db7722011-08-26 17:32:56 -0700163 }
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700164}
165
Shih-wei Liao303b01e2011-09-14 00:46:13 -0700166TEST_F(CompilerTest, DISABLED_AbstractMethodErrorStub) {
167 const ClassLoader* class_loader = LoadDex("AbstractMethod");
168 EnsureCompiled(class_loader, "AbstractMethod", "callme", "()V", true);
169
170 // Create a jobj_ of class "B", NOT class "AbstractMethod".
171 jclass b_class = env_->FindClass("B");
172 jmethodID constructor = env_->GetMethodID(b_class, "<init>", "()V");
173 jobject jobj_ = env_->NewObject(b_class, constructor);
174 ASSERT_TRUE(jobj_ != NULL);
175
176#if defined(__arm__)
177 // Will throw AbstractMethodError exception.
178 env_->CallNonvirtualVoidMethod(jobj_, class_, mid_);
179#endif // __arm__
180}
181
buzbee2a475e72011-09-07 17:19:17 -0700182// TODO: need check-cast test (when stub complete & we can throw/catch
183
buzbeec143c552011-08-20 17:38:58 -0700184} // namespace art