Skip to content

Commit e7ed7b6

Browse files
committed
Add IntervalMap::iterator::set{Start,Stop,Value} methods that allow limited
editing of the current interval. These methods may cause coalescing, there are corresponding set*Unchecked methods for editing without coalescing. The non-coalescing methods are useful for applying monotonic transforms to all keys or values in a map without accidentally coalescing transformed and untransformed intervals. llvm-svn: 120829
1 parent 0e05e24 commit e7ed7b6

File tree

3 files changed

+294
-31
lines changed

3 files changed

+294
-31
lines changed

llvm/include/llvm/ADT/IntervalMap.h

+172-21
Original file line numberDiff line numberDiff line change
@@ -904,10 +904,10 @@ class Path {
904904
return true;
905905
}
906906

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.
909909
/// @param Level Node to examine.
910-
bool atLastBranch(unsigned Level) const {
910+
bool atLastEntry(unsigned Level) const {
911911
return path[Level].offset == path[Level].size - 1;
912912
}
913913

@@ -1365,37 +1365,44 @@ class IntervalMap<KeyT, ValT, N, Traits>::const_iterator :
13651365
void treeFind(KeyT x);
13661366
void treeAdvanceTo(KeyT x);
13671367

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 {
13771370
assert(valid() && "Cannot access invalid iterator");
13781371
return branched() ? path.leaf<Leaf>().start(path.leafOffset()) :
13791372
path.leaf<RootLeaf>().start(path.leafOffset());
13801373
}
13811374

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 {
13841377
assert(valid() && "Cannot access invalid iterator");
13851378
return branched() ? path.leaf<Leaf>().stop(path.leafOffset()) :
13861379
path.leaf<RootLeaf>().stop(path.leafOffset());
13871380
}
13881381

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 {
13911384
assert(valid() && "Cannot access invalid iterator");
13921385
return branched() ? path.leaf<Leaf>().value(path.leafOffset()) :
13931386
path.leaf<RootLeaf>().value(path.leafOffset());
13941387
}
13951388

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(); }
13991406

14001407
bool operator==(const const_iterator &RHS) const {
14011408
assert(map == RHS.map && "Cannot compare iterators from different maps");
@@ -1554,10 +1561,50 @@ class IntervalMap<KeyT, ValT, N, Traits>::iterator : public const_iterator {
15541561
void treeInsert(KeyT a, KeyT b, ValT y);
15551562
void eraseNode(unsigned Level);
15561563
void treeErase(bool UpdateRoot = true);
1564+
bool canCoalesceLeft(KeyT Start, ValT x);
1565+
bool canCoalesceRight(KeyT Stop, ValT x);
1566+
15571567
public:
15581568
/// iterator - Create null iterator.
15591569
iterator() {}
15601570

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+
15611608
/// insert - Insert mapping [a;b] -> y before the current position.
15621609
void insert(KeyT a, KeyT b, ValT y);
15631610

@@ -1588,6 +1635,62 @@ class IntervalMap<KeyT, ValT, N, Traits>::iterator : public const_iterator {
15881635

15891636
};
15901637

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+
15911694
/// setNodeStop - Update the stop key of the current node at level and above.
15921695
template <typename KeyT, typename ValT, unsigned N, typename Traits>
15931696
void IntervalMap<KeyT, ValT, N, Traits>::
@@ -1599,13 +1702,61 @@ iterator::setNodeStop(unsigned Level, KeyT Stop) {
15991702
// Update nodes pointing to the current node.
16001703
while (--Level) {
16011704
P.node<Branch>(Level).stop(P.offset(Level)) = Stop;
1602-
if (!P.atLastBranch(Level))
1705+
if (!P.atLastEntry(Level))
16031706
return;
16041707
}
16051708
// Update root separately since it has a different layout.
16061709
P.node<RootBranch>(Level).stop(P.offset(Level)) = Stop;
16071710
}
16081711

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+
16091760
/// insertNode - insert a node before the current path at level.
16101761
/// Leave the current path pointing at the new node.
16111762
/// @param Level path index of the node to be inserted.
@@ -1650,7 +1801,7 @@ iterator::insertNode(unsigned Level, IntervalMapImpl::NodeRef Node, KeyT Stop) {
16501801
}
16511802
P.node<Branch>(Level).insert(P.offset(Level), P.size(Level), Node, Stop);
16521803
P.setSize(Level, P.size(Level) + 1);
1653-
if (P.atLastBranch(Level))
1804+
if (P.atLastEntry(Level))
16541805
setNodeStop(Level, Stop);
16551806
P.reset(Level + 1);
16561807
return SplitRoot;

llvm/lib/Support/IntervalMap.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -79,11 +79,11 @@ NodeRef Path::getRightSibling(unsigned Level) const {
7979

8080
// Go up the tree until we can go right.
8181
unsigned l = Level - 1;
82-
while (l && atLastBranch(l))
82+
while (l && atLastEntry(l))
8383
--l;
8484

8585
// We can't go right.
86-
if (atLastBranch(l))
86+
if (atLastEntry(l))
8787
return NodeRef();
8888

8989
// NR is the subtree containing our right sibling.
@@ -100,7 +100,7 @@ void Path::moveRight(unsigned Level) {
100100

101101
// Go up the tree until we can go right.
102102
unsigned l = Level - 1;
103-
while (l && atLastBranch(l))
103+
while (l && atLastEntry(l))
104104
--l;
105105

106106
// NR is the subtree containing our right sibling. If we hit end(), we have

0 commit comments

Comments
 (0)