/* * Copyright 2013-2021 Max Kellermann * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the * distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef CONST_BUFFER_HXX #define CONST_BUFFER_HXX #include #include template struct ConstBuffer; template<> struct ConstBuffer { typedef std::size_t size_type; using value_type = void; using pointer = const void *; using const_pointer = pointer; using iterator = pointer; using const_iterator = pointer; pointer data; size_type size; ConstBuffer() = default; constexpr ConstBuffer(std::nullptr_t) noexcept :data(nullptr), size(0) {} constexpr ConstBuffer(pointer _data, size_type _size) noexcept :data(_data), size(_size) {} constexpr static ConstBuffer FromVoid(ConstBuffer other) noexcept { return other; } constexpr ConstBuffer ToVoid() const noexcept { return *this; } constexpr bool IsNull() const noexcept { return data == nullptr; } constexpr bool operator==(std::nullptr_t) const noexcept { return data == nullptr; } constexpr bool operator!=(std::nullptr_t) const noexcept { return data != nullptr; } constexpr bool empty() const noexcept { return size == 0; } }; /** * A reference to a memory area that is read-only. */ template struct ConstBuffer { using size_type = std::size_t; using value_type = T; using reference = const T &; using const_reference = reference; using pointer = const T *; using const_pointer = pointer; using iterator = pointer; using const_iterator = pointer; pointer data; size_type size; ConstBuffer() = default; constexpr ConstBuffer(std::nullptr_t) noexcept :data(nullptr), size(0) {} constexpr ConstBuffer(pointer _data, size_type _size) noexcept :data(_data), size(_size) {} constexpr ConstBuffer(pointer _data, pointer _end) noexcept :data(_data), size(_end - _data) {} /** * Convert array to ConstBuffer instance. */ template constexpr ConstBuffer(const T (&_data)[_size]) noexcept :data(_data), size(_size) {} /** * Cast a ConstBuffer to a ConstBuffer, rounding down * to the next multiple of T's size. */ static constexpr ConstBuffer FromVoidFloor(ConstBuffer other) noexcept { static_assert(sizeof(T) > 0, "Empty base type"); return ConstBuffer(pointer(other.data), other.size / sizeof(T)); } /** * Cast a ConstBuffer to a ConstBuffer. A "void" * buffer records its size in bytes, and when casting to "T", * the assertion below ensures that the size is a multiple of * sizeof(T). */ constexpr static ConstBuffer FromVoid(ConstBuffer other) noexcept { static_assert(sizeof(T) > 0, "Empty base type"); assert(other.size % sizeof(T) == 0); return FromVoidFloor(other); } constexpr ConstBuffer ToVoid() const noexcept { static_assert(sizeof(T) > 0, "Empty base type"); return ConstBuffer(data, size * sizeof(T)); } constexpr bool IsNull() const noexcept { return data == nullptr; } constexpr bool operator==(std::nullptr_t) const noexcept { return data == nullptr; } constexpr bool operator!=(std::nullptr_t) const noexcept { return data != nullptr; } constexpr bool empty() const noexcept { return size == 0; } template constexpr bool Contains(U &&u) const noexcept { for (const auto &i : *this) if (u == i) return true; return false; } constexpr iterator begin() const noexcept { return data; } constexpr iterator end() const noexcept { return data + size; } constexpr const_iterator cbegin() const noexcept { return data; } constexpr const_iterator cend() const noexcept { return data + size; } constexpr reference operator[](size_type i) const noexcept { assert(i < size); return data[i]; } /** * Returns a reference to the first element. Buffer must not * be empty. */ constexpr reference front() const noexcept { assert(!empty()); return data[0]; } /** * Returns a reference to the last element. Buffer must not * be empty. */ constexpr reference back() const noexcept { assert(!empty()); return data[size - 1]; } /** * Remove the first element (by moving the head pointer, does * not actually modify the buffer). Buffer must not be empty. */ constexpr void pop_front() noexcept { assert(!empty()); ++data; --size; } /** * Remove the last element (by moving the tail pointer, does * not actually modify the buffer). Buffer must not be empty. */ constexpr void pop_back() noexcept { assert(!empty()); --size; } /** * Remove the first element and return a reference to it. * Buffer must not be empty. */ constexpr reference shift() noexcept { reference result = front(); pop_front(); return result; } constexpr void skip_front(size_type n) noexcept { assert(size >= n); data += n; size -= n; } /** * Move the front pointer to the given address, and adjust the * size attribute to retain the old end address. */ void MoveFront(pointer new_data) noexcept { assert(IsNull() == (new_data == nullptr)); assert(new_data <= end()); size = end() - new_data; data = new_data; } /** * Move the end pointer to the given address (by adjusting the * size). */ void SetEnd(pointer new_end) noexcept { assert(IsNull() == (new_end == nullptr)); assert(new_end >= begin()); size = new_end - data; } }; #endif