@@ -904,10 +904,10 @@ class Path {
904
904
return true ;
905
905
}
906
906
907
- // / atLastBranch - Return true if the path is at the last branch of the node
908
- // / at Level.
907
+ // / atLastEntry - Return true if the path is at the last entry of the node at
908
+ // / Level.
909
909
// / @param Level Node to examine.
910
- bool atLastBranch (unsigned Level) const {
910
+ bool atLastEntry (unsigned Level) const {
911
911
return path[Level].offset == path[Level].size - 1 ;
912
912
}
913
913
@@ -1365,37 +1365,44 @@ class IntervalMap<KeyT, ValT, N, Traits>::const_iterator :
1365
1365
void treeFind (KeyT x);
1366
1366
void treeAdvanceTo (KeyT x);
1367
1367
1368
- public:
1369
- // / const_iterator - Create an iterator that isn't pointing anywhere.
1370
- const_iterator () : map(0 ) {}
1371
-
1372
- // / valid - Return true if the current position is valid, false for end().
1373
- bool valid () const { return path.valid (); }
1374
-
1375
- // / start - Return the beginning of the current interval.
1376
- const KeyT &start () const {
1368
+ // / unsafeStart - Writable access to start() for iterator.
1369
+ KeyT &unsafeStart () const {
1377
1370
assert (valid () && " Cannot access invalid iterator" );
1378
1371
return branched () ? path.leaf <Leaf>().start (path.leafOffset ()) :
1379
1372
path.leaf <RootLeaf>().start (path.leafOffset ());
1380
1373
}
1381
1374
1382
- // / stop - Return the end of the current interval .
1383
- const KeyT &stop () const {
1375
+ // / unsafeStop - Writable access to stop() for iterator .
1376
+ KeyT &unsafeStop () const {
1384
1377
assert (valid () && " Cannot access invalid iterator" );
1385
1378
return branched () ? path.leaf <Leaf>().stop (path.leafOffset ()) :
1386
1379
path.leaf <RootLeaf>().stop (path.leafOffset ());
1387
1380
}
1388
1381
1389
- // / value - Return the mapped value at the current interval .
1390
- const ValT &value () const {
1382
+ // / unsafeValue - Writable access to value() for iterator .
1383
+ ValT &unsafeValue () const {
1391
1384
assert (valid () && " Cannot access invalid iterator" );
1392
1385
return branched () ? path.leaf <Leaf>().value (path.leafOffset ()) :
1393
1386
path.leaf <RootLeaf>().value (path.leafOffset ());
1394
1387
}
1395
1388
1396
- const ValT &operator *() const {
1397
- return value ();
1398
- }
1389
+ public:
1390
+ // / const_iterator - Create an iterator that isn't pointing anywhere.
1391
+ const_iterator () : map(0 ) {}
1392
+
1393
+ // / valid - Return true if the current position is valid, false for end().
1394
+ bool valid () const { return path.valid (); }
1395
+
1396
+ // / start - Return the beginning of the current interval.
1397
+ const KeyT &start () const { return unsafeStart (); }
1398
+
1399
+ // / stop - Return the end of the current interval.
1400
+ const KeyT &stop () const { return unsafeStop (); }
1401
+
1402
+ // / value - Return the mapped value at the current interval.
1403
+ const ValT &value () const { return unsafeValue (); }
1404
+
1405
+ const ValT &operator *() const { return value (); }
1399
1406
1400
1407
bool operator ==(const const_iterator &RHS) const {
1401
1408
assert (map == RHS.map && " Cannot compare iterators from different maps" );
@@ -1554,10 +1561,50 @@ class IntervalMap<KeyT, ValT, N, Traits>::iterator : public const_iterator {
1554
1561
void treeInsert (KeyT a, KeyT b, ValT y);
1555
1562
void eraseNode (unsigned Level);
1556
1563
void treeErase (bool UpdateRoot = true );
1564
+ bool canCoalesceLeft (KeyT Start, ValT x);
1565
+ bool canCoalesceRight (KeyT Stop, ValT x);
1566
+
1557
1567
public:
1558
1568
// / iterator - Create null iterator.
1559
1569
iterator () {}
1560
1570
1571
+ // / setStart - Move the start of the current interval.
1572
+ // / This may cause coalescing with the previous interval.
1573
+ // / @param a New start key, must not overlap the previous interval.
1574
+ void setStart (KeyT a);
1575
+
1576
+ // / setStop - Move the end of the current interval.
1577
+ // / This may cause coalescing with the following interval.
1578
+ // / @param b New stop key, must not overlap the following interval.
1579
+ void setStop (KeyT b);
1580
+
1581
+ // / setValue - Change the mapped value of the current interval.
1582
+ // / This may cause coalescing with the previous and following intervals.
1583
+ // / @param x New value.
1584
+ void setValue (ValT x);
1585
+
1586
+ // / setStartUnchecked - Move the start of the current interval without
1587
+ // / checking for coalescing or overlaps.
1588
+ // / This should only be used when it is known that coalescing is not required.
1589
+ // / @param a New start key.
1590
+ void setStartUnchecked (KeyT a) { this ->unsafeStart () = a; }
1591
+
1592
+ // / setStopUnchecked - Move the end of the current interval without checking
1593
+ // / for coalescing or overlaps.
1594
+ // / This should only be used when it is known that coalescing is not required.
1595
+ // / @param b New stop key.
1596
+ void setStopUnchecked (KeyT b) {
1597
+ this ->unsafeStop () = b;
1598
+ // Update keys in branch nodes as well.
1599
+ if (this ->path .atLastEntry (this ->path .height ()))
1600
+ setNodeStop (this ->path .height (), b);
1601
+ }
1602
+
1603
+ // / setValueUnchecked - Change the mapped value of the current interval
1604
+ // / without checking for coalescing.
1605
+ // / @param x New value.
1606
+ void setValueUnchecked (ValT x) { this ->unsafeValue () = x; }
1607
+
1561
1608
// / insert - Insert mapping [a;b] -> y before the current position.
1562
1609
void insert (KeyT a, KeyT b, ValT y);
1563
1610
@@ -1588,6 +1635,62 @@ class IntervalMap<KeyT, ValT, N, Traits>::iterator : public const_iterator {
1588
1635
1589
1636
};
1590
1637
1638
+ // / canCoalesceLeft - Can the current interval coalesce to the left after
1639
+ // / changing start or value?
1640
+ // / @param Start New start of current interval.
1641
+ // / @param Value New value for current interval.
1642
+ // / @return True when updating the current interval would enable coalescing.
1643
+ template <typename KeyT, typename ValT, unsigned N, typename Traits>
1644
+ bool IntervalMap<KeyT, ValT, N, Traits>::
1645
+ iterator::canCoalesceLeft (KeyT Start, ValT Value) {
1646
+ using namespace IntervalMapImpl ;
1647
+ Path &P = this ->path ;
1648
+ if (!this ->branched ()) {
1649
+ unsigned i = P.leafOffset ();
1650
+ RootLeaf &Node = P.leaf <RootLeaf>();
1651
+ return i && Node.value (i-1 ) == Value &&
1652
+ Traits::adjacent (Node.stop (i-1 ), Start);
1653
+ }
1654
+ // Branched.
1655
+ if (unsigned i = P.leafOffset ()) {
1656
+ Leaf &Node = P.leaf <Leaf>();
1657
+ return Node.value (i-1 ) == Value && Traits::adjacent (Node.stop (i-1 ), Start);
1658
+ } else if (NodeRef NR = P.getLeftSibling (P.height ())) {
1659
+ unsigned i = NR.size () - 1 ;
1660
+ Leaf &Node = NR.get <Leaf>();
1661
+ return Node.value (i) == Value && Traits::adjacent (Node.stop (i), Start);
1662
+ }
1663
+ return false ;
1664
+ }
1665
+
1666
+ // / canCoalesceRight - Can the current interval coalesce to the right after
1667
+ // / changing stop or value?
1668
+ // / @param Stop New stop of current interval.
1669
+ // / @param Value New value for current interval.
1670
+ // / @return True when updating the current interval would enable coalescing.
1671
+ template <typename KeyT, typename ValT, unsigned N, typename Traits>
1672
+ bool IntervalMap<KeyT, ValT, N, Traits>::
1673
+ iterator::canCoalesceRight (KeyT Stop, ValT Value) {
1674
+ using namespace IntervalMapImpl ;
1675
+ Path &P = this ->path ;
1676
+ unsigned i = P.leafOffset () + 1 ;
1677
+ if (!this ->branched ()) {
1678
+ if (i >= P.leafSize ())
1679
+ return false ;
1680
+ RootLeaf &Node = P.leaf <RootLeaf>();
1681
+ return Node.value (i) == Value && Traits::adjacent (Stop, Node.start (i));
1682
+ }
1683
+ // Branched.
1684
+ if (i < P.leafSize ()) {
1685
+ Leaf &Node = P.leaf <Leaf>();
1686
+ return Node.value (i) == Value && Traits::adjacent (Stop, Node.start (i));
1687
+ } else if (NodeRef NR = P.getRightSibling (P.height ())) {
1688
+ Leaf &Node = NR.get <Leaf>();
1689
+ return Node.value (0 ) == Value && Traits::adjacent (Stop, Node.start (0 ));
1690
+ }
1691
+ return false ;
1692
+ }
1693
+
1591
1694
// / setNodeStop - Update the stop key of the current node at level and above.
1592
1695
template <typename KeyT, typename ValT, unsigned N, typename Traits>
1593
1696
void IntervalMap<KeyT, ValT, N, Traits>::
@@ -1599,13 +1702,61 @@ iterator::setNodeStop(unsigned Level, KeyT Stop) {
1599
1702
// Update nodes pointing to the current node.
1600
1703
while (--Level) {
1601
1704
P.node <Branch>(Level).stop (P.offset (Level)) = Stop;
1602
- if (!P.atLastBranch (Level))
1705
+ if (!P.atLastEntry (Level))
1603
1706
return ;
1604
1707
}
1605
1708
// Update root separately since it has a different layout.
1606
1709
P.node <RootBranch>(Level).stop (P.offset (Level)) = Stop;
1607
1710
}
1608
1711
1712
+ template <typename KeyT, typename ValT, unsigned N, typename Traits>
1713
+ void IntervalMap<KeyT, ValT, N, Traits>::
1714
+ iterator::setStart (KeyT a) {
1715
+ assert (Traits::stopLess (a, this ->stop ()) && " Cannot move start beyond stop" );
1716
+ KeyT &CurStart = this ->unsafeStart ();
1717
+ if (!Traits::startLess (a, CurStart) || !canCoalesceLeft (a, this ->value ())) {
1718
+ CurStart = a;
1719
+ return ;
1720
+ }
1721
+ // Coalesce with the interval to the left.
1722
+ --*this ;
1723
+ a = this ->start ();
1724
+ erase ();
1725
+ setStartUnchecked (a);
1726
+ }
1727
+
1728
+ template <typename KeyT, typename ValT, unsigned N, typename Traits>
1729
+ void IntervalMap<KeyT, ValT, N, Traits>::
1730
+ iterator::setStop (KeyT b) {
1731
+ assert (Traits::stopLess (this ->start (), b) && " Cannot move stop beyond start" );
1732
+ if (Traits::startLess (b, this ->stop ()) ||
1733
+ !canCoalesceRight (b, this ->value ())) {
1734
+ setStopUnchecked (b);
1735
+ return ;
1736
+ }
1737
+ // Coalesce with interval to the right.
1738
+ KeyT a = this ->start ();
1739
+ erase ();
1740
+ setStartUnchecked (a);
1741
+ }
1742
+
1743
+ template <typename KeyT, typename ValT, unsigned N, typename Traits>
1744
+ void IntervalMap<KeyT, ValT, N, Traits>::
1745
+ iterator::setValue (ValT x) {
1746
+ setValueUnchecked (x);
1747
+ if (canCoalesceRight (this ->stop (), x)) {
1748
+ KeyT a = this ->start ();
1749
+ erase ();
1750
+ setStartUnchecked (a);
1751
+ }
1752
+ if (canCoalesceLeft (this ->start (), x)) {
1753
+ --*this ;
1754
+ KeyT a = this ->start ();
1755
+ erase ();
1756
+ setStartUnchecked (a);
1757
+ }
1758
+ }
1759
+
1609
1760
// / insertNode - insert a node before the current path at level.
1610
1761
// / Leave the current path pointing at the new node.
1611
1762
// / @param Level path index of the node to be inserted.
@@ -1650,7 +1801,7 @@ iterator::insertNode(unsigned Level, IntervalMapImpl::NodeRef Node, KeyT Stop) {
1650
1801
}
1651
1802
P.node <Branch>(Level).insert (P.offset (Level), P.size (Level), Node, Stop);
1652
1803
P.setSize (Level, P.size (Level) + 1 );
1653
- if (P.atLastBranch (Level))
1804
+ if (P.atLastEntry (Level))
1654
1805
setNodeStop (Level, Stop);
1655
1806
P.reset (Level + 1 );
1656
1807
return SplitRoot;
0 commit comments