cannam@62: // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors cannam@62: // Licensed under the MIT License: cannam@62: // cannam@62: // Permission is hereby granted, free of charge, to any person obtaining a copy cannam@62: // of this software and associated documentation files (the "Software"), to deal cannam@62: // in the Software without restriction, including without limitation the rights cannam@62: // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell cannam@62: // copies of the Software, and to permit persons to whom the Software is cannam@62: // furnished to do so, subject to the following conditions: cannam@62: // cannam@62: // The above copyright notice and this permission notice shall be included in cannam@62: // all copies or substantial portions of the Software. cannam@62: // cannam@62: // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR cannam@62: // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, cannam@62: // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE cannam@62: // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER cannam@62: // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, cannam@62: // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN cannam@62: // THE SOFTWARE. cannam@62: cannam@62: #ifndef CAPNP_LIST_H_ cannam@62: #define CAPNP_LIST_H_ cannam@62: cannam@62: #if defined(__GNUC__) && !defined(CAPNP_HEADER_WARNINGS) cannam@62: #pragma GCC system_header cannam@62: #endif cannam@62: cannam@62: #include "layout.h" cannam@62: #include "orphan.h" cannam@62: #include cannam@62: #ifdef KJ_STD_COMPAT cannam@62: #include cannam@62: #endif // KJ_STD_COMPAT cannam@62: cannam@62: namespace capnp { cannam@62: namespace _ { // private cannam@62: cannam@62: template cannam@62: class TemporaryPointer { cannam@62: // This class is a little hack which lets us define operator->() in cases where it needs to cannam@62: // return a pointer to a temporary value. We instead construct a TemporaryPointer and return that cannam@62: // (by value). The compiler then invokes operator->() on the TemporaryPointer, which itself is cannam@62: // able to return a real pointer to its member. cannam@62: cannam@62: public: cannam@62: TemporaryPointer(T&& value): value(kj::mv(value)) {} cannam@62: TemporaryPointer(const T& value): value(value) {} cannam@62: cannam@62: inline T* operator->() { return &value; } cannam@62: private: cannam@62: T value; cannam@62: }; cannam@62: cannam@62: template cannam@62: class IndexingIterator { cannam@62: public: cannam@62: IndexingIterator() = default; cannam@62: cannam@62: inline Element operator*() const { return (*container)[index]; } cannam@62: inline TemporaryPointer operator->() const { cannam@62: return TemporaryPointer((*container)[index]); cannam@62: } cannam@62: inline Element operator[]( int off) const { return (*container)[index]; } cannam@62: inline Element operator[](uint off) const { return (*container)[index]; } cannam@62: cannam@62: inline IndexingIterator& operator++() { ++index; return *this; } cannam@62: inline IndexingIterator operator++(int) { IndexingIterator other = *this; ++index; return other; } cannam@62: inline IndexingIterator& operator--() { --index; return *this; } cannam@62: inline IndexingIterator operator--(int) { IndexingIterator other = *this; --index; return other; } cannam@62: cannam@62: inline IndexingIterator operator+(uint amount) const { return IndexingIterator(container, index + amount); } cannam@62: inline IndexingIterator operator-(uint amount) const { return IndexingIterator(container, index - amount); } cannam@62: inline IndexingIterator operator+( int amount) const { return IndexingIterator(container, index + amount); } cannam@62: inline IndexingIterator operator-( int amount) const { return IndexingIterator(container, index - amount); } cannam@62: cannam@62: inline int operator-(const IndexingIterator& other) const { return index - other.index; } cannam@62: cannam@62: inline IndexingIterator& operator+=(uint amount) { index += amount; return *this; } cannam@62: inline IndexingIterator& operator-=(uint amount) { index -= amount; return *this; } cannam@62: inline IndexingIterator& operator+=( int amount) { index += amount; return *this; } cannam@62: inline IndexingIterator& operator-=( int amount) { index -= amount; return *this; } cannam@62: cannam@62: // STL says comparing iterators of different containers is not allowed, so we only compare cannam@62: // indices here. cannam@62: inline bool operator==(const IndexingIterator& other) const { return index == other.index; } cannam@62: inline bool operator!=(const IndexingIterator& other) const { return index != other.index; } cannam@62: inline bool operator<=(const IndexingIterator& other) const { return index <= other.index; } cannam@62: inline bool operator>=(const IndexingIterator& other) const { return index >= other.index; } cannam@62: inline bool operator< (const IndexingIterator& other) const { return index < other.index; } cannam@62: inline bool operator> (const IndexingIterator& other) const { return index > other.index; } cannam@62: cannam@62: private: cannam@62: Container* container; cannam@62: uint index; cannam@62: cannam@62: friend Container; cannam@62: inline IndexingIterator(Container* container, uint index) cannam@62: : container(container), index(index) {} cannam@62: }; cannam@62: cannam@62: } // namespace _ (private) cannam@62: cannam@62: template cannam@62: struct List { cannam@62: // List of primitives. cannam@62: cannam@62: List() = delete; cannam@62: cannam@62: class Reader { cannam@62: public: cannam@62: typedef List Reads; cannam@62: cannam@62: inline Reader(): reader(_::elementSizeForType()) {} cannam@62: inline explicit Reader(_::ListReader reader): reader(reader) {} cannam@62: cannam@62: inline uint size() const { return unbound(reader.size() / ELEMENTS); } cannam@62: inline T operator[](uint index) const { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return reader.template getDataElement(bounded(index) * ELEMENTS); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() const { return Iterator(this, 0); } cannam@62: inline Iterator end() const { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListReader reader; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: template cannam@62: friend struct List; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Builder { cannam@62: public: cannam@62: typedef List Builds; cannam@62: cannam@62: inline Builder(): builder(_::elementSizeForType()) {} cannam@62: inline Builder(decltype(nullptr)): Builder() {} cannam@62: inline explicit Builder(_::ListBuilder builder): builder(builder) {} cannam@62: cannam@62: inline operator Reader() const { return Reader(builder.asReader()); } cannam@62: inline Reader asReader() const { return Reader(builder.asReader()); } cannam@62: cannam@62: inline uint size() const { return unbound(builder.size() / ELEMENTS); } cannam@62: inline T operator[](uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return builder.template getDataElement(bounded(index) * ELEMENTS); cannam@62: } cannam@62: inline void set(uint index, T value) { cannam@62: // Alas, it is not possible to make operator[] return a reference to which you can assign, cannam@62: // since the encoded representation does not necessarily match the compiler's representation cannam@62: // of the type. We can't even return a clever class that implements operator T() and cannam@62: // operator=() because it will lead to surprising behavior when using type inference (e.g. cannam@62: // calling a template function with inferred argument types, or using "auto" or "decltype"). cannam@62: cannam@62: builder.template setDataElement(bounded(index) * ELEMENTS, value); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() { return Iterator(this, 0); } cannam@62: inline Iterator end() { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListBuilder builder; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Pipeline {}; cannam@62: cannam@62: private: cannam@62: inline static _::ListBuilder initPointer(_::PointerBuilder builder, uint size) { cannam@62: return builder.initList(_::elementSizeForType(), bounded(size) * ELEMENTS); cannam@62: } cannam@62: inline static _::ListBuilder getFromPointer(_::PointerBuilder builder, const word* defaultValue) { cannam@62: return builder.getList(_::elementSizeForType(), defaultValue); cannam@62: } cannam@62: inline static _::ListReader getFromPointer( cannam@62: const _::PointerReader& reader, const word* defaultValue) { cannam@62: return reader.getList(_::elementSizeForType(), defaultValue); cannam@62: } cannam@62: cannam@62: template cannam@62: friend struct List; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: }; cannam@62: cannam@62: template cannam@62: struct List: public List {}; cannam@62: cannam@62: template cannam@62: struct List { cannam@62: // List of structs. cannam@62: cannam@62: List() = delete; cannam@62: cannam@62: class Reader { cannam@62: public: cannam@62: typedef List Reads; cannam@62: cannam@62: inline Reader(): reader(ElementSize::INLINE_COMPOSITE) {} cannam@62: inline explicit Reader(_::ListReader reader): reader(reader) {} cannam@62: cannam@62: inline uint size() const { return unbound(reader.size() / ELEMENTS); } cannam@62: inline typename T::Reader operator[](uint index) const { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return typename T::Reader(reader.getStructElement(bounded(index) * ELEMENTS)); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() const { return Iterator(this, 0); } cannam@62: inline Iterator end() const { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListReader reader; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: template cannam@62: friend struct List; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Builder { cannam@62: public: cannam@62: typedef List Builds; cannam@62: cannam@62: inline Builder(): builder(ElementSize::INLINE_COMPOSITE) {} cannam@62: inline Builder(decltype(nullptr)): Builder() {} cannam@62: inline explicit Builder(_::ListBuilder builder): builder(builder) {} cannam@62: cannam@62: inline operator Reader() const { return Reader(builder.asReader()); } cannam@62: inline Reader asReader() const { return Reader(builder.asReader()); } cannam@62: cannam@62: inline uint size() const { return unbound(builder.size() / ELEMENTS); } cannam@62: inline typename T::Builder operator[](uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return typename T::Builder(builder.getStructElement(bounded(index) * ELEMENTS)); cannam@62: } cannam@62: cannam@62: inline void adoptWithCaveats(uint index, Orphan&& orphan) { cannam@62: // Mostly behaves like you'd expect `adopt` to behave, but with two caveats originating from cannam@62: // the fact that structs in a struct list are allocated inline rather than by pointer: cannam@62: // * This actually performs a shallow copy, effectively adopting each of the orphan's cannam@62: // children rather than adopting the orphan itself. The orphan ends up being discarded, cannam@62: // possibly wasting space in the message object. cannam@62: // * If the orphan is larger than the target struct -- say, because the orphan was built cannam@62: // using a newer version of the schema that has additional fields -- it will be truncated, cannam@62: // losing data. cannam@62: cannam@62: KJ_IREQUIRE(index < size()); cannam@62: cannam@62: // We pass a zero-valued StructSize to asStruct() because we do not want the struct to be cannam@62: // expanded under any circumstances. We're just going to throw it away anyway, and cannam@62: // transferContentFrom() already carefully compares the struct sizes before transferring. cannam@62: builder.getStructElement(bounded(index) * ELEMENTS).transferContentFrom( cannam@62: orphan.builder.asStruct(_::StructSize(ZERO * WORDS, ZERO * POINTERS))); cannam@62: } cannam@62: inline void setWithCaveats(uint index, const typename T::Reader& reader) { cannam@62: // Mostly behaves like you'd expect `set` to behave, but with a caveat originating from cannam@62: // the fact that structs in a struct list are allocated inline rather than by pointer: cannam@62: // If the source struct is larger than the target struct -- say, because the source was built cannam@62: // using a newer version of the schema that has additional fields -- it will be truncated, cannam@62: // losing data. cannam@62: // cannam@62: // Note: If you are trying to concatenate some lists, use Orphanage::newOrphanConcat() to cannam@62: // do it without losing any data in case the source lists come from a newer version of the cannam@62: // protocol. (Plus, it's easier to use anyhow.) cannam@62: cannam@62: KJ_IREQUIRE(index < size()); cannam@62: builder.getStructElement(bounded(index) * ELEMENTS).copyContentFrom(reader._reader); cannam@62: } cannam@62: cannam@62: // There are no init(), set(), adopt(), or disown() methods for lists of structs because the cannam@62: // elements of the list are inlined and are initialized when the list is initialized. This cannam@62: // means that init() would be redundant, and set() would risk data loss if the input struct cannam@62: // were from a newer version of the protocol. cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() { return Iterator(this, 0); } cannam@62: inline Iterator end() { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListBuilder builder; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Pipeline {}; cannam@62: cannam@62: private: cannam@62: inline static _::ListBuilder initPointer(_::PointerBuilder builder, uint size) { cannam@62: return builder.initStructList(bounded(size) * ELEMENTS, _::structSize()); cannam@62: } cannam@62: inline static _::ListBuilder getFromPointer(_::PointerBuilder builder, const word* defaultValue) { cannam@62: return builder.getStructList(_::structSize(), defaultValue); cannam@62: } cannam@62: inline static _::ListReader getFromPointer( cannam@62: const _::PointerReader& reader, const word* defaultValue) { cannam@62: return reader.getList(ElementSize::INLINE_COMPOSITE, defaultValue); cannam@62: } cannam@62: cannam@62: template cannam@62: friend struct List; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: }; cannam@62: cannam@62: template cannam@62: struct List, Kind::LIST> { cannam@62: // List of lists. cannam@62: cannam@62: List() = delete; cannam@62: cannam@62: class Reader { cannam@62: public: cannam@62: typedef List> Reads; cannam@62: cannam@62: inline Reader(): reader(ElementSize::POINTER) {} cannam@62: inline explicit Reader(_::ListReader reader): reader(reader) {} cannam@62: cannam@62: inline uint size() const { return unbound(reader.size() / ELEMENTS); } cannam@62: inline typename List::Reader operator[](uint index) const { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return typename List::Reader(_::PointerHelpers>::get( cannam@62: reader.getPointerElement(bounded(index) * ELEMENTS))); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator::Reader> Iterator; cannam@62: inline Iterator begin() const { return Iterator(this, 0); } cannam@62: inline Iterator end() const { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListReader reader; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: template cannam@62: friend struct List; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Builder { cannam@62: public: cannam@62: typedef List> Builds; cannam@62: cannam@62: inline Builder(): builder(ElementSize::POINTER) {} cannam@62: inline Builder(decltype(nullptr)): Builder() {} cannam@62: inline explicit Builder(_::ListBuilder builder): builder(builder) {} cannam@62: cannam@62: inline operator Reader() const { return Reader(builder.asReader()); } cannam@62: inline Reader asReader() const { return Reader(builder.asReader()); } cannam@62: cannam@62: inline uint size() const { return unbound(builder.size() / ELEMENTS); } cannam@62: inline typename List::Builder operator[](uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return typename List::Builder(_::PointerHelpers>::get( cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS))); cannam@62: } cannam@62: inline typename List::Builder init(uint index, uint size) { cannam@62: KJ_IREQUIRE(index < this->size()); cannam@62: return typename List::Builder(_::PointerHelpers>::init( cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS), size)); cannam@62: } cannam@62: inline void set(uint index, typename List::Reader value) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS).setList(value.reader); cannam@62: } cannam@62: void set(uint index, std::initializer_list> value) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: auto l = init(index, value.size()); cannam@62: uint i = 0; cannam@62: for (auto& element: value) { cannam@62: l.set(i++, element); cannam@62: } cannam@62: } cannam@62: inline void adopt(uint index, Orphan&& value) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS).adopt(kj::mv(value.builder)); cannam@62: } cannam@62: inline Orphan disown(uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return Orphan(builder.getPointerElement(bounded(index) * ELEMENTS).disown()); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator::Builder> Iterator; cannam@62: inline Iterator begin() { return Iterator(this, 0); } cannam@62: inline Iterator end() { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListBuilder builder; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Pipeline {}; cannam@62: cannam@62: private: cannam@62: inline static _::ListBuilder initPointer(_::PointerBuilder builder, uint size) { cannam@62: return builder.initList(ElementSize::POINTER, bounded(size) * ELEMENTS); cannam@62: } cannam@62: inline static _::ListBuilder getFromPointer(_::PointerBuilder builder, const word* defaultValue) { cannam@62: return builder.getList(ElementSize::POINTER, defaultValue); cannam@62: } cannam@62: inline static _::ListReader getFromPointer( cannam@62: const _::PointerReader& reader, const word* defaultValue) { cannam@62: return reader.getList(ElementSize::POINTER, defaultValue); cannam@62: } cannam@62: cannam@62: template cannam@62: friend struct List; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: }; cannam@62: cannam@62: template cannam@62: struct List { cannam@62: List() = delete; cannam@62: cannam@62: class Reader { cannam@62: public: cannam@62: typedef List Reads; cannam@62: cannam@62: inline Reader(): reader(ElementSize::POINTER) {} cannam@62: inline explicit Reader(_::ListReader reader): reader(reader) {} cannam@62: cannam@62: inline uint size() const { return unbound(reader.size() / ELEMENTS); } cannam@62: inline typename T::Reader operator[](uint index) const { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return reader.getPointerElement(bounded(index) * ELEMENTS) cannam@62: .template getBlob(nullptr, ZERO * BYTES); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() const { return Iterator(this, 0); } cannam@62: inline Iterator end() const { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListReader reader; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: template cannam@62: friend struct List; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Builder { cannam@62: public: cannam@62: typedef List Builds; cannam@62: cannam@62: inline Builder(): builder(ElementSize::POINTER) {} cannam@62: inline Builder(decltype(nullptr)): Builder() {} cannam@62: inline explicit Builder(_::ListBuilder builder): builder(builder) {} cannam@62: cannam@62: inline operator Reader() const { return Reader(builder.asReader()); } cannam@62: inline Reader asReader() const { return Reader(builder.asReader()); } cannam@62: cannam@62: inline uint size() const { return unbound(builder.size() / ELEMENTS); } cannam@62: inline typename T::Builder operator[](uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return builder.getPointerElement(bounded(index) * ELEMENTS) cannam@62: .template getBlob(nullptr, ZERO * BYTES); cannam@62: } cannam@62: inline void set(uint index, typename T::Reader value) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS).template setBlob(value); cannam@62: } cannam@62: inline typename T::Builder init(uint index, uint size) { cannam@62: KJ_IREQUIRE(index < this->size()); cannam@62: return builder.getPointerElement(bounded(index) * ELEMENTS) cannam@62: .template initBlob(bounded(size) * BYTES); cannam@62: } cannam@62: inline void adopt(uint index, Orphan&& value) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: builder.getPointerElement(bounded(index) * ELEMENTS).adopt(kj::mv(value.builder)); cannam@62: } cannam@62: inline Orphan disown(uint index) { cannam@62: KJ_IREQUIRE(index < size()); cannam@62: return Orphan(builder.getPointerElement(bounded(index) * ELEMENTS).disown()); cannam@62: } cannam@62: cannam@62: typedef _::IndexingIterator Iterator; cannam@62: inline Iterator begin() { return Iterator(this, 0); } cannam@62: inline Iterator end() { return Iterator(this, size()); } cannam@62: cannam@62: private: cannam@62: _::ListBuilder builder; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: friend class Orphanage; cannam@62: template cannam@62: friend struct ToDynamic_; cannam@62: }; cannam@62: cannam@62: class Pipeline {}; cannam@62: cannam@62: private: cannam@62: inline static _::ListBuilder initPointer(_::PointerBuilder builder, uint size) { cannam@62: return builder.initList(ElementSize::POINTER, bounded(size) * ELEMENTS); cannam@62: } cannam@62: inline static _::ListBuilder getFromPointer(_::PointerBuilder builder, const word* defaultValue) { cannam@62: return builder.getList(ElementSize::POINTER, defaultValue); cannam@62: } cannam@62: inline static _::ListReader getFromPointer( cannam@62: const _::PointerReader& reader, const word* defaultValue) { cannam@62: return reader.getList(ElementSize::POINTER, defaultValue); cannam@62: } cannam@62: cannam@62: template cannam@62: friend struct List; cannam@62: template cannam@62: friend struct _::PointerHelpers; cannam@62: }; cannam@62: cannam@62: } // namespace capnp cannam@62: cannam@62: #ifdef KJ_STD_COMPAT cannam@62: namespace std { cannam@62: cannam@62: template cannam@62: struct iterator_traits> cannam@62: : public std::iterator {}; cannam@62: cannam@62: } // namespace std cannam@62: #endif // KJ_STD_COMPAT cannam@62: cannam@62: #endif // CAPNP_LIST_H_