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/tagged_tree-inl.h"
17 #include "ecmascript/js_object-inl.h"
18 #include "ecmascript/object_factory.h"
19 #include "libpandabase/utils/bit_utils.h"
20
21 namespace panda::ecmascript {
22 template<typename Derived>
Create(const JSThread * thread,int numberOfElements)23 JSHandle<Derived> TaggedTree<Derived>::Create(const JSThread *thread, int numberOfElements)
24 {
25 ASSERT_PRINT(numberOfElements > 0, "size must be a non-negative integer");
26 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
27 auto capacity = numberOfElements > MIN_CAPACITY ? static_cast<uint32_t>(numberOfElements) : MIN_CAPACITY;
28 int length = ELEMENTS_START_INDEX + numberOfElements * (Derived::ENTRY_SIZE);
29
30 auto tree = JSHandle<Derived>::Cast(factory->NewTaggedArray(length));
31 tree->SetNumberOfElements(thread, 0);
32 tree->SetNumberOfDeletedElements(thread, 0);
33 tree->SetRootEntries(thread, -1);
34 tree->SetCompare(thread, JSTaggedValue::Hole());
35 tree->SetCapacity(thread, capacity);
36 return tree;
37 }
38
39 template<typename Derived>
InsertRebalance(const JSThread * thread,int index)40 void TaggedTree<Derived>::InsertRebalance(const JSThread *thread, int index)
41 {
42 while (IsVaildIndex(index) && GetColor(GetParent(index)) == TreeColor::RED) {
43 if (IsLeft(GetParent(index))) {
44 int bro = GetLeftBrother(GetParent(index));
45 if (GetColor(bro)) {
46 SetColor(thread, GetParent(index), TreeColor::BLACK);
47 SetColor(thread, bro, TreeColor::BLACK);
48 SetColor(thread, GetParent(GetParent(index)), TreeColor::RED);
49 index = GetParent(GetParent(index));
50 } else {
51 if (IsRight(index)) {
52 index = GetParent(index);
53 LeftRotate(thread, index);
54 }
55 SetColor(thread, GetParent(index), TreeColor::BLACK);
56 SetColor(thread, GetParent(GetParent(index)), TreeColor::RED);
57 RightRotate(thread, GetParent(GetParent(index)));
58 }
59 } else {
60 int bro = GetRightBrother(GetParent(index));
61 if (GetColor(bro)) {
62 SetColor(thread, GetParent(index), TreeColor::BLACK);
63 SetColor(thread, bro, TreeColor::BLACK);
64 SetColor(thread, GetParent(GetParent(index)), TreeColor::RED);
65 index = GetParent(GetParent(index));
66 } else {
67 if (IsLeft(index)) {
68 index = GetParent(index);
69 RightRotate(thread, index);
70 }
71 SetColor(thread, GetParent(index), TreeColor::BLACK);
72 SetColor(thread, GetParent(GetParent(index)), TreeColor::RED);
73 LeftRotate(thread, GetParent(GetParent(index)));
74 }
75 }
76 }
77 SetColor(thread, GetRootEntries(), TreeColor::BLACK);
78 }
79
80 template<typename Derived>
LeftRotate(const JSThread * thread,int index)81 void TaggedTree<Derived>::LeftRotate(const JSThread *thread, int index)
82 {
83 if (index >= 0) {
84 int right = GetRightChild(index).GetInt();
85 JSTaggedValue leftOfRight = GetLeftChild(right);
86 SetRightChild(thread, index, leftOfRight);
87 if (!leftOfRight.IsHole()) {
88 SetParent(thread, leftOfRight.GetInt(), JSTaggedValue(index));
89 }
90 int parentOfIndex = GetParent(index);
91 SetParent(thread, right, JSTaggedValue(parentOfIndex));
92 if (parentOfIndex < 0) {
93 SetRootEntries(thread, right);
94 } else {
95 JSTaggedValue left = GetLeftChild(parentOfIndex);
96 if (!left.IsHole() && left.GetInt() == index) { // change to isleft
97 SetLeftChild(thread, parentOfIndex, JSTaggedValue(right));
98 } else {
99 SetRightChild(thread, parentOfIndex, JSTaggedValue(right));
100 }
101 }
102 SetLeftChild(thread, right, JSTaggedValue(index));
103 SetParent(thread, index, JSTaggedValue(right));
104 }
105 }
106
107 template<typename Derived>
RightRotate(const JSThread * thread,int index)108 void TaggedTree<Derived>::RightRotate(const JSThread *thread, int index)
109 {
110 if (index >= 0) {
111 int left = GetLeftChild(index).GetInt();
112 JSTaggedValue rightOfLeft = GetRightChild(left);
113 SetLeftChild(thread, index, rightOfLeft);
114 if (!rightOfLeft.IsHole()) {
115 SetParent(thread, rightOfLeft.GetInt(), JSTaggedValue(index));
116 }
117 int parentOfIndex = GetParent(index);
118 SetParent(thread, left, JSTaggedValue(parentOfIndex));
119 if (parentOfIndex < 0) {
120 SetRootEntries(thread, left);
121 } else {
122 JSTaggedValue right = GetRightChild(parentOfIndex);
123 if (!right.IsHole() && right.GetInt() == index) { // change to isright
124 SetRightChild(thread, parentOfIndex, JSTaggedValue(left));
125 } else {
126 SetLeftChild(thread, parentOfIndex, JSTaggedValue(left));
127 }
128 }
129 SetRightChild(thread, left, JSTaggedValue(index));
130 SetParent(thread, index, JSTaggedValue(left));
131 }
132 }
133
134 template<typename Derived>
AdjustTaggedTree(const JSThread * thread,const JSHandle<Derived> & tree,int len)135 JSHandle<Derived> TaggedTree<Derived>::AdjustTaggedTree(const JSThread *thread, const JSHandle<Derived> &tree, int len)
136 {
137 JSMutableHandle<Derived> newTree(thread, JSTaggedValue::Undefined());
138 ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
139 if (tree->NumberOfDeletedElements() == 0) {
140 newTree.Update(factory->ExtendArray(JSHandle<TaggedArray>::Cast(tree), len).GetTaggedValue());
141 return newTree;
142 }
143
144 int elements = tree->NumberOfElements();
145 newTree.Update(factory->NewTaggedArray(len).GetTaggedValue());
146 newTree->SetNumberOfElements(thread, elements);
147 newTree->SetNumberOfDeletedElements(thread, 0);
148
149 newTree->SetRootEntries(thread, 0);
150 CQueue<int> entries;
151 entries.push(tree->GetRootEntries());
152 int index = 0;
153 newTree->SetParent(thread, index, JSTaggedValue(-1));
154 int child = 1;
155 while (!entries.empty()) {
156 int parent = entries.front();
157 JSTaggedValue left = tree->GetLeftChild(parent);
158 if (!left.IsHole()) {
159 entries.push(left.GetInt());
160 newTree->SetLeftChild(thread, index, JSTaggedValue(child));
161 newTree->SetParent(thread, child++, JSTaggedValue(index));
162 }
163 JSTaggedValue right = tree->GetRightChild(parent);
164 if (!right.IsHole()) {
165 entries.push(right.GetInt());
166 newTree->SetRightChild(thread, index, JSTaggedValue(child));
167 newTree->SetParent(thread, child++, JSTaggedValue(index));
168 }
169 tree->CopyEntry(thread, parent, newTree, index);
170 entries.pop();
171 index++;
172 }
173 return newTree;
174 }
175
176 template<typename Derived>
Transplant(const JSThread * thread,int dst,int src)177 void TaggedTree<Derived>::Transplant(const JSThread *thread, int dst, int src)
178 {
179 int parent = GetParent(dst);
180 if (parent < 0) {
181 SetRootEntries(thread, src);
182 } else if (IsLeft(dst)) {
183 JSTaggedValue child = src < 0 ? JSTaggedValue::Hole() : JSTaggedValue(src);
184 SetLeftChild(thread, parent, child);
185 } else {
186 JSTaggedValue child = src < 0 ? JSTaggedValue::Hole() : JSTaggedValue(src);
187 SetRightChild(thread, parent, child);
188 }
189 SetParent(thread, src, JSTaggedValue(parent));
190 }
191
192 template<typename Derived>
Remove(const JSThread * thread,const JSHandle<Derived> & tree,int entry)193 void TaggedTree<Derived>::Remove(const JSThread *thread, const JSHandle<Derived> &tree, int entry)
194 {
195 int successor = entry;
196 if (!tree->GetLeftChild(entry).IsHole() && !tree->GetRightChild(entry).IsHole()) {
197 successor = tree->GetSuccessor(entry);
198 tree->CopyData(thread, entry, successor);
199 }
200 JSTaggedValue left = tree->GetLeftChild(successor);
201 JSTaggedValue right = tree->GetRightChild(successor);
202 int child = left.IsHole() ? (right.IsHole() ? -1 : right.GetInt()) : left.GetInt();
203 if (child < 0) {
204 if (tree->GetColor(successor) == TreeColor::BLACK) {
205 tree->DeleteRebalance(thread, successor);
206 }
207 }
208 tree->Transplant(thread, successor, child);
209
210 if (child >= 0) {
211 if (tree->GetColor(successor) == TreeColor::BLACK) {
212 tree->DeleteRebalance(thread, child);
213 }
214 }
215 tree->RemoveEntry(thread, successor);
216 tree->SetNumberOfElements(thread, tree->NumberOfElements() - 1);
217 tree->SetNumberOfDeletedElements(thread, tree->NumberOfDeletedElements() + 1);
218 }
219
220 template<typename Derived>
DeleteRebalance(const JSThread * thread,int index)221 void TaggedTree<Derived>::DeleteRebalance(const JSThread *thread, int index)
222 {
223 while (index != GetRootEntries() && GetColor(index) == TreeColor::BLACK) {
224 if (IsLeft(index)) {
225 int bro = GetLeftBrother(index);
226 if (GetColor(bro)) {
227 SetColor(thread, bro, TreeColor::BLACK);
228 SetColor(thread, GetParent(index), TreeColor::RED);
229 LeftRotate(thread, GetParent(index));
230 bro = GetLeftBrother(index);
231 }
232 if (GetColor(GetLeftChildIndex(bro)) == TreeColor::BLACK &&
233 GetColor(GetRightChildIndex(bro)) == TreeColor::BLACK) {
234 SetColor(thread, bro, TreeColor::RED);
235 index = GetParent(index);
236 } else {
237 if (GetColor(GetRightChildIndex(bro)) == TreeColor::BLACK) {
238 SetColor(thread, GetLeftChildIndex(bro), TreeColor::BLACK);
239 SetColor(thread, bro, TreeColor::RED);
240 RightRotate(thread, bro);
241 bro = GetLeftBrother(index);
242 }
243 SetColor(thread, bro, GetColor(GetParent(index)));
244 SetColor(thread, GetParent(index), TreeColor::BLACK);
245 SetColor(thread, GetRightChildIndex(bro), TreeColor::BLACK);
246 LeftRotate(thread, GetParent(index));
247 index = GetRootEntries();
248 }
249 } else {
250 int bro = GetRightBrother(index);
251 if (GetColor(bro)) {
252 SetColor(thread, bro, TreeColor::BLACK);
253 SetColor(thread, GetParent(index), TreeColor::RED);
254 RightRotate(thread, GetParent(index));
255 bro = GetRightBrother(index);
256 }
257 if (GetColor(GetRightChildIndex(bro)) == TreeColor::BLACK &&
258 GetColor(GetLeftChildIndex(bro)) == TreeColor::BLACK) {
259 SetColor(thread, bro, TreeColor::RED);
260 index = GetParent(index);
261 } else {
262 if (GetColor(GetLeftChildIndex(bro)) == TreeColor::BLACK) {
263 SetColor(thread, GetRightChildIndex(bro), TreeColor::BLACK);
264 SetColor(thread, bro, TreeColor::RED);
265 LeftRotate(thread, bro);
266 bro = GetRightBrother(index);
267 }
268 SetColor(thread, bro, GetColor(GetParent(index)));
269 SetColor(thread, GetParent(index), TreeColor::BLACK);
270 SetColor(thread, GetLeftChildIndex(bro), TreeColor::BLACK);
271 RightRotate(thread, GetParent(index));
272 index = GetRootEntries();
273 }
274 }
275 }
276 SetColor(thread, index, TreeColor::BLACK);
277 }
278
279 template<typename Derived>
GetPreDecessor(int entry) const280 int TaggedTree<Derived>::GetPreDecessor(int entry) const
281 {
282 int child = GetLeftChildIndex(entry);
283 if (child >= 0) {
284 return GetMaximum(child);
285 }
286 int parent = GetParent(entry);
287 while (parent >= 0 && (GetLeftChildIndex(parent) == entry)) {
288 entry = parent;
289 parent = GetParent(entry);
290 }
291 return parent;
292 }
293
294 template<typename Derived>
GetSuccessor(int entry) const295 int TaggedTree<Derived>::GetSuccessor(int entry) const
296 {
297 int child = GetRightChildIndex(entry);
298 if (child >= 0) {
299 return GetMinimum(child);
300 }
301 int parent = GetParent(entry);
302 while (parent >= 0 && (GetRightChildIndex(parent) == entry)) {
303 entry = parent;
304 parent = GetParent(entry);
305 }
306 return parent;
307 }
308
309 template<typename Derived>
FindEntry(JSThread * thread,const JSHandle<Derived> & tree,const JSHandle<JSTaggedValue> & key)310 int TaggedTree<Derived>::FindEntry(JSThread *thread, const JSHandle<Derived> &tree, const JSHandle<JSTaggedValue> &key)
311 {
312 int parentIndex = tree->GetRootEntries();
313 JSMutableHandle<JSTaggedValue> parentKey(thread, tree->GetKey(parentIndex));
314 ComparisonResult res;
315 while (!parentKey->IsHole()) {
316 res = EntryCompare(thread, key, parentKey, tree);
317 RETURN_VALUE_IF_ABRUPT_COMPLETION(thread, 0);
318 if (res == ComparisonResult::EQUAL) {
319 return parentIndex;
320 } else if (res == ComparisonResult::LESS) {
321 JSTaggedValue child = tree->GetLeftChild(parentIndex);
322 if (child.IsHole()) break;
323 parentIndex = child.GetInt();
324 } else {
325 JSTaggedValue child = tree->GetRightChild(parentIndex);
326 if (child.IsHole()) break;
327 parentIndex = child.GetInt();
328 }
329 parentKey.Update(tree->GetKey(parentIndex));
330 }
331 return -1;
332 }
333
334 template<typename Derived>
GetSortArray(const JSThread * thread,const JSHandle<Derived> & tree)335 JSHandle<TaggedArray> TaggedTree<Derived>::GetSortArray(const JSThread *thread, const JSHandle<Derived> &tree)
336 {
337 JSHandle<TaggedArray> sortArray = thread->GetEcmaVM()->GetFactory()->NewTaggedArray(tree->NumberOfElements());
338 CStack<int> entries;
339 int index = tree->GetRootEntries();
340 int aid = 0;
341 while (index >= 0 || !entries.empty()) {
342 while (index >= 0) {
343 entries.emplace(index);
344 index = tree->GetLeftChildIndex(index);
345 }
346 if (!entries.empty()) {
347 sortArray->Set(thread, aid++, JSTaggedValue(entries.top()));
348 index = tree->GetRightChildIndex(entries.top());
349 entries.pop();
350 }
351 }
352 return sortArray;
353 }
354
355 template<typename Derived>
Insert(JSThread * thread,JSHandle<Derived> & tree,const JSHandle<JSTaggedValue> & key,const JSHandle<JSTaggedValue> & value)356 JSHandle<Derived> TaggedTree<Derived>::Insert(JSThread *thread, JSHandle<Derived> &tree,
357 const JSHandle<JSTaggedValue> &key, const JSHandle<JSTaggedValue> &value)
358 {
359 ASSERT(IsKey(key.GetTaggedValue()));
360 JSMutableHandle<JSTaggedValue> parentKey(thread, tree->GetRootKey());
361 if (parentKey->IsHole()) {
362 tree->SetRoot(thread, 0, key.GetTaggedValue(), value.GetTaggedValue());
363 tree->SetNumberOfElements(thread, tree->NumberOfElements() + 1);
364 return tree;
365 }
366
367 JSHandle<Derived> newTree = GrowCapacity(thread, tree);
368 int parentIndex = newTree->GetRootEntries();
369 ComparisonResult res;
370 while (!parentKey->IsHole()) {
371 res = EntryCompare(thread, key, parentKey, tree);
372 RETURN_VALUE_IF_ABRUPT_COMPLETION(thread, JSHandle<Derived>(thread, JSTaggedValue::Exception()));
373 if (res == ComparisonResult::EQUAL) {
374 newTree->SetValue(thread, parentIndex, value.GetTaggedValue());
375 return tree;
376 } else if (res == ComparisonResult::LESS) {
377 JSTaggedValue child = newTree->GetLeftChild(parentIndex);
378 if (child.IsHole()) break;
379 parentIndex = child.GetInt();
380 } else {
381 JSTaggedValue child = newTree->GetRightChild(parentIndex);
382 if (child.IsHole()) break;
383 parentIndex = child.GetInt();
384 }
385 parentKey.Update(newTree->GetKey(parentIndex));
386 }
387
388 int entry = newTree->NumberOfElements() + newTree->NumberOfDeletedElements();
389 if (static_cast<bool>(res)) {
390 newTree->InsertRightEntry(thread, parentIndex, entry, key.GetTaggedValue(), value.GetTaggedValue());
391 } else {
392 newTree->InsertLeftEntry(thread, parentIndex, entry, key.GetTaggedValue(), value.GetTaggedValue());
393 }
394 newTree->SetNumberOfElements(thread, newTree->NumberOfElements() + 1);
395 newTree->InsertRebalance(thread, entry);
396 return newTree;
397 }
398
399 template<typename Derived>
GrowCapacity(const JSThread * thread,JSHandle<Derived> & tree)400 JSHandle<Derived> TaggedTree<Derived>::GrowCapacity(const JSThread *thread, JSHandle<Derived> &tree)
401 {
402 int nof = tree->NumberOfElements() + tree->NumberOfDeletedElements();
403 int oldCapacity = tree->Capacity();
404 if (nof + 1 <= oldCapacity) {
405 return tree;
406 }
407
408 int newCapacity = ComputeCapacity(oldCapacity);
409 int length = ELEMENTS_START_INDEX + newCapacity * (Derived::ENTRY_SIZE);
410 JSHandle<Derived> newTree = AdjustTaggedTree(thread, tree, length);
411 newTree->SetCapacity(thread, newCapacity);
412 return newTree;
413 }
414
415 template<typename Derived>
GetLowerKey(JSThread * thread,const JSHandle<Derived> & tree,const JSHandle<JSTaggedValue> & key)416 JSTaggedValue TaggedTree<Derived>::GetLowerKey(JSThread *thread, const JSHandle<Derived> &tree,
417 const JSHandle<JSTaggedValue> &key)
418 {
419 int parentIndex = tree->GetRootEntries();
420 JSMutableHandle<JSTaggedValue> parentKey(thread, tree->GetKey(parentIndex));
421 int resultIndex = -1;
422 ComparisonResult res;
423 while (parentIndex >= 0) {
424 res = EntryCompare(thread, key, parentKey, tree);
425 RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
426 if (res == ComparisonResult::GREAT) {
427 resultIndex = parentIndex;
428 parentIndex = tree->GetRightChildIndex(parentIndex);
429 } else {
430 parentIndex = tree->GetLeftChildIndex(parentIndex);
431 }
432 parentKey.Update(tree->GetKey(parentIndex));
433 }
434 JSTaggedValue lowerKey = tree->GetKey(resultIndex);
435 return tree->Transform(lowerKey);
436 }
437
438 template<typename Derived>
GetHigherKey(JSThread * thread,const JSHandle<Derived> & tree,const JSHandle<JSTaggedValue> & key)439 JSTaggedValue TaggedTree<Derived>::GetHigherKey(JSThread *thread, const JSHandle<Derived> &tree,
440 const JSHandle<JSTaggedValue> &key)
441 {
442 int parentIndex = tree->GetRootEntries();
443 JSMutableHandle<JSTaggedValue> parentKey(thread, tree->GetKey(parentIndex));
444 int resultIndex = -1;
445 ComparisonResult res;
446 while (parentIndex >= 0) {
447 res = EntryCompare(thread, key, parentKey, tree);
448 RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
449 if (res == ComparisonResult::LESS) {
450 resultIndex = parentIndex;
451 parentIndex = tree->GetLeftChildIndex(parentIndex);
452 } else {
453 parentIndex = tree->GetRightChildIndex(parentIndex);
454 }
455 parentKey.Update(tree->GetKey(parentIndex));
456 }
457 JSTaggedValue lowerKey = tree->GetKey(resultIndex);
458 return tree->Transform(lowerKey);
459 }
460
461 template<typename Derived>
Shrink(const JSThread * thread,const JSHandle<Derived> & tree)462 JSHandle<Derived> TaggedTree<Derived>::Shrink(const JSThread *thread, const JSHandle<Derived> &tree)
463 {
464 int oldCapacity = tree->Capacity();
465 if (tree->NumberOfElements() >= (oldCapacity + 1) / 4) { // 4: quarter
466 return tree;
467 }
468 int newCapacity = (oldCapacity - 1) >> 1;
469 if (newCapacity < Derived::MIN_SHRINK_CAPACITY) {
470 return tree;
471 }
472
473 int length = ELEMENTS_START_INDEX + newCapacity * (Derived::ENTRY_SIZE);
474 JSHandle<Derived> newTree = AdjustTaggedTree(thread, tree, length);
475 newTree->SetCapacity(thread, newCapacity);
476 return newTree;
477 }
478
479 // TaggedTreeMap
Create(const JSThread * thread,int numberOfElements)480 JSTaggedValue TaggedTreeMap::Create(const JSThread *thread, int numberOfElements)
481 {
482 return RBTree::Create(thread, numberOfElements).GetTaggedValue();
483 }
484
GetArrayFromMap(const JSThread * thread,const JSHandle<TaggedTreeMap> & map)485 JSHandle<TaggedArray> TaggedTreeMap::GetArrayFromMap(const JSThread *thread, const JSHandle<TaggedTreeMap> &map)
486 {
487 return RBTree::GetSortArray(thread, map);
488 }
489
Set(JSThread * thread,JSHandle<TaggedTreeMap> & obj,const JSHandle<JSTaggedValue> & key,const JSHandle<JSTaggedValue> & value)490 JSTaggedValue TaggedTreeMap::Set(JSThread *thread, JSHandle<TaggedTreeMap> &obj,
491 const JSHandle<JSTaggedValue> &key, const JSHandle<JSTaggedValue> &value)
492 {
493 return RBTree::Insert(thread, obj, key, value).GetTaggedValue();
494 }
495
Delete(JSThread * thread,const JSHandle<TaggedTreeMap> & map,int entry)496 JSTaggedValue TaggedTreeMap::Delete(JSThread *thread, const JSHandle<TaggedTreeMap> &map, int entry)
497 {
498 RBTree::Remove(thread, map, entry);
499 return RBTree::Shrink(thread, map).GetTaggedValue();
500 }
501
HasValue(const JSThread * thread,JSTaggedValue value) const502 bool TaggedTreeMap::HasValue(const JSThread *thread, JSTaggedValue value) const
503 {
504 int root = GetRootEntries();
505 if (root < 0) {
506 return false;
507 }
508
509 CQueue<int> entries;
510 entries.push(root);
511 while (!entries.empty()) {
512 int parent = entries.front();
513 if (JSTaggedValue::SameValue(GetValue(parent), value)) {
514 return true;
515 }
516 int left = GetLeftChildIndex(parent);
517 if (left >= 0) {
518 entries.push(left);
519 }
520 int right = GetRightChildIndex(parent);
521 if (right >= 0) {
522 entries.push(right);
523 }
524 entries.pop();
525 }
526 return false;
527 }
528
SetAll(JSThread * thread,JSHandle<TaggedTreeMap> & dst,const JSHandle<TaggedTreeMap> & src)529 JSTaggedValue TaggedTreeMap::SetAll(JSThread *thread, JSHandle<TaggedTreeMap> &dst, const JSHandle<TaggedTreeMap> &src)
530 {
531 CQueue<int> entries;
532 entries.push(src->GetRootEntries());
533 JSMutableHandle<TaggedTreeMap> map(dst);
534 while (!entries.empty()) {
535 int parent = entries.front();
536 auto tmap = Insert(thread, map, JSHandle<JSTaggedValue>(thread, src->GetKey(parent)),
537 JSHandle<JSTaggedValue>(thread, src->GetValue(parent)));
538 RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
539 map.Update(tmap.GetTaggedValue());
540 int left = src->GetLeftChildIndex(parent);
541 if (left >= 0) {
542 entries.push(left);
543 }
544 int right = src->GetRightChildIndex(parent);
545 if (right >= 0) {
546 entries.push(right);
547 }
548 entries.pop();
549 }
550 return map.GetTaggedValue();
551 }
552
GetLowerKey(JSThread * thread,const JSHandle<TaggedTreeMap> & map,const JSHandle<JSTaggedValue> & key)553 JSTaggedValue TaggedTreeMap::GetLowerKey(JSThread *thread, const JSHandle<TaggedTreeMap> &map,
554 const JSHandle<JSTaggedValue> &key)
555 {
556 return RBTree::GetLowerKey(thread, map, key);
557 }
558
GetHigherKey(JSThread * thread,const JSHandle<TaggedTreeMap> & map,const JSHandle<JSTaggedValue> & key)559 JSTaggedValue TaggedTreeMap::GetHigherKey(JSThread *thread, const JSHandle<TaggedTreeMap> &map,
560 const JSHandle<JSTaggedValue> &key)
561 {
562 return RBTree::GetHigherKey(thread, map, key);
563 }
564
FindEntry(JSThread * thread,const JSHandle<TaggedTreeMap> & map,const JSHandle<JSTaggedValue> & key)565 int TaggedTreeMap::FindEntry(JSThread *thread, const JSHandle<TaggedTreeMap> &map, const JSHandle<JSTaggedValue> &key)
566 {
567 return RBTree::FindEntry(thread, map, key);
568 }
569
570 // TaggedTreeSet
Create(const JSThread * thread,int numberOfElements)571 JSTaggedValue TaggedTreeSet::Create(const JSThread *thread, int numberOfElements)
572 {
573 return RBTree::Create(thread, numberOfElements).GetTaggedValue();
574 }
575
GetArrayFromSet(const JSThread * thread,const JSHandle<TaggedTreeSet> & set)576 JSHandle<TaggedArray> TaggedTreeSet::GetArrayFromSet(const JSThread *thread, const JSHandle<TaggedTreeSet> &set)
577 {
578 return RBTree::GetSortArray(thread, set);
579 }
580
Add(JSThread * thread,JSHandle<TaggedTreeSet> & obj,const JSHandle<JSTaggedValue> & value)581 JSTaggedValue TaggedTreeSet::Add(JSThread *thread, JSHandle<TaggedTreeSet> &obj, const JSHandle<JSTaggedValue> &value)
582 {
583 return RBTree::Insert(thread, obj, value, value).GetTaggedValue();
584 }
585
Delete(JSThread * thread,const JSHandle<TaggedTreeSet> & set,int entry)586 JSTaggedValue TaggedTreeSet::Delete(JSThread *thread, const JSHandle<TaggedTreeSet> &set, int entry)
587 {
588 RBTree::Remove(thread, set, entry);
589 return RBTree::Shrink(thread, set).GetTaggedValue();
590 }
591
GetLowerKey(JSThread * thread,const JSHandle<TaggedTreeSet> & set,const JSHandle<JSTaggedValue> & key)592 JSTaggedValue TaggedTreeSet::GetLowerKey(JSThread *thread, const JSHandle<TaggedTreeSet> &set,
593 const JSHandle<JSTaggedValue> &key)
594 {
595 return RBTree::GetLowerKey(thread, set, key);
596 }
597
GetHigherKey(JSThread * thread,const JSHandle<TaggedTreeSet> & set,const JSHandle<JSTaggedValue> & key)598 JSTaggedValue TaggedTreeSet::GetHigherKey(JSThread *thread, const JSHandle<TaggedTreeSet> &set,
599 const JSHandle<JSTaggedValue> &key)
600 {
601 return RBTree::GetHigherKey(thread, set, key);
602 }
603
FindEntry(JSThread * thread,const JSHandle<TaggedTreeSet> & set,const JSHandle<JSTaggedValue> & key)604 int TaggedTreeSet::FindEntry(JSThread *thread, const JSHandle<TaggedTreeSet> &set, const JSHandle<JSTaggedValue> &key)
605 {
606 return RBTree::FindEntry(thread, set, key);
607 }
608 } // namespace panda::ecmascript
609