// Copyright 2014 The Chromium 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 SANDBOX_LINUX_BPF_DSL_CONS_H_ #define SANDBOX_LINUX_BPF_DSL_CONS_H_ #include <memory> #include "base/macros.h" #include "sandbox/sandbox_export.h" namespace sandbox { namespace cons { // Namespace cons provides an abstraction for immutable "cons list" // data structures as commonly provided in functional programming // languages like Lisp or Haskell. // // A cons list is a linked list consisting of "cells", each of which // have a "head" and a "tail" element. A cell's head element contains // a user specified value, while the tail element contains a (possibly // null) pointer to another cell. // // An empty list (idiomatically referred to as "nil") can be // constructed as "cons::List<Foo>()" or simply as "nullptr" if Foo // can be inferred from context (e.g., calling a function that has a // "cons::List<Foo>" parameter). // // Existing lists (including empty lists) can be extended by // prepending new values to the front using the "Cons(head, tail)" // function, which will allocate a new cons cell. Notably, cons lists // support creating multiple lists that share a common tail sequence. // // Lastly, lists support iteration via C++11's range-based for loop // construct. // // Examples: // // // basic construction // const cons::List<char> kNil = nullptr; // cons::List<char> ba = Cons('b', Cons('a', kNil)); // // // common tail sequence // cons::List<char> cba = Cons('c', ba); // cons::List<char> dba = Cons('d', ba); // // // iteration // for (const char& ch : cba) { // // iterates 'c', 'b', 'a' // } // for (const char& ch : dba) { // // iterates 'd', 'b', 'a' // } // Forward declarations. template <typename T> class Cell; template <typename T> class ListIterator; // List represents a (possibly null) pointer to a cons cell. template <typename T> using List = std::shared_ptr<const Cell<T>>; // Cons extends a cons list by prepending a new value to the front. template <typename T> List<T> Cons(const T& head, List<T> tail) { return std::make_shared<Cell<T>>(head, std::move(tail)); } // Cell represents an individual "cons cell" within a cons list. template <typename T> class Cell { public: Cell(const T& head, List<T> tail) : head_(head), tail_(std::move(tail)) {} // Head returns this cell's head element. const T& head() const { return head_; } // Tail returns this cell's tail element. const List<T>& tail() const { return tail_; } private: T head_; List<T> tail_; DISALLOW_COPY_AND_ASSIGN(Cell); }; // Begin returns a list iterator pointing to the first element of the // cons list. It's provided to support range-based for loops. template <typename T> ListIterator<T> begin(const List<T>& list) { return ListIterator<T>(list); } // End returns a list iterator pointing to the "past-the-end" element // of the cons list (i.e., nil). It's provided to support range-based // for loops. template <typename T> ListIterator<T> end(const List<T>& list) { return ListIterator<T>(); } // ListIterator provides C++ forward iterator semantics for traversing // a cons list. template <typename T> class ListIterator { public: ListIterator() : list_() {} explicit ListIterator(const List<T>& list) : list_(list) {} const T& operator*() const { return list_->head(); } ListIterator& operator++() { list_ = list_->tail(); return *this; } friend bool operator==(const ListIterator& lhs, const ListIterator& rhs) { return lhs.list_ == rhs.list_; } private: List<T> list_; }; template <typename T> bool operator!=(const ListIterator<T>& lhs, const ListIterator<T>& rhs) { return !(lhs == rhs); } } // namespace cons } // namespace sandbox #endif // SANDBOX_LINUX_BPF_DSL_CONS_H_