-
Notifications
You must be signed in to change notification settings - Fork 8
/
cartesianTree.h
54 lines (50 loc) · 2.12 KB
/
cartesianTree.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
// This code is part of the Problem Based Benchmark Suite (PBBS)
// Copyright (c) 2011 Guy Blelloch, Julian Shun and the PBBS team
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights (to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
struct node { int parent; int value;};
void merge(node* N, int left, int right) {
int head;
if (N[left].value > N[right].value) {
head = left; left = N[left].parent;}
else {head = right; right= N[right].parent;}
while(1) {
if (left == 0) {N[head].parent = right; break;}
if (right == 0) {N[head].parent = left; break;}
if (N[left].value > N[right].value) {
N[head].parent = left; left = N[left].parent;}
else {N[head].parent = right; right = N[right].parent;}
head = N[head].parent;}}
void cartesianTree(node* Nodes, int s, int n) {
if (n < 2) return;
if(n == 2) {
if (Nodes[s].value > Nodes[s+1].value) Nodes[s].parent=s+1;
else Nodes[s+1].parent=s;
return;
}
if (n > 1000){
parallel_spawn cartesianTree(Nodes,s,n/2);
cartesianTree(Nodes,s+n/2,n-n/2);
parallel_sync;
} else {
cartesianTree(Nodes,s,n/2);
cartesianTree(Nodes,s+n/2,n-n/2);
}
merge(Nodes,s+n/2-1,s+n/2);}