1141cc406Sopenharmony_ci/* sane - Scanner Access Now Easy. 2141cc406Sopenharmony_ci 3141cc406Sopenharmony_ci Copyright (C) 2019 Povilas Kanapickas <povilas@radix.lt> 4141cc406Sopenharmony_ci 5141cc406Sopenharmony_ci This file is part of the SANE package. 6141cc406Sopenharmony_ci 7141cc406Sopenharmony_ci This program is free software; you can redistribute it and/or 8141cc406Sopenharmony_ci modify it under the terms of the GNU General Public License as 9141cc406Sopenharmony_ci published by the Free Software Foundation; either version 2 of the 10141cc406Sopenharmony_ci License, or (at your option) any later version. 11141cc406Sopenharmony_ci 12141cc406Sopenharmony_ci This program is distributed in the hope that it will be useful, but 13141cc406Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 14141cc406Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15141cc406Sopenharmony_ci General Public License for more details. 16141cc406Sopenharmony_ci 17141cc406Sopenharmony_ci You should have received a copy of the GNU General Public License 18141cc406Sopenharmony_ci along with this program. If not, see <https://www.gnu.org/licenses/>. 19141cc406Sopenharmony_ci*/ 20141cc406Sopenharmony_ci 21141cc406Sopenharmony_ci#ifndef BACKEND_GENESYS_LINE_BUFFER_H 22141cc406Sopenharmony_ci#define BACKEND_GENESYS_LINE_BUFFER_H 23141cc406Sopenharmony_ci 24141cc406Sopenharmony_ci#include "error.h" 25141cc406Sopenharmony_ci 26141cc406Sopenharmony_ci#include <algorithm> 27141cc406Sopenharmony_ci#include <cstdint> 28141cc406Sopenharmony_ci#include <cstddef> 29141cc406Sopenharmony_ci#include <vector> 30141cc406Sopenharmony_ci 31141cc406Sopenharmony_cinamespace genesys { 32141cc406Sopenharmony_ci 33141cc406Sopenharmony_ciclass RowBuffer 34141cc406Sopenharmony_ci{ 35141cc406Sopenharmony_cipublic: 36141cc406Sopenharmony_ci RowBuffer(std::size_t line_bytes) : row_bytes_{line_bytes} {} 37141cc406Sopenharmony_ci RowBuffer(const RowBuffer&) = default; 38141cc406Sopenharmony_ci RowBuffer& operator=(const RowBuffer&) = default; 39141cc406Sopenharmony_ci ~RowBuffer() = default; 40141cc406Sopenharmony_ci 41141cc406Sopenharmony_ci const std::uint8_t* get_row_ptr(std::size_t y) const 42141cc406Sopenharmony_ci { 43141cc406Sopenharmony_ci if (y >= height()) { 44141cc406Sopenharmony_ci throw SaneException("y %zu is out of range", y); 45141cc406Sopenharmony_ci } 46141cc406Sopenharmony_ci return data_.data() + row_bytes_ * get_row_index(y); 47141cc406Sopenharmony_ci } 48141cc406Sopenharmony_ci 49141cc406Sopenharmony_ci std::uint8_t* get_row_ptr(std::size_t y) 50141cc406Sopenharmony_ci { 51141cc406Sopenharmony_ci if (y >= height()) { 52141cc406Sopenharmony_ci throw SaneException("y %zu is out of range", y); 53141cc406Sopenharmony_ci } 54141cc406Sopenharmony_ci return data_.data() + row_bytes_ * get_row_index(y); 55141cc406Sopenharmony_ci } 56141cc406Sopenharmony_ci 57141cc406Sopenharmony_ci const std::uint8_t* get_front_row_ptr() const { return get_row_ptr(0); } 58141cc406Sopenharmony_ci std::uint8_t* get_front_row_ptr() { return get_row_ptr(0); } 59141cc406Sopenharmony_ci const std::uint8_t* get_back_row_ptr() const { return get_row_ptr(height() - 1); } 60141cc406Sopenharmony_ci std::uint8_t* get_back_row_ptr() { return get_row_ptr(height() - 1); } 61141cc406Sopenharmony_ci 62141cc406Sopenharmony_ci bool empty() const { return is_linear_ && first_ == last_; } 63141cc406Sopenharmony_ci 64141cc406Sopenharmony_ci bool full() 65141cc406Sopenharmony_ci { 66141cc406Sopenharmony_ci if (is_linear_) { 67141cc406Sopenharmony_ci return last_ == buffer_end_; 68141cc406Sopenharmony_ci } 69141cc406Sopenharmony_ci return first_ == last_; 70141cc406Sopenharmony_ci } 71141cc406Sopenharmony_ci 72141cc406Sopenharmony_ci bool is_linear() const { return is_linear_; } 73141cc406Sopenharmony_ci 74141cc406Sopenharmony_ci void linearize() 75141cc406Sopenharmony_ci { 76141cc406Sopenharmony_ci if (!is_linear_) { 77141cc406Sopenharmony_ci std::rotate(data_.begin(), data_.begin() + row_bytes_ * first_, data_.end()); 78141cc406Sopenharmony_ci last_ = height(); 79141cc406Sopenharmony_ci first_ = 0; 80141cc406Sopenharmony_ci is_linear_ = true; 81141cc406Sopenharmony_ci } 82141cc406Sopenharmony_ci } 83141cc406Sopenharmony_ci 84141cc406Sopenharmony_ci void pop_front() 85141cc406Sopenharmony_ci { 86141cc406Sopenharmony_ci if (empty()) { 87141cc406Sopenharmony_ci throw SaneException("Trying to pop out of empty() line buffer"); 88141cc406Sopenharmony_ci } 89141cc406Sopenharmony_ci 90141cc406Sopenharmony_ci first_++; 91141cc406Sopenharmony_ci if (first_ == last_) { 92141cc406Sopenharmony_ci first_ = 0; 93141cc406Sopenharmony_ci last_ = 0; 94141cc406Sopenharmony_ci is_linear_ = true; 95141cc406Sopenharmony_ci } else if (first_ == buffer_end_) { 96141cc406Sopenharmony_ci first_ = 0; 97141cc406Sopenharmony_ci is_linear_ = true; 98141cc406Sopenharmony_ci } 99141cc406Sopenharmony_ci } 100141cc406Sopenharmony_ci 101141cc406Sopenharmony_ci void push_front() 102141cc406Sopenharmony_ci { 103141cc406Sopenharmony_ci if (height() + 1 >= height_capacity()) { 104141cc406Sopenharmony_ci ensure_capacity(std::max<std::size_t>(1, height() * 2)); 105141cc406Sopenharmony_ci } 106141cc406Sopenharmony_ci 107141cc406Sopenharmony_ci if (first_ == 0) { 108141cc406Sopenharmony_ci is_linear_ = false; 109141cc406Sopenharmony_ci first_ = buffer_end_; 110141cc406Sopenharmony_ci } 111141cc406Sopenharmony_ci first_--; 112141cc406Sopenharmony_ci } 113141cc406Sopenharmony_ci 114141cc406Sopenharmony_ci void pop_back() 115141cc406Sopenharmony_ci { 116141cc406Sopenharmony_ci if (empty()) { 117141cc406Sopenharmony_ci throw SaneException("Trying to pop out of empty() line buffer"); 118141cc406Sopenharmony_ci } 119141cc406Sopenharmony_ci if (last_ == 0) { 120141cc406Sopenharmony_ci last_ = buffer_end_; 121141cc406Sopenharmony_ci is_linear_ = true; 122141cc406Sopenharmony_ci } 123141cc406Sopenharmony_ci last_--; 124141cc406Sopenharmony_ci if (first_ == last_) { 125141cc406Sopenharmony_ci first_ = 0; 126141cc406Sopenharmony_ci last_ = 0; 127141cc406Sopenharmony_ci is_linear_ = true; 128141cc406Sopenharmony_ci } 129141cc406Sopenharmony_ci } 130141cc406Sopenharmony_ci 131141cc406Sopenharmony_ci void push_back() 132141cc406Sopenharmony_ci { 133141cc406Sopenharmony_ci if (height() + 1 >= height_capacity()) { 134141cc406Sopenharmony_ci ensure_capacity(std::max<std::size_t>(1, height() * 2)); 135141cc406Sopenharmony_ci } 136141cc406Sopenharmony_ci 137141cc406Sopenharmony_ci if (last_ == buffer_end_) { 138141cc406Sopenharmony_ci is_linear_ = false; 139141cc406Sopenharmony_ci last_ = 0; 140141cc406Sopenharmony_ci } 141141cc406Sopenharmony_ci last_++; 142141cc406Sopenharmony_ci } 143141cc406Sopenharmony_ci 144141cc406Sopenharmony_ci std::size_t row_bytes() const { return row_bytes_; } 145141cc406Sopenharmony_ci 146141cc406Sopenharmony_ci std::size_t height() const 147141cc406Sopenharmony_ci { 148141cc406Sopenharmony_ci if (!is_linear_) { 149141cc406Sopenharmony_ci return last_ + buffer_end_ - first_; 150141cc406Sopenharmony_ci } 151141cc406Sopenharmony_ci return last_ - first_; 152141cc406Sopenharmony_ci } 153141cc406Sopenharmony_ci 154141cc406Sopenharmony_ci std::size_t height_capacity() const { return buffer_end_; } 155141cc406Sopenharmony_ci 156141cc406Sopenharmony_ci void clear() 157141cc406Sopenharmony_ci { 158141cc406Sopenharmony_ci first_ = 0; 159141cc406Sopenharmony_ci last_ = 0; 160141cc406Sopenharmony_ci } 161141cc406Sopenharmony_ci 162141cc406Sopenharmony_ciprivate: 163141cc406Sopenharmony_ci std::size_t get_row_index(std::size_t index) const 164141cc406Sopenharmony_ci { 165141cc406Sopenharmony_ci if (index >= buffer_end_ - first_) { 166141cc406Sopenharmony_ci return index - (buffer_end_ - first_); 167141cc406Sopenharmony_ci } 168141cc406Sopenharmony_ci return index + first_; 169141cc406Sopenharmony_ci } 170141cc406Sopenharmony_ci 171141cc406Sopenharmony_ci void ensure_capacity(std::size_t capacity) 172141cc406Sopenharmony_ci { 173141cc406Sopenharmony_ci if (capacity < height_capacity()) 174141cc406Sopenharmony_ci return; 175141cc406Sopenharmony_ci linearize(); 176141cc406Sopenharmony_ci data_.resize(capacity * row_bytes_); 177141cc406Sopenharmony_ci buffer_end_ = capacity; 178141cc406Sopenharmony_ci } 179141cc406Sopenharmony_ci 180141cc406Sopenharmony_ciprivate: 181141cc406Sopenharmony_ci std::size_t row_bytes_ = 0; 182141cc406Sopenharmony_ci std::size_t first_ = 0; 183141cc406Sopenharmony_ci std::size_t last_ = 0; 184141cc406Sopenharmony_ci std::size_t buffer_end_ = 0; 185141cc406Sopenharmony_ci bool is_linear_ = true; 186141cc406Sopenharmony_ci std::vector<std::uint8_t> data_; 187141cc406Sopenharmony_ci}; 188141cc406Sopenharmony_ci 189141cc406Sopenharmony_ci} // namespace genesys 190141cc406Sopenharmony_ci 191141cc406Sopenharmony_ci#endif // BACKEND_GENESYS_LINE_BUFFER_H 192