1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *  * Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  *  * Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in
12  *    the documentation and/or other materials provided with the
13  *    distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  */
28 
29 #pragma once
30 
31 #include <android-base/macros.h>
32 
33 template<typename T>
34 struct LinkedListEntry {
35   LinkedListEntry<T>* next;
36   T* element;
37 };
38 
39 // ForwardInputIterator
40 template<typename T>
41 class LinkedListIterator {
42  public:
LinkedListIterator()43   LinkedListIterator() : entry_(nullptr) {}
LinkedListIterator(const LinkedListIterator<T> & that)44   LinkedListIterator(const LinkedListIterator<T>& that) : entry_(that.entry_) {}
LinkedListIterator(LinkedListEntry<T> * entry)45   explicit LinkedListIterator(LinkedListEntry<T>* entry) : entry_(entry) {}
46 
47   LinkedListIterator<T>& operator=(const LinkedListIterator<T>& that) {
48     entry_ = that.entry_;
49     return *this;
50   }
51 
52   LinkedListIterator<T>& operator++() {
53     entry_ = entry_->next;
54     return *this;
55   }
56 
57   T* const operator*() {
58     return entry_->element;
59   }
60 
61   bool operator==(const LinkedListIterator<T>& that) const {
62     return entry_ == that.entry_;
63   }
64 
65   bool operator!=(const LinkedListIterator<T>& that) const {
66     return entry_ != that.entry_;
67   }
68 
69  private:
70   LinkedListEntry<T> *entry_;
71 };
72 
73 /*
74  * Represents linked list of objects of type T
75  */
76 template<typename T, typename Allocator>
77 class LinkedList {
78  public:
79   typedef LinkedListIterator<T> iterator;
80   typedef T* value_type;
81 
LinkedList()82   LinkedList() : head_(nullptr), tail_(nullptr) {}
~LinkedList()83   ~LinkedList() {
84     clear();
85   }
86 
LinkedList(LinkedList && that)87   LinkedList(LinkedList&& that) noexcept {
88     this->head_ = that.head_;
89     this->tail_ = that.tail_;
90     that.head_ = that.tail_ = nullptr;
91   }
92 
push_front(T * const element)93   void push_front(T* const element) {
94     LinkedListEntry<T>* new_entry = Allocator::alloc();
95     new_entry->next = head_;
96     new_entry->element = element;
97     head_ = new_entry;
98     if (tail_ == nullptr) {
99       tail_ = new_entry;
100     }
101   }
102 
push_back(T * const element)103   void push_back(T* const element) {
104     LinkedListEntry<T>* new_entry = Allocator::alloc();
105     new_entry->next = nullptr;
106     new_entry->element = element;
107     if (tail_ == nullptr) {
108       tail_ = head_ = new_entry;
109     } else {
110       tail_->next = new_entry;
111       tail_ = new_entry;
112     }
113   }
114 
pop_front()115   T* pop_front() {
116     if (head_ == nullptr) {
117       return nullptr;
118     }
119 
120     LinkedListEntry<T>* entry = head_;
121     T* element = entry->element;
122     head_ = entry->next;
123     Allocator::free(entry);
124 
125     if (head_ == nullptr) {
126       tail_ = nullptr;
127     }
128 
129     return element;
130   }
131 
front()132   T* front() const {
133     if (head_ == nullptr) {
134       return nullptr;
135     }
136 
137     return head_->element;
138   }
139 
clear()140   void clear() {
141     while (head_ != nullptr) {
142       LinkedListEntry<T>* p = head_;
143       head_ = head_->next;
144       Allocator::free(p);
145     }
146 
147     tail_ = nullptr;
148   }
149 
empty()150   bool empty() {
151     return (head_ == nullptr);
152   }
153 
154   template<typename F>
for_each(F action)155   void for_each(F action) const {
156     visit([&] (T* si) {
157       action(si);
158       return true;
159     });
160   }
161 
162   template<typename F>
visit(F action)163   bool visit(F action) const {
164     for (LinkedListEntry<T>* e = head_; e != nullptr; e = e->next) {
165       if (!action(e->element)) {
166         return false;
167       }
168     }
169     return true;
170   }
171 
172   template<typename F>
remove_if(F predicate)173   void remove_if(F predicate) {
174     for (LinkedListEntry<T>* e = head_, *p = nullptr; e != nullptr;) {
175       if (predicate(e->element)) {
176         LinkedListEntry<T>* next = e->next;
177         if (p == nullptr) {
178           head_ = next;
179         } else {
180           p->next = next;
181         }
182 
183         if (tail_ == e) {
184           tail_ = p;
185         }
186 
187         Allocator::free(e);
188 
189         e = next;
190       } else {
191         p = e;
192         e = e->next;
193       }
194     }
195   }
196 
remove(T * element)197   void remove(T* element) {
198     remove_if([&](T* e) {
199       return e == element;
200     });
201   }
202 
203   template<typename F>
find_if(F predicate)204   T* find_if(F predicate) const {
205     for (LinkedListEntry<T>* e = head_; e != nullptr; e = e->next) {
206       if (predicate(e->element)) {
207         return e->element;
208       }
209     }
210 
211     return nullptr;
212   }
213 
begin()214   iterator begin() const {
215     return iterator(head_);
216   }
217 
end()218   iterator end() const {
219     return iterator(nullptr);
220   }
221 
find(T * value)222   iterator find(T* value) const {
223     for (LinkedListEntry<T>* e = head_; e != nullptr; e = e->next) {
224       if (e->element == value) {
225         return iterator(e);
226       }
227     }
228 
229     return end();
230   }
231 
copy_to_array(T * array[],size_t array_length)232   size_t copy_to_array(T* array[], size_t array_length) const {
233     size_t sz = 0;
234     for (LinkedListEntry<T>* e = head_; sz < array_length && e != nullptr; e = e->next) {
235       array[sz++] = e->element;
236     }
237 
238     return sz;
239   }
240 
contains(const T * el)241   bool contains(const T* el) const {
242     for (LinkedListEntry<T>* e = head_; e != nullptr; e = e->next) {
243       if (e->element == el) {
244         return true;
245       }
246     }
247     return false;
248   }
249 
make_list(T * const element)250   static LinkedList make_list(T* const element) {
251     LinkedList<T, Allocator> one_element_list;
252     one_element_list.push_back(element);
253     return one_element_list;
254   }
255 
size()256   size_t size() const {
257     size_t result = 0;
258     for_each([&](T*) { ++result; });
259     return result;
260   }
261 
262  private:
263   LinkedListEntry<T>* head_;
264   LinkedListEntry<T>* tail_;
265   DISALLOW_COPY_AND_ASSIGN(LinkedList);
266 };
267