1 /*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17
18 #ifndef ANDROID_VINTF_MAP_VALUE_ITERATOR_H
19 #define ANDROID_VINTF_MAP_VALUE_ITERATOR_H
20
21 #include <iterator>
22 #include <map>
23
24 namespace android {
25 namespace vintf {
26
27 template<typename Map>
28 struct MapIterTypes {
29 using K = typename Map::key_type;
30 using V = typename Map::mapped_type;
31
32 // Iterator over all values of a Map
33 template<bool is_const>
34 struct IteratorImpl : public std::iterator <
35 std::bidirectional_iterator_tag, /* Category */
36 V,
37 ptrdiff_t, /* Distance */
38 typename std::conditional<is_const, const V *, V *>::type /* Pointer */,
39 typename std::conditional<is_const, const V &, V &>::type /* Reference */
40 >
41 {
42 using traits = std::iterator_traits<IteratorImpl>;
43 using ptr_type = typename traits::pointer;
44 using ref_type = typename traits::reference;
45 using diff_type = typename traits::difference_type;
46
47 using map_iter = typename std::conditional<is_const,
48 typename Map::const_iterator, typename Map::iterator>::type;
49
IteratorImplMapIterTypes::IteratorImpl50 IteratorImpl(map_iter i) : mIter(i) {}
51
52 inline IteratorImpl &operator++() {
53 mIter++;
54 return *this;
55 };
56 inline IteratorImpl operator++(int) {
57 IteratorImpl i = *this;
58 mIter++;
59 return i;
60 }
61 inline IteratorImpl &operator--() {
62 mIter--;
63 return *this;
64 }
65 inline IteratorImpl operator--(int) {
66 IteratorImpl i = *this;
67 mIter--;
68 return i;
69 }
70 inline ref_type operator*() const { return mIter->second; }
71 inline ptr_type operator->() const { return &(mIter->second); }
72 inline bool operator==(const IteratorImpl &rhs) const { return mIter == rhs.mIter; }
73 inline bool operator!=(const IteratorImpl &rhs) const { return mIter != rhs.mIter; }
74
75 private:
76 map_iter mIter;
77 };
78
79 using ValueIterator = IteratorImpl<false>;
80 using ConstValueIterator = IteratorImpl<true>;
81
82 template<bool is_const>
83 struct IterableImpl {
84 using map_ref = typename std::conditional<is_const, const Map &, Map &>::type;
IterableImplMapIterTypes::IterableImpl85 IterableImpl(map_ref map) : mMap(map) {}
86
beginMapIterTypes::IterableImpl87 IteratorImpl<is_const> begin() const {
88 return IteratorImpl<is_const>(mMap.begin());
89 }
90
endMapIterTypes::IterableImpl91 IteratorImpl<is_const> end() const {
92 return IteratorImpl<is_const>(mMap.end());
93 }
94
95 private:
96 map_ref mMap;
97 };
98
99 using ValueIterable = IterableImpl<false>;
100 using ConstValueIterable = IterableImpl<true>;
101 };
102
103 template<typename K, typename V>
104 using ConstMapValueIterable = typename MapIterTypes<std::map<K, V>>::ConstValueIterable;
105 template<typename K, typename V>
106 using ConstMultiMapValueIterable = typename MapIterTypes<std::multimap<K, V>>::ConstValueIterable;
107
108 template<typename K, typename V>
iterateValues(const std::map<K,V> & map)109 ConstMapValueIterable<K, V> iterateValues(const std::map<K, V> &map) {
110 return map;
111 }
112 template<typename K, typename V>
iterateValues(const std::multimap<K,V> & map)113 ConstMultiMapValueIterable<K, V> iterateValues(const std::multimap<K, V> &map) {
114 return map;
115 }
116
117 } // namespace vintf
118 } // namespace android
119
120 #endif // ANDROID_VINTF_MAP_VALUE_ITERATOR_H
121