1 /*
2 * Copyright (c) 2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "ecmascript/base/builtins_base.h"
17
18 #include "ecmascript/ecma_runtime_call_info.h"
19 #include "ecmascript/ecma_string.h"
20 #include "ecmascript/ecma_vm.h"
21 #include "ecmascript/global_env.h"
22 #include "ecmascript/ic/proto_change_details.h"
23 #include "ecmascript/js_function.h"
24 #include "ecmascript/js_handle.h"
25 #include "ecmascript/js_hclass.h"
26 #include "ecmascript/js_object-inl.h"
27 #include "ecmascript/js_thread.h"
28 #include "ecmascript/mem/verification.h"
29 #include "ecmascript/object_factory.h"
30 #include "ecmascript/object_operator.h"
31 #include "ecmascript/property_attributes.h"
32 #include "ecmascript/tagged_array-inl.h"
33 #include "ecmascript/tagged_dictionary.h"
34 #include "ecmascript/tests/test_helper.h"
35 #include "ecmascript/weak_vector.h"
36
37 using namespace panda::ecmascript;
38 using namespace panda::ecmascript::base;
39
40 namespace panda::test {
41 class JSObjectTest : public BaseTestWithScope<false> {
42 };
43
JSObjectTestCreate(JSThread * thread)44 static JSFunction *JSObjectTestCreate(JSThread *thread)
45 {
46 JSHandle<GlobalEnv> globalEnv = thread->GetEcmaVM()->GetGlobalEnv();
47 return globalEnv->GetObjectFunction().GetObject<JSFunction>();
48 }
49
HWTEST_F_L0(JSObjectTest,Create)50 HWTEST_F_L0(JSObjectTest, Create)
51 {
52 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
53 JSHandle<JSObject> jsobject =
54 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
55 EXPECT_TRUE(*jsobject != nullptr);
56 }
57
HWTEST_F_L0(JSObjectTest,SetProperty)58 HWTEST_F_L0(JSObjectTest, SetProperty)
59 {
60 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
61 JSHandle<JSObject> jsobject =
62 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
63 EXPECT_TRUE(*jsobject != nullptr);
64
65 char array[] = "x";
66 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
67 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
68
69 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(jsobject), key, value);
70 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(jsobject), key).GetValue()->GetInt(), 1);
71
72 JSHandle<JSTaggedValue> value2(thread, JSTaggedValue(2));
73 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(jsobject), key, value2);
74 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(jsobject), key).GetValue()->GetInt(), 2);
75 }
76
HWTEST_F_L0(JSObjectTest,GetProperty)77 HWTEST_F_L0(JSObjectTest, GetProperty)
78 {
79 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
80 JSHandle<JSObject> obj =
81 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
82 EXPECT_TRUE(*obj != nullptr);
83
84 char array[] = "x";
85 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
86 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
87
88 EXPECT_TRUE(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->IsUndefined());
89
90 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, value);
91 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt(), 1);
92 }
93
HWTEST_F_L0(JSObjectTest,DeleteProperty)94 HWTEST_F_L0(JSObjectTest, DeleteProperty)
95 {
96 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
97 JSHandle<JSObject> obj =
98 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
99 EXPECT_TRUE(*obj != nullptr);
100
101 char array[] = "print";
102 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
103 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
104
105 JSObject::DeleteProperty(thread, (obj), key);
106 EXPECT_TRUE(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->IsUndefined());
107
108 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, value);
109 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt(), 1);
110
111 JSHandle<JSTaggedValue> key2(thread->GetEcmaVM()->GetFactory()->NewFromASCII("print_test"));
112 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key2,
113 JSHandle<JSTaggedValue>(thread, JSTaggedValue(10)));
114 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key2).GetValue()->GetInt(), 10);
115
116 JSObject::DeleteProperty(thread, (obj), key);
117 EXPECT_TRUE(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->IsUndefined());
118 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key2).GetValue()->GetInt(), 10);
119 }
120
HWTEST_F_L0(JSObjectTest,DeletePropertyGlobal)121 HWTEST_F_L0(JSObjectTest, DeletePropertyGlobal)
122 {
123 JSHandle<GlobalEnv> globalEnv = thread->GetEcmaVM()->GetGlobalEnv();
124 JSHandle<JSTaggedValue> global(thread, globalEnv->GetGlobalObject());
125 JSHandle<JSTaggedValue> printKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("print"));
126 JSHandle<JSTaggedValue> printTestKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("print_test"));
127
128 JSHandle<JSTaggedValue> value = JSObject::GetProperty(thread, global, printKey).GetValue();
129
130 JSObject::SetProperty(thread, global, printTestKey, value);
131
132 JSTaggedValue val2 = JSObject::GetProperty(thread, global, printTestKey).GetValue().GetTaggedValue();
133 EXPECT_EQ(val2, value.GetTaggedValue());
134 JSTaggedValue::DeletePropertyOrThrow(thread, global, printTestKey);
135 JSTaggedValue val3 = JSObject::GetProperty(thread, global, printKey).GetValue().GetTaggedValue();
136 EXPECT_NE(val3, JSTaggedValue::Undefined());
137 }
138
GetPropertyHasOwnPeroperty(JSThread * thread,bool hasOwnProperty=false)139 void GetPropertyHasOwnPeroperty(JSThread *thread, bool hasOwnProperty = false)
140 {
141 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
142 JSHandle<JSObject> grandfather = JSObject::ObjectCreate(thread, nullHandle);
143 JSHandle<JSObject> father = JSObject::ObjectCreate(thread, grandfather);
144 JSHandle<JSObject> son = JSObject::ObjectCreate(thread, father);
145
146 JSHandle<JSTaggedValue> sonKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
147 JSHandle<JSTaggedValue> fatherKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
148 JSHandle<JSTaggedValue> grandfatherKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
149 JSHandle<JSTaggedValue> sonValue(thread, JSTaggedValue(1)); // 1 : value
150 JSHandle<JSTaggedValue> fatherValue(thread, JSTaggedValue(2)); // 2 : value
151 JSHandle<JSTaggedValue> grandfatherValue(thread, JSTaggedValue(3)); // 3 : value
152
153 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(son), sonKey, sonValue);
154 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(father), fatherKey, fatherValue);
155 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(grandfather), grandfatherKey, grandfatherValue);
156
157 if (hasOwnProperty) {
158 bool flag = JSTaggedValue::HasOwnProperty(thread, JSHandle<JSTaggedValue>::Cast(son), sonKey);
159 EXPECT_TRUE(flag);
160 flag = JSTaggedValue::HasOwnProperty(thread, JSHandle<JSTaggedValue>::Cast(son), fatherKey);
161 EXPECT_FALSE(flag);
162 flag = JSTaggedValue::HasOwnProperty(thread, JSHandle<JSTaggedValue>::Cast(son), grandfatherKey);
163 EXPECT_FALSE(flag);
164 } else {
165 EXPECT_EQ(sonValue.GetTaggedValue(),
166 JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(son), sonKey).GetValue().GetTaggedValue());
167 EXPECT_EQ(fatherValue.GetTaggedValue(),
168 JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(son), fatherKey).GetValue().GetTaggedValue());
169 EXPECT_EQ(
170 grandfatherValue.GetTaggedValue(),
171 JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(son), grandfatherKey).GetValue().GetTaggedValue());
172 }
173 }
174
HWTEST_F_L0(JSObjectTest,GetPropertyInPrototypeChain)175 HWTEST_F_L0(JSObjectTest, GetPropertyInPrototypeChain)
176 {
177 GetPropertyHasOwnPeroperty(thread);
178 }
179
HWTEST_F_L0(JSObjectTest,PropertyAttribute)180 HWTEST_F_L0(JSObjectTest, PropertyAttribute)
181 {
182 JSHandle<JSTaggedValue> constructor(thread, JSObjectTestCreate(thread));
183 JSHandle<JSObject> obj1 =
184 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(constructor), constructor);
185 JSHandle<JSObject> obj2 =
186 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(constructor), constructor);
187
188 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
189 JSHandle<JSTaggedValue> key2(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
190
191 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
192 JSHandle<JSTaggedValue> value2(thread, JSTaggedValue(2));
193
194 // test set property
195 PropertyDescriptor desc(thread);
196 desc.SetValue(value1);
197 desc.SetWritable(false);
198 JSObject::DefineOwnProperty(thread, obj1, key1, desc);
199 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value2);
200 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key1, value1);
201 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key1, value2);
202 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1).GetValue().GetTaggedValue(),
203 value1.GetTaggedValue());
204 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj2), key1).GetValue().GetTaggedValue(),
205 value2.GetTaggedValue());
206
207 // test delete property
208 PropertyDescriptor desc1(thread);
209 desc1.SetValue(value1);
210 desc1.SetConfigurable(false);
211 JSObject::DefineOwnProperty(thread, obj1, key2, desc1);
212 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key2, value1);
213 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key2, value1);
214 JSObject::DeleteProperty(thread, (obj1), key2);
215 JSObject::DeleteProperty(thread, (obj2), key2);
216 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key2).GetValue().GetTaggedValue(),
217 value1.GetTaggedValue());
218 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj2), key2).GetValue().GetTaggedValue(),
219 JSTaggedValue::Undefined());
220 }
221
HWTEST_F_L0(JSObjectTest,CreateDataProperty)222 HWTEST_F_L0(JSObjectTest, CreateDataProperty)
223 {
224 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
225 JSHandle<JSObject> obj =
226 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
227 EXPECT_TRUE(*obj != nullptr);
228
229 char array[] = "x";
230 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
231 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
232
233 bool success = JSObject::CreateDataProperty(thread, obj, key, value);
234 EXPECT_TRUE(success);
235
236 success = JSTaggedValue::HasOwnProperty(thread, JSHandle<JSTaggedValue>::Cast(obj), key);
237 EXPECT_TRUE(success);
238
239 PropertyDescriptor desc(thread);
240 success = JSObject::GetOwnProperty(thread, obj, key, desc);
241 EXPECT_TRUE(success);
242 EXPECT_EQ(true, desc.IsWritable());
243 EXPECT_EQ(true, desc.IsEnumerable());
244 EXPECT_EQ(true, desc.IsConfigurable());
245 }
246
HWTEST_F_L0(JSObjectTest,CreateMethodProperty)247 HWTEST_F_L0(JSObjectTest, CreateMethodProperty)
248 {
249 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
250 JSHandle<JSObject> obj =
251 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
252 EXPECT_TRUE(*obj != nullptr);
253
254 char array[] = "x";
255 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
256 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
257
258 bool success = JSObject::CreateMethodProperty(thread, obj, key, value);
259 EXPECT_TRUE(success);
260
261 success = JSTaggedValue::HasOwnProperty(thread, JSHandle<JSTaggedValue>::Cast(obj), key);
262 EXPECT_TRUE(success);
263
264 PropertyDescriptor desc(thread);
265 success = JSObject::GetOwnProperty(thread, obj, key, desc);
266 EXPECT_TRUE(success);
267 EXPECT_EQ(true, desc.IsWritable());
268 EXPECT_EQ(false, desc.IsEnumerable());
269 EXPECT_EQ(true, desc.IsConfigurable());
270 }
271
HWTEST_F_L0(JSObjectTest,DefinePropertyOrThrow)272 HWTEST_F_L0(JSObjectTest, DefinePropertyOrThrow)
273 {
274 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
275 JSHandle<JSObject> obj =
276 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
277 EXPECT_TRUE(*obj != nullptr);
278
279 char array[] = "x";
280 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
281
282 PropertyDescriptor desc1(thread, JSHandle<JSTaggedValue>(thread, JSTaggedValue(1)), true, true, true);
283 bool success = JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, desc1);
284 EXPECT_TRUE(success);
285 PropertyDescriptor descRes1(thread);
286 success = JSObject::GetOwnProperty(thread, obj, key, descRes1);
287 EXPECT_TRUE(success);
288 EXPECT_EQ(1, descRes1.GetValue()->GetInt());
289 EXPECT_EQ(true, descRes1.IsWritable());
290 EXPECT_EQ(true, descRes1.IsEnumerable());
291 EXPECT_EQ(true, descRes1.IsConfigurable());
292
293 PropertyDescriptor desc2(thread, false, true, true);
294 success = JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, desc2);
295 EXPECT_TRUE(success);
296 PropertyDescriptor descRes2(thread);
297 success = JSObject::GetOwnProperty(thread, obj, key, descRes2);
298 EXPECT_TRUE(success);
299 EXPECT_EQ(1, descRes2.GetValue()->GetInt());
300 EXPECT_EQ(false, descRes2.IsWritable());
301 EXPECT_EQ(true, descRes2.IsEnumerable());
302 EXPECT_EQ(true, descRes2.IsConfigurable());
303
304 PropertyDescriptor desc3(thread);
305 desc3.SetWritable(false);
306 desc3.SetEnumerable(false);
307 desc3.SetConfigurable(false);
308 success = JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, desc3);
309 EXPECT_TRUE(success);
310 PropertyDescriptor descRes3(thread);
311 success = JSObject::GetOwnProperty(thread, obj, key, descRes3);
312 EXPECT_TRUE(success);
313 EXPECT_EQ(1, descRes3.GetValue()->GetInt());
314 EXPECT_EQ(false, descRes3.IsWritable());
315 EXPECT_EQ(false, descRes3.IsEnumerable());
316 EXPECT_EQ(false, descRes3.IsConfigurable());
317
318 PropertyDescriptor desc4(thread, JSHandle<JSTaggedValue>(thread, JSTaggedValue(2)));
319 success = JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, desc4);
320 EXPECT_FALSE(success);
321 }
322
HWTEST_F_L0(JSObjectTest,HasProperty)323 HWTEST_F_L0(JSObjectTest, HasProperty)
324 {
325 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
326 JSHandle<JSObject> obj =
327 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
328 EXPECT_TRUE(*obj != nullptr);
329
330 char array[] = "x";
331 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII(array));
332 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
333
334 bool flag = JSObject::HasProperty(thread, obj, key);
335 EXPECT_FALSE(flag);
336
337 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, value);
338 flag = JSObject::HasProperty(thread, obj, key);
339 EXPECT_TRUE(flag);
340
341 JSObject::DeleteProperty(thread, (obj), key);
342 flag = JSObject::HasProperty(thread, obj, key);
343 EXPECT_FALSE(flag);
344 }
345
HWTEST_F_L0(JSObjectTest,HasPropertyWithProtoType)346 HWTEST_F_L0(JSObjectTest, HasPropertyWithProtoType)
347 {
348 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
349 JSHandle<JSObject> grandfather = JSObject::ObjectCreate(thread, nullHandle);
350 JSHandle<JSObject> father = JSObject::ObjectCreate(thread, grandfather);
351 JSHandle<JSObject> son = JSObject::ObjectCreate(thread, father);
352
353 JSHandle<JSTaggedValue> testGrand(thread,
354 JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(grandfather)));
355 JSHandle<JSTaggedValue> testFather(thread, JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(father)));
356 JSHandle<JSTaggedValue> testSon(thread, JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(son)));
357 EXPECT_TRUE(testSon.GetTaggedValue() != testFather.GetTaggedValue());
358 EXPECT_TRUE(testGrand.GetTaggedValue() != testFather.GetTaggedValue());
359 JSHandle<JSTaggedValue> sonKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
360 JSHandle<JSTaggedValue> fatherKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
361 JSHandle<JSTaggedValue> grandfatherKey(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
362 JSHandle<JSTaggedValue> sonValue(thread, JSTaggedValue(1));
363 JSHandle<JSTaggedValue> fatherValue(thread, JSTaggedValue(2));
364 JSHandle<JSTaggedValue> grandfatherValue(thread, JSTaggedValue(3));
365
366 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(son), sonKey, sonValue);
367 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(father), fatherKey, fatherValue);
368 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(grandfather), grandfatherKey, grandfatherValue);
369
370 bool flag = JSObject::HasProperty(thread, son, sonKey);
371 EXPECT_TRUE(flag);
372 flag = JSObject::HasProperty(thread, son, fatherKey);
373 EXPECT_TRUE(flag);
374 flag = JSObject::HasProperty(thread, son, grandfatherKey);
375 EXPECT_TRUE(flag);
376 }
377
HWTEST_F_L0(JSObjectTest,HasOwnProperty)378 HWTEST_F_L0(JSObjectTest, HasOwnProperty)
379 {
380 GetPropertyHasOwnPeroperty(thread, true);
381 }
382
HWTEST_F_L0(JSObjectTest,GetOwnPropertyKeys)383 HWTEST_F_L0(JSObjectTest, GetOwnPropertyKeys)
384 {
385 JSHandle<JSTaggedValue> constructor(thread, JSObjectTestCreate(thread));
386 JSHandle<JSObject> obj =
387 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(constructor), constructor);
388
389 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII("x"));
390 JSHandle<JSTaggedValue> key2(thread->GetEcmaVM()->GetFactory()->NewFromASCII("y"));
391 JSHandle<JSTaggedValue> key3(thread->GetEcmaVM()->GetFactory()->NewFromASCII("3"));
392 JSHandle<JSTaggedValue> key4(thread->GetEcmaVM()->GetFactory()->NewFromASCII("4"));
393 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
394 JSHandle<JSTaggedValue> value2(thread, JSTaggedValue(2));
395 JSHandle<JSTaggedValue> value3(thread, JSTaggedValue(3));
396 JSHandle<JSTaggedValue> value4(thread, JSTaggedValue(4));
397
398 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key1, value1);
399 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key2, value2);
400 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key3, value3);
401 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key4, value4);
402
403 JSHandle<TaggedArray> array = JSObject::GetOwnPropertyKeys(thread, obj);
404 uint32_t length = array->GetLength();
405 EXPECT_EQ(length, 4U);
406 int sum = 0;
407 for (uint32_t i = 0; i < length; i++) {
408 JSHandle<JSTaggedValue> key(thread, array->Get(i));
409 sum += JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt();
410 }
411 EXPECT_EQ(sum, 10);
412 }
413
HWTEST_F_L0(JSObjectTest,ObjectCreateMethod)414 HWTEST_F_L0(JSObjectTest, ObjectCreateMethod)
415 {
416 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
417 JSHandle<JSObject> grandfather = JSObject::ObjectCreate(thread, nullHandle);
418 JSHandle<JSObject> father = JSObject::ObjectCreate(thread, grandfather);
419 JSHandle<JSObject> son = JSObject::ObjectCreate(thread, father);
420
421 EXPECT_EQ(JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(son)), father.GetTaggedValue());
422 EXPECT_EQ(JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(father)), grandfather.GetTaggedValue());
423 EXPECT_EQ(JSTaggedValue::GetPrototype(thread, JSHandle<JSTaggedValue>(grandfather)), JSTaggedValue::Null());
424 }
425
HWTEST_F_L0(JSObjectTest,GetMethod)426 HWTEST_F_L0(JSObjectTest, GetMethod)
427 {
428 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
429 JSHandle<JSObject> obj = JSObject::ObjectCreate(thread, nullHandle);
430 JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
431 JSHandle<JSTaggedValue> func(thread->GetEcmaVM()->GetFactory()->NewJSFunction(env));
432 JSHandle<JSFunction>::Cast(func)->GetJSHClass()->SetCallable(true);
433 EXPECT_TRUE(*func != nullptr);
434 JSHandle<JSTaggedValue> key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("1"));
435 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, func);
436 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue().GetTaggedValue(),
437 func.GetTaggedValue());
438 }
439
HWTEST_F_L0(JSObjectTest,EnumerableOwnNames)440 HWTEST_F_L0(JSObjectTest, EnumerableOwnNames)
441 {
442 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
443 JSHandle<JSObject> obj =
444 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
445 EXPECT_TRUE(*obj != nullptr);
446
447 CString tagCStr = "x";
448 JSHandle<EcmaString> tagString = thread->GetEcmaVM()->GetFactory()->NewFromASCII(&tagCStr[0]);
449 JSHandle<JSTaggedValue> key(tagString);
450
451 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
452
453 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, value);
454 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt(), 1);
455
456 JSHandle<TaggedArray> names = JSObject::EnumerableOwnNames(thread, obj);
457
458 JSHandle<JSTaggedValue> keyFromNames(thread, JSTaggedValue(names->Get(0)));
459 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), keyFromNames).GetValue()->GetInt(), 1);
460
461 PropertyDescriptor descNoEnum(thread);
462 descNoEnum.SetEnumerable(false);
463 JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, descNoEnum);
464 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt(), 1);
465
466 JSHandle<TaggedArray> namesNoEnum = JSObject::EnumerableOwnNames(thread, obj);
467 EXPECT_TRUE(namesNoEnum->GetLength() == 0U);
468
469 PropertyDescriptor descEnum(thread);
470 descEnum.SetConfigurable(false);
471 descEnum.SetEnumerable(true);
472 JSTaggedValue::DefinePropertyOrThrow(thread, JSHandle<JSTaggedValue>(obj), key, descEnum);
473 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue()->GetInt(), 1);
474
475 JSHandle<TaggedArray> namesNoConfig = JSObject::EnumerableOwnNames(thread, obj);
476
477 JSHandle<JSTaggedValue> keyNoConfig(thread, JSTaggedValue(namesNoConfig->Get(0)));
478 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), keyNoConfig).GetValue()->GetInt(), 1);
479 }
480
HWTEST_F_L0(JSObjectTest,SetIntegrityLevelSealed)481 HWTEST_F_L0(JSObjectTest, SetIntegrityLevelSealed)
482 {
483 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
484 JSHandle<JSObject> obj1 =
485 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
486 EXPECT_TRUE(*obj1 != nullptr);
487 CString undefinedCStr = "x";
488 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII(&undefinedCStr[0]));
489 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
490 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value1);
491
492 // test SetIntegrityLevel::SEALED
493 JSHandle<JSObject> jsobject(obj1);
494 bool status1 = JSObject::SetIntegrityLevel(thread, jsobject, IntegrityLevel::SEALED);
495 EXPECT_TRUE(status1);
496 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1).GetValue().GetTaggedValue(),
497 value1.GetTaggedValue());
498 PropertyDescriptor desc1(thread);
499 bool success1 = JSObject::GetOwnProperty(thread, jsobject, key1, desc1);
500 EXPECT_TRUE(success1);
501 EXPECT_EQ(true, desc1.IsWritable());
502 EXPECT_EQ(true, desc1.IsEnumerable());
503 EXPECT_EQ(false, desc1.IsConfigurable());
504 }
505
HWTEST_F_L0(JSObjectTest,SetIntegrityLevelFrozen)506 HWTEST_F_L0(JSObjectTest, SetIntegrityLevelFrozen)
507 {
508 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
509 JSHandle<JSObject> obj1 =
510 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
511 EXPECT_TRUE(*obj1 != nullptr);
512
513 CString undefinedCStr = "x";
514 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII(&undefinedCStr[0]));
515 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
516 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value1);
517
518 // test SetIntegrityLevel::FROZEN
519 bool status1 = JSObject::SetIntegrityLevel(thread, obj1, IntegrityLevel::FROZEN);
520 EXPECT_TRUE(status1);
521 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1).GetValue().GetTaggedValue(),
522 value1.GetTaggedValue());
523 PropertyDescriptor desc1(thread);
524 bool success1 = JSObject::GetOwnProperty(thread, obj1, key1, desc1);
525 EXPECT_TRUE(success1);
526 EXPECT_EQ(false, desc1.IsWritable());
527 EXPECT_EQ(true, desc1.IsEnumerable());
528 EXPECT_EQ(false, desc1.IsConfigurable());
529 }
530
HWTEST_F_L0(JSObjectTest,TestIntegrityLevelSealed)531 HWTEST_F_L0(JSObjectTest, TestIntegrityLevelSealed)
532 {
533 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
534 JSHandle<JSObject> obj1 =
535 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
536 CString undefinedCStr = "level";
537 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII(&undefinedCStr[0]));
538 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
539 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value1);
540 obj1->GetJSHClass()->SetExtensible(false);
541
542 // test SetIntegrityLevel::SEALED
543 bool status1 = JSObject::SetIntegrityLevel(thread, obj1, IntegrityLevel::SEALED);
544 EXPECT_TRUE(status1);
545 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1).GetValue().GetTaggedValue(),
546 value1.GetTaggedValue());
547
548 PropertyDescriptor desc1(thread);
549 bool success1 = JSObject::GetOwnProperty(thread, obj1, key1, desc1);
550 EXPECT_TRUE(success1);
551 EXPECT_EQ(true, JSObject::TestIntegrityLevel(thread, obj1, IntegrityLevel::SEALED));
552 EXPECT_EQ(false, JSObject::TestIntegrityLevel(thread, obj1, IntegrityLevel::FROZEN));
553 }
554
HWTEST_F_L0(JSObjectTest,TestIntegrityLevelFrozen)555 HWTEST_F_L0(JSObjectTest, TestIntegrityLevelFrozen)
556 {
557 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
558 JSHandle<JSObject> obj1 =
559 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
560 CString undefinedCStr = "level";
561 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII(&undefinedCStr[0]));
562 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
563 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value1);
564 obj1->GetJSHClass()->SetExtensible(false);
565
566 // test SetIntegrityLevel::FROZEN
567 bool status1 = JSObject::SetIntegrityLevel(thread, obj1, IntegrityLevel::FROZEN);
568 EXPECT_TRUE(status1);
569 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1).GetValue().GetTaggedValue(),
570 value1.GetTaggedValue());
571
572 PropertyDescriptor desc1(thread);
573 bool success1 = JSObject::GetOwnProperty(thread, obj1, key1, desc1);
574 EXPECT_TRUE(success1);
575 EXPECT_EQ(true, JSObject::TestIntegrityLevel(thread, obj1, IntegrityLevel::SEALED));
576 EXPECT_EQ(true, JSObject::TestIntegrityLevel(thread, obj1, IntegrityLevel::FROZEN));
577 }
578
HWTEST_F_L0(JSObjectTest,TestIntegrityLevelWithoutProperty)579 HWTEST_F_L0(JSObjectTest, TestIntegrityLevelWithoutProperty)
580 {
581 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
582 JSHandle<JSTaggedValue> obj1(
583 thread->GetEcmaVM()->GetFactory()->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1));
584 JSHandle<JSObject>::Cast(obj1)->GetJSHClass()->SetExtensible(false);
585 CString undefinedCStr = "level";
586 JSHandle<JSTaggedValue> key1(thread->GetEcmaVM()->GetFactory()->NewFromASCII(&undefinedCStr[0]));
587
588 // test SetIntegrityLevel::FROZEN
589 JSHandle<JSObject> jsobject(obj1);
590 bool status1 = JSObject::SetIntegrityLevel(thread, jsobject, IntegrityLevel::SEALED);
591 EXPECT_TRUE(status1);
592
593 PropertyDescriptor desc1(thread);
594 bool success1 = JSObject::GetOwnProperty(thread, jsobject, key1, desc1);
595 EXPECT_TRUE(!success1);
596 EXPECT_EQ(true, JSObject::TestIntegrityLevel(thread, jsobject, IntegrityLevel::SEALED));
597 EXPECT_EQ(true, JSObject::TestIntegrityLevel(thread, jsobject, IntegrityLevel::FROZEN));
598 }
599
TestGetter(EcmaRuntimeCallInfo * argv)600 JSTaggedValue TestGetter(EcmaRuntimeCallInfo *argv)
601 {
602 auto thread = argv->GetThread();
603 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
604 JSHandle<JSObject> obj(BuiltinsBase::GetThis(argv));
605 JSHandle<JSTaggedValue> key(factory->NewFromASCII("y"));
606 JSTaggedValue value = JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue().GetTaggedValue();
607
608 return JSTaggedValue(value.GetInt() + 1);
609 }
610
HWTEST_F_L0(JSObjectTest,Getter)611 HWTEST_F_L0(JSObjectTest, Getter)
612 {
613 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
614 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
615 JSHandle<JSObject> obj = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
616 JSHandle<JSTaggedValue> key1(factory->NewFromASCII("x"));
617 JSHandle<JSTaggedValue> key2(factory->NewFromASCII("y"));
618 JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
619 JSHandle<JSFunction> getter =
620 thread->GetEcmaVM()->GetFactory()->NewJSFunction(env, reinterpret_cast<void *>(TestGetter));
621
622 PropertyDescriptor desc1(thread);
623 desc1.SetGetter(JSHandle<JSTaggedValue>::Cast(getter));
624 bool success1 = JSObject::DefineOwnProperty(thread, obj, key1, desc1);
625 EXPECT_TRUE(success1);
626
627 PropertyDescriptor desc2(thread);
628 desc2.SetValue(JSHandle<JSTaggedValue>(thread, JSTaggedValue(1)));
629 success1 = JSObject::DefineOwnProperty(thread, obj, key2, desc2);
630 EXPECT_TRUE(success1);
631
632 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key1).GetValue().GetTaggedValue(),
633 JSTaggedValue(2));
634 }
635
TestSetter(EcmaRuntimeCallInfo * argv)636 JSTaggedValue TestSetter(EcmaRuntimeCallInfo *argv)
637 {
638 JSThread *thread = argv->GetThread();
639 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
640 JSHandle<JSObject> obj(BuiltinsBase::GetThis(argv));
641 JSHandle<JSTaggedValue> key(factory->NewFromASCII("y"));
642 JSTaggedValue value(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue().GetTaggedValue());
643 JSHandle<JSTaggedValue> valueHandle(thread, JSTaggedValue(value.GetInt() + 1));
644 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key, valueHandle);
645
646 return JSTaggedValue(JSTaggedValue::True());
647 }
648
HWTEST_F_L0(JSObjectTest,Setter)649 HWTEST_F_L0(JSObjectTest, Setter)
650 {
651 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
652 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
653 JSHandle<JSObject> obj = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
654 JSHandle<JSTaggedValue> key1(factory->NewFromASCII("x"));
655 JSHandle<JSTaggedValue> key2(factory->NewFromASCII("y"));
656 JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
657 JSHandle<JSFunction> setter =
658 thread->GetEcmaVM()->GetFactory()->NewJSFunction(env, reinterpret_cast<void *>(TestSetter));
659
660 PropertyDescriptor desc1(thread);
661 desc1.SetSetter(JSHandle<JSTaggedValue>::Cast(setter));
662 bool success1 = JSObject::DefineOwnProperty(thread, obj, key1, desc1);
663 EXPECT_TRUE(success1);
664
665 PropertyDescriptor desc2(thread, JSHandle<JSTaggedValue>(thread, JSTaggedValue(1)), true, true, true);
666 success1 = JSObject::DefineOwnProperty(thread, obj, key2, desc2);
667 EXPECT_TRUE(success1);
668
669 JSHandle<JSTaggedValue> valueHandle(thread, JSTaggedValue::Undefined());
670 EXPECT_TRUE(JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj), key1, valueHandle));
671 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key2).GetValue().GetTaggedValue(),
672 JSTaggedValue(2));
673 }
674
HWTEST_F_L0(JSObjectTest,SpeciesConstructor)675 HWTEST_F_L0(JSObjectTest, SpeciesConstructor)
676 {
677 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
678 JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
679 const GlobalEnvConstants *globalConst = thread->GlobalConstants();
680 JSHandle<JSFunction> constructorFunc =
681 factory->NewJSFunction(env, static_cast<void *>(nullptr), FunctionKind::BASE_CONSTRUCTOR);
682 JSHandle<JSTaggedValue> constructorFuncValue(constructorFunc);
683 constructorFunc->GetJSHClass()->SetExtensible(true);
684 JSFunction::NewJSFunctionPrototype(thread, constructorFunc);
685
686 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
687 JSHandle<JSTaggedValue> undefinedValue(thread, JSTaggedValue::Undefined());
688 JSHandle<JSObject> protoObj = JSObject::ObjectCreate(thread, nullHandle);
689 JSHandle<JSTaggedValue> protoObjValue(protoObj);
690
691 JSHandle<JSTaggedValue> constructorKey = globalConst->GetHandledConstructorString();
692 JSObject::SetProperty(thread, protoObjValue, constructorKey, constructorFuncValue);
693
694 factory->NewJSObjectByConstructor(constructorFunc, JSHandle<JSTaggedValue>::Cast(constructorFunc));
695 JSHandle<JSFunction> speciesConstruct =
696 factory->NewJSFunction(env, static_cast<void *>(nullptr), FunctionKind::BASE_CONSTRUCTOR);
697 JSHandle<JSTaggedValue> speciesConstructValue(speciesConstruct);
698 constructorFunc->GetJSHClass()->SetExtensible(true);
699 JSFunction::MakeConstructor(thread, speciesConstruct, undefinedValue);
700
701 JSHandle<JSTaggedValue> speciesSymbol = env->GetSpeciesSymbol();
702 JSObject::SetProperty(thread, constructorFuncValue, speciesSymbol, speciesConstructValue);
703
704 JSTaggedValue speciesValue =
705 JSObject::SpeciesConstructor(thread, protoObj, constructorFuncValue).GetTaggedValue();
706 EXPECT_EQ(speciesValue, speciesConstructValue.GetTaggedValue());
707 }
708
TestUndefinedGetter(EcmaRuntimeCallInfo * argv)709 JSTaggedValue TestUndefinedGetter([[maybe_unused]] EcmaRuntimeCallInfo *argv)
710 {
711 // 10 : test case
712 return JSTaggedValue(10);
713 }
714
TestUndefinedSetter(EcmaRuntimeCallInfo * argv)715 JSTaggedValue TestUndefinedSetter([[maybe_unused]] EcmaRuntimeCallInfo *argv)
716 {
717 // 10 : test case
718 return JSTaggedValue(10);
719 }
720
GetterOrSetterIsUndefined(JSThread * thread,JSHandle<JSTaggedValue> & key)721 JSHandle<JSObject> GetterOrSetterIsUndefined(JSThread *thread, JSHandle<JSTaggedValue>& key)
722 {
723 JSHandle<JSTaggedValue> hclass1(thread, JSObjectTestCreate(thread));
724 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
725 JSHandle<JSObject> obj = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(hclass1), hclass1);
726
727 JSHandle<GlobalEnv> env = thread->GetEcmaVM()->GetGlobalEnv();
728 JSHandle<JSFunction> getter =
729 thread->GetEcmaVM()->GetFactory()->NewJSFunction(env, reinterpret_cast<void *>(TestUndefinedGetter));
730 JSHandle<JSFunction> setter =
731 thread->GetEcmaVM()->GetFactory()->NewJSFunction(env, reinterpret_cast<void *>(TestUndefinedSetter));
732
733 PropertyDescriptor desc1(thread);
734 desc1.SetGetter(JSHandle<JSTaggedValue>::Cast(getter));
735 desc1.SetSetter(JSHandle<JSTaggedValue>::Cast(setter));
736 desc1.SetConfigurable(true);
737 desc1.SetEnumerable(true);
738 bool success1 = JSObject::DefineOwnProperty(thread, obj, key, desc1);
739 EXPECT_TRUE(success1);
740 return obj;
741 }
742
HWTEST_F_L0(JSObjectTest,GetterIsUndefined)743 HWTEST_F_L0(JSObjectTest, GetterIsUndefined)
744 {
745 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
746 JSHandle<JSTaggedValue> unGetter(thread, JSTaggedValue::Undefined());
747 JSHandle<JSTaggedValue> key(factory->NewFromASCII("property"));
748 auto obj = GetterOrSetterIsUndefined(thread, key);
749 PropertyDescriptor desc2(thread);
750 desc2.SetGetter(unGetter);
751 bool success2 = JSObject::DefineOwnProperty(thread, obj, key, desc2);
752 EXPECT_TRUE(success2);
753
754 PropertyDescriptor desc(thread);
755 bool success = JSObject::GetOwnProperty(thread, obj, key, desc);
756 EXPECT_TRUE(success);
757 EXPECT_TRUE(desc.GetSetter()->IsJSFunction());
758 EXPECT_TRUE(desc.GetGetter()->IsUndefined());
759 }
760
HWTEST_F_L0(JSObjectTest,SetterIsUndefined)761 HWTEST_F_L0(JSObjectTest, SetterIsUndefined)
762 {
763 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
764 JSHandle<JSTaggedValue> unSetter(thread, JSTaggedValue::Undefined());
765 JSHandle<JSTaggedValue> key(factory->NewFromASCII("property"));
766 auto obj = GetterOrSetterIsUndefined(thread, key);
767 PropertyDescriptor desc2(thread);
768 desc2.SetSetter(unSetter);
769 bool success2 = JSObject::DefineOwnProperty(thread, obj, key, desc2);
770 EXPECT_TRUE(success2);
771
772 PropertyDescriptor desc(thread);
773 bool success = JSObject::GetOwnProperty(thread, obj, key, desc);
774 EXPECT_TRUE(success);
775 EXPECT_TRUE(desc.GetSetter()->IsUndefined());
776
777 EXPECT_EQ(JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj), key).GetValue().GetTaggedValue(),
778 JSTaggedValue(10));
779 }
780
HWTEST_F_L0(JSObjectTest,Transitions)781 HWTEST_F_L0(JSObjectTest, Transitions)
782 {
783 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
784 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
785 JSHandle<JSObject> obj1 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
786 JSHandle<JSObject> obj2 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
787
788 JSHandle<JSHClass> hc1(thread, obj1->GetJSHClass());
789 JSHandle<JSHClass> hc2(thread, obj2->GetJSHClass());
790 EXPECT_EQ(hc1.GetTaggedValue(), hc2.GetTaggedValue());
791
792 JSHandle<JSTaggedValue> key1(factory->NewFromASCII("x"));
793 JSHandle<JSTaggedValue> key2(factory->NewFromASCII("y"));
794 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
795
796 // key1
797 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, value);
798 JSHandle<JSHClass> hc3(thread, obj1->GetJSHClass());
799 EXPECT_NE(hc1.GetTaggedValue(), hc3.GetTaggedValue());
800
801 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key1, value);
802 JSHandle<JSHClass> hc4(thread, obj2->GetJSHClass());
803 EXPECT_EQ(hc3.GetTaggedValue(), hc4.GetTaggedValue());
804
805 // key2
806 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key2, value);
807 JSHandle<JSHClass> hc5(thread, obj1->GetJSHClass());
808 EXPECT_NE(hc3.GetTaggedValue(), hc5.GetTaggedValue());
809
810 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key2, value);
811 JSHandle<JSHClass> hc6(thread, obj2->GetJSHClass());
812 EXPECT_EQ(hc5.GetTaggedValue(), hc6.GetTaggedValue());
813 }
814
HWTEST_F_L0(JSObjectTest,FastToSlow)815 HWTEST_F_L0(JSObjectTest, FastToSlow)
816 {
817 auto ecmaVM = thread->GetEcmaVM();
818 ObjectFactory *factory = ecmaVM->GetFactory();
819
820 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
821 JSHandle<JSObject> obj1 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
822
823 JSMutableHandle<EcmaString> key(thread, factory->NewFromASCII("x"));
824 JSMutableHandle<JSTaggedValue> number(thread, JSTaggedValue(0));
825 JSMutableHandle<JSTaggedValue> newkey(thread, JSTaggedValue(0));
826 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
827
828 ecmaVM->SetEnableForceGC(false);
829 for (uint32_t i = 0; i < PropertyAttributes::MAX_FAST_PROPS_CAPACITY; i++) {
830 number.Update(JSTaggedValue(i));
831 number.Update(JSTaggedValue::ToString(thread, number).GetTaggedValue());
832 EcmaString *newString = *factory->ConcatFromString(key, JSTaggedValue::ToString(thread, number));
833 newkey.Update(JSTaggedValue(newString));
834 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), newkey, value);
835 }
836 ecmaVM->SetEnableForceGC(true);
837
838 EXPECT_FALSE(TaggedArray::Cast(obj1->GetProperties().GetTaggedObject())->IsDictionaryMode());
839
840 number.Update(JSTaggedValue(PropertyAttributes::MAX_FAST_PROPS_CAPACITY));
841 number.Update(JSTaggedValue::ToString(thread, number).GetTaggedValue());
842 EcmaString *newString = *factory->ConcatFromString(key, JSTaggedValue::ToString(thread, number));
843 newkey.Update(JSTaggedValue(newString));
844 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), newkey, value);
845
846 EXPECT_TRUE(TaggedArray::Cast(obj1->GetProperties().GetTaggedObject())->IsDictionaryMode());
847 NameDictionary *dict = NameDictionary::Cast(obj1->GetProperties().GetTaggedObject());
848 EXPECT_EQ(dict->EntriesCount(), static_cast<int>(PropertyAttributes::MAX_FAST_PROPS_CAPACITY + 1));
849 EXPECT_EQ(dict->NextEnumerationIndex(thread),
850 static_cast<int>(PropertyAttributes::MAX_FAST_PROPS_CAPACITY + 1));
851 }
852
HWTEST_F_L0(JSObjectTest,DeleteMiddle)853 HWTEST_F_L0(JSObjectTest, DeleteMiddle)
854 {
855 auto ecmaVM = thread->GetEcmaVM();
856 ObjectFactory *factory = ecmaVM->GetFactory();
857
858 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
859 JSHandle<JSObject> obj1 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
860
861 JSMutableHandle<EcmaString> key(thread, factory->NewFromASCII("x"));
862 JSMutableHandle<JSTaggedValue> number(thread, JSTaggedValue(0));
863 JSMutableHandle<JSTaggedValue> newkey(thread, JSTaggedValue(0));
864 JSHandle<JSTaggedValue> value(thread, JSTaggedValue(1));
865
866 for (uint32_t i = 0; i < 10; i++) {
867 number.Update(JSTaggedValue(i));
868 number.Update(JSTaggedValue::ToString(thread, number).GetTaggedValue());
869 EcmaString *newString = *factory->ConcatFromString(key, JSTaggedValue::ToString(thread, number));
870 newkey.Update(JSTaggedValue(newString));
871 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), newkey, value);
872 }
873
874 EXPECT_FALSE(TaggedArray::Cast(obj1->GetProperties().GetTaggedObject())->IsDictionaryMode());
875
876 JSMutableHandle<JSTaggedValue> key5(thread, factory->NewFromASCII("x5"));
877 JSObject::DeleteProperty(thread, (obj1), key5);
878
879 EXPECT_TRUE(TaggedArray::Cast(obj1->GetProperties().GetTaggedObject())->IsDictionaryMode());
880 NameDictionary *dict = NameDictionary::Cast(obj1->GetProperties().GetTaggedObject());
881 EXPECT_EQ(dict->EntriesCount(), 9);
882 EXPECT_FALSE(JSObject::HasProperty(thread, obj1, key5));
883 }
884
HWTEST_F_L0(JSObjectTest,ElementFastToSlow)885 HWTEST_F_L0(JSObjectTest, ElementFastToSlow)
886 {
887 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
888 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
889 JSHandle<JSTaggedValue> key0(thread, JSTaggedValue(0));
890 JSHandle<JSTaggedValue> key1(thread, JSTaggedValue(1));
891 JSHandle<JSTaggedValue> key2(thread, JSTaggedValue(2));
892 JSHandle<JSTaggedValue> key2000(thread, JSTaggedValue(2000));
893 JSHandle<JSTaggedValue> keyStr(factory->NewFromASCII("str"));
894
895 // test dictionary [0,1,2,...,2000]
896 JSHandle<JSObject> obj1 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
897 EXPECT_TRUE(!TaggedArray::Cast(obj1->GetElements().GetTaggedObject())->IsDictionaryMode());
898 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), keyStr, key2);
899 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key0, key0);
900 EXPECT_TRUE(!TaggedArray::Cast(obj1->GetElements().GetTaggedObject())->IsDictionaryMode());
901 JSHandle<JSHClass> hclass(thread, obj1->GetJSHClass());
902 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key1, key1);
903 EXPECT_TRUE(!TaggedArray::Cast(obj1->GetElements().GetTaggedObject())->IsDictionaryMode());
904 EXPECT_EQ(obj1->GetJSHClass(), *hclass);
905
906 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), key2000, key2000);
907 EXPECT_TRUE(TaggedArray::Cast(obj1->GetElements().GetTaggedObject())->IsDictionaryMode());
908 JSTaggedValue value =
909 JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj1), keyStr).GetValue().GetTaggedValue();
910 EXPECT_EQ(value, key2.GetTaggedValue());
911 // test holey [0,,2]
912 JSHandle<JSObject> obj2 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
913 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key0, key0);
914 EXPECT_TRUE(!TaggedArray::Cast(obj2->GetElements().GetTaggedObject())->IsDictionaryMode());
915 JSHandle<JSHClass> hclass2(thread, obj2->GetJSHClass());
916 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), key2, key2);
917 EXPECT_TRUE(!TaggedArray::Cast(obj2->GetElements().GetTaggedObject())->IsDictionaryMode());
918 EXPECT_EQ(obj2->GetJSHClass(), *hclass2);
919 // test change attr
920 JSHandle<JSObject> obj3 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
921 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), key0, key0);
922 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), key1, key1);
923 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), key2, key2);
924 EXPECT_TRUE(!TaggedArray::Cast(obj3->GetElements().GetTaggedObject())->IsDictionaryMode());
925 PropertyDescriptor desc(thread);
926 desc.SetValue(key1);
927 desc.SetWritable(false);
928 JSObject::DefineOwnProperty(thread, obj3, key1, desc);
929 EXPECT_TRUE(TaggedArray::Cast(obj3->GetElements().GetTaggedObject())->IsDictionaryMode());
930 // test delete element
931 JSHandle<JSObject> obj4 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
932 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), key0, key0);
933 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), key1, key1);
934 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), key2, key2);
935 EXPECT_TRUE(!TaggedArray::Cast(obj4->GetElements().GetTaggedObject())->IsDictionaryMode());
936 JSObject::DeleteProperty(thread, (obj4), key1);
937 EXPECT_TRUE(TaggedArray::Cast(obj4->GetElements().GetTaggedObject())->IsDictionaryMode());
938
939 JSHandle<JSTaggedValue> value1001(thread, JSTaggedValue(1001));
940 JSHandle<JSObject> obj100 = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
941 PropertyDescriptor desc1(thread);
942 desc1.SetValue(value1001);
943 desc1.SetWritable(false);
944 desc1.SetEnumerable(false);
945 desc1.SetConfigurable(false);
946 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj100), key0, key1);
947 JSObject::DefineOwnProperty(thread, obj100, key0, desc1);
948 JSTaggedValue result1001 =
949 JSObject::GetProperty(thread, JSHandle<JSTaggedValue>(obj100), key0).GetValue().GetTaggedValue();
950 EXPECT_EQ(result1001, value1001.GetTaggedValue());
951 }
952
HWTEST_F_L0(JSObjectTest,EnableProtoChangeMarker)953 HWTEST_F_L0(JSObjectTest, EnableProtoChangeMarker)
954 {
955 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
956 JSHandle<JSObject> obj1 = JSObject::ObjectCreate(thread, nullHandle);
957 JSHandle<JSObject> obj2 = JSObject::ObjectCreate(thread, obj1);
958 JSHandle<JSObject> obj3 = JSObject::ObjectCreate(thread, obj2);
959
960 JSHandle<JSTaggedValue> obj1Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
961 JSHandle<JSTaggedValue> obj2Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
962 JSHandle<JSTaggedValue> obj3Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
963 JSHandle<JSTaggedValue> obj1Value(thread, JSTaggedValue(1));
964 JSHandle<JSTaggedValue> obj2Value(thread, JSTaggedValue(2));
965 JSHandle<JSTaggedValue> obj3Value(thread, JSTaggedValue(3));
966
967 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), obj1Key, obj1Value);
968 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), obj2Key, obj2Value);
969 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), obj3Key, obj3Value);
970 JSHandle<JSHClass> obj3Class(thread, obj3->GetJSHClass());
971 JSHandle<JSTaggedValue> resultMarker = JSHClass::EnableProtoChangeMarker(thread, obj3Class);
972 EXPECT_TRUE(resultMarker->IsProtoChangeMarker());
973 bool hasChanged = ProtoChangeMarker::Cast(resultMarker->GetTaggedObject())->GetHasChanged();
974 EXPECT_TRUE(!hasChanged);
975
976 JSHandle<JSHClass> obj1Class(thread, obj1->GetJSHClass());
977 JSHandle<JSHClass> obj2Class(thread, obj2->GetJSHClass());
978 JSTaggedValue obj2Marker = obj2Class->GetProtoChangeMarker();
979 EXPECT_TRUE(obj2Marker.IsProtoChangeMarker());
980 bool hasChanged2 = ProtoChangeMarker::Cast(obj2Marker.GetTaggedObject())->GetHasChanged();
981 EXPECT_TRUE(!hasChanged2);
982
983 JSTaggedValue obj1Marker = obj1Class->GetProtoChangeMarker();
984 EXPECT_TRUE(!obj1Marker.IsProtoChangeMarker());
985
986 JSTaggedValue protoDetails2 = obj2Class->GetProtoChangeDetails();
987 EXPECT_TRUE(protoDetails2.IsProtoChangeDetails());
988 JSTaggedValue protoDetails1 = obj1Class->GetProtoChangeDetails();
989 EXPECT_TRUE(protoDetails1.IsProtoChangeDetails());
990 JSTaggedValue listeners1 = ProtoChangeDetails::Cast(protoDetails1.GetTaggedObject())->GetChangeListener();
991 EXPECT_TRUE(!listeners1.IsUndefined());
992 JSTaggedValue listeners2 = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetChangeListener();
993 EXPECT_TRUE(listeners2.IsUndefined());
994 uint32_t index = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetRegisterIndex();
995 JSTaggedValue listenersResult = ChangeListener::Cast(listeners1.GetTaggedObject())->Get(index);
996 EXPECT_TRUE(listenersResult == obj2Class.GetTaggedValue());
997 }
998
HWTEST_F_L0(JSObjectTest,BuildRegisterTree)999 HWTEST_F_L0(JSObjectTest, BuildRegisterTree)
1000 {
1001 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
1002 JSHandle<JSObject> obj1 = JSObject::ObjectCreate(thread, nullHandle);
1003 JSHandle<JSObject> obj2 = JSObject::ObjectCreate(thread, obj1);
1004 JSHandle<JSObject> obj3 = JSObject::ObjectCreate(thread, obj2);
1005 JSHandle<JSObject> obj4 = JSObject::ObjectCreate(thread, obj2);
1006 JSHandle<JSObject> obj5 = JSObject::ObjectCreate(thread, obj4);
1007 JSHandle<JSObject> obj6 = JSObject::ObjectCreate(thread, obj2);
1008 JSHandle<JSObject> obj7 = JSObject::ObjectCreate(thread, obj6);
1009
1010 JSHandle<JSTaggedValue> obj1Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
1011 JSHandle<JSTaggedValue> obj2Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
1012 JSHandle<JSTaggedValue> obj3Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
1013 JSHandle<JSTaggedValue> obj4Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key4"));
1014 JSHandle<JSTaggedValue> obj5Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key5"));
1015 JSHandle<JSTaggedValue> obj6Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key6"));
1016 JSHandle<JSTaggedValue> obj7Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key7"));
1017
1018 JSHandle<JSTaggedValue> obj1Value(thread, JSTaggedValue(1));
1019 JSHandle<JSTaggedValue> obj2Value(thread, JSTaggedValue(2));
1020 JSHandle<JSTaggedValue> obj3Value(thread, JSTaggedValue(3));
1021 JSHandle<JSTaggedValue> obj4Value(thread, JSTaggedValue(4));
1022 JSHandle<JSTaggedValue> obj5Value(thread, JSTaggedValue(5));
1023 JSHandle<JSTaggedValue> obj6Value(thread, JSTaggedValue(6));
1024 JSHandle<JSTaggedValue> obj7Value(thread, JSTaggedValue(7));
1025
1026 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), obj1Key, obj1Value);
1027 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), obj2Key, obj2Value);
1028 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), obj3Key, obj3Value);
1029 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), obj4Key, obj4Value);
1030 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj5), obj5Key, obj5Value);
1031 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj6), obj6Key, obj6Value);
1032 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj7), obj7Key, obj7Value);
1033
1034 JSHandle<JSHClass> obj1Class(thread, obj1->GetJSHClass());
1035 JSHandle<JSHClass> obj2Class(thread, obj2->GetJSHClass());
1036 JSHandle<JSHClass> obj3Class(thread, obj3->GetJSHClass());
1037 JSHandle<JSHClass> obj4Class(thread, obj4->GetJSHClass());
1038 JSHandle<JSHClass> obj5Class(thread, obj5->GetJSHClass());
1039 JSHandle<JSHClass> obj6Class(thread, obj6->GetJSHClass());
1040 JSHandle<JSHClass> obj7Class(thread, obj7->GetJSHClass());
1041
1042 JSHandle<JSTaggedValue> result3Marker = JSHClass::EnableProtoChangeMarker(thread, obj3Class);
1043 JSHandle<JSTaggedValue> result5Marker = JSHClass::EnableProtoChangeMarker(thread, obj5Class);
1044 EXPECT_TRUE(result3Marker->IsProtoChangeMarker());
1045 EXPECT_TRUE(!(ProtoChangeMarker::Cast(result3Marker->GetTaggedObject())->GetHasChanged()));
1046 EXPECT_TRUE(result5Marker->IsProtoChangeMarker());
1047 EXPECT_TRUE(!(ProtoChangeMarker::Cast(result5Marker->GetTaggedObject())->GetHasChanged()));
1048
1049 EXPECT_TRUE(obj4Class->GetProtoChangeMarker().IsProtoChangeMarker());
1050 EXPECT_TRUE(!obj6Class->GetProtoChangeMarker().IsProtoChangeMarker());
1051
1052 JSHandle<JSTaggedValue> result7Marker = JSHClass::EnableProtoChangeMarker(thread, obj7Class);
1053 EXPECT_TRUE(result7Marker->IsProtoChangeMarker());
1054 EXPECT_TRUE(!(ProtoChangeMarker::Cast(result7Marker->GetTaggedObject())->GetHasChanged()));
1055
1056 JSTaggedValue protoDetails1 = obj1Class->GetProtoChangeDetails();
1057 EXPECT_TRUE(protoDetails1.IsProtoChangeDetails());
1058 JSTaggedValue listeners1Value = ProtoChangeDetails::Cast(protoDetails1.GetTaggedObject())->GetChangeListener();
1059 EXPECT_TRUE(listeners1Value != JSTaggedValue(0));
1060 JSHandle<ChangeListener> listeners1(thread, listeners1Value.GetTaggedObject());
1061 JSTaggedValue protoDetails2 = obj2Class->GetProtoChangeDetails();
1062 EXPECT_TRUE(protoDetails2.IsProtoChangeDetails());
1063 uint32_t index2 = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetRegisterIndex();
1064 EXPECT_TRUE(listeners1->Get(index2) == obj2Class.GetTaggedValue());
1065
1066 JSTaggedValue listeners2Value = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetChangeListener();
1067 EXPECT_TRUE(listeners2Value != JSTaggedValue(0));
1068 JSHandle<ChangeListener> listeners2(thread, listeners2Value.GetTaggedObject());
1069 JSTaggedValue protoDetails4 = obj4Class->GetProtoChangeDetails();
1070 JSTaggedValue protoDetails6 = obj6Class->GetProtoChangeDetails();
1071 EXPECT_TRUE(protoDetails4.IsProtoChangeDetails());
1072 EXPECT_TRUE(protoDetails6.IsProtoChangeDetails());
1073 uint32_t index4 = ProtoChangeDetails::Cast(protoDetails4.GetTaggedObject())->GetRegisterIndex();
1074 EXPECT_TRUE(listeners2->Get(index4) == obj4Class.GetTaggedValue());
1075 uint32_t index6 = ProtoChangeDetails::Cast(protoDetails6.GetTaggedObject())->GetRegisterIndex();
1076 EXPECT_TRUE(listeners2->Get(index6) == obj6Class.GetTaggedValue());
1077
1078 EXPECT_TRUE(listeners1->GetEnd() == 1U);
1079 EXPECT_TRUE(listeners2->GetEnd() == 2U);
1080 }
1081
HWTEST_F_L0(JSObjectTest,NoticeThroughChain)1082 HWTEST_F_L0(JSObjectTest, NoticeThroughChain)
1083 {
1084 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
1085 JSHandle<JSObject> obj1 = JSObject::ObjectCreate(thread, nullHandle);
1086 JSHandle<JSObject> obj2 = JSObject::ObjectCreate(thread, obj1);
1087 JSHandle<JSObject> obj3 = JSObject::ObjectCreate(thread, obj2);
1088 JSHandle<JSObject> obj4 = JSObject::ObjectCreate(thread, obj2);
1089 JSHandle<JSObject> obj5 = JSObject::ObjectCreate(thread, obj4);
1090 JSHandle<JSObject> obj6 = JSObject::ObjectCreate(thread, obj2);
1091 JSHandle<JSObject> obj7 = JSObject::ObjectCreate(thread, obj6);
1092
1093 JSHandle<JSTaggedValue> obj1Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
1094 JSHandle<JSTaggedValue> obj2Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
1095 JSHandle<JSTaggedValue> obj3Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
1096 JSHandle<JSTaggedValue> obj4Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key4"));
1097 JSHandle<JSTaggedValue> obj5Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key5"));
1098 JSHandle<JSTaggedValue> obj6Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key6"));
1099 JSHandle<JSTaggedValue> obj7Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key7"));
1100
1101 JSHandle<JSTaggedValue> obj1Value(thread, JSTaggedValue(1));
1102 JSHandle<JSTaggedValue> obj2Value(thread, JSTaggedValue(2));
1103 JSHandle<JSTaggedValue> obj3Value(thread, JSTaggedValue(3));
1104 JSHandle<JSTaggedValue> obj4Value(thread, JSTaggedValue(4));
1105 JSHandle<JSTaggedValue> obj5Value(thread, JSTaggedValue(5));
1106 JSHandle<JSTaggedValue> obj6Value(thread, JSTaggedValue(6));
1107 JSHandle<JSTaggedValue> obj7Value(thread, JSTaggedValue(7));
1108
1109 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), obj1Key, obj1Value);
1110 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), obj2Key, obj2Value);
1111 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), obj3Key, obj3Value);
1112 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), obj4Key, obj4Value);
1113 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj5), obj5Key, obj5Value);
1114 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj6), obj6Key, obj6Value);
1115 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj7), obj7Key, obj7Value);
1116
1117 JSHandle<JSHClass> obj1Class(thread, obj1->GetJSHClass());
1118 JSHandle<JSHClass> obj2Class(thread, obj2->GetJSHClass());
1119 JSHandle<JSHClass> obj3Class(thread, obj3->GetJSHClass());
1120 JSHandle<JSHClass> obj4Class(thread, obj4->GetJSHClass());
1121 JSHandle<JSHClass> obj5Class(thread, obj5->GetJSHClass());
1122 JSHandle<JSHClass> obj6Class(thread, obj6->GetJSHClass());
1123 JSHandle<JSHClass> obj7Class(thread, obj7->GetJSHClass());
1124
1125 JSHClass::EnableProtoChangeMarker(thread, obj3Class);
1126 JSHClass::EnableProtoChangeMarker(thread, obj7Class);
1127 JSHClass::EnableProtoChangeMarker(thread, obj5Class);
1128
1129 JSHClass::NoticeThroughChain(thread, obj2Class);
1130 JSHClass::UnregisterOnProtoChain(thread, obj2Class);
1131 JSTaggedValue protoDetails1 = obj1Class->GetProtoChangeDetails();
1132 EXPECT_TRUE(protoDetails1.IsProtoChangeDetails());
1133 JSTaggedValue listeners1Value = ProtoChangeDetails::Cast(protoDetails1.GetTaggedObject())->GetChangeListener();
1134 EXPECT_TRUE(listeners1Value != JSTaggedValue(0));
1135 JSHandle<ChangeListener> listeners1(thread, listeners1Value.GetTaggedObject());
1136 uint32_t holeIndex = ChangeListener::CheckHole(listeners1);
1137 EXPECT_TRUE(holeIndex == 0U);
1138
1139 JSTaggedValue protoDetails2 = obj2Class->GetProtoChangeDetails();
1140 EXPECT_TRUE(protoDetails2.IsProtoChangeDetails());
1141 JSTaggedValue listeners2Value = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetChangeListener();
1142 EXPECT_TRUE(listeners2Value != JSTaggedValue(0));
1143 uint32_t index2 = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetRegisterIndex();
1144 EXPECT_TRUE(listeners1->Get(index2).IsHole());
1145
1146 JSTaggedValue obj6Marker = obj6Class->GetProtoChangeMarker();
1147 EXPECT_TRUE(obj6Marker.IsProtoChangeMarker());
1148 bool hasChanged6 = ProtoChangeMarker::Cast(obj6Marker.GetTaggedObject())->GetHasChanged();
1149 EXPECT_TRUE(hasChanged6);
1150
1151 JSTaggedValue obj4Marker = obj4Class->GetProtoChangeMarker();
1152 EXPECT_TRUE(obj4Marker.IsProtoChangeMarker());
1153 bool hasChanged4 = ProtoChangeMarker::Cast(obj4Marker.GetTaggedObject())->GetHasChanged();
1154 EXPECT_TRUE(hasChanged4);
1155 }
1156
HWTEST_F_L0(JSObjectTest,ChangeProtoAndNoticeTheChain)1157 HWTEST_F_L0(JSObjectTest, ChangeProtoAndNoticeTheChain)
1158 {
1159 JSHandle<JSObject> nullHandle(thread, JSTaggedValue::Null());
1160 JSHandle<JSObject> obj1 = JSObject::ObjectCreate(thread, nullHandle);
1161 JSHandle<JSObject> obj2 = JSObject::ObjectCreate(thread, obj1);
1162 JSHandle<JSObject> obj3 = JSObject::ObjectCreate(thread, obj1);
1163 JSHandle<JSObject> obj4 = JSObject::ObjectCreate(thread, obj2);
1164 JSHandle<JSObject> obj5 = JSObject::ObjectCreate(thread, obj4);
1165 JSHandle<JSObject> obj6 = JSObject::ObjectCreate(thread, obj2);
1166 JSHandle<JSObject> obj7 = JSObject::ObjectCreate(thread, obj6);
1167
1168 JSHandle<JSTaggedValue> obj1Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key1"));
1169 JSHandle<JSTaggedValue> obj2Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key2"));
1170 JSHandle<JSTaggedValue> obj3Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key3"));
1171 JSHandle<JSTaggedValue> obj4Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key4"));
1172 JSHandle<JSTaggedValue> obj5Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key5"));
1173 JSHandle<JSTaggedValue> obj6Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key6"));
1174 JSHandle<JSTaggedValue> obj7Key(thread->GetEcmaVM()->GetFactory()->NewFromASCII("key7"));
1175
1176 JSHandle<JSTaggedValue> obj1Value(thread, JSTaggedValue(1));
1177 JSHandle<JSTaggedValue> obj2Value(thread, JSTaggedValue(2));
1178 JSHandle<JSTaggedValue> obj3Value(thread, JSTaggedValue(3));
1179 JSHandle<JSTaggedValue> obj4Value(thread, JSTaggedValue(4));
1180 JSHandle<JSTaggedValue> obj5Value(thread, JSTaggedValue(5));
1181 JSHandle<JSTaggedValue> obj6Value(thread, JSTaggedValue(6));
1182 JSHandle<JSTaggedValue> obj7Value(thread, JSTaggedValue(7));
1183
1184 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj1), obj1Key, obj1Value);
1185 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj2), obj2Key, obj2Value);
1186 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj3), obj3Key, obj3Value);
1187 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj4), obj4Key, obj4Value);
1188 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj5), obj5Key, obj5Value);
1189 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj6), obj6Key, obj6Value);
1190 JSObject::SetProperty(thread, JSHandle<JSTaggedValue>(obj7), obj7Key, obj7Value);
1191
1192 JSHandle<JSHClass> obj5Class(thread, obj5->GetJSHClass());
1193 JSHandle<JSHClass> obj7Class(thread, obj7->GetJSHClass());
1194
1195 JSHClass::EnableProtoChangeMarker(thread, obj7Class);
1196 JSHClass::EnableProtoChangeMarker(thread, obj5Class);
1197
1198 JSObject::SetPrototype(thread, obj2, JSHandle<JSTaggedValue>(obj3));
1199
1200 JSHandle<JSHClass> obj1Class(thread, obj1->GetJSHClass());
1201 JSHandle<JSHClass> obj2Class(thread, obj2->GetJSHClass());
1202 JSHandle<JSHClass> obj3Class(thread, obj3->GetJSHClass());
1203 JSHandle<JSHClass> obj4Class(thread, obj4->GetJSHClass());
1204 JSHandle<JSHClass> obj6Class(thread, obj6->GetJSHClass());
1205
1206 JSTaggedValue obj6Marker = obj6Class->GetProtoChangeMarker();
1207 EXPECT_TRUE(obj6Marker.IsProtoChangeMarker());
1208 bool hasChanged6 = ProtoChangeMarker::Cast(obj6Marker.GetTaggedObject())->GetHasChanged();
1209 EXPECT_TRUE(hasChanged6);
1210
1211 JSTaggedValue obj4Marker = obj4Class->GetProtoChangeMarker();
1212 EXPECT_TRUE(obj4Marker.IsProtoChangeMarker());
1213 bool hasChanged4 = ProtoChangeMarker::Cast(obj4Marker.GetTaggedObject())->GetHasChanged();
1214 EXPECT_TRUE(hasChanged4);
1215
1216 JSTaggedValue protoDetails1 = obj1Class->GetProtoChangeDetails();
1217 EXPECT_TRUE(protoDetails1.IsProtoChangeDetails());
1218 JSTaggedValue protoDetails2 = obj2Class->GetProtoChangeDetails();
1219 EXPECT_TRUE(protoDetails2.IsProtoChangeDetails());
1220 JSTaggedValue protoDetails3 = obj3Class->GetProtoChangeDetails();
1221 EXPECT_TRUE(protoDetails3.IsProtoChangeDetails());
1222 JSTaggedValue protoDetails4 = obj4Class->GetProtoChangeDetails();
1223 EXPECT_TRUE(protoDetails4.IsProtoChangeDetails());
1224 JSTaggedValue protoDetails6 = obj6Class->GetProtoChangeDetails();
1225 EXPECT_TRUE(protoDetails6.IsProtoChangeDetails());
1226
1227 JSTaggedValue listeners1 = ProtoChangeDetails::Cast(protoDetails1.GetTaggedObject())->GetChangeListener();
1228 EXPECT_TRUE(listeners1 != JSTaggedValue(0));
1229 JSTaggedValue listeners2 = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetChangeListener();
1230 EXPECT_TRUE(listeners2 != JSTaggedValue(0));
1231 JSTaggedValue listeners3 = ProtoChangeDetails::Cast(protoDetails3.GetTaggedObject())->GetChangeListener();
1232 EXPECT_TRUE(listeners3 != JSTaggedValue(0));
1233
1234 uint32_t index2 = ProtoChangeDetails::Cast(protoDetails2.GetTaggedObject())->GetRegisterIndex();
1235 uint32_t index3 = ProtoChangeDetails::Cast(protoDetails3.GetTaggedObject())->GetRegisterIndex();
1236 uint32_t index4 = ProtoChangeDetails::Cast(protoDetails4.GetTaggedObject())->GetRegisterIndex();
1237 uint32_t index6 = ProtoChangeDetails::Cast(protoDetails6.GetTaggedObject())->GetRegisterIndex();
1238
1239 JSTaggedValue result2 = ChangeListener::Cast(listeners3.GetTaggedObject())->Get(index2);
1240 JSTaggedValue result3 = ChangeListener::Cast(listeners1.GetTaggedObject())->Get(index3);
1241 JSTaggedValue result4 = ChangeListener::Cast(listeners2.GetTaggedObject())->Get(index4);
1242 JSTaggedValue result6 = ChangeListener::Cast(listeners2.GetTaggedObject())->Get(index6);
1243
1244 EXPECT_TRUE(result2 == obj2Class.GetTaggedValue());
1245 EXPECT_TRUE(result3 == obj3Class.GetTaggedValue());
1246 EXPECT_TRUE(result4 == obj4Class.GetTaggedValue());
1247 EXPECT_TRUE(result6 == obj6Class.GetTaggedValue());
1248 }
1249
HWTEST_F_L0(JSObjectTest,NativePointerField)1250 HWTEST_F_L0(JSObjectTest, NativePointerField)
1251 {
1252 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
1253 JSHandle<JSTaggedValue> objFunc(thread, JSObjectTestCreate(thread));
1254 JSHandle<JSObject> obj = factory->NewJSObjectByConstructor(JSHandle<JSFunction>(objFunc), objFunc);
1255 ECMAObject::SetHash(thread, 87, JSHandle<ECMAObject>::Cast(obj));
1256 EXPECT_TRUE(obj->GetHash() == 87);
1257
1258 obj->SetNativePointerFieldCount(thread, 1);
1259 char array[] = "Hello World!";
1260 obj->SetNativePointerField(thread, 0, array, nullptr, nullptr);
1261 int32_t count = obj->GetNativePointerFieldCount();
1262 EXPECT_TRUE(count == 1);
1263 void *pointer = obj->GetNativePointerField(0);
1264 EXPECT_TRUE(pointer == array);
1265 }
1266
CreateTestHClass(JSThread * thread)1267 static JSHandle<JSHClass> CreateTestHClass(JSThread *thread)
1268 {
1269 JSHandle<GlobalEnv> globalEnv = thread->GetEcmaVM()->GetGlobalEnv();
1270 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
1271 JSHandle<JSTaggedValue> objectFuncPrototype = globalEnv->GetObjectFunctionPrototype();
1272 JSHandle<JSHClass> hclass = factory->NewEcmaHClass(JSObject::SIZE, JSType::JS_OBJECT, objectFuncPrototype);
1273 return hclass;
1274 }
1275
HWTEST_F_L0(JSObjectTest,UpdateWeakTransitions)1276 HWTEST_F_L0(JSObjectTest, UpdateWeakTransitions)
1277 {
1278 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
1279 EcmaVM *vm = thread->GetEcmaVM();
1280 PropertyAttributes attr = PropertyAttributes::Default();
1281
1282 // Initialize three objects by hc0
1283 JSHandle<JSHClass> hc0 = CreateTestHClass(thread);
1284 JSMutableHandle<JSHClass> hca(thread, JSTaggedValue::Undefined());
1285 JSMutableHandle<JSHClass> hcb(thread, JSTaggedValue::Undefined());
1286 [[maybe_unused]] JSHandle<JSObject> obj0 = factory->NewJSObject(hc0); // need it to ensure hc0 not be collected
1287 JSHandle<JSObject> obj1 = factory->NewJSObject(hc0);
1288 JSHandle<JSObject> obj2 = factory->NewJSObject(hc0);
1289
1290 JSHandle<JSTaggedValue> keyA(factory->NewFromASCII("a"));
1291 JSHandle<JSTaggedValue> keyB(factory->NewFromASCII("b"));
1292
1293 {
1294 // need a new handle scope to ensure no scope refers hc1, hc2.
1295 [[maybe_unused]] EcmaHandleScope handleScope(thread);
1296
1297 // / hc1 (obj1)
1298 // occur transition, hc0 (obj0)
1299 // \ hc2 (obj2)
1300 JSObject::SetProperty(thread, obj1, keyA, JSHandle<JSTaggedValue>(thread, JSTaggedValue(1)));
1301 JSObject::SetProperty(thread, obj2, keyB, JSHandle<JSTaggedValue>(thread, JSTaggedValue(2)));
1302
1303 EXPECT_TRUE(hc0->GetTransitions().IsTaggedArray());
1304 EXPECT_EQ(hc0->FindTransitions(keyA.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1305 obj1->GetClass());
1306 EXPECT_EQ(hc0->FindTransitions(keyB.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1307 obj2->GetClass());
1308 hca.Update(JSTaggedValue(obj1->GetClass()));
1309 hcb.Update(JSTaggedValue(obj2->GetClass()));
1310
1311 // / hc1 --> hc3 (obj1)
1312 // hc0 (obj0)
1313 // \ hc2 --> hc4 (obj2)
1314 JSObject::SetProperty(thread, obj1, keyB, JSHandle<JSTaggedValue>(thread, JSTaggedValue(3)));
1315 JSObject::SetProperty(thread, obj2, keyA, JSHandle<JSTaggedValue>(thread, JSTaggedValue(4)));
1316 }
1317
1318 // collect hc1, hc2
1319 vm->CollectGarbage(TriggerGCType::FULL_GC);
1320
1321 EXPECT_EQ(hc0->FindTransitions(keyA.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1322 hca.GetObject<JSHClass>());
1323 EXPECT_EQ(hc0->FindTransitions(keyB.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1324 hcb.GetObject<JSHClass>());
1325
1326 JSHandle<JSObject> obj3 = factory->NewJSObject(hc0);
1327 JSHandle<JSObject> obj4 = factory->NewJSObject(hc0);
1328
1329 // / hc5 (obj3)
1330 // re-occur transition, hc0 (obj0)
1331 // \ hc6 (obj4)
1332 JSObject::SetProperty(thread, obj3, keyA, JSHandle<JSTaggedValue>(thread, JSTaggedValue(5)));
1333 JSObject::SetProperty(thread, obj4, keyB, JSHandle<JSTaggedValue>(thread, JSTaggedValue(6)));
1334
1335 EXPECT_TRUE(hc0->GetTransitions().IsTaggedArray());
1336 EXPECT_EQ(hc0->FindTransitions(keyA.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1337 obj3->GetClass());
1338 EXPECT_EQ(hc0->FindTransitions(keyB.GetTaggedValue(), attr.GetTaggedValue(), Representation::NONE),
1339 obj4->GetClass());
1340 EXPECT_EQ(obj3->GetClass(), hca.GetObject<JSHClass>());
1341 EXPECT_EQ(obj4->GetClass(), hcb.GetObject<JSHClass>());
1342 }
1343
HWTEST_F_L0(JSObjectTest,CreateObjectFromProperties)1344 HWTEST_F_L0(JSObjectTest, CreateObjectFromProperties)
1345 {
1346 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
1347 JSHandle<TaggedArray> properties = factory->NewTaggedArray(2060); // 2060: array length
1348
1349 JSHandle<EcmaString> key1(factory->NewFromASCII("aa"));
1350 JSHandle<JSTaggedValue> value1(thread, JSTaggedValue(1));
1351
1352 for (int i = 0; i < 1030; i++) {
1353 JSHandle<EcmaString> key2(factory->NewFromASCII(std::to_string(i)));
1354 JSHandle<EcmaString> key3 =
1355 JSHandle<EcmaString>(thread, EcmaStringAccessor::Concat(thread->GetEcmaVM(), key1, key2));
1356 properties->Set(thread, 2 * i, key3.GetTaggedValue());
1357 properties->Set(thread, 2 * i + 1, value1.GetTaggedValue());
1358 }
1359
1360 JSHandle<JSObject> newObj = JSObject::CreateObjectFromProperties(thread, properties);
1361 Heap *heap = const_cast<Heap *>(thread->GetEcmaVM()->GetHeap());
1362 heap->GetSweeper()->EnsureAllTaskFinished();
1363 Verification(heap).VerifyAll();
1364 auto sHeap = SharedHeap::GetInstance();
1365 SuspendAllScope suspendScope(thread);
1366 SharedHeapVerification(sHeap, VerifyKind::VERIFY_PRE_SHARED_GC).VerifyAll();
1367 EXPECT_TRUE(newObj->GetClass()->IsDictionaryMode());
1368 }
1369 } // namespace panda::test
1370