/src/3rdparty/webkit/Source/JavaScriptCore/wtf/Deque.h

https://bitbucket.org/ultra_iter/qt-vtl · C Header · 688 lines · 549 code · 96 blank · 43 comment · 61 complexity · 2817ff20ff67b233c38c4e6513345e2b MD5 · raw file

  1. /*
  2. * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
  3. * Copyright (C) 2009 Google Inc. 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. *
  9. * 1. Redistributions of source code must retain the above copyright
  10. * notice, this list of conditions and the following disclaimer.
  11. * 2. Redistributions in binary form must reproduce the above copyright
  12. * notice, this list of conditions and the following disclaimer in the
  13. * documentation and/or other materials provided with the distribution.
  14. * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
  15. * its contributors may be used to endorse or promote products derived
  16. * from this software without specific prior written permission.
  17. *
  18. * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
  19. * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  20. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  21. * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
  22. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  23. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  24. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  25. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  27. * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. */
  29. #ifndef WTF_Deque_h
  30. #define WTF_Deque_h
  31. // FIXME: Could move what Vector and Deque share into a separate file.
  32. // Deque doesn't actually use Vector.
  33. #include "PassTraits.h"
  34. #include "Vector.h"
  35. namespace WTF {
  36. template<typename T, size_t inlineCapacity> class DequeIteratorBase;
  37. template<typename T, size_t inlineCapacity> class DequeIterator;
  38. template<typename T, size_t inlineCapacity> class DequeConstIterator;
  39. template<typename T, size_t inlineCapacity> class DequeReverseIterator;
  40. template<typename T, size_t inlineCapacity> class DequeConstReverseIterator;
  41. template<typename T, size_t inlineCapacity = 0>
  42. class Deque {
  43. WTF_MAKE_FAST_ALLOCATED;
  44. public:
  45. typedef DequeIterator<T, inlineCapacity> iterator;
  46. typedef DequeConstIterator<T, inlineCapacity> const_iterator;
  47. typedef DequeReverseIterator<T, inlineCapacity> reverse_iterator;
  48. typedef DequeConstReverseIterator<T, inlineCapacity> const_reverse_iterator;
  49. typedef PassTraits<T> Pass;
  50. typedef typename PassTraits<T>::PassType PassType;
  51. Deque();
  52. Deque(const Deque<T, inlineCapacity>&);
  53. Deque& operator=(const Deque<T, inlineCapacity>&);
  54. ~Deque();
  55. void swap(Deque<T, inlineCapacity>&);
  56. size_t size() const { return m_start <= m_end ? m_end - m_start : m_end + m_buffer.capacity() - m_start; }
  57. bool isEmpty() const { return m_start == m_end; }
  58. iterator begin() { return iterator(this, m_start); }
  59. iterator end() { return iterator(this, m_end); }
  60. const_iterator begin() const { return const_iterator(this, m_start); }
  61. const_iterator end() const { return const_iterator(this, m_end); }
  62. reverse_iterator rbegin() { return reverse_iterator(this, m_end); }
  63. reverse_iterator rend() { return reverse_iterator(this, m_start); }
  64. const_reverse_iterator rbegin() const { return const_reverse_iterator(this, m_end); }
  65. const_reverse_iterator rend() const { return const_reverse_iterator(this, m_start); }
  66. T& first() { ASSERT(m_start != m_end); return m_buffer.buffer()[m_start]; }
  67. const T& first() const { ASSERT(m_start != m_end); return m_buffer.buffer()[m_start]; }
  68. PassType takeFirst();
  69. template<typename U> void append(const U&);
  70. template<typename U> void prepend(const U&);
  71. void removeFirst();
  72. void remove(iterator&);
  73. void remove(const_iterator&);
  74. void clear();
  75. template<typename Predicate>
  76. iterator findIf(Predicate&);
  77. private:
  78. friend class DequeIteratorBase<T, inlineCapacity>;
  79. typedef VectorBuffer<T, inlineCapacity> Buffer;
  80. typedef VectorTypeOperations<T> TypeOperations;
  81. typedef DequeIteratorBase<T, inlineCapacity> IteratorBase;
  82. void remove(size_t position);
  83. void invalidateIterators();
  84. void destroyAll();
  85. void checkValidity() const;
  86. void checkIndexValidity(size_t) const;
  87. void expandCapacityIfNeeded();
  88. void expandCapacity();
  89. size_t m_start;
  90. size_t m_end;
  91. Buffer m_buffer;
  92. #ifndef NDEBUG
  93. mutable IteratorBase* m_iterators;
  94. #endif
  95. };
  96. template<typename T, size_t inlineCapacity = 0>
  97. class DequeIteratorBase {
  98. private:
  99. typedef DequeIteratorBase<T, inlineCapacity> Base;
  100. protected:
  101. DequeIteratorBase();
  102. DequeIteratorBase(const Deque<T, inlineCapacity>*, size_t);
  103. DequeIteratorBase(const Base&);
  104. Base& operator=(const Base&);
  105. ~DequeIteratorBase();
  106. void assign(const Base& other) { *this = other; }
  107. void increment();
  108. void decrement();
  109. T* before() const;
  110. T* after() const;
  111. bool isEqual(const Base&) const;
  112. private:
  113. void addToIteratorsList();
  114. void removeFromIteratorsList();
  115. void checkValidity() const;
  116. void checkValidity(const Base&) const;
  117. Deque<T, inlineCapacity>* m_deque;
  118. size_t m_index;
  119. friend class Deque<T, inlineCapacity>;
  120. #ifndef NDEBUG
  121. mutable DequeIteratorBase* m_next;
  122. mutable DequeIteratorBase* m_previous;
  123. #endif
  124. };
  125. template<typename T, size_t inlineCapacity = 0>
  126. class DequeIterator : public DequeIteratorBase<T, inlineCapacity> {
  127. private:
  128. typedef DequeIteratorBase<T, inlineCapacity> Base;
  129. typedef DequeIterator<T, inlineCapacity> Iterator;
  130. public:
  131. DequeIterator(Deque<T, inlineCapacity>* deque, size_t index) : Base(deque, index) { }
  132. DequeIterator(const Iterator& other) : Base(other) { }
  133. DequeIterator& operator=(const Iterator& other) { Base::assign(other); return *this; }
  134. T& operator*() const { return *Base::after(); }
  135. T* operator->() const { return Base::after(); }
  136. bool operator==(const Iterator& other) const { return Base::isEqual(other); }
  137. bool operator!=(const Iterator& other) const { return !Base::isEqual(other); }
  138. Iterator& operator++() { Base::increment(); return *this; }
  139. // postfix ++ intentionally omitted
  140. Iterator& operator--() { Base::decrement(); return *this; }
  141. // postfix -- intentionally omitted
  142. };
  143. template<typename T, size_t inlineCapacity = 0>
  144. class DequeConstIterator : public DequeIteratorBase<T, inlineCapacity> {
  145. private:
  146. typedef DequeIteratorBase<T, inlineCapacity> Base;
  147. typedef DequeConstIterator<T, inlineCapacity> Iterator;
  148. typedef DequeIterator<T, inlineCapacity> NonConstIterator;
  149. public:
  150. DequeConstIterator(const Deque<T, inlineCapacity>* deque, size_t index) : Base(deque, index) { }
  151. DequeConstIterator(const Iterator& other) : Base(other) { }
  152. DequeConstIterator(const NonConstIterator& other) : Base(other) { }
  153. DequeConstIterator& operator=(const Iterator& other) { Base::assign(other); return *this; }
  154. DequeConstIterator& operator=(const NonConstIterator& other) { Base::assign(other); return *this; }
  155. const T& operator*() const { return *Base::after(); }
  156. const T* operator->() const { return Base::after(); }
  157. bool operator==(const Iterator& other) const { return Base::isEqual(other); }
  158. bool operator!=(const Iterator& other) const { return !Base::isEqual(other); }
  159. Iterator& operator++() { Base::increment(); return *this; }
  160. // postfix ++ intentionally omitted
  161. Iterator& operator--() { Base::decrement(); return *this; }
  162. // postfix -- intentionally omitted
  163. };
  164. template<typename T, size_t inlineCapacity = 0>
  165. class DequeReverseIterator : public DequeIteratorBase<T, inlineCapacity> {
  166. private:
  167. typedef DequeIteratorBase<T, inlineCapacity> Base;
  168. typedef DequeReverseIterator<T, inlineCapacity> Iterator;
  169. public:
  170. DequeReverseIterator(const Deque<T, inlineCapacity>* deque, size_t index) : Base(deque, index) { }
  171. DequeReverseIterator(const Iterator& other) : Base(other) { }
  172. DequeReverseIterator& operator=(const Iterator& other) { Base::assign(other); return *this; }
  173. T& operator*() const { return *Base::before(); }
  174. T* operator->() const { return Base::before(); }
  175. bool operator==(const Iterator& other) const { return Base::isEqual(other); }
  176. bool operator!=(const Iterator& other) const { return !Base::isEqual(other); }
  177. Iterator& operator++() { Base::decrement(); return *this; }
  178. // postfix ++ intentionally omitted
  179. Iterator& operator--() { Base::increment(); return *this; }
  180. // postfix -- intentionally omitted
  181. };
  182. template<typename T, size_t inlineCapacity = 0>
  183. class DequeConstReverseIterator : public DequeIteratorBase<T, inlineCapacity> {
  184. private:
  185. typedef DequeIteratorBase<T, inlineCapacity> Base;
  186. typedef DequeConstReverseIterator<T, inlineCapacity> Iterator;
  187. typedef DequeReverseIterator<T, inlineCapacity> NonConstIterator;
  188. public:
  189. DequeConstReverseIterator(const Deque<T, inlineCapacity>* deque, size_t index) : Base(deque, index) { }
  190. DequeConstReverseIterator(const Iterator& other) : Base(other) { }
  191. DequeConstReverseIterator(const NonConstIterator& other) : Base(other) { }
  192. DequeConstReverseIterator& operator=(const Iterator& other) { Base::assign(other); return *this; }
  193. DequeConstReverseIterator& operator=(const NonConstIterator& other) { Base::assign(other); return *this; }
  194. const T& operator*() const { return *Base::before(); }
  195. const T* operator->() const { return Base::before(); }
  196. bool operator==(const Iterator& other) const { return Base::isEqual(other); }
  197. bool operator!=(const Iterator& other) const { return !Base::isEqual(other); }
  198. Iterator& operator++() { Base::decrement(); return *this; }
  199. // postfix ++ intentionally omitted
  200. Iterator& operator--() { Base::increment(); return *this; }
  201. // postfix -- intentionally omitted
  202. };
  203. #ifdef NDEBUG
  204. template<typename T, size_t inlineCapacity> inline void Deque<T, inlineCapacity>::checkValidity() const { }
  205. template<typename T, size_t inlineCapacity> inline void Deque<T, inlineCapacity>::checkIndexValidity(size_t) const { }
  206. template<typename T, size_t inlineCapacity> inline void Deque<T, inlineCapacity>::invalidateIterators() { }
  207. #else
  208. template<typename T, size_t inlineCapacity>
  209. void Deque<T, inlineCapacity>::checkValidity() const
  210. {
  211. // In this implementation a capacity of 1 would confuse append() and
  212. // other places that assume the index after capacity - 1 is 0.
  213. ASSERT(m_buffer.capacity() != 1);
  214. if (!m_buffer.capacity()) {
  215. ASSERT(!m_start);
  216. ASSERT(!m_end);
  217. } else {
  218. ASSERT(m_start < m_buffer.capacity());
  219. ASSERT(m_end < m_buffer.capacity());
  220. }
  221. }
  222. template<typename T, size_t inlineCapacity>
  223. void Deque<T, inlineCapacity>::checkIndexValidity(size_t index) const
  224. {
  225. ASSERT(index <= m_buffer.capacity());
  226. if (m_start <= m_end) {
  227. ASSERT(index >= m_start);
  228. ASSERT(index <= m_end);
  229. } else {
  230. ASSERT(index >= m_start || index <= m_end);
  231. }
  232. }
  233. template<typename T, size_t inlineCapacity>
  234. void Deque<T, inlineCapacity>::invalidateIterators()
  235. {
  236. IteratorBase* next;
  237. for (IteratorBase* p = m_iterators; p; p = next) {
  238. next = p->m_next;
  239. p->m_deque = 0;
  240. p->m_next = 0;
  241. p->m_previous = 0;
  242. }
  243. m_iterators = 0;
  244. }
  245. #endif
  246. template<typename T, size_t inlineCapacity>
  247. inline Deque<T, inlineCapacity>::Deque()
  248. : m_start(0)
  249. , m_end(0)
  250. #ifndef NDEBUG
  251. , m_iterators(0)
  252. #endif
  253. {
  254. checkValidity();
  255. }
  256. template<typename T, size_t inlineCapacity>
  257. inline Deque<T, inlineCapacity>::Deque(const Deque<T, inlineCapacity>& other)
  258. : m_start(other.m_start)
  259. , m_end(other.m_end)
  260. , m_buffer(other.m_buffer.capacity())
  261. #ifndef NDEBUG
  262. , m_iterators(0)
  263. #endif
  264. {
  265. const T* otherBuffer = other.m_buffer.buffer();
  266. if (m_start <= m_end)
  267. TypeOperations::uninitializedCopy(otherBuffer + m_start, otherBuffer + m_end, m_buffer.buffer() + m_start);
  268. else {
  269. TypeOperations::uninitializedCopy(otherBuffer, otherBuffer + m_end, m_buffer.buffer());
  270. TypeOperations::uninitializedCopy(otherBuffer + m_start, otherBuffer + m_buffer.capacity(), m_buffer.buffer() + m_start);
  271. }
  272. }
  273. template<typename T, size_t inlineCapacity>
  274. void deleteAllValues(const Deque<T, inlineCapacity>& collection)
  275. {
  276. typedef typename Deque<T, inlineCapacity>::const_iterator iterator;
  277. iterator end = collection.end();
  278. for (iterator it = collection.begin(); it != end; ++it)
  279. delete *it;
  280. }
  281. template<typename T, size_t inlineCapacity>
  282. inline Deque<T, inlineCapacity>& Deque<T, inlineCapacity>::operator=(const Deque<T, inlineCapacity>& other)
  283. {
  284. // FIXME: This is inefficient if we're using an inline buffer and T is
  285. // expensive to copy since it will copy the buffer twice instead of once.
  286. Deque<T> copy(other);
  287. swap(copy);
  288. return *this;
  289. }
  290. template<typename T, size_t inlineCapacity>
  291. inline void Deque<T, inlineCapacity>::destroyAll()
  292. {
  293. if (m_start <= m_end)
  294. TypeOperations::destruct(m_buffer.buffer() + m_start, m_buffer.buffer() + m_end);
  295. else {
  296. TypeOperations::destruct(m_buffer.buffer(), m_buffer.buffer() + m_end);
  297. TypeOperations::destruct(m_buffer.buffer() + m_start, m_buffer.buffer() + m_buffer.capacity());
  298. }
  299. }
  300. template<typename T, size_t inlineCapacity>
  301. inline Deque<T, inlineCapacity>::~Deque()
  302. {
  303. checkValidity();
  304. invalidateIterators();
  305. destroyAll();
  306. }
  307. template<typename T, size_t inlineCapacity>
  308. inline void Deque<T, inlineCapacity>::swap(Deque<T, inlineCapacity>& other)
  309. {
  310. checkValidity();
  311. other.checkValidity();
  312. invalidateIterators();
  313. std::swap(m_start, other.m_start);
  314. std::swap(m_end, other.m_end);
  315. m_buffer.swap(other.m_buffer);
  316. checkValidity();
  317. other.checkValidity();
  318. }
  319. template<typename T, size_t inlineCapacity>
  320. inline void Deque<T, inlineCapacity>::clear()
  321. {
  322. checkValidity();
  323. invalidateIterators();
  324. destroyAll();
  325. m_start = 0;
  326. m_end = 0;
  327. checkValidity();
  328. }
  329. template<typename T, size_t inlineCapacity>
  330. template<typename Predicate>
  331. inline DequeIterator<T, inlineCapacity> Deque<T, inlineCapacity>::findIf(Predicate& predicate)
  332. {
  333. iterator end_iterator = end();
  334. for (iterator it = begin(); it != end_iterator; ++it) {
  335. if (predicate(*it))
  336. return it;
  337. }
  338. return end_iterator;
  339. }
  340. template<typename T, size_t inlineCapacity>
  341. inline void Deque<T, inlineCapacity>::expandCapacityIfNeeded()
  342. {
  343. if (m_start) {
  344. if (m_end + 1 != m_start)
  345. return;
  346. } else if (m_end) {
  347. if (m_end != m_buffer.capacity() - 1)
  348. return;
  349. } else if (m_buffer.capacity())
  350. return;
  351. expandCapacity();
  352. }
  353. template<typename T, size_t inlineCapacity>
  354. void Deque<T, inlineCapacity>::expandCapacity()
  355. {
  356. checkValidity();
  357. size_t oldCapacity = m_buffer.capacity();
  358. size_t newCapacity = max(static_cast<size_t>(16), oldCapacity + oldCapacity / 4 + 1);
  359. T* oldBuffer = m_buffer.buffer();
  360. m_buffer.allocateBuffer(newCapacity);
  361. if (m_start <= m_end)
  362. TypeOperations::move(oldBuffer + m_start, oldBuffer + m_end, m_buffer.buffer() + m_start);
  363. else {
  364. TypeOperations::move(oldBuffer, oldBuffer + m_end, m_buffer.buffer());
  365. size_t newStart = newCapacity - (oldCapacity - m_start);
  366. TypeOperations::move(oldBuffer + m_start, oldBuffer + oldCapacity, m_buffer.buffer() + newStart);
  367. m_start = newStart;
  368. }
  369. m_buffer.deallocateBuffer(oldBuffer);
  370. checkValidity();
  371. }
  372. template<typename T, size_t inlineCapacity>
  373. inline typename Deque<T, inlineCapacity>::PassType Deque<T, inlineCapacity>::takeFirst()
  374. {
  375. T oldFirst = Pass::transfer(first());
  376. removeFirst();
  377. return Pass::transfer(oldFirst);
  378. }
  379. template<typename T, size_t inlineCapacity> template<typename U>
  380. inline void Deque<T, inlineCapacity>::append(const U& value)
  381. {
  382. checkValidity();
  383. expandCapacityIfNeeded();
  384. new (&m_buffer.buffer()[m_end]) T(value);
  385. if (m_end == m_buffer.capacity() - 1)
  386. m_end = 0;
  387. else
  388. ++m_end;
  389. checkValidity();
  390. }
  391. template<typename T, size_t inlineCapacity> template<typename U>
  392. inline void Deque<T, inlineCapacity>::prepend(const U& value)
  393. {
  394. checkValidity();
  395. expandCapacityIfNeeded();
  396. if (!m_start)
  397. m_start = m_buffer.capacity() - 1;
  398. else
  399. --m_start;
  400. new (&m_buffer.buffer()[m_start]) T(value);
  401. checkValidity();
  402. }
  403. template<typename T, size_t inlineCapacity>
  404. inline void Deque<T, inlineCapacity>::removeFirst()
  405. {
  406. checkValidity();
  407. invalidateIterators();
  408. ASSERT(!isEmpty());
  409. TypeOperations::destruct(&m_buffer.buffer()[m_start], &m_buffer.buffer()[m_start + 1]);
  410. if (m_start == m_buffer.capacity() - 1)
  411. m_start = 0;
  412. else
  413. ++m_start;
  414. checkValidity();
  415. }
  416. template<typename T, size_t inlineCapacity>
  417. inline void Deque<T, inlineCapacity>::remove(iterator& it)
  418. {
  419. it.checkValidity();
  420. remove(it.m_index);
  421. }
  422. template<typename T, size_t inlineCapacity>
  423. inline void Deque<T, inlineCapacity>::remove(const_iterator& it)
  424. {
  425. it.checkValidity();
  426. remove(it.m_index);
  427. }
  428. template<typename T, size_t inlineCapacity>
  429. inline void Deque<T, inlineCapacity>::remove(size_t position)
  430. {
  431. if (position == m_end)
  432. return;
  433. checkValidity();
  434. invalidateIterators();
  435. T* buffer = m_buffer.buffer();
  436. TypeOperations::destruct(&buffer[position], &buffer[position + 1]);
  437. // Find which segment of the circular buffer contained the remove element, and only move elements in that part.
  438. if (position >= m_start) {
  439. TypeOperations::moveOverlapping(buffer + m_start, buffer + position, buffer + m_start + 1);
  440. m_start = (m_start + 1) % m_buffer.capacity();
  441. } else {
  442. TypeOperations::moveOverlapping(buffer + position + 1, buffer + m_end, buffer + position);
  443. m_end = (m_end - 1 + m_buffer.capacity()) % m_buffer.capacity();
  444. }
  445. checkValidity();
  446. }
  447. #ifdef NDEBUG
  448. template<typename T, size_t inlineCapacity> inline void DequeIteratorBase<T, inlineCapacity>::checkValidity() const { }
  449. template<typename T, size_t inlineCapacity> inline void DequeIteratorBase<T, inlineCapacity>::checkValidity(const DequeIteratorBase<T, inlineCapacity>&) const { }
  450. template<typename T, size_t inlineCapacity> inline void DequeIteratorBase<T, inlineCapacity>::addToIteratorsList() { }
  451. template<typename T, size_t inlineCapacity> inline void DequeIteratorBase<T, inlineCapacity>::removeFromIteratorsList() { }
  452. #else
  453. template<typename T, size_t inlineCapacity>
  454. void DequeIteratorBase<T, inlineCapacity>::checkValidity() const
  455. {
  456. ASSERT(m_deque);
  457. m_deque->checkIndexValidity(m_index);
  458. }
  459. template<typename T, size_t inlineCapacity>
  460. void DequeIteratorBase<T, inlineCapacity>::checkValidity(const Base& other) const
  461. {
  462. checkValidity();
  463. other.checkValidity();
  464. ASSERT(m_deque == other.m_deque);
  465. }
  466. template<typename T, size_t inlineCapacity>
  467. void DequeIteratorBase<T, inlineCapacity>::addToIteratorsList()
  468. {
  469. if (!m_deque)
  470. m_next = 0;
  471. else {
  472. m_next = m_deque->m_iterators;
  473. m_deque->m_iterators = this;
  474. if (m_next)
  475. m_next->m_previous = this;
  476. }
  477. m_previous = 0;
  478. }
  479. template<typename T, size_t inlineCapacity>
  480. void DequeIteratorBase<T, inlineCapacity>::removeFromIteratorsList()
  481. {
  482. if (!m_deque) {
  483. ASSERT(!m_next);
  484. ASSERT(!m_previous);
  485. } else {
  486. if (m_next) {
  487. ASSERT(m_next->m_previous == this);
  488. m_next->m_previous = m_previous;
  489. }
  490. if (m_previous) {
  491. ASSERT(m_deque->m_iterators != this);
  492. ASSERT(m_previous->m_next == this);
  493. m_previous->m_next = m_next;
  494. } else {
  495. ASSERT(m_deque->m_iterators == this);
  496. m_deque->m_iterators = m_next;
  497. }
  498. }
  499. m_next = 0;
  500. m_previous = 0;
  501. }
  502. #endif
  503. template<typename T, size_t inlineCapacity>
  504. inline DequeIteratorBase<T, inlineCapacity>::DequeIteratorBase()
  505. : m_deque(0)
  506. {
  507. }
  508. template<typename T, size_t inlineCapacity>
  509. inline DequeIteratorBase<T, inlineCapacity>::DequeIteratorBase(const Deque<T, inlineCapacity>* deque, size_t index)
  510. : m_deque(const_cast<Deque<T, inlineCapacity>*>(deque))
  511. , m_index(index)
  512. {
  513. addToIteratorsList();
  514. checkValidity();
  515. }
  516. template<typename T, size_t inlineCapacity>
  517. inline DequeIteratorBase<T, inlineCapacity>::DequeIteratorBase(const Base& other)
  518. : m_deque(other.m_deque)
  519. , m_index(other.m_index)
  520. {
  521. addToIteratorsList();
  522. checkValidity();
  523. }
  524. template<typename T, size_t inlineCapacity>
  525. inline DequeIteratorBase<T, inlineCapacity>& DequeIteratorBase<T, inlineCapacity>::operator=(const Base& other)
  526. {
  527. checkValidity();
  528. other.checkValidity();
  529. removeFromIteratorsList();
  530. m_deque = other.m_deque;
  531. m_index = other.m_index;
  532. addToIteratorsList();
  533. checkValidity();
  534. return *this;
  535. }
  536. template<typename T, size_t inlineCapacity>
  537. inline DequeIteratorBase<T, inlineCapacity>::~DequeIteratorBase()
  538. {
  539. #ifndef NDEBUG
  540. removeFromIteratorsList();
  541. m_deque = 0;
  542. #endif
  543. }
  544. template<typename T, size_t inlineCapacity>
  545. inline bool DequeIteratorBase<T, inlineCapacity>::isEqual(const Base& other) const
  546. {
  547. checkValidity(other);
  548. return m_index == other.m_index;
  549. }
  550. template<typename T, size_t inlineCapacity>
  551. inline void DequeIteratorBase<T, inlineCapacity>::increment()
  552. {
  553. checkValidity();
  554. ASSERT(m_index != m_deque->m_end);
  555. ASSERT(m_deque->m_buffer.capacity());
  556. if (m_index == m_deque->m_buffer.capacity() - 1)
  557. m_index = 0;
  558. else
  559. ++m_index;
  560. checkValidity();
  561. }
  562. template<typename T, size_t inlineCapacity>
  563. inline void DequeIteratorBase<T, inlineCapacity>::decrement()
  564. {
  565. checkValidity();
  566. ASSERT(m_index != m_deque->m_start);
  567. ASSERT(m_deque->m_buffer.capacity());
  568. if (!m_index)
  569. m_index = m_deque->m_buffer.capacity() - 1;
  570. else
  571. --m_index;
  572. checkValidity();
  573. }
  574. template<typename T, size_t inlineCapacity>
  575. inline T* DequeIteratorBase<T, inlineCapacity>::after() const
  576. {
  577. checkValidity();
  578. ASSERT(m_index != m_deque->m_end);
  579. return &m_deque->m_buffer.buffer()[m_index];
  580. }
  581. template<typename T, size_t inlineCapacity>
  582. inline T* DequeIteratorBase<T, inlineCapacity>::before() const
  583. {
  584. checkValidity();
  585. ASSERT(m_index != m_deque->m_start);
  586. if (!m_index)
  587. return &m_deque->m_buffer.buffer()[m_deque->m_buffer.capacity() - 1];
  588. return &m_deque->m_buffer.buffer()[m_index - 1];
  589. }
  590. } // namespace WTF
  591. using WTF::Deque;
  592. #endif // WTF_Deque_h