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