95
95
def iter_topo_order(self):
96
96
"""Yield the nodes of the graph in a topological order.
98
98
After finishing iteration the sorter is empty and you cannot continue
102
visitable = set(graph)
104
# this is a stack storing the depth first search into the graph.
105
pending_node_stack = []
106
# at each level of 'recursion' we have to check each parent. This
107
# stack stores the parents we have not yet checked for the node at the
108
# matching depth in pending_node_stack
109
pending_parents_stack = []
111
# this is a set of the completed nodes for fast checking whether a
112
# parent in a node we are processing on the stack has already been
113
# emitted and thus can be skipped.
114
completed_node_names = set()
102
117
# now pick a random node in the source graph, and transfer it to the
103
# top of the depth first search stack.
104
node_name, parents = self._graph.popitem()
105
self._push_node(node_name, parents)
106
while self._node_name_stack:
107
# loop until this call completes.
108
parents_to_visit = self._pending_parents_stack[-1]
109
# if all parents are done, the revision is done
118
# top of the depth first search stack of pending nodes.
119
node_name, parents = graph.popitem()
120
pending_node_stack.append(node_name)
121
pending_parents_stack.append(list(parents))
123
# loop until pending_node_stack is empty
124
while pending_node_stack:
125
parents_to_visit = pending_parents_stack[-1]
126
# if there are no parents left, the revision is done
110
127
if not parents_to_visit:
111
128
# append the revision to the topo sorted list
112
# all the nodes parents have been added to the output, now
113
# we can add it to the output.
114
yield self._pop_node()
129
# all the nodes parents have been added to the output,
130
# now we can add it to the output.
131
popped_node = pending_node_stack.pop()
132
pending_parents_stack.pop()
133
completed_node_names.add(popped_node)
116
while self._pending_parents_stack[-1]:
117
# recurse depth first into a single parent
118
next_node_name = self._pending_parents_stack[-1].pop()
119
if next_node_name in self._completed_node_names:
120
# this parent was completed by a child on the
121
# call stack. skip it.
123
# otherwise transfer it from the source graph into the
124
# top of the current depth first search stack.
126
parents = self._graph.pop(next_node_name)
128
# if the next node is not in the source graph it has
129
# already been popped from it and placed into the
130
# current search stack (but not completed or we would
131
# have hit the continue 4 lines up.
132
# this indicates a cycle.
133
raise errors.GraphCycleError(self._node_name_stack)
134
self._push_node(next_node_name, parents)
135
# and do not continue processing parents until this 'call'
139
def _push_node(self, node_name, parents):
140
"""Add node_name to the pending node stack.
142
Names in this stack will get emitted into the output as they are popped
145
self._node_name_stack.append(node_name)
146
self._pending_parents_stack.append(list(parents))
149
"""Pop the top node off the stack
151
The node is appended to the sorted output.
153
# we are returning from the flattened call frame:
154
# pop off the local variables
155
node_name = self._node_name_stack.pop()
156
self._pending_parents_stack.pop()
158
self._completed_node_names.add(node_name)
136
# recurse depth first into a single parent
137
next_node_name = parents_to_visit.pop()
139
if next_node_name in completed_node_names:
140
# parent was already completed by a child, skip it.
142
if next_node_name not in visitable:
143
# parent is not a node in the original graph, skip it.
146
# transfer it along with its parents from the source graph
147
# into the top of the current depth first search stack.
149
parents = graph.pop(next_node_name)
151
# if the next node is not in the source graph it has
152
# already been popped from it and placed into the
153
# current search stack (but not completed or we would
154
# have hit the continue 6 lines up). this indicates a
156
raise errors.GraphCycleError(pending_node_stack)
157
pending_node_stack.append(next_node_name)
158
pending_parents_stack.append(list(parents))
162
161
def merge_sort(graph, branch_tip, mainline_revisions=None, generate_revno=False):
163
162
"""Topological sort a graph which groups merges.
165
164
:param graph: sequence of pairs of node->parents_list.
166
:param branch_tip: the tip of the branch to graph. Revisions not
165
:param branch_tip: the tip of the branch to graph. Revisions not
167
166
reachable from branch_tip are not included in the
169
168
:param mainline_revisions: If not None this forces a mainline to be
379
# the scheduling order is: F, E, D, C, B, A
400
# the scheduling order is: F, E, D, C, B, A
380
401
# that is - 'left subtree, right subtree, node'
381
402
# which would mean that when we schedule A we can emit the entire tree.
382
403
self._scheduled_nodes = []
383
# This records for each node when we have processed its left most
404
# This records for each node when we have processed its left most
384
405
# unmerged subtree. After this subtree is scheduled, all other subtrees
385
406
# have their merge depth increased by one from this nodes merge depth.
386
407
# it contains tuples - name, merge_depth
387
408
self._left_subtree_pushed_stack = []
389
410
# seed the search with the tip of the branch
390
if branch_tip is not None:
411
if (branch_tip is not None and
412
branch_tip != _mod_revision.NULL_REVISION and
413
branch_tip != (_mod_revision.NULL_REVISION,)):
391
414
parents = self._graph.pop(branch_tip)
392
415
self._push_node(branch_tip, 0, parents)
394
417
def sorted(self):
395
418
"""Sort the graph and return as a list.
397
420
After calling this the sorter is empty and you must create a new one.
399
422
return list(self.iter_topo_order())
401
424
def iter_topo_order(self):
402
425
"""Yield the nodes of the graph in a topological order.
404
427
After finishing iteration the sorter is empty and you cannot continue
407
while self._node_name_stack:
430
# These are safe to offload to local variables, because they are used
431
# as a stack and modified in place, never assigned to.
432
node_name_stack = self._node_name_stack
433
node_merge_depth_stack = self._node_merge_depth_stack
434
pending_parents_stack = self._pending_parents_stack
435
left_subtree_pushed_stack = self._left_subtree_pushed_stack
436
completed_node_names = self._completed_node_names
437
scheduled_nodes = self._scheduled_nodes
439
graph_pop = self._graph.pop
441
def push_node(node_name, merge_depth, parents,
442
node_name_stack_append=node_name_stack.append,
443
node_merge_depth_stack_append=node_merge_depth_stack.append,
444
left_subtree_pushed_stack_append=left_subtree_pushed_stack.append,
445
pending_parents_stack_append=pending_parents_stack.append,
446
first_child_stack_append=self._first_child_stack.append,
449
"""Add node_name to the pending node stack.
451
Names in this stack will get emitted into the output as they are popped
454
This inlines a lot of self._variable.append functions as local
457
node_name_stack_append(node_name)
458
node_merge_depth_stack_append(merge_depth)
459
left_subtree_pushed_stack_append(False)
460
pending_parents_stack_append(list(parents))
461
# as we push it, check if it is the first child
464
# node has parents, assign from the left most parent.
466
parent_info = revnos[parents[0]]
468
# Left-hand parent is a ghost, consider it not to exist
470
if parent_info is not None:
471
first_child = parent_info[1]
472
parent_info[1] = False
474
# We don't use the same algorithm here, but we need to keep the
477
first_child_stack_append(first_child)
479
def pop_node(node_name_stack_pop=node_name_stack.pop,
480
node_merge_depth_stack_pop=node_merge_depth_stack.pop,
481
first_child_stack_pop=self._first_child_stack.pop,
482
left_subtree_pushed_stack_pop=left_subtree_pushed_stack.pop,
483
pending_parents_stack_pop=pending_parents_stack.pop,
484
original_graph=self._original_graph,
486
completed_node_names_add=self._completed_node_names.add,
487
scheduled_nodes_append=scheduled_nodes.append,
488
revno_to_branch_count=self._revno_to_branch_count,
490
"""Pop the top node off the stack
492
The node is appended to the sorted output.
494
# we are returning from the flattened call frame:
495
# pop off the local variables
496
node_name = node_name_stack_pop()
497
merge_depth = node_merge_depth_stack_pop()
498
first_child = first_child_stack_pop()
499
# remove this node from the pending lists:
500
left_subtree_pushed_stack_pop()
501
pending_parents_stack_pop()
503
parents = original_graph[node_name]
506
# node has parents, assign from the left most parent.
508
parent_revno = revnos[parents[0]][0]
510
# left-hand parent is a ghost, treat it as not existing
512
if parent_revno is not None:
514
# not the first child, make a new branch
515
base_revno = parent_revno[0]
516
branch_count = revno_to_branch_count.get(base_revno, 0)
518
revno_to_branch_count[base_revno] = branch_count
519
revno = (parent_revno[0], branch_count, 1)
520
# revno = (parent_revno[0], branch_count, parent_revno[-1]+1)
522
# as the first child, we just increase the final revision
524
revno = parent_revno[:-1] + (parent_revno[-1] + 1,)
526
# no parents, use the root sequence
527
root_count = revno_to_branch_count.get(0, -1)
530
revno = (0, root_count, 1)
533
revno_to_branch_count[0] = root_count
535
# store the revno for this node for future reference
536
revnos[node_name][0] = revno
537
completed_node_names_add(node_name)
538
scheduled_nodes_append((node_name, merge_depth, revno))
542
while node_name_stack:
408
543
# loop until this call completes.
409
parents_to_visit = self._pending_parents_stack[-1]
544
parents_to_visit = pending_parents_stack[-1]
410
545
# if all parents are done, the revision is done
411
546
if not parents_to_visit:
412
547
# append the revision to the topo sorted scheduled list:
413
548
# all the nodes parents have been scheduled added, now
414
549
# we can add it to the output.
417
while self._pending_parents_stack[-1]:
418
if not self._left_subtree_pushed_stack[-1]:
552
while pending_parents_stack[-1]:
553
if not left_subtree_pushed_stack[-1]:
419
554
# recurse depth first into the primary parent
420
next_node_name = self._pending_parents_stack[-1].pop(0)
555
next_node_name = pending_parents_stack[-1].pop(0)
556
is_left_subtree = True
557
left_subtree_pushed_stack[-1] = True
422
559
# place any merges in right-to-left order for scheduling
423
560
# which gives us left-to-right order after we reverse
424
# the scheduled queue. XXX: This has the effect of
561
# the scheduled queue. XXX: This has the effect of
425
562
# allocating common-new revisions to the right-most
426
# subtree rather than the left most, which will
563
# subtree rather than the left most, which will
427
564
# display nicely (you get smaller trees at the top
428
565
# of the combined merge).
429
next_node_name = self._pending_parents_stack[-1].pop()
430
if next_node_name in self._completed_node_names:
566
next_node_name = pending_parents_stack[-1].pop()
567
is_left_subtree = False
568
if next_node_name in completed_node_names:
431
569
# this parent was completed by a child on the
432
570
# call stack. skip it.
434
572
# otherwise transfer it from the source graph into the
435
573
# top of the current depth first search stack.
437
parents = self._graph.pop(next_node_name)
575
parents = graph_pop(next_node_name)
439
577
# if the next node is not in the source graph it has
440
578
# already been popped from it and placed into the
441
579
# current search stack (but not completed or we would
442
580
# have hit the continue 4 lines up.
443
581
# this indicates a cycle.
444
raise errors.GraphCycleError(self._node_name_stack)
582
if next_node_name in self._original_graph:
583
raise errors.GraphCycleError(node_name_stack)
585
# This is just a ghost parent, ignore it
445
587
next_merge_depth = 0
446
if self._left_subtree_pushed_stack[-1]:
447
589
# a new child branch from name_stack[-1]
448
592
next_merge_depth = 1
451
self._left_subtree_pushed_stack[-1] = True
452
593
next_merge_depth = (
453
self._node_merge_depth_stack[-1] + next_merge_depth)
594
node_merge_depth_stack[-1] + next_merge_depth)
456
597
next_merge_depth,
458
# and do not continue processing parents until this 'call'
599
# and do not continue processing parents until this 'call'
461
603
# We have scheduled the graph. Now deliver the ordered output:
462
604
sequence_number = 0
463
while self._scheduled_nodes:
464
node_name, merge_depth, revno = self._scheduled_nodes.pop()
465
if node_name == self._stop_revision:
605
stop_revision = self._stop_revision
606
generate_revno = self._generate_revno
607
original_graph = self._original_graph
609
while scheduled_nodes:
610
node_name, merge_depth, revno = scheduled_nodes.pop()
611
if node_name == stop_revision:
467
if not len(self._scheduled_nodes):
613
if not len(scheduled_nodes):
468
614
# last revision is the end of a merge
469
615
end_of_merge = True
470
elif self._scheduled_nodes[-1][1] < merge_depth:
616
elif scheduled_nodes[-1][1] < merge_depth:
471
617
# the next node is to our left
472
618
end_of_merge = True
473
elif (self._scheduled_nodes[-1][1] == merge_depth and
474
(self._scheduled_nodes[-1][0] not in
475
self._original_graph[node_name])):
619
elif (scheduled_nodes[-1][1] == merge_depth and
620
(scheduled_nodes[-1][0] not in
621
original_graph[node_name])):
476
622
# the next node was part of a multiple-merge.
477
623
end_of_merge = True
479
625
end_of_merge = False
480
if self._generate_revno:
481
627
yield (sequence_number, node_name, merge_depth, revno, end_of_merge)
483
629
yield (sequence_number, node_name, merge_depth, end_of_merge)
515
666
# pop off the local variables
516
667
node_name = self._node_name_stack.pop()
517
668
merge_depth = self._node_merge_depth_stack.pop()
518
sequence = self._assigned_sequence_stack.pop()
669
first_child = self._first_child_stack.pop()
519
670
# remove this node from the pending lists:
520
671
self._left_subtree_pushed_stack.pop()
521
672
self._pending_parents_stack.pop()
523
674
parents = self._original_graph[node_name]
525
677
# node has parents, assign from the left most parent.
526
parent_revno = self._revnos[parents[0]]
679
parent_revno = self._revnos[parents[0]][0]
681
# left-hand parent is a ghost, treat it as not existing
683
if parent_revno is not None:
528
685
# not the first child, make a new branch
529
revno = parent_revno[0] + (sequence, 1)
686
base_revno = parent_revno[0]
687
branch_count = self._revno_to_branch_count.get(base_revno, 0)
689
self._revno_to_branch_count[base_revno] = branch_count
690
revno = (parent_revno[0], branch_count, 1)
691
# revno = (parent_revno[0], branch_count, parent_revno[-1]+1)
531
# increment the sequence number within the branch
532
revno = parent_revno[0][:-1] + (parent_revno[0][-1] + 1,)
693
# as the first child, we just increase the final revision
695
revno = parent_revno[:-1] + (parent_revno[-1] + 1,)
534
697
# no parents, use the root sequence
536
# make a parallel import revision number
537
revno = (0, sequence, 1)
698
root_count = self._revno_to_branch_count.get(0, 0)
699
root_count = self._revno_to_branch_count.get(0, -1)
702
revno = (0, root_count, 1)
705
self._revno_to_branch_count[0] = root_count
541
707
# store the revno for this node for future reference
542
708
self._revnos[node_name][0] = revno