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#define DEBUG_DECLARE_ONLY
22141cc406Sopenharmony_ci
23141cc406Sopenharmony_ci#include "image_buffer.h"
24141cc406Sopenharmony_ci#include "image.h"
25141cc406Sopenharmony_ci#include "utilities.h"
26141cc406Sopenharmony_ci
27141cc406Sopenharmony_cinamespace genesys {
28141cc406Sopenharmony_ci
29141cc406Sopenharmony_ciImageBuffer::ImageBuffer(std::size_t size, ProducerCallback producer) :
30141cc406Sopenharmony_ci    producer_{producer},
31141cc406Sopenharmony_ci    size_{size}
32141cc406Sopenharmony_ci{
33141cc406Sopenharmony_ci    buffer_.resize(size_);
34141cc406Sopenharmony_ci}
35141cc406Sopenharmony_ci
36141cc406Sopenharmony_cibool ImageBuffer::get_data(std::size_t size, std::uint8_t* out_data)
37141cc406Sopenharmony_ci{
38141cc406Sopenharmony_ci    const std::uint8_t* out_data_end = out_data + size;
39141cc406Sopenharmony_ci
40141cc406Sopenharmony_ci    auto copy_buffer = [&]()
41141cc406Sopenharmony_ci    {
42141cc406Sopenharmony_ci        std::size_t bytes_copy = std::min<std::size_t>(out_data_end - out_data, available());
43141cc406Sopenharmony_ci        std::memcpy(out_data, buffer_.data() + buffer_offset_, bytes_copy);
44141cc406Sopenharmony_ci        out_data += bytes_copy;
45141cc406Sopenharmony_ci        buffer_offset_ += bytes_copy;
46141cc406Sopenharmony_ci    };
47141cc406Sopenharmony_ci
48141cc406Sopenharmony_ci    // first, read remaining data from buffer
49141cc406Sopenharmony_ci    if (available() > 0) {
50141cc406Sopenharmony_ci        copy_buffer();
51141cc406Sopenharmony_ci    }
52141cc406Sopenharmony_ci
53141cc406Sopenharmony_ci    if (out_data == out_data_end) {
54141cc406Sopenharmony_ci        return true;
55141cc406Sopenharmony_ci    }
56141cc406Sopenharmony_ci
57141cc406Sopenharmony_ci    // now the buffer is empty and there's more data to be read
58141cc406Sopenharmony_ci    bool got_data = true;
59141cc406Sopenharmony_ci    do {
60141cc406Sopenharmony_ci        buffer_offset_ = 0;
61141cc406Sopenharmony_ci
62141cc406Sopenharmony_ci        std::size_t size_to_read = size_;
63141cc406Sopenharmony_ci        if (remaining_size_ != BUFFER_SIZE_UNSET) {
64141cc406Sopenharmony_ci            size_to_read = std::min<std::uint64_t>(size_to_read, remaining_size_);
65141cc406Sopenharmony_ci            remaining_size_ -= size_to_read;
66141cc406Sopenharmony_ci        }
67141cc406Sopenharmony_ci
68141cc406Sopenharmony_ci        std::size_t aligned_size_to_read = size_to_read;
69141cc406Sopenharmony_ci        if (remaining_size_ == 0 && last_read_multiple_ != BUFFER_SIZE_UNSET) {
70141cc406Sopenharmony_ci            aligned_size_to_read = align_multiple_ceil(size_to_read, last_read_multiple_);
71141cc406Sopenharmony_ci        }
72141cc406Sopenharmony_ci
73141cc406Sopenharmony_ci        got_data &= producer_(aligned_size_to_read, buffer_.data());
74141cc406Sopenharmony_ci        curr_size_ = size_to_read;
75141cc406Sopenharmony_ci
76141cc406Sopenharmony_ci        copy_buffer();
77141cc406Sopenharmony_ci
78141cc406Sopenharmony_ci        if (remaining_size_ == 0 && out_data < out_data_end) {
79141cc406Sopenharmony_ci            got_data = false;
80141cc406Sopenharmony_ci        }
81141cc406Sopenharmony_ci
82141cc406Sopenharmony_ci    } while (out_data < out_data_end && got_data);
83141cc406Sopenharmony_ci
84141cc406Sopenharmony_ci    return got_data;
85141cc406Sopenharmony_ci}
86141cc406Sopenharmony_ci
87141cc406Sopenharmony_ci} // namespace genesys
88