line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
############################################################################# |
2
|
|
|
|
|
|
|
# One chain of nodes in a Graph::Easy - used internally for layouts. |
3
|
|
|
|
|
|
|
# |
4
|
|
|
|
|
|
|
# (c) by Tels 2004-2006. Part of Graph::Easy |
5
|
|
|
|
|
|
|
############################################################################# |
6
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
package Graph::Easy::Layout::Chain; |
8
|
|
|
|
|
|
|
|
9
|
48
|
|
|
48
|
|
14205
|
use Graph::Easy::Base; |
|
48
|
|
|
|
|
55
|
|
|
48
|
|
|
|
|
1831
|
|
10
|
|
|
|
|
|
|
$VERSION = '0.76'; |
11
|
|
|
|
|
|
|
@ISA = qw/Graph::Easy::Base/; |
12
|
|
|
|
|
|
|
|
13
|
48
|
|
|
48
|
|
159
|
use strict; |
|
48
|
|
|
|
|
58
|
|
|
48
|
|
|
|
|
986
|
|
14
|
48
|
|
|
48
|
|
145
|
use warnings; |
|
48
|
|
|
|
|
57
|
|
|
48
|
|
|
|
|
1200
|
|
15
|
|
|
|
|
|
|
|
16
|
48
|
|
|
48
|
|
335
|
use Graph::Easy::Util qw(ord_values); |
|
48
|
|
|
|
|
54
|
|
|
48
|
|
|
|
|
2240
|
|
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
use constant { |
19
|
48
|
|
|
|
|
66975
|
_ACTION_NODE => 0, # place node somewhere |
20
|
|
|
|
|
|
|
_ACTION_TRACE => 1, # trace path from src to dest |
21
|
|
|
|
|
|
|
_ACTION_CHAIN => 2, # place node in chain (with parent) |
22
|
|
|
|
|
|
|
_ACTION_EDGES => 3, # trace all edges (shortes connect. first) |
23
|
48
|
|
|
48
|
|
173
|
}; |
|
48
|
|
|
|
|
49
|
|
24
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
############################################################################# |
26
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
sub _init |
28
|
|
|
|
|
|
|
{ |
29
|
|
|
|
|
|
|
# Generic init routine, to be overriden in subclasses. |
30
|
794
|
|
|
794
|
|
802
|
my ($self,$args) = @_; |
31
|
|
|
|
|
|
|
|
32
|
794
|
|
|
|
|
2100
|
foreach my $k (sort keys %$args) |
33
|
|
|
|
|
|
|
{ |
34
|
1588
|
50
|
|
|
|
4044
|
if ($k !~ /^(start|graph)\z/) |
35
|
|
|
|
|
|
|
{ |
36
|
0
|
|
|
|
|
0
|
require Carp; |
37
|
0
|
|
|
|
|
0
|
Carp::confess ("Invalid argument '$k' passed to __PACKAGE__->new()"); |
38
|
|
|
|
|
|
|
} |
39
|
1588
|
|
|
|
|
2099
|
$self->{$k} = $args->{$k}; |
40
|
|
|
|
|
|
|
} |
41
|
|
|
|
|
|
|
|
42
|
794
|
|
|
|
|
1061
|
$self->{end} = $self->{start}; |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
# store chain at node (to lookup node => chain info) |
45
|
794
|
|
|
|
|
914
|
$self->{start}->{_chain} = $self; |
46
|
794
|
|
|
|
|
1496
|
$self->{start}->{_next} = undef; |
47
|
|
|
|
|
|
|
|
48
|
794
|
|
|
|
|
792
|
$self->{len} = 1; |
49
|
|
|
|
|
|
|
|
50
|
794
|
|
|
|
|
1664
|
$self; |
51
|
|
|
|
|
|
|
} |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
sub start |
54
|
|
|
|
|
|
|
{ |
55
|
|
|
|
|
|
|
# return first node in the chain |
56
|
6
|
|
|
6
|
1
|
8
|
my $self = shift; |
57
|
|
|
|
|
|
|
|
58
|
6
|
|
|
|
|
19
|
$self->{start}; |
59
|
|
|
|
|
|
|
} |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
sub end |
62
|
|
|
|
|
|
|
{ |
63
|
|
|
|
|
|
|
# return last node in the chain |
64
|
6
|
|
|
6
|
1
|
7
|
my $self = shift; |
65
|
|
|
|
|
|
|
|
66
|
6
|
|
|
|
|
17
|
$self->{end}; |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
sub add_node |
70
|
|
|
|
|
|
|
{ |
71
|
|
|
|
|
|
|
# add a node at the end of the chain |
72
|
357
|
|
|
357
|
1
|
382
|
my ($self, $node) = @_; |
73
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
# store at end |
75
|
357
|
|
|
|
|
436
|
$self->{end}->{_next} = $node; |
76
|
357
|
|
|
|
|
327
|
$self->{end} = $node; |
77
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
# store chain at node (to lookup node => chain info) |
79
|
357
|
|
|
|
|
567
|
$node->{_chain} = $self; |
80
|
357
|
|
|
|
|
601
|
$node->{_next} = undef; |
81
|
|
|
|
|
|
|
|
82
|
357
|
|
|
|
|
318
|
$self->{len} ++; |
83
|
|
|
|
|
|
|
|
84
|
357
|
|
|
|
|
438
|
$self; |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
sub length |
88
|
|
|
|
|
|
|
{ |
89
|
|
|
|
|
|
|
# Return the length of the chain in nodes. Takes optional |
90
|
|
|
|
|
|
|
# node from where to calculate length. |
91
|
11
|
|
|
11
|
1
|
16
|
my ($self, $node) = @_; |
92
|
|
|
|
|
|
|
|
93
|
11
|
100
|
|
|
|
37
|
return $self->{len} unless defined $node; |
94
|
|
|
|
|
|
|
|
95
|
3
|
|
|
|
|
3
|
my $len = 0; |
96
|
3
|
|
|
|
|
6
|
while (defined $node) |
97
|
|
|
|
|
|
|
{ |
98
|
4
|
|
|
|
|
12
|
$len++; $node = $node->{_next}; |
|
4
|
|
|
|
|
7
|
|
99
|
|
|
|
|
|
|
} |
100
|
|
|
|
|
|
|
|
101
|
3
|
|
|
|
|
8
|
$len; |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
sub nodes |
105
|
|
|
|
|
|
|
{ |
106
|
|
|
|
|
|
|
# return all the nodes in the chain as a list, in order. |
107
|
3
|
|
|
3
|
1
|
728
|
my $self = shift; |
108
|
|
|
|
|
|
|
|
109
|
3
|
|
|
|
|
5
|
my @nodes = (); |
110
|
3
|
|
|
|
|
3
|
my $n = $self->{start}; |
111
|
3
|
|
|
|
|
5
|
while (defined $n) |
112
|
|
|
|
|
|
|
{ |
113
|
12
|
|
|
|
|
9
|
push @nodes, $n; |
114
|
12
|
|
|
|
|
17
|
$n = $n->{_next}; |
115
|
|
|
|
|
|
|
} |
116
|
|
|
|
|
|
|
|
117
|
3
|
|
|
|
|
8
|
@nodes; |
118
|
|
|
|
|
|
|
} |
119
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
sub layout |
121
|
|
|
|
|
|
|
{ |
122
|
|
|
|
|
|
|
# Return an action stack containing the nec. actions to |
123
|
|
|
|
|
|
|
# lay out the nodes in the chain, plus any connections between |
124
|
|
|
|
|
|
|
# them. |
125
|
605
|
|
|
605
|
1
|
678
|
my ($self, $edge) = @_; |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
# prevent doing it twice |
128
|
605
|
50
|
|
|
|
922
|
return [] if $self->{_done}; $self->{_done} = 1; |
|
605
|
|
|
|
|
641
|
|
129
|
|
|
|
|
|
|
|
130
|
605
|
|
|
|
|
587
|
my @TODO = (); |
131
|
|
|
|
|
|
|
|
132
|
605
|
|
|
|
|
497
|
my $g = $self->{graph}; |
133
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
# first, layout all the nodes in the chain: |
135
|
|
|
|
|
|
|
|
136
|
|
|
|
|
|
|
# start with first node |
137
|
605
|
|
|
|
|
487
|
my $pre = $self->{start}; my $n = $pre->{_next}; |
|
605
|
|
|
|
|
600
|
|
138
|
605
|
100
|
|
|
|
929
|
if (exists $pre->{_todo}) |
139
|
|
|
|
|
|
|
{ |
140
|
|
|
|
|
|
|
# edges with a flow attribute must be handled differently |
141
|
|
|
|
|
|
|
# XXX TODO: the test for attribute('flow') might be wrong (raw_attribute()?) |
142
|
204
|
100
|
100
|
|
|
896
|
if ($edge && ($edge->{to} == $pre) && ($edge->attribute('flow') || $edge->has_ports())) |
|
|
|
33
|
|
|
|
|
|
|
|
66
|
|
|
|
|
143
|
|
|
|
|
|
|
{ |
144
|
103
|
|
|
|
|
281
|
push @TODO, $g->_action( _ACTION_CHAIN, $pre, 0, $edge->{from}, $edge); |
145
|
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
else |
147
|
|
|
|
|
|
|
{ |
148
|
101
|
|
|
|
|
237
|
push @TODO, $g->_action( _ACTION_NODE, $pre, 0, $edge ); |
149
|
|
|
|
|
|
|
} |
150
|
|
|
|
|
|
|
} |
151
|
|
|
|
|
|
|
|
152
|
605
|
50
|
|
|
|
986
|
print STDERR "# Stack after first:\n" if $g->{debug}; |
153
|
605
|
50
|
|
|
|
797
|
$g->_dump_stack(@TODO) if $g->{debug}; |
154
|
|
|
|
|
|
|
|
155
|
605
|
|
|
|
|
932
|
while (defined $n) |
156
|
|
|
|
|
|
|
{ |
157
|
585
|
100
|
|
|
|
876
|
if (exists $n->{_todo}) |
158
|
|
|
|
|
|
|
{ |
159
|
|
|
|
|
|
|
# CHAIN means if $n isn't placed yet, it will be done with |
160
|
|
|
|
|
|
|
# $pre as parent: |
161
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
# in case there are multiple edges to the target node, use the first |
163
|
|
|
|
|
|
|
# one to determine the flow: |
164
|
516
|
|
|
|
|
1082
|
my @edges = $g->edge($pre,$n); |
165
|
|
|
|
|
|
|
|
166
|
516
|
|
|
|
|
1068
|
push @TODO, $g->_action( _ACTION_CHAIN, $n, 0, $pre, $edges[0] ); |
167
|
|
|
|
|
|
|
} |
168
|
585
|
|
|
|
|
517
|
$pre = $n; |
169
|
585
|
|
|
|
|
943
|
$n = $n->{_next}; |
170
|
|
|
|
|
|
|
} |
171
|
|
|
|
|
|
|
|
172
|
605
|
50
|
|
|
|
960
|
print STDERR "# Stack after chaining:\n" if $g->{debug}; |
173
|
605
|
50
|
|
|
|
877
|
$g->_dump_stack(@TODO) if $g->{debug}; |
174
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
# link from each node to the next |
176
|
605
|
|
|
|
|
530
|
$pre = $self->{start}; $n = $pre->{_next}; |
|
605
|
|
|
|
|
588
|
|
177
|
605
|
|
|
|
|
892
|
while (defined $n) |
178
|
|
|
|
|
|
|
{ |
179
|
|
|
|
|
|
|
# first do edges going from P to N |
180
|
|
|
|
|
|
|
#for my $e (sort { $a->{to}->{name} cmp $b->{to}->{name} } values %{$pre->{edges}}) |
181
|
585
|
|
|
|
|
908
|
for my $e (ord_values ( $pre->{edges})) |
182
|
|
|
|
|
|
|
{ |
183
|
|
|
|
|
|
|
# skip selfloops and backward links, these will be done later |
184
|
1250
|
100
|
|
|
|
2072
|
next if $e->{to} != $n; |
185
|
|
|
|
|
|
|
|
186
|
602
|
100
|
|
|
|
896
|
next unless exists $e->{_todo}; |
187
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
# skip links from/to groups |
189
|
|
|
|
|
|
|
next if $e->{to}->isa('Graph::Easy::Group') || |
190
|
582
|
50
|
33
|
|
|
3236
|
$e->{from}->isa('Graph::Easy::Group'); |
191
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
# # skip edges with a flow |
193
|
|
|
|
|
|
|
# next if exists $e->{att}->{start} || exist $e->{att}->{end}; |
194
|
|
|
|
|
|
|
|
195
|
582
|
|
|
|
|
894
|
push @TODO, [ _ACTION_TRACE, $e ]; |
196
|
582
|
|
|
|
|
788
|
delete $e->{_todo}; |
197
|
|
|
|
|
|
|
} |
198
|
|
|
|
|
|
|
|
199
|
585
|
|
|
|
|
458
|
} continue { $pre = $n; $n = $n->{_next}; } |
|
585
|
|
|
|
|
1040
|
|
200
|
|
|
|
|
|
|
|
201
|
605
|
50
|
|
|
|
848
|
print STDERR "# Stack after chain-linking:\n" if $g->{debug}; |
202
|
605
|
50
|
|
|
|
855
|
$g->_dump_stack(@TODO) if $g->{debug}; |
203
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
# Do all other links inside the chain (backwards, going forward more than |
205
|
|
|
|
|
|
|
# one node etc) |
206
|
|
|
|
|
|
|
|
207
|
605
|
|
|
|
|
584
|
$n = $self->{start}; |
208
|
605
|
|
|
|
|
858
|
while (defined $n) |
209
|
|
|
|
|
|
|
{ |
210
|
1190
|
|
|
|
|
837
|
my @edges; |
211
|
|
|
|
|
|
|
|
212
|
|
|
|
|
|
|
my @count; |
213
|
|
|
|
|
|
|
|
214
|
1190
|
50
|
|
|
|
1520
|
print STDERR "# inter-chain link from $n->{name}\n" if $g->{debug}; |
215
|
|
|
|
|
|
|
|
216
|
|
|
|
|
|
|
# gather all edges starting at $n, but do the ones with a flow first |
217
|
|
|
|
|
|
|
# for my $e (sort { $a->{to}->{name} cmp $b->{to}->{name} } values %{$n->{edges}}) |
218
|
1190
|
|
|
|
|
1959
|
for my $e (ord_values ( $n->{edges})) |
219
|
|
|
|
|
|
|
{ |
220
|
|
|
|
|
|
|
# skip selfloops, these will be done later |
221
|
1941
|
100
|
|
|
|
3352
|
next if $e->{to} == $n; |
222
|
|
|
|
|
|
|
|
223
|
889
|
100
|
|
|
|
1509
|
next if !ref($e->{to}->{_chain}); |
224
|
883
|
50
|
|
|
|
1359
|
next if !ref($e->{from}->{_chain}); |
225
|
|
|
|
|
|
|
|
226
|
883
|
100
|
|
|
|
1670
|
next if $e->has_ports(); |
227
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
# skip links from/to groups |
229
|
|
|
|
|
|
|
next if $e->{to}->isa('Graph::Easy::Group') || |
230
|
815
|
50
|
33
|
|
|
5020
|
$e->{from}->isa('Graph::Easy::Group'); |
231
|
|
|
|
|
|
|
|
232
|
815
|
50
|
|
|
|
1322
|
print STDERR "# inter-chain link from $n->{name} to $e->{to}->{name}\n" if $g->{debug}; |
233
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
# leaving the chain? |
235
|
815
|
100
|
|
|
|
1410
|
next if $e->{to}->{_chain} != $self; |
236
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
# print STDERR "# trying for $n->{name}:\t $e->{from}->{name} to $e->{to}->{name}\n"; |
238
|
587
|
100
|
|
|
|
1165
|
next unless exists $e->{_todo}; |
239
|
|
|
|
|
|
|
|
240
|
|
|
|
|
|
|
# calculate for this edge, how far it goes |
241
|
72
|
|
|
|
|
93
|
my $count = 0; |
242
|
72
|
|
|
|
|
69
|
my $curr = $n; |
243
|
72
|
|
100
|
|
|
290
|
while (defined $curr && $curr != $e->{to}) |
244
|
|
|
|
|
|
|
{ |
245
|
185
|
|
|
|
|
162
|
$curr = $curr->{_next}; $count ++; |
|
185
|
|
|
|
|
493
|
|
246
|
|
|
|
|
|
|
} |
247
|
72
|
100
|
|
|
|
145
|
if (!defined $curr) |
248
|
|
|
|
|
|
|
{ |
249
|
|
|
|
|
|
|
# edge goes backward |
250
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
# start at $to |
252
|
15
|
|
|
|
|
23
|
$curr = $e->{to}; |
253
|
15
|
|
|
|
|
21
|
$count = 0; |
254
|
15
|
|
66
|
|
|
77
|
while (defined $curr && $curr != $e->{from}) |
255
|
|
|
|
|
|
|
{ |
256
|
32
|
|
|
|
|
27
|
$curr = $curr->{_next}; $count ++; |
|
32
|
|
|
|
|
88
|
|
257
|
|
|
|
|
|
|
} |
258
|
15
|
50
|
|
|
|
29
|
$count = 100000 if !defined $curr; # should not happen |
259
|
|
|
|
|
|
|
} |
260
|
72
|
|
|
|
|
125
|
push @edges, [ $count, $e ]; |
261
|
72
|
|
|
|
|
216
|
push @count, [ $count, $e->{from}->{name}, $e->{to}->{name} ]; |
262
|
|
|
|
|
|
|
} |
263
|
|
|
|
|
|
|
|
264
|
|
|
|
|
|
|
# use Data::Dumper; print STDERR "count\n", Dumper(@count); |
265
|
|
|
|
|
|
|
|
266
|
|
|
|
|
|
|
# do edges, shortest first |
267
|
1190
|
|
|
|
|
1799
|
for my $e (sort { $a->[0] <=> $b->[0] } @edges) |
|
8
|
|
|
|
|
14
|
|
268
|
|
|
|
|
|
|
{ |
269
|
72
|
|
|
|
|
129
|
push @TODO, [ _ACTION_TRACE, $e->[1] ]; |
270
|
72
|
|
|
|
|
104
|
delete $e->[1]->{_todo}; |
271
|
|
|
|
|
|
|
} |
272
|
|
|
|
|
|
|
|
273
|
1190
|
|
|
|
|
2087
|
$n = $n->{_next}; |
274
|
|
|
|
|
|
|
} |
275
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
# also do all selfloops on $n |
277
|
605
|
|
|
|
|
563
|
$n = $self->{start}; |
278
|
605
|
|
|
|
|
889
|
while (defined $n) |
279
|
|
|
|
|
|
|
{ |
280
|
|
|
|
|
|
|
# for my $e (sort { $a->{to}->{name} cmp $b->{to}->{name} } values %{$n->{edges}}) |
281
|
1190
|
|
|
|
|
1632
|
for my $e (ord_values $n->{edges}) |
282
|
|
|
|
|
|
|
{ |
283
|
1941
|
100
|
|
|
|
2723
|
next unless exists $e->{_todo}; |
284
|
|
|
|
|
|
|
|
285
|
|
|
|
|
|
|
# print STDERR "# $e->{from}->{name} to $e->{to}->{name} on $n->{name}\n"; |
286
|
|
|
|
|
|
|
# print STDERR "# ne $e->{to} $n $e->{id}\n" |
287
|
|
|
|
|
|
|
# if $e->{from} != $n || $e->{to} != $n; # no selfloop? |
288
|
|
|
|
|
|
|
|
289
|
515
|
100
|
100
|
|
|
1462
|
next if $e->{from} != $n || $e->{to} != $n; # no selfloop? |
290
|
|
|
|
|
|
|
|
291
|
31
|
|
|
|
|
61
|
push @TODO, [ _ACTION_TRACE, $e ]; |
292
|
31
|
|
|
|
|
40
|
delete $e->{_todo}; |
293
|
|
|
|
|
|
|
} |
294
|
1190
|
|
|
|
|
1933
|
$n = $n->{_next}; |
295
|
|
|
|
|
|
|
} |
296
|
|
|
|
|
|
|
|
297
|
605
|
50
|
|
|
|
895
|
print STDERR "# Stack after self-loops:\n" if $g->{debug}; |
298
|
605
|
50
|
|
|
|
869
|
$g->_dump_stack(@TODO) if $g->{debug}; |
299
|
|
|
|
|
|
|
|
300
|
|
|
|
|
|
|
# XXX TODO |
301
|
|
|
|
|
|
|
# now we should do any links that start or end at this chain, recursively |
302
|
|
|
|
|
|
|
|
303
|
605
|
|
|
|
|
498
|
$n = $self->{start}; |
304
|
605
|
|
|
|
|
865
|
while (defined $n) |
305
|
|
|
|
|
|
|
{ |
306
|
|
|
|
|
|
|
|
307
|
|
|
|
|
|
|
# all chains that start at this node |
308
|
1190
|
|
|
|
|
931
|
for my $e (sort { $a->{to}->{name} cmp $b->{to}->{name} } values %{$n->{edges}}) |
|
1176
|
|
|
|
|
1464
|
|
|
1190
|
|
|
|
|
2047
|
|
309
|
|
|
|
|
|
|
{ |
310
|
1941
|
|
|
|
|
1542
|
my $to = $e->{to}; |
311
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
# skip links to groups |
313
|
1941
|
50
|
|
|
|
4185
|
next if $to->isa('Graph::Easy::Group'); |
314
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
# print STDERR "# chain-tracking to: $to->{name} $to->{_chain}\n"; |
316
|
|
|
|
|
|
|
|
317
|
1941
|
100
|
66
|
|
|
6173
|
next unless exists $to->{_chain} && ref($to->{_chain}) =~ /Chain/; |
318
|
1935
|
|
|
|
|
1453
|
my $chain = $to->{_chain}; |
319
|
1935
|
100
|
|
|
|
2922
|
next if $chain->{_done}; |
320
|
|
|
|
|
|
|
|
321
|
|
|
|
|
|
|
# print STDERR "# chain-tracking to: $to->{name}\n"; |
322
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
# pass the edge along, in case it has a flow |
324
|
|
|
|
|
|
|
# my @pass = (); |
325
|
|
|
|
|
|
|
# push @pass, $e if $chain->{_first} && $e->{to} == $chain->{_first}; |
326
|
165
|
50
|
|
|
|
296
|
push @TODO, @{ $chain->layout($e) } unless $chain->{_done}; |
|
165
|
|
|
|
|
353
|
|
327
|
|
|
|
|
|
|
|
328
|
|
|
|
|
|
|
# link the edges to $to |
329
|
165
|
100
|
|
|
|
348
|
next unless exists $e->{_todo}; # was already done above? |
330
|
|
|
|
|
|
|
|
331
|
|
|
|
|
|
|
# next if $e->has_ports(); |
332
|
|
|
|
|
|
|
|
333
|
148
|
|
|
|
|
232
|
push @TODO, [ _ACTION_TRACE, $e ]; |
334
|
148
|
|
|
|
|
213
|
delete $e->{_todo}; |
335
|
|
|
|
|
|
|
} |
336
|
1190
|
|
|
|
|
1737
|
$n = $n->{_next}; |
337
|
|
|
|
|
|
|
} |
338
|
|
|
|
|
|
|
|
339
|
605
|
|
|
|
|
1687
|
\@TODO; |
340
|
|
|
|
|
|
|
} |
341
|
|
|
|
|
|
|
|
342
|
|
|
|
|
|
|
sub dump |
343
|
|
|
|
|
|
|
{ |
344
|
|
|
|
|
|
|
# dump the chain to STDERR |
345
|
0
|
|
|
0
|
1
|
0
|
my ($self, $indent) = @_; |
346
|
|
|
|
|
|
|
|
347
|
0
|
0
|
|
|
|
0
|
$indent = '' unless defined $indent; |
348
|
|
|
|
|
|
|
|
349
|
0
|
|
|
|
|
0
|
print STDERR "#$indent chain id $self->{id} (len $self->{len}):\n"; |
350
|
0
|
0
|
0
|
|
|
0
|
print STDERR "#$indent is empty\n" and return if $self->{len} == 0; |
351
|
|
|
|
|
|
|
|
352
|
0
|
|
|
|
|
0
|
my $n = $self->{start}; |
353
|
0
|
|
|
|
|
0
|
while (defined $n) |
354
|
|
|
|
|
|
|
{ |
355
|
0
|
|
|
|
|
0
|
print STDERR "#$indent $n->{name} (chain id: $n->{_chain}->{id})\n"; |
356
|
0
|
|
|
|
|
0
|
$n = $n->{_next}; |
357
|
|
|
|
|
|
|
} |
358
|
0
|
|
|
|
|
0
|
$self; |
359
|
|
|
|
|
|
|
} |
360
|
|
|
|
|
|
|
|
361
|
|
|
|
|
|
|
sub merge |
362
|
|
|
|
|
|
|
{ |
363
|
|
|
|
|
|
|
# take another chain, and merge it into ourselves. If $where is defined, |
364
|
|
|
|
|
|
|
# absorb only the nodes from $where onwards (instead of all of them). |
365
|
214
|
|
|
214
|
1
|
212
|
my ($self, $other, $where) = @_; |
366
|
|
|
|
|
|
|
|
367
|
214
|
|
|
|
|
191
|
my $g = $self->{graph}; |
368
|
|
|
|
|
|
|
|
369
|
214
|
50
|
|
|
|
315
|
print STDERR "# panik: ", join(" \n",caller()),"\n" if !defined $other; |
370
|
|
|
|
|
|
|
|
371
|
|
|
|
|
|
|
print STDERR |
372
|
|
|
|
|
|
|
"# Merging chain $other->{id} (len $other->{len}) into $self->{id} (len $self->{len})\n" |
373
|
214
|
50
|
|
|
|
364
|
if $g->{debug}; |
374
|
|
|
|
|
|
|
|
375
|
|
|
|
|
|
|
print STDERR |
376
|
|
|
|
|
|
|
"# Merging from $where->{name} onwards\n" |
377
|
214
|
50
|
33
|
|
|
390
|
if $g->{debug} && ref($where); |
378
|
|
|
|
|
|
|
|
379
|
|
|
|
|
|
|
# cannot merge myself into myself (without allocating infinitely memory) |
380
|
214
|
50
|
|
|
|
333
|
return if $self == $other; |
381
|
|
|
|
|
|
|
|
382
|
|
|
|
|
|
|
# start at start as default |
383
|
214
|
50
|
66
|
|
|
1018
|
$where = undef unless ref($where) && exists $where->{_chain} && $where->{_chain} == $other; |
|
|
|
33
|
|
|
|
|
384
|
|
|
|
|
|
|
|
385
|
214
|
100
|
|
|
|
318
|
$where = $other->{start} unless defined $where; |
386
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
# make all nodes from chain #1 belong to it (to detect loops) |
388
|
214
|
|
|
|
|
207
|
my $n = $self->{start}; |
389
|
214
|
|
|
|
|
346
|
while (defined $n) |
390
|
|
|
|
|
|
|
{ |
391
|
232
|
|
|
|
|
213
|
$n->{_chain} = $self; |
392
|
232
|
|
|
|
|
402
|
$n = $n->{_next}; |
393
|
|
|
|
|
|
|
} |
394
|
|
|
|
|
|
|
|
395
|
214
|
50
|
|
|
|
333
|
print STDERR "# changed nodes\n" if $g->{debug}; |
396
|
214
|
50
|
|
|
|
313
|
$self->dump() if $g->{debug}; |
397
|
|
|
|
|
|
|
|
398
|
|
|
|
|
|
|
# terminate at $where |
399
|
214
|
|
|
|
|
244
|
$self->{end}->{_next} = $where; |
400
|
214
|
|
|
|
|
209
|
$self->{end} = $other->{end}; |
401
|
|
|
|
|
|
|
|
402
|
|
|
|
|
|
|
# start at joiner |
403
|
214
|
|
|
|
|
194
|
$n = $where; |
404
|
214
|
|
|
|
|
327
|
while (ref($n)) |
405
|
|
|
|
|
|
|
{ |
406
|
496
|
|
|
|
|
383
|
$n->{_chain} = $self; |
407
|
496
|
|
|
|
|
331
|
my $pre = $n; |
408
|
496
|
|
|
|
|
369
|
$n = $n->{_next}; |
409
|
|
|
|
|
|
|
|
410
|
|
|
|
|
|
|
# sleep(1); |
411
|
|
|
|
|
|
|
# print "# at $n->{name} $n->{_chain}\n" if ref($n); |
412
|
496
|
50
|
66
|
|
|
1942
|
if (ref($n) && defined $n->{_chain} && $n->{_chain} == $self) # already points into ourself? |
|
|
|
66
|
|
|
|
|
413
|
|
|
|
|
|
|
{ |
414
|
|
|
|
|
|
|
# sleep(1); |
415
|
|
|
|
|
|
|
# print "# pre $pre->{name} $pre->{_chain}\n"; |
416
|
0
|
|
|
|
|
0
|
$pre->{_next} = undef; # terminate |
417
|
0
|
|
|
|
|
0
|
$self->{end} = $pre; |
418
|
0
|
|
|
|
|
0
|
last; |
419
|
|
|
|
|
|
|
} |
420
|
|
|
|
|
|
|
} |
421
|
|
|
|
|
|
|
|
422
|
|
|
|
|
|
|
# could speed this up |
423
|
214
|
|
|
|
|
196
|
$self->{len} = 0; $n = $self->{start}; |
|
214
|
|
|
|
|
188
|
|
424
|
214
|
|
|
|
|
340
|
while (defined $n) |
425
|
|
|
|
|
|
|
{ |
426
|
728
|
|
|
|
|
487
|
$self->{len}++; $n = $n->{_next}; |
|
728
|
|
|
|
|
937
|
|
427
|
|
|
|
|
|
|
} |
428
|
|
|
|
|
|
|
|
429
|
|
|
|
|
|
|
# print "done merging, dumping result:\n"; |
430
|
|
|
|
|
|
|
# $self->dump(); sleep(10); |
431
|
|
|
|
|
|
|
|
432
|
214
|
100
|
66
|
|
|
716
|
if (defined $other->{start} && $where == $other->{start}) |
433
|
|
|
|
|
|
|
{ |
434
|
|
|
|
|
|
|
# we absorbed the other chain completely, so drop it |
435
|
165
|
|
|
|
|
166
|
$other->{end} = undef; |
436
|
165
|
|
|
|
|
137
|
$other->{start} = undef; |
437
|
165
|
|
|
|
|
162
|
$other->{len} = 0; |
438
|
|
|
|
|
|
|
# caller is responsible for cleaning it up |
439
|
|
|
|
|
|
|
} |
440
|
|
|
|
|
|
|
|
441
|
214
|
50
|
|
|
|
341
|
print STDERR "# after merging\n" if $g->{debug}; |
442
|
214
|
50
|
|
|
|
298
|
$self->dump() if $g->{debug}; |
443
|
|
|
|
|
|
|
|
444
|
214
|
|
|
|
|
309
|
$self; |
445
|
|
|
|
|
|
|
} |
446
|
|
|
|
|
|
|
|
447
|
|
|
|
|
|
|
1; |
448
|
|
|
|
|
|
|
__END__ |