2020-07-27 10:48:54 +08:00
|
|
|
#ifndef BASE_RUN_SET_H
|
|
|
|
#define BASE_RUN_SET_H
|
2013-07-08 21:44:09 +01:00
|
|
|
|
2020-07-27 10:48:54 +08:00
|
|
|
#include "base/run.h"
|
2013-07-08 21:44:09 +01:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <set>
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
|
|
|
namespace base {
|
|
|
|
template <typename T>
|
|
|
|
class run_set {
|
|
|
|
public:
|
2013-07-09 10:36:30 +01:00
|
|
|
void clear() {
|
|
|
|
runs_.clear();
|
|
|
|
}
|
|
|
|
|
2013-07-08 21:44:09 +01:00
|
|
|
void add(T const &b) {
|
|
|
|
add(run<T>(b, b + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
void add(T const &b, T const &e) {
|
|
|
|
add(run<T>(b, e));
|
|
|
|
}
|
|
|
|
|
|
|
|
void add(run<T> const &r_) {
|
|
|
|
run<T> r(r_);
|
|
|
|
|
|
|
|
if (runs_.size()) {
|
|
|
|
// Skip all blocks that end before r
|
2013-07-09 13:55:21 +01:00
|
|
|
const_iterator it = runs_.lower_bound(r);
|
|
|
|
if (it != runs_.begin())
|
|
|
|
--it;
|
|
|
|
|
2013-07-09 10:36:30 +01:00
|
|
|
while (it != runs_.end() && it->end_ < r.begin_)
|
2013-07-08 21:44:09 +01:00
|
|
|
++it;
|
|
|
|
|
|
|
|
// work out which runs overlap
|
|
|
|
if (it != runs_.end()) {
|
|
|
|
r.begin_ = min_maybe(it->begin_, r.begin_);
|
|
|
|
const_iterator first = it;
|
2013-07-09 10:36:30 +01:00
|
|
|
while (it != runs_.end() && it->begin_ <= r.end_) {
|
2013-07-08 21:44:09 +01:00
|
|
|
r.end_ = max_maybe(it->end_, r.end_);
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove overlapping runs
|
|
|
|
runs_.erase(first, it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
runs_.insert(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
void merge(run_set<T> const &rhs) {
|
|
|
|
for (const_iterator it = rhs.begin(); it != rhs.end(); ++it)
|
|
|
|
add(*it);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool member(T const &v) const {
|
|
|
|
if (!runs_.size())
|
|
|
|
return false;
|
|
|
|
|
2013-08-08 10:49:59 +01:00
|
|
|
typename rset::const_iterator it = runs_.lower_bound(run<T>(v));
|
2013-07-08 21:44:09 +01:00
|
|
|
|
2018-09-05 18:21:09 +08:00
|
|
|
if (it != runs_.end() && it->begin_ == v)
|
2013-07-08 21:44:09 +01:00
|
|
|
return true;
|
|
|
|
|
2018-09-05 18:21:09 +08:00
|
|
|
if (it != runs_.begin()) {
|
|
|
|
it--;
|
2013-07-08 21:44:09 +01:00
|
|
|
return it->contains(v);
|
2018-09-05 18:21:09 +08:00
|
|
|
}
|
2013-07-08 21:44:09 +01:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct compare_begin {
|
|
|
|
bool operator ()(run<T> const &r1, run<T> const &r2) const {
|
|
|
|
return r1.begin_ < r2.begin_;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef std::set<run<T>, compare_begin> rset;
|
|
|
|
typedef typename rset::const_iterator const_iterator;
|
|
|
|
|
|
|
|
const_iterator begin() const {
|
|
|
|
return runs_.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
const_iterator end() const {
|
|
|
|
return runs_.end();
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:08:00 +08:00
|
|
|
const_iterator lower_bound(T const &b) const {
|
|
|
|
return runs_.lower_bound(run<T>(b, b + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
const_iterator upper_bound(T const &b) const {
|
|
|
|
return runs_.upper_bound(run<T>(b, b + 1));
|
|
|
|
}
|
|
|
|
|
2013-07-08 23:21:48 +01:00
|
|
|
void negate() {
|
|
|
|
rset replacement;
|
|
|
|
|
|
|
|
if (runs_.begin() == runs_.end())
|
|
|
|
replacement.insert(run<T>());
|
|
|
|
else {
|
2013-08-08 10:49:59 +01:00
|
|
|
typename rset::const_iterator b = runs_.begin();
|
2014-06-16 13:41:45 +01:00
|
|
|
|
2014-09-08 11:07:46 +01:00
|
|
|
// Some versions of gcc give a spurious
|
|
|
|
// warning here. So we initialize it to
|
|
|
|
// get round it.
|
|
|
|
maybe last(0);
|
|
|
|
last = b->end_;
|
2013-07-08 23:21:48 +01:00
|
|
|
|
|
|
|
if (b->begin_)
|
|
|
|
replacement.insert(run<T>(maybe(), *(b->begin_)));
|
|
|
|
|
|
|
|
++b;
|
|
|
|
while (b != runs_.end()) {
|
|
|
|
replacement.insert(run<T>(last, b->begin_));
|
|
|
|
last = b->end_;
|
|
|
|
++b;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (last)
|
|
|
|
replacement.insert(run<T>(last, maybe()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
runs_ = replacement;
|
|
|
|
}
|
|
|
|
|
2013-07-08 21:44:09 +01:00
|
|
|
private:
|
|
|
|
typedef typename run<T>::maybe maybe;
|
|
|
|
|
|
|
|
static maybe min_maybe(maybe const &m1, maybe const &m2) {
|
|
|
|
if (!m1 || !m2)
|
|
|
|
return maybe();
|
|
|
|
|
|
|
|
return maybe(std::min<T>(*m1, *m2));
|
|
|
|
}
|
|
|
|
|
|
|
|
static maybe max_maybe(maybe const &m1, maybe const &m2) {
|
|
|
|
if (!m1 || !m2)
|
|
|
|
return maybe();
|
|
|
|
|
|
|
|
return maybe(std::max<T>(*m1, *m2));
|
|
|
|
}
|
|
|
|
|
2013-07-08 23:21:48 +01:00
|
|
|
rset runs_;
|
2013-07-08 21:44:09 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
|
|
|
#endif
|