xref: /aosp_15_r20/external/libchrome/base/containers/unique_ptr_adapters.h (revision 635a864187cb8b6c713ff48b7e790a6b21769273)
1*635a8641SAndroid Build Coastguard Worker // Copyright 2017 The Chromium Authors. All rights reserved.
2*635a8641SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*635a8641SAndroid Build Coastguard Worker // found in the LICENSE file.
4*635a8641SAndroid Build Coastguard Worker 
5*635a8641SAndroid Build Coastguard Worker #ifndef BASE_CONTAINERS_UNIQUE_PTR_ADAPTERS_H_
6*635a8641SAndroid Build Coastguard Worker #define BASE_CONTAINERS_UNIQUE_PTR_ADAPTERS_H_
7*635a8641SAndroid Build Coastguard Worker 
8*635a8641SAndroid Build Coastguard Worker #include <memory>
9*635a8641SAndroid Build Coastguard Worker 
10*635a8641SAndroid Build Coastguard Worker namespace base {
11*635a8641SAndroid Build Coastguard Worker 
12*635a8641SAndroid Build Coastguard Worker // This transparent comparator allows to lookup by raw pointer in
13*635a8641SAndroid Build Coastguard Worker // a container of unique pointers. This functionality is based on C++14
14*635a8641SAndroid Build Coastguard Worker // extensions to std::set/std::map interface, and can also be used
15*635a8641SAndroid Build Coastguard Worker // with base::flat_set/base::flat_map.
16*635a8641SAndroid Build Coastguard Worker //
17*635a8641SAndroid Build Coastguard Worker // Example usage:
18*635a8641SAndroid Build Coastguard Worker //   Foo* foo = ...
19*635a8641SAndroid Build Coastguard Worker //   std::set<std::unique_ptr<Foo>, base::UniquePtrComparator> set;
20*635a8641SAndroid Build Coastguard Worker //   set.insert(std::unique_ptr<Foo>(foo));
21*635a8641SAndroid Build Coastguard Worker //   ...
22*635a8641SAndroid Build Coastguard Worker //   auto it = set.find(foo);
23*635a8641SAndroid Build Coastguard Worker //   EXPECT_EQ(foo, it->get());
24*635a8641SAndroid Build Coastguard Worker //
25*635a8641SAndroid Build Coastguard Worker // You can find more information about transparent comparisons here:
26*635a8641SAndroid Build Coastguard Worker // http://en.cppreference.com/w/cpp/utility/functional/less_void
27*635a8641SAndroid Build Coastguard Worker struct UniquePtrComparator {
28*635a8641SAndroid Build Coastguard Worker   using is_transparent = int;
29*635a8641SAndroid Build Coastguard Worker 
30*635a8641SAndroid Build Coastguard Worker   template <typename T>
operatorUniquePtrComparator31*635a8641SAndroid Build Coastguard Worker   bool operator()(const std::unique_ptr<T>& lhs,
32*635a8641SAndroid Build Coastguard Worker                   const std::unique_ptr<T>& rhs) const {
33*635a8641SAndroid Build Coastguard Worker     return lhs < rhs;
34*635a8641SAndroid Build Coastguard Worker   }
35*635a8641SAndroid Build Coastguard Worker 
36*635a8641SAndroid Build Coastguard Worker   template <typename T>
operatorUniquePtrComparator37*635a8641SAndroid Build Coastguard Worker   bool operator()(const T* lhs, const std::unique_ptr<T>& rhs) const {
38*635a8641SAndroid Build Coastguard Worker     return lhs < rhs.get();
39*635a8641SAndroid Build Coastguard Worker   }
40*635a8641SAndroid Build Coastguard Worker 
41*635a8641SAndroid Build Coastguard Worker   template <typename T>
operatorUniquePtrComparator42*635a8641SAndroid Build Coastguard Worker   bool operator()(const std::unique_ptr<T>& lhs, const T* rhs) const {
43*635a8641SAndroid Build Coastguard Worker     return lhs.get() < rhs;
44*635a8641SAndroid Build Coastguard Worker   }
45*635a8641SAndroid Build Coastguard Worker };
46*635a8641SAndroid Build Coastguard Worker 
47*635a8641SAndroid Build Coastguard Worker // UniquePtrMatcher is useful for finding an element in a container of
48*635a8641SAndroid Build Coastguard Worker // unique_ptrs when you have the raw pointer.
49*635a8641SAndroid Build Coastguard Worker //
50*635a8641SAndroid Build Coastguard Worker // Example usage:
51*635a8641SAndroid Build Coastguard Worker //   std::vector<std::unique_ptr<Foo>> vector;
52*635a8641SAndroid Build Coastguard Worker //   Foo* element = ...
53*635a8641SAndroid Build Coastguard Worker //   auto iter = std::find_if(vector.begin(), vector.end(),
54*635a8641SAndroid Build Coastguard Worker //                            MatchesUniquePtr(element));
55*635a8641SAndroid Build Coastguard Worker //
56*635a8641SAndroid Build Coastguard Worker // Example of erasing from container:
57*635a8641SAndroid Build Coastguard Worker //   EraseIf(v, MatchesUniquePtr(element));
58*635a8641SAndroid Build Coastguard Worker //
59*635a8641SAndroid Build Coastguard Worker template <class T, class Deleter = std::default_delete<T>>
60*635a8641SAndroid Build Coastguard Worker struct UniquePtrMatcher {
UniquePtrMatcherUniquePtrMatcher61*635a8641SAndroid Build Coastguard Worker   explicit UniquePtrMatcher(T* t) : t_(t) {}
62*635a8641SAndroid Build Coastguard Worker 
operatorUniquePtrMatcher63*635a8641SAndroid Build Coastguard Worker   bool operator()(const std::unique_ptr<T, Deleter>& o) {
64*635a8641SAndroid Build Coastguard Worker     return o.get() == t_;
65*635a8641SAndroid Build Coastguard Worker   }
66*635a8641SAndroid Build Coastguard Worker 
67*635a8641SAndroid Build Coastguard Worker  private:
68*635a8641SAndroid Build Coastguard Worker   T* const t_;
69*635a8641SAndroid Build Coastguard Worker };
70*635a8641SAndroid Build Coastguard Worker 
71*635a8641SAndroid Build Coastguard Worker template <class T, class Deleter = std::default_delete<T>>
MatchesUniquePtr(T * t)72*635a8641SAndroid Build Coastguard Worker UniquePtrMatcher<T, Deleter> MatchesUniquePtr(T* t) {
73*635a8641SAndroid Build Coastguard Worker   return UniquePtrMatcher<T, Deleter>(t);
74*635a8641SAndroid Build Coastguard Worker }
75*635a8641SAndroid Build Coastguard Worker 
76*635a8641SAndroid Build Coastguard Worker }  // namespace base
77*635a8641SAndroid Build Coastguard Worker 
78*635a8641SAndroid Build Coastguard Worker #endif  // BASE_CONTAINERS_UNIQUE_PTR_ADAPTERS_H_
79