• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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