|  | // Copyright 2018 the V8 project authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef V8_COMPILER_FUNCTIONAL_LIST_H_ | 
|  | #define V8_COMPILER_FUNCTIONAL_LIST_H_ | 
|  |  | 
|  | #include <iterator> | 
|  | #include "src/zone/zone.h" | 
|  |  | 
|  | namespace v8 { | 
|  | namespace internal { | 
|  | namespace compiler { | 
|  |  | 
|  | // A generic stack implemented as a purely functional singly-linked list, which | 
|  | // results in an O(1) copy operation. It is the equivalent of functional lists | 
|  | // in ML-like languages, with the only difference that it also caches the length | 
|  | // of the list in each node. | 
|  | // TODO(tebbi): Use this implementation also for RedundancyElimination. | 
|  | template <class A> | 
|  | class FunctionalList { | 
|  | private: | 
|  | struct Cons : ZoneObject { | 
|  | Cons(A top, Cons* rest) | 
|  | : top(std::move(top)), rest(rest), size(1 + (rest ? rest->size : 0)) {} | 
|  | A const top; | 
|  | Cons* const rest; | 
|  | size_t const size; | 
|  | }; | 
|  |  | 
|  | public: | 
|  | FunctionalList() : elements_(nullptr) {} | 
|  |  | 
|  | bool operator==(const FunctionalList<A>& other) const { | 
|  | if (Size() != other.Size()) return false; | 
|  | iterator it = begin(); | 
|  | iterator other_it = other.begin(); | 
|  | while (true) { | 
|  | if (it == other_it) return true; | 
|  | if (*it != *other_it) return false; | 
|  | ++it; | 
|  | ++other_it; | 
|  | } | 
|  | } | 
|  | bool operator!=(const FunctionalList<A>& other) const { | 
|  | return !(*this == other); | 
|  | } | 
|  |  | 
|  | bool TriviallyEquals(const FunctionalList<A>& other) const { | 
|  | return elements_ == other.elements_; | 
|  | } | 
|  |  | 
|  | const A& Front() const { | 
|  | DCHECK_GT(Size(), 0); | 
|  | return elements_->top; | 
|  | } | 
|  |  | 
|  | FunctionalList Rest() const { | 
|  | FunctionalList result = *this; | 
|  | result.DropFront(); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void DropFront() { | 
|  | CHECK_GT(Size(), 0); | 
|  | elements_ = elements_->rest; | 
|  | } | 
|  |  | 
|  | void PushFront(A a, Zone* zone) { | 
|  | elements_ = zone->New<Cons>(std::move(a), elements_); | 
|  | } | 
|  |  | 
|  | // If {hint} happens to be exactly what we want to allocate, avoid allocation | 
|  | // by reusing {hint}. | 
|  | void PushFront(A a, Zone* zone, FunctionalList hint) { | 
|  | if (hint.Size() == Size() + 1 && hint.Front() == a && | 
|  | hint.Rest() == *this) { | 
|  | *this = hint; | 
|  | } else { | 
|  | PushFront(a, zone); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Drop elements until the current stack is equal to the tail shared with | 
|  | // {other}. The shared tail must not only be equal, but also refer to the | 
|  | // same memory. | 
|  | void ResetToCommonAncestor(FunctionalList other) { | 
|  | while (other.Size() > Size()) other.DropFront(); | 
|  | while (other.Size() < Size()) DropFront(); | 
|  | while (elements_ != other.elements_) { | 
|  | DropFront(); | 
|  | other.DropFront(); | 
|  | } | 
|  | } | 
|  |  | 
|  | size_t Size() const { return elements_ ? elements_->size : 0; } | 
|  |  | 
|  | void Clear() { elements_ = nullptr; } | 
|  |  | 
|  | class iterator : public std::iterator<std::forward_iterator_tag, A> { | 
|  | public: | 
|  | explicit iterator(Cons* cur) : current_(cur) {} | 
|  |  | 
|  | const A& operator*() const { return current_->top; } | 
|  | iterator& operator++() { | 
|  | current_ = current_->rest; | 
|  | return *this; | 
|  | } | 
|  | bool operator==(const iterator& other) const { | 
|  | return this->current_ == other.current_; | 
|  | } | 
|  | bool operator!=(const iterator& other) const { return !(*this == other); } | 
|  |  | 
|  | // Implemented so that std::find and friends can use std::iterator_traits | 
|  | // for this iterator type. | 
|  | typedef std::forward_iterator_tag iterator_category; | 
|  | typedef ptrdiff_t difference_type; | 
|  | typedef A value_type; | 
|  | typedef A* pointer; | 
|  | typedef A& reference; | 
|  |  | 
|  | private: | 
|  | Cons* current_; | 
|  | }; | 
|  |  | 
|  | iterator begin() const { return iterator(elements_); } | 
|  | iterator end() const { return iterator(nullptr); } | 
|  |  | 
|  | private: | 
|  | Cons* elements_; | 
|  | }; | 
|  |  | 
|  | }  // namespace compiler | 
|  | }  // namespace internal | 
|  | }  // namespace v8 | 
|  |  | 
|  | #endif  // V8_COMPILER_FUNCTIONAL_LIST_H_ |