mpd/src/io/BufferedReader.cxx

131 lines
3.1 KiB
C++
Raw Normal View History

/*
* Copyright 2014-2022 Max Kellermann <max.kellermann@gmail.com>
*
2019-12-16 17:02:35 +01:00
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
2019-12-16 17:02:35 +01:00
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
2019-12-16 17:02:35 +01:00
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "BufferedReader.hxx"
#include "Reader.hxx"
#include "util/TextFile.hxx"
2023-03-05 08:43:32 +01:00
#include <algorithm> // for std::copy_n()
2021-12-07 11:49:59 +01:00
#include <cassert>
#include <cstdint>
#include <stdexcept>
bool
BufferedReader::Fill(bool need_more)
{
if (eof)
return !need_more;
auto w = buffer.Write();
if (w.empty()) {
if (buffer.GetCapacity() >= MAX_SIZE)
return !need_more;
buffer.Grow(buffer.GetCapacity() * 2);
w = buffer.Write();
assert(!w.empty());
}
std::size_t nbytes = reader.Read(w.data(), w.size());
if (nbytes == 0) {
eof = true;
return !need_more;
}
buffer.Append(nbytes);
return true;
}
void *
2021-12-07 11:49:59 +01:00
BufferedReader::ReadFull(std::size_t size)
{
while (true) {
auto r = Read();
if (r.size() >= size)
return r.data();
if (!Fill(true))
throw std::runtime_error("Premature end of file");
}
}
2021-12-07 11:49:59 +01:00
std::size_t
BufferedReader::ReadFromBuffer(std::span<std::byte> dest) noexcept
{
const auto src = Read();
std::size_t nbytes = std::min(src.size(), dest.size());
std::copy_n(src.data(), nbytes, dest.data());
Consume(nbytes);
return nbytes;
}
void
BufferedReader::ReadFull(std::span<std::byte> dest)
{
while (true) {
std::size_t nbytes = ReadFromBuffer(dest);
dest = dest.subspan(nbytes);
if (dest.empty())
break;
if (!Fill(true))
throw std::runtime_error("Premature end of file");
}
}
char *
BufferedReader::ReadLine()
{
do {
char *line = ReadBufferedLine(buffer);
if (line != nullptr) {
++line_number;
return line;
}
} while (Fill(true));
if (!eof || buffer.empty())
return nullptr;
auto w = buffer.Write();
if (w.empty()) {
buffer.Grow(buffer.GetCapacity() + 1);
w = buffer.Write();
assert(!w.empty());
}
/* terminate the last line */
w[0] = 0;
char *line = buffer.Read().data();
buffer.Clear();
++line_number;
return line;
}