xref: /aosp_15_r20/external/pdfium/testing/range_set.cpp (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2017 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker 
5*3ac0a46fSAndroid Build Coastguard Worker #include "testing/range_set.h"
6*3ac0a46fSAndroid Build Coastguard Worker 
7*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
8*3ac0a46fSAndroid Build Coastguard Worker 
9*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_system.h"
10*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check.h"
11*3ac0a46fSAndroid Build Coastguard Worker 
12*3ac0a46fSAndroid Build Coastguard Worker RangeSet::RangeSet() = default;
13*3ac0a46fSAndroid Build Coastguard Worker 
14*3ac0a46fSAndroid Build Coastguard Worker RangeSet::~RangeSet() = default;
15*3ac0a46fSAndroid Build Coastguard Worker 
Contains(const Range & range) const16*3ac0a46fSAndroid Build Coastguard Worker bool RangeSet::Contains(const Range& range) const {
17*3ac0a46fSAndroid Build Coastguard Worker   if (IsEmptyRange(range))
18*3ac0a46fSAndroid Build Coastguard Worker     return false;
19*3ac0a46fSAndroid Build Coastguard Worker 
20*3ac0a46fSAndroid Build Coastguard Worker   const Range fixed_range = FixDirection(range);
21*3ac0a46fSAndroid Build Coastguard Worker   auto it = ranges().upper_bound(fixed_range);
22*3ac0a46fSAndroid Build Coastguard Worker 
23*3ac0a46fSAndroid Build Coastguard Worker   if (it == ranges().begin())
24*3ac0a46fSAndroid Build Coastguard Worker     return false;  // No ranges includes range.first.
25*3ac0a46fSAndroid Build Coastguard Worker 
26*3ac0a46fSAndroid Build Coastguard Worker   --it;  // Now it starts equal or before range.first.
27*3ac0a46fSAndroid Build Coastguard Worker   return it->second >= fixed_range.second;
28*3ac0a46fSAndroid Build Coastguard Worker }
29*3ac0a46fSAndroid Build Coastguard Worker 
Union(const Range & range)30*3ac0a46fSAndroid Build Coastguard Worker void RangeSet::Union(const Range& range) {
31*3ac0a46fSAndroid Build Coastguard Worker   if (IsEmptyRange(range))
32*3ac0a46fSAndroid Build Coastguard Worker     return;
33*3ac0a46fSAndroid Build Coastguard Worker 
34*3ac0a46fSAndroid Build Coastguard Worker   Range fixed_range = FixDirection(range);
35*3ac0a46fSAndroid Build Coastguard Worker   if (IsEmpty()) {
36*3ac0a46fSAndroid Build Coastguard Worker     ranges_.insert(fixed_range);
37*3ac0a46fSAndroid Build Coastguard Worker     return;
38*3ac0a46fSAndroid Build Coastguard Worker   }
39*3ac0a46fSAndroid Build Coastguard Worker 
40*3ac0a46fSAndroid Build Coastguard Worker   auto start = ranges_.upper_bound(fixed_range);
41*3ac0a46fSAndroid Build Coastguard Worker   if (start != ranges_.begin())
42*3ac0a46fSAndroid Build Coastguard Worker     --start;  // start now points to the key equal or lower than offset.
43*3ac0a46fSAndroid Build Coastguard Worker 
44*3ac0a46fSAndroid Build Coastguard Worker   if (start->second < fixed_range.first)
45*3ac0a46fSAndroid Build Coastguard Worker     ++start;  // start element is entirely before current range, skip it.
46*3ac0a46fSAndroid Build Coastguard Worker 
47*3ac0a46fSAndroid Build Coastguard Worker   auto end = ranges_.upper_bound(Range(fixed_range.second, fixed_range.second));
48*3ac0a46fSAndroid Build Coastguard Worker 
49*3ac0a46fSAndroid Build Coastguard Worker   if (start == end) {  // No ranges to merge.
50*3ac0a46fSAndroid Build Coastguard Worker     ranges_.insert(fixed_range);
51*3ac0a46fSAndroid Build Coastguard Worker     return;
52*3ac0a46fSAndroid Build Coastguard Worker   }
53*3ac0a46fSAndroid Build Coastguard Worker 
54*3ac0a46fSAndroid Build Coastguard Worker   --end;
55*3ac0a46fSAndroid Build Coastguard Worker 
56*3ac0a46fSAndroid Build Coastguard Worker   const size_t new_start = std::min(start->first, fixed_range.first);
57*3ac0a46fSAndroid Build Coastguard Worker   const size_t new_end = std::max(end->second, fixed_range.second);
58*3ac0a46fSAndroid Build Coastguard Worker   ranges_.erase(start, ++end);
59*3ac0a46fSAndroid Build Coastguard Worker   ranges_.insert(Range(new_start, new_end));
60*3ac0a46fSAndroid Build Coastguard Worker }
61*3ac0a46fSAndroid Build Coastguard Worker 
Union(const RangeSet & range_set)62*3ac0a46fSAndroid Build Coastguard Worker void RangeSet::Union(const RangeSet& range_set) {
63*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(&range_set != this);
64*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& it : range_set.ranges())
65*3ac0a46fSAndroid Build Coastguard Worker     Union(it);
66*3ac0a46fSAndroid Build Coastguard Worker }
67*3ac0a46fSAndroid Build Coastguard Worker 
FixDirection(const Range & range) const68*3ac0a46fSAndroid Build Coastguard Worker RangeSet::Range RangeSet::FixDirection(const Range& range) const {
69*3ac0a46fSAndroid Build Coastguard Worker   return range.first <= range.second ? range
70*3ac0a46fSAndroid Build Coastguard Worker                                      : Range(range.second + 1, range.first + 1);
71*3ac0a46fSAndroid Build Coastguard Worker }
72*3ac0a46fSAndroid Build Coastguard Worker 
IsEmptyRange(const Range & range) const73*3ac0a46fSAndroid Build Coastguard Worker bool RangeSet::IsEmptyRange(const Range& range) const {
74*3ac0a46fSAndroid Build Coastguard Worker   return range.first == range.second;
75*3ac0a46fSAndroid Build Coastguard Worker }
76