xref: /aosp_15_r20/external/cronet/base/stl_util.h (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2011 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker // Derived from google3/util/gtl/stl_util.h
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #ifndef BASE_STL_UTIL_H_
8*6777b538SAndroid Build Coastguard Worker #define BASE_STL_UTIL_H_
9*6777b538SAndroid Build Coastguard Worker 
10*6777b538SAndroid Build Coastguard Worker #include <algorithm>
11*6777b538SAndroid Build Coastguard Worker #include <forward_list>
12*6777b538SAndroid Build Coastguard Worker #include <iterator>
13*6777b538SAndroid Build Coastguard Worker #include <type_traits>
14*6777b538SAndroid Build Coastguard Worker 
15*6777b538SAndroid Build Coastguard Worker #include "base/check.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/ranges/algorithm.h"
17*6777b538SAndroid Build Coastguard Worker 
18*6777b538SAndroid Build Coastguard Worker namespace base {
19*6777b538SAndroid Build Coastguard Worker 
20*6777b538SAndroid Build Coastguard Worker namespace internal {
21*6777b538SAndroid Build Coastguard Worker 
22*6777b538SAndroid Build Coastguard Worker template <typename Iter>
23*6777b538SAndroid Build Coastguard Worker constexpr bool IsRandomAccessIter =
24*6777b538SAndroid Build Coastguard Worker     std::is_same_v<typename std::iterator_traits<Iter>::iterator_category,
25*6777b538SAndroid Build Coastguard Worker                    std::random_access_iterator_tag>;
26*6777b538SAndroid Build Coastguard Worker 
27*6777b538SAndroid Build Coastguard Worker }  // namespace internal
28*6777b538SAndroid Build Coastguard Worker 
29*6777b538SAndroid Build Coastguard Worker // Returns a const reference to the underlying container of a container adapter.
30*6777b538SAndroid Build Coastguard Worker // Works for std::priority_queue, std::queue, and std::stack.
31*6777b538SAndroid Build Coastguard Worker template <class A>
GetUnderlyingContainer(const A & adapter)32*6777b538SAndroid Build Coastguard Worker const typename A::container_type& GetUnderlyingContainer(const A& adapter) {
33*6777b538SAndroid Build Coastguard Worker   struct ExposedAdapter : A {
34*6777b538SAndroid Build Coastguard Worker     using A::c;
35*6777b538SAndroid Build Coastguard Worker   };
36*6777b538SAndroid Build Coastguard Worker   return adapter.*&ExposedAdapter::c;
37*6777b538SAndroid Build Coastguard Worker }
38*6777b538SAndroid Build Coastguard Worker 
39*6777b538SAndroid Build Coastguard Worker // Clears internal memory of an STL object.
40*6777b538SAndroid Build Coastguard Worker // STL clear()/reserve(0) does not always free internal memory allocated
41*6777b538SAndroid Build Coastguard Worker // This function uses swap/destructor to ensure the internal memory is freed.
42*6777b538SAndroid Build Coastguard Worker template<class T>
STLClearObject(T * obj)43*6777b538SAndroid Build Coastguard Worker void STLClearObject(T* obj) {
44*6777b538SAndroid Build Coastguard Worker   T tmp;
45*6777b538SAndroid Build Coastguard Worker   tmp.swap(*obj);
46*6777b538SAndroid Build Coastguard Worker   // Sometimes "T tmp" allocates objects with memory (arena implementation?).
47*6777b538SAndroid Build Coastguard Worker   // Hence using additional reserve(0) even if it doesn't always work.
48*6777b538SAndroid Build Coastguard Worker   obj->reserve(0);
49*6777b538SAndroid Build Coastguard Worker }
50*6777b538SAndroid Build Coastguard Worker 
51*6777b538SAndroid Build Coastguard Worker // Returns a new ResultType containing the difference of two sorted containers.
52*6777b538SAndroid Build Coastguard Worker template <typename ResultType, typename Arg1, typename Arg2>
STLSetDifference(const Arg1 & a1,const Arg2 & a2)53*6777b538SAndroid Build Coastguard Worker ResultType STLSetDifference(const Arg1& a1, const Arg2& a2) {
54*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a1));
55*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a2));
56*6777b538SAndroid Build Coastguard Worker   ResultType difference;
57*6777b538SAndroid Build Coastguard Worker   std::set_difference(a1.begin(), a1.end(),
58*6777b538SAndroid Build Coastguard Worker                       a2.begin(), a2.end(),
59*6777b538SAndroid Build Coastguard Worker                       std::inserter(difference, difference.end()));
60*6777b538SAndroid Build Coastguard Worker   return difference;
61*6777b538SAndroid Build Coastguard Worker }
62*6777b538SAndroid Build Coastguard Worker 
63*6777b538SAndroid Build Coastguard Worker // Returns a new ResultType containing the union of two sorted containers.
64*6777b538SAndroid Build Coastguard Worker template <typename ResultType, typename Arg1, typename Arg2>
STLSetUnion(const Arg1 & a1,const Arg2 & a2)65*6777b538SAndroid Build Coastguard Worker ResultType STLSetUnion(const Arg1& a1, const Arg2& a2) {
66*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a1));
67*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a2));
68*6777b538SAndroid Build Coastguard Worker   ResultType result;
69*6777b538SAndroid Build Coastguard Worker   std::set_union(a1.begin(), a1.end(),
70*6777b538SAndroid Build Coastguard Worker                  a2.begin(), a2.end(),
71*6777b538SAndroid Build Coastguard Worker                  std::inserter(result, result.end()));
72*6777b538SAndroid Build Coastguard Worker   return result;
73*6777b538SAndroid Build Coastguard Worker }
74*6777b538SAndroid Build Coastguard Worker 
75*6777b538SAndroid Build Coastguard Worker // Returns a new ResultType containing the intersection of two sorted
76*6777b538SAndroid Build Coastguard Worker // containers.
77*6777b538SAndroid Build Coastguard Worker template <typename ResultType, typename Arg1, typename Arg2>
STLSetIntersection(const Arg1 & a1,const Arg2 & a2)78*6777b538SAndroid Build Coastguard Worker ResultType STLSetIntersection(const Arg1& a1, const Arg2& a2) {
79*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a1));
80*6777b538SAndroid Build Coastguard Worker   DCHECK(ranges::is_sorted(a2));
81*6777b538SAndroid Build Coastguard Worker   ResultType result;
82*6777b538SAndroid Build Coastguard Worker   std::set_intersection(a1.begin(), a1.end(),
83*6777b538SAndroid Build Coastguard Worker                         a2.begin(), a2.end(),
84*6777b538SAndroid Build Coastguard Worker                         std::inserter(result, result.end()));
85*6777b538SAndroid Build Coastguard Worker   return result;
86*6777b538SAndroid Build Coastguard Worker }
87*6777b538SAndroid Build Coastguard Worker 
88*6777b538SAndroid Build Coastguard Worker // A helper class to be used as the predicate with |EraseIf| to implement
89*6777b538SAndroid Build Coastguard Worker // in-place set intersection. Helps implement the algorithm of going through
90*6777b538SAndroid Build Coastguard Worker // each container an element at a time, erasing elements from the first
91*6777b538SAndroid Build Coastguard Worker // container if they aren't in the second container. Requires each container be
92*6777b538SAndroid Build Coastguard Worker // sorted. Note that the logic below appears inverted since it is returning
93*6777b538SAndroid Build Coastguard Worker // whether an element should be erased.
94*6777b538SAndroid Build Coastguard Worker template <class Collection>
95*6777b538SAndroid Build Coastguard Worker class IsNotIn {
96*6777b538SAndroid Build Coastguard Worker  public:
IsNotIn(const Collection & collection)97*6777b538SAndroid Build Coastguard Worker   explicit IsNotIn(const Collection& collection)
98*6777b538SAndroid Build Coastguard Worker       : i_(collection.begin()), end_(collection.end()) {}
99*6777b538SAndroid Build Coastguard Worker 
operator()100*6777b538SAndroid Build Coastguard Worker   bool operator()(const typename Collection::value_type& x) {
101*6777b538SAndroid Build Coastguard Worker     while (i_ != end_ && *i_ < x)
102*6777b538SAndroid Build Coastguard Worker       ++i_;
103*6777b538SAndroid Build Coastguard Worker     if (i_ == end_)
104*6777b538SAndroid Build Coastguard Worker       return true;
105*6777b538SAndroid Build Coastguard Worker     if (*i_ == x) {
106*6777b538SAndroid Build Coastguard Worker       ++i_;
107*6777b538SAndroid Build Coastguard Worker       return false;
108*6777b538SAndroid Build Coastguard Worker     }
109*6777b538SAndroid Build Coastguard Worker     return true;
110*6777b538SAndroid Build Coastguard Worker   }
111*6777b538SAndroid Build Coastguard Worker 
112*6777b538SAndroid Build Coastguard Worker  private:
113*6777b538SAndroid Build Coastguard Worker   typename Collection::const_iterator i_;
114*6777b538SAndroid Build Coastguard Worker   const typename Collection::const_iterator end_;
115*6777b538SAndroid Build Coastguard Worker };
116*6777b538SAndroid Build Coastguard Worker 
117*6777b538SAndroid Build Coastguard Worker }  // namespace base
118*6777b538SAndroid Build Coastguard Worker 
119*6777b538SAndroid Build Coastguard Worker #endif  // BASE_STL_UTIL_H_
120