blob: c7572979fea000d278a4ed884417a3a98d4b4256 [file] [log] [blame]
Carl Shapiro894d0fa2011-06-30 14:48:49 -07001// Copyright 2011 Google Inc. All Rights Reserved.
Carl Shapiro894d0fa2011-06-30 14:48:49 -07002
Brian Carlstrom578bbdc2011-07-21 14:07:47 -07003#include "class_linker.h"
4#include "common_test.h"
5#include "dex_file.h"
6#include "heap.h"
7#include "object.h"
8#include "scoped_ptr.h"
Carl Shapiro894d0fa2011-06-30 14:48:49 -07009
Brian Carlstrom0b138b22011-07-27 15:19:17 -070010#include <stdint.h>
Carl Shapiro894d0fa2011-06-30 14:48:49 -070011#include <stdio.h>
12#include "gtest/gtest.h"
13
14namespace art {
15
Brian Carlstromf734cf52011-08-17 16:28:14 -070016class ObjectTest : public CommonTest {
Brian Carlstrom0b138b22011-07-27 15:19:17 -070017 protected:
Carl Shapirof88c9522011-08-06 15:47:38 -070018 void AssertString(size_t length,
Brian Carlstrom0b138b22011-07-27 15:19:17 -070019 const char* utf8_in,
20 const char* utf16_expected_le,
Carl Shapirof88c9522011-08-06 15:47:38 -070021 uint32_t hash_expected) {
Brian Carlstrom0b138b22011-07-27 15:19:17 -070022 uint16_t utf16_expected[length];
Carl Shapirof88c9522011-08-06 15:47:38 -070023 for (size_t i = 0; i < length; i++) {
Brian Carlstrom0b138b22011-07-27 15:19:17 -070024 uint16_t ch = (((utf16_expected_le[i*2 + 0] & 0xff) << 8) |
25 ((utf16_expected_le[i*2 + 1] & 0xff) << 0));
26 utf16_expected[i] = ch;
27 }
28
Jesse Wilson8989d992011-08-02 13:39:42 -070029 String* string = String::AllocFromModifiedUtf8(length, utf8_in);
Carl Shapiro8860c0e2011-08-04 17:36:16 -070030 ASSERT_EQ(length, string->GetLength());
31 ASSERT_TRUE(string->GetCharArray() != NULL);
Jesse Wilsonfd687c52011-08-04 19:27:35 -070032 ASSERT_TRUE(string->GetCharArray()->GetData() != NULL);
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -040033 // strlen is necessary because the 1-character string "\0" is interpreted as ""
Carl Shapirof88c9522011-08-06 15:47:38 -070034 ASSERT_TRUE(string->Equals(utf8_in) || length != strlen(utf8_in));
35 for (size_t i = 0; i < length; i++) {
Jesse Wilsonfd687c52011-08-04 19:27:35 -070036 EXPECT_EQ(utf16_expected[i], string->CharAt(i));
Brian Carlstrom0b138b22011-07-27 15:19:17 -070037 }
Carl Shapiro8860c0e2011-08-04 17:36:16 -070038 EXPECT_EQ(hash_expected, string->GetHashCode());
Brian Carlstrom0b138b22011-07-27 15:19:17 -070039 }
40};
Brian Carlstroma331b3c2011-07-18 17:47:56 -070041
42TEST_F(ObjectTest, IsInSamePackage) {
Carl Shapiro894d0fa2011-06-30 14:48:49 -070043 // Matches
Brian Carlstrom9cff8e12011-08-18 16:47:29 -070044 EXPECT_TRUE(Class::IsInSamePackage(String::AllocFromModifiedUtf8("Ljava/lang/Object;"),
45 String::AllocFromModifiedUtf8("Ljava/lang/Class")));
46 EXPECT_TRUE(Class::IsInSamePackage(String::AllocFromModifiedUtf8("LFoo;"),
47 String::AllocFromModifiedUtf8("LBar;")));
Carl Shapiro894d0fa2011-06-30 14:48:49 -070048
49 // Mismatches
Brian Carlstrom9cff8e12011-08-18 16:47:29 -070050 EXPECT_FALSE(Class::IsInSamePackage(String::AllocFromModifiedUtf8("Ljava/lang/Object;"),
51 String::AllocFromModifiedUtf8("Ljava/io/File;")));
52 EXPECT_FALSE(Class::IsInSamePackage(String::AllocFromModifiedUtf8("Ljava/lang/Object;"),
53 String::AllocFromModifiedUtf8("Ljava/lang/reflect/Method;")));
Carl Shapiro894d0fa2011-06-30 14:48:49 -070054}
55
Brian Carlstrom578bbdc2011-07-21 14:07:47 -070056TEST_F(ObjectTest, AllocObjectArray) {
Brian Carlstromf7ed11a2011-08-09 17:55:51 -070057 ObjectArray<Object>* oa = class_linker_->AllocObjectArray<Object>(2);
Elliott Hughesd8ddfd52011-08-15 14:32:53 -070058 EXPECT_EQ(2, oa->GetLength());
Brian Carlstromf7ed11a2011-08-09 17:55:51 -070059 EXPECT_TRUE(oa->Get(0) == NULL);
60 EXPECT_TRUE(oa->Get(1) == NULL);
61 oa->Set(0, oa);
62 EXPECT_TRUE(oa->Get(0) == oa);
63 EXPECT_TRUE(oa->Get(1) == NULL);
64 oa->Set(1, oa);
65 EXPECT_TRUE(oa->Get(0) == oa);
66 EXPECT_TRUE(oa->Get(1) == oa);
67
Elliott Hughes710a0cb2011-08-16 14:32:37 -070068 Thread* self = Thread::Current();
69 Class* aioobe = class_linker_->FindSystemClass("Ljava/lang/ArrayIndexOutOfBoundsException;");
70
71 EXPECT_TRUE(oa->Get(-1) == NULL);
72 EXPECT_TRUE(self->IsExceptionPending());
73 EXPECT_EQ(aioobe, self->GetException()->GetClass());
74 self->ClearException();
75
76 EXPECT_TRUE(oa->Get(2) == NULL);
77 EXPECT_TRUE(self->IsExceptionPending());
78 EXPECT_EQ(aioobe, self->GetException()->GetClass());
79 self->ClearException();
80
Brian Carlstromf7ed11a2011-08-09 17:55:51 -070081 ASSERT_TRUE(oa->GetClass() != NULL);
82 ASSERT_EQ(2U, oa->GetClass()->NumInterfaces());
83 EXPECT_EQ(class_linker_->FindSystemClass("Ljava/lang/Cloneable;"),
84 oa->GetClass()->GetInterface(0));
85 EXPECT_EQ(class_linker_->FindSystemClass("Ljava/io/Serializable;"),
86 oa->GetClass()->GetInterface(1));
Carl Shapiro0e5d75d2011-07-06 18:28:37 -070087}
88
Elliott Hughes710a0cb2011-08-16 14:32:37 -070089template<typename ArrayT>
90void TestPrimitiveArray(ClassLinker* cl) {
91 typedef typename ArrayT::ElementType T;
92
93 ArrayT* a = ArrayT::Alloc(2);
94 EXPECT_EQ(2, a->GetLength());
95 EXPECT_EQ(0, a->Get(0));
96 EXPECT_EQ(0, a->Get(1));
97 a->Set(0, T(123));
98 EXPECT_EQ(T(123), a->Get(0));
99 EXPECT_EQ(0, a->Get(1));
100 a->Set(1, T(321));
101 EXPECT_EQ(T(123), a->Get(0));
102 EXPECT_EQ(T(321), a->Get(1));
103
104 Thread* self = Thread::Current();
105 Class* aioobe = cl->FindSystemClass("Ljava/lang/ArrayIndexOutOfBoundsException;");
106
107 EXPECT_EQ(0, a->Get(-1));
108 EXPECT_TRUE(self->IsExceptionPending());
109 EXPECT_EQ(aioobe, self->GetException()->GetClass());
110 self->ClearException();
111
112 EXPECT_EQ(0, a->Get(2));
113 EXPECT_TRUE(self->IsExceptionPending());
114 EXPECT_EQ(aioobe, self->GetException()->GetClass());
115 self->ClearException();
116}
117
118TEST_F(ObjectTest, PrimitiveArray_Boolean_Alloc) {
119 TestPrimitiveArray<BooleanArray>(class_linker_);
120}
121TEST_F(ObjectTest, PrimitiveArray_Byte_Alloc) {
122 TestPrimitiveArray<ByteArray>(class_linker_);
123}
124TEST_F(ObjectTest, PrimitiveArray_Char_Alloc) {
125 TestPrimitiveArray<CharArray>(class_linker_);
126}
127TEST_F(ObjectTest, PrimitiveArray_Double_Alloc) {
128 TestPrimitiveArray<DoubleArray>(class_linker_);
129}
130TEST_F(ObjectTest, PrimitiveArray_Float_Alloc) {
131 TestPrimitiveArray<FloatArray>(class_linker_);
132}
133TEST_F(ObjectTest, PrimitiveArray_Int_Alloc) {
134 TestPrimitiveArray<IntArray>(class_linker_);
135}
136TEST_F(ObjectTest, PrimitiveArray_Long_Alloc) {
137 TestPrimitiveArray<LongArray>(class_linker_);
138}
139TEST_F(ObjectTest, PrimitiveArray_Short_Alloc) {
140 TestPrimitiveArray<ShortArray>(class_linker_);
141}
142
Brian Carlstrom0b138b22011-07-27 15:19:17 -0700143TEST_F(ObjectTest, String) {
144 // Test the empty string.
145 AssertString(0, "", "", 0);
146
147 // Test one-byte characters.
148 AssertString(1, " ", "\x00\x20", 0x20);
149 AssertString(1, "", "\x00\x00", 0);
150 AssertString(1, "\x7f", "\x00\x7f", 0x7f);
151 AssertString(2, "hi", "\x00\x68\x00\x69", (31 * 0x68) + 0x69);
152
153 // Test two-byte characters.
154 AssertString(1, "\xc2\x80", "\x00\x80", 0x80);
155 AssertString(1, "\xd9\xa6", "\x06\x66", 0x0666);
156 AssertString(1, "\xdf\xbf", "\x07\xff", 0x07ff);
157 AssertString(3, "h\xd9\xa6i", "\x00\x68\x06\x66\x00\x69", (31 * ((31 * 0x68) + 0x0666)) + 0x69);
158
159 // Test three-byte characters.
160 AssertString(1, "\xe0\xa0\x80", "\x08\x00", 0x0800);
161 AssertString(1, "\xe1\x88\xb4", "\x12\x34", 0x1234);
162 AssertString(1, "\xef\xbf\xbf", "\xff\xff", 0xffff);
163 AssertString(3, "h\xe1\x88\xb4i", "\x00\x68\x12\x34\x00\x69", (31 * ((31 * 0x68) + 0x1234)) + 0x69);
164}
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400165
Jesse Wilsonf7e85a52011-08-01 18:45:58 -0700166TEST_F(ObjectTest, StringEqualsUtf8) {
Elliott Hughesbfaadc82011-08-18 17:36:58 -0700167 String* string = String::AllocFromModifiedUtf8("android");
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700168 EXPECT_TRUE(string->Equals("android"));
169 EXPECT_FALSE(string->Equals("Android"));
170 EXPECT_FALSE(string->Equals("ANDROID"));
171 EXPECT_FALSE(string->Equals(""));
172 EXPECT_FALSE(string->Equals("and"));
173 EXPECT_FALSE(string->Equals("androids"));
Jesse Wilsonf7e85a52011-08-01 18:45:58 -0700174
Elliott Hughesbfaadc82011-08-18 17:36:58 -0700175 String* empty = String::AllocFromModifiedUtf8("");
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700176 EXPECT_TRUE(empty->Equals(""));
177 EXPECT_FALSE(empty->Equals("a"));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400178}
179
180TEST_F(ObjectTest, StringEquals) {
Elliott Hughesbfaadc82011-08-18 17:36:58 -0700181 String* string = String::AllocFromModifiedUtf8("android");
182 EXPECT_TRUE(string->Equals(String::AllocFromModifiedUtf8("android")));
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700183 EXPECT_FALSE(string->Equals("Android"));
184 EXPECT_FALSE(string->Equals("ANDROID"));
185 EXPECT_FALSE(string->Equals(""));
186 EXPECT_FALSE(string->Equals("and"));
187 EXPECT_FALSE(string->Equals("androids"));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400188
Elliott Hughesbfaadc82011-08-18 17:36:58 -0700189 String* empty = String::AllocFromModifiedUtf8("");
Carl Shapiro8860c0e2011-08-04 17:36:16 -0700190 EXPECT_TRUE(empty->Equals(""));
191 EXPECT_FALSE(empty->Equals("a"));
192}
193
194TEST_F(ObjectTest, DescriptorCompare) {
195 ClassLinker* linker = class_linker_;
196
197 scoped_ptr<DexFile> proto1_dex_file(OpenDexFileBase64(kProtoCompareDex));
198 PathClassLoader* class_loader_1 = AllocPathClassLoader(proto1_dex_file.get());
199 scoped_ptr<DexFile> proto2_dex_file(OpenDexFileBase64(kProtoCompare2Dex));
200 PathClassLoader* class_loader_2 = AllocPathClassLoader(proto2_dex_file.get());
201
202 Class* klass1 = linker->FindClass("LProtoCompare;", class_loader_1);
203 ASSERT_TRUE(klass1 != NULL);
204 Class* klass2 = linker->FindClass("LProtoCompare2;", class_loader_2);
205 ASSERT_TRUE(klass2 != NULL);
206
207 Method* m1_1 = klass1->GetVirtualMethod(0);
208 EXPECT_TRUE(m1_1->GetName()->Equals("m1"));
209 Method* m2_1 = klass1->GetVirtualMethod(1);
210 EXPECT_TRUE(m2_1->GetName()->Equals("m2"));
211 Method* m3_1 = klass1->GetVirtualMethod(2);
212 EXPECT_TRUE(m3_1->GetName()->Equals("m3"));
213 Method* m4_1 = klass1->GetVirtualMethod(3);
214 EXPECT_TRUE(m4_1->GetName()->Equals("m4"));
215
216 Method* m1_2 = klass2->GetVirtualMethod(0);
217 EXPECT_TRUE(m1_2->GetName()->Equals("m1"));
218 Method* m2_2 = klass2->GetVirtualMethod(1);
219 EXPECT_TRUE(m2_2->GetName()->Equals("m2"));
220 Method* m3_2 = klass2->GetVirtualMethod(2);
221 EXPECT_TRUE(m3_2->GetName()->Equals("m3"));
222 Method* m4_2 = klass2->GetVirtualMethod(3);
223 EXPECT_TRUE(m4_2->GetName()->Equals("m4"));
224
225 EXPECT_TRUE(m1_1->HasSameNameAndDescriptor(m1_2));
226 EXPECT_TRUE(m1_2->HasSameNameAndDescriptor(m1_1));
227
228 EXPECT_TRUE(m2_1->HasSameNameAndDescriptor(m2_2));
229 EXPECT_TRUE(m2_2->HasSameNameAndDescriptor(m2_1));
230
231 EXPECT_TRUE(m3_1->HasSameNameAndDescriptor(m3_2));
232 EXPECT_TRUE(m3_2->HasSameNameAndDescriptor(m3_1));
233
234 EXPECT_TRUE(m4_1->HasSameNameAndDescriptor(m4_2));
235 EXPECT_TRUE(m4_2->HasSameNameAndDescriptor(m4_1));
Jesse Wilsoncbe9fc02011-07-29 18:59:50 -0400236}
237
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700238
239TEST_F(ObjectTest, StringHashCode) {
Elliott Hughesbfaadc82011-08-18 17:36:58 -0700240 EXPECT_EQ(0U, String::AllocFromModifiedUtf8("")->GetHashCode());
241 EXPECT_EQ(65U, String::AllocFromModifiedUtf8("A")->GetHashCode());
242 EXPECT_EQ(64578U, String::AllocFromModifiedUtf8("ABC")->GetHashCode());
Jesse Wilsonfd687c52011-08-04 19:27:35 -0700243}
244
Brian Carlstromf7ed11a2011-08-09 17:55:51 -0700245TEST_F(ObjectTest, InstanceOf) {
246 scoped_ptr<DexFile> dex(OpenDexFileBase64(kXandY));
247 PathClassLoader* class_loader = AllocPathClassLoader(dex.get());
248 Class* X = class_linker_->FindClass("LX;", class_loader);
249 Class* Y = class_linker_->FindClass("LY;", class_loader);
250 ASSERT_TRUE(X != NULL);
251 ASSERT_TRUE(Y != NULL);
252
253 EXPECT_FALSE(Object::InstanceOf(NULL, X));
254 EXPECT_FALSE(Object::InstanceOf(NULL, Y));
255
256 Object* x = X->NewInstance();
257 Object* y = Y->NewInstance();
258 ASSERT_TRUE(x != NULL);
259 ASSERT_TRUE(y != NULL);
260
261 EXPECT_TRUE(Object::InstanceOf(x, X));
262 EXPECT_FALSE(Object::InstanceOf(x, Y));
263 EXPECT_TRUE(Object::InstanceOf(y, X));
264 EXPECT_TRUE(Object::InstanceOf(y, Y));
265
266 EXPECT_TRUE(x->InstanceOf(X));
267 EXPECT_FALSE(x->InstanceOf(Y));
268 EXPECT_TRUE(y->InstanceOf(X));
269 EXPECT_TRUE(y->InstanceOf(Y));
270}
271
272TEST_F(ObjectTest, IsAssignableFrom) {
273 scoped_ptr<DexFile> dex(OpenDexFileBase64(kXandY));
274 PathClassLoader* class_loader = AllocPathClassLoader(dex.get());
275 Class* X = class_linker_->FindClass("LX;", class_loader);
276 Class* Y = class_linker_->FindClass("LY;", class_loader);
277
278 EXPECT_TRUE(X->IsAssignableFrom(X));
279 EXPECT_TRUE(X->IsAssignableFrom(Y));
280 EXPECT_FALSE(Y->IsAssignableFrom(X));
281 EXPECT_TRUE(Y->IsAssignableFrom(Y));
282}
283
284TEST_F(ObjectTest, IsAssignableFromArray) {
285 scoped_ptr<DexFile> dex(OpenDexFileBase64(kXandY));
286 PathClassLoader* class_loader = AllocPathClassLoader(dex.get());
287 Class* X = class_linker_->FindClass("LX;", class_loader);
288 Class* Y = class_linker_->FindClass("LY;", class_loader);
289 ASSERT_TRUE(X != NULL);
290 ASSERT_TRUE(Y != NULL);
291
292 Class* YA = class_linker_->FindClass("[LY;", class_loader);
293 Class* YAA = class_linker_->FindClass("[[LY;", class_loader);
294 ASSERT_TRUE(YA != NULL);
295 ASSERT_TRUE(YAA != NULL);
296
297 Class* XAA = class_linker_->FindClass("[[LX;", class_loader);
298 ASSERT_TRUE(XAA != NULL);
299
300 Class* O = class_linker_->FindSystemClass("Ljava/lang/Object;");
301 Class* OA = class_linker_->FindSystemClass("[Ljava/lang/Object;");
302 Class* OAA = class_linker_->FindSystemClass("[[Ljava/lang/Object;");
303 Class* OAAA = class_linker_->FindSystemClass("[[[Ljava/lang/Object;");
304 ASSERT_TRUE(O != NULL);
305 ASSERT_TRUE(OA != NULL);
306 ASSERT_TRUE(OAA != NULL);
307 ASSERT_TRUE(OAAA != NULL);
308
309 Class* S = class_linker_->FindSystemClass("Ljava/io/Serializable;");
310 Class* SA = class_linker_->FindSystemClass("[Ljava/io/Serializable;");
311 Class* SAA = class_linker_->FindSystemClass("[[Ljava/io/Serializable;");
312 ASSERT_TRUE(S != NULL);
313 ASSERT_TRUE(SA != NULL);
314 ASSERT_TRUE(SAA != NULL);
315
316 Class* IA = class_linker_->FindSystemClass("[I");
317 ASSERT_TRUE(IA != NULL);
318
319 EXPECT_TRUE(YAA->IsAssignableFrom(YAA)); // identity
320 EXPECT_TRUE(XAA->IsAssignableFrom(YAA)); // element superclass
321 EXPECT_FALSE(YAA->IsAssignableFrom(XAA));
322 EXPECT_FALSE(Y->IsAssignableFrom(YAA));
323 EXPECT_FALSE(YA->IsAssignableFrom(YAA));
324 EXPECT_TRUE(O->IsAssignableFrom(YAA)); // everything is an Object
325 EXPECT_TRUE(OA->IsAssignableFrom(YAA));
326 EXPECT_TRUE(OAA->IsAssignableFrom(YAA));
327 EXPECT_TRUE(S->IsAssignableFrom(YAA)); // all arrays are Serializable
328 EXPECT_TRUE(SA->IsAssignableFrom(YAA));
329 EXPECT_FALSE(SAA->IsAssignableFrom(YAA)); // unless Y was Serializable
330
331 EXPECT_FALSE(IA->IsAssignableFrom(OA));
332 EXPECT_FALSE(OA->IsAssignableFrom(IA));
333 EXPECT_TRUE(O->IsAssignableFrom(IA));
334}
335
Elliott Hughescdf53122011-08-19 15:46:09 -0700336TEST_F(ObjectTest, FindInstanceField) {
337 String* s = String::AllocFromModifiedUtf8("ABC");
338 ASSERT_TRUE(s != NULL);
339 Class* c = s->GetClass();
340 ASSERT_TRUE(c != NULL);
341
342 // Wrong type.
343 EXPECT_TRUE(c->FindDeclaredInstanceField("count", "J") == NULL);
344 EXPECT_TRUE(c->FindInstanceField("count", "J") == NULL);
345
346 // Wrong name.
347 EXPECT_TRUE(c->FindDeclaredInstanceField("Count", "I") == NULL);
348 EXPECT_TRUE(c->FindInstanceField("Count", "I") == NULL);
349
350 // Right name and type.
351 Field* f1 = c->FindDeclaredInstanceField("count", "I");
352 Field* f2 = c->FindInstanceField("count", "I");
353 EXPECT_TRUE(f1 != NULL);
354 EXPECT_TRUE(f2 != NULL);
355 EXPECT_EQ(f1, f2);
356
357 // TODO: check that s.count == 3.
358
359 // Ensure that we handle superclass fields correctly...
360 c = class_linker_->FindSystemClass("Ljava/lang/StringBuilder;");
361 ASSERT_TRUE(c != NULL);
362 // No StringBuilder.count...
363 EXPECT_TRUE(c->FindDeclaredInstanceField("count", "I") == NULL);
364 // ...but there is an AbstractStringBuilder.count.
365 EXPECT_TRUE(c->FindInstanceField("count", "I") != NULL);
366}
367
368TEST_F(ObjectTest, FindStaticField) {
369 String* s = String::AllocFromModifiedUtf8("ABC");
370 ASSERT_TRUE(s != NULL);
371 Class* c = s->GetClass();
372 ASSERT_TRUE(c != NULL);
373
374 // Wrong type.
375 EXPECT_TRUE(c->FindDeclaredStaticField("CASE_INSENSITIVE_ORDER", "I") == NULL);
376 EXPECT_TRUE(c->FindStaticField("CASE_INSENSITIVE_ORDER", "I") == NULL);
377
378 // Wrong name.
379 EXPECT_TRUE(c->FindDeclaredStaticField("cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;") == NULL);
380 EXPECT_TRUE(c->FindStaticField("cASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;") == NULL);
381
382 // Right name and type.
383 Field* f1 = c->FindDeclaredStaticField("CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
384 Field* f2 = c->FindStaticField("CASE_INSENSITIVE_ORDER", "Ljava/util/Comparator;");
385 EXPECT_TRUE(f1 != NULL);
386 EXPECT_TRUE(f2 != NULL);
387 EXPECT_EQ(f1, f2);
388
389 // TODO: test static fields via superclasses.
390 // TODO: test static fields via interfaces.
391 // TODO: test that interfaces trump superclasses.
392}
393
Carl Shapiro894d0fa2011-06-30 14:48:49 -0700394} // namespace art