#pragma once #include namespace DB { namespace ErrorCodes { extern const int LOGICAL_ERROR; } /* Deque with fixed memory size. Allows pushing gaps. * frontGap() returns the number of gaps were inserted before front. * * This structure may be implemented via std::deque, but * - Deque uses fixed amount of memory which is allocated in constructor. No more allocations are performed. * - Gaps are not stored as separate values in queue, which is more memory efficient. * - Deque is responsible for gaps invariant: after removing element, moves gaps into neighbor cell. * * Note: empty deque may have non-zero front gap. */ template class FixedSizeDequeWithGaps { public: struct ValueWithGap { /// The number of gaps before current element. The number of gaps after last element stores into end cell. size_t gap; /// Store char[] instead of T in order to make ValueWithGap POD. /// Call placement constructors after push and and destructors after pop. char value[sizeof(T)]; }; explicit FixedSizeDequeWithGaps(size_t size) { container.resize_fill(size + 1); } ~FixedSizeDequeWithGaps() { auto destruct_range = [this](size_t from, size_t to) { for (size_t i = from; i < to; ++i) destructValue(i); }; if (begin <= end) destruct_range(begin, end); else { destruct_range(0, end); destruct_range(begin, container.size()); } } void pushBack(const T & value) { checkEnoughSpaceToInsert(); constructValue(end, value); moveRight(end); container[end].gap = 0; } void pushGap(size_t count) { container[end].gap += count; } void popBack() { checkHasValuesToRemove(); size_t curr_gap = container[end].gap; moveLeft(end); destructValue(end); container[end].gap += curr_gap; } void popFront() { checkHasValuesToRemove(); destructValue(begin); moveRight(begin); } T & front() { checkHasValuesToGet(); return getValue(begin); } const T & front() const { checkHasValuesToGet(); return getValue(begin); } const T & back() const { size_t ps = end; moveLeft(ps); return getValue(ps); } size_t & frontGap() { return container[begin].gap; } const size_t & frontGap() const { return container[begin].gap; } size_t size() const { if (begin <= end) return end - begin; return end + (container.size() - begin); } bool empty() const { return begin == end; } private: PODArray container; size_t gap_before_first = 0; size_t begin = 0; size_t end = 0; void constructValue(size_t index, const T & value) { new (container[index].value) T(value); } void destructValue(size_t index) { reinterpret_cast(container[index].value)->~T(); } T & getValue(size_t index) { return *reinterpret_cast(container[index].value); } const T & getValue(size_t index) const { return *reinterpret_cast(container[index].value); } void moveRight(size_t & index) const { ++index; if (index == container.size()) index = 0; } void moveLeft(size_t & index) const { if (index == 0) index = container.size(); --index; } void checkEnoughSpaceToInsert() const { if (size() + 1 == container.size()) throw Exception(ErrorCodes::LOGICAL_ERROR, "Not enough space to insert into FixedSizeDequeWithGaps with capacity {}", container.size() - 1); } void checkHasValuesToRemove() const { if (empty()) throw Exception(ErrorCodes::LOGICAL_ERROR, "Cannot remove from empty FixedSizeDequeWithGaps"); } void checkHasValuesToGet() const { if (empty()) throw Exception(ErrorCodes::LOGICAL_ERROR, "Cannot get value from empty FixedSizeDequeWithGaps"); } }; }