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