/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
*/
//include
#include "common/std/algorithm.h"
#include "ags/shared/util/buffered_stream.h"
#include "ags/shared/util/stdio_compat.h"
#include "ags/shared/util/string.h"
namespace AGS3 {
namespace AGS {
namespace Shared {
//-----------------------------------------------------------------------------
// BufferedStream
//-----------------------------------------------------------------------------
const size_t BufferedStream::BufferSize;
BufferedStream::BufferedStream(const String &file_name, FileOpenMode open_mode, FileWorkMode work_mode, DataEndianess stream_endianess)
: FileStream(file_name, open_mode, work_mode, stream_endianess) {
if (IsValid()) {
soff_t end_pos = FileStream::Seek(0, kSeekEnd);
if (end_pos >= 0) {
_start = 0;
_end = end_pos;
if (FileStream::Seek(0, kSeekBegin) < 0)
_end = -1;
}
if (_end == -1) {
FileStream::Close();
error("Error determining stream end.");
}
}
}
BufferedStream::~BufferedStream() {
BufferedStream::Close();
}
void BufferedStream::FillBufferFromPosition(soff_t position) {
FileStream::Seek(position, kSeekBegin);
// remember to restrict to the end position!
size_t fill_size = std::min(BufferSize, static_cast(_end - position));
_buffer.resize(fill_size);
auto sz = FileStream::Read(_buffer.data(), fill_size);
_buffer.resize(sz);
_bufferPosition = position;
}
void BufferedStream::FlushBuffer(soff_t position) {
size_t sz = _buffer.size() > 0 ? FileStream::Write(_buffer.data(), _buffer.size()) : 0u;
_buffer.clear(); // will start from the clean buffer next time
_bufferPosition += sz;
if (position != _bufferPosition) {
FileStream::Seek(position, kSeekBegin);
_bufferPosition = position;
}
}
bool BufferedStream::EOS() const {
return _position == _end;
}
soff_t BufferedStream::GetLength() const {
return _end - _start;
}
soff_t BufferedStream::GetPosition() const {
return _position - _start;
}
void BufferedStream::Close() {
if (GetWorkMode() == kFile_Write)
FlushBuffer(_position);
FileStream::Close();
}
bool BufferedStream::Flush() {
if (GetWorkMode() == kFile_Write)
FlushBuffer(_position);
return FileStream::Flush();
}
size_t BufferedStream::Read(void *buffer, size_t size) {
// If the read size is larger than the internal buffer size,
// then read directly into the user buffer and bail out.
if (size >= BufferSize) {
FileStream::Seek(_position, kSeekBegin);
// remember to restrict to the end position!
size_t fill_size = std::min(size, static_cast(_end - _position));
size_t sz = FileStream::Read(buffer, fill_size);
_position += sz;
return sz;
}
auto *to = static_cast(buffer);
while(size > 0) {
if (_position < _bufferPosition || _position >= _bufferPosition + _buffer.size()) {
FillBufferFromPosition(_position);
}
if (_buffer.empty()) { break; } // reached EOS
assert(_position >= _bufferPosition && _position < _bufferPosition + _buffer.size());
soff_t bufferOffset = _position - _bufferPosition;
assert(bufferOffset >= 0);
size_t bytesLeft = _buffer.size() - (size_t)bufferOffset;
size_t chunkSize = MIN(bytesLeft, size);
memcpy(to, _buffer.data() + bufferOffset, chunkSize);
to += chunkSize;
_position += chunkSize;
size -= chunkSize;
}
return to - static_cast(buffer);
}
int32_t BufferedStream::ReadByte() {
uint8_t ch;
auto bytesRead = Read(&ch, 1);
if (bytesRead != 1) {
return EOF;
}
return ch;
}
size_t BufferedStream::Write(const void *buffer, size_t size) {
const uint8_t *from = static_cast(buffer);
while (size > 0) {
if (_position < _bufferPosition || // seek'd before buffer pos
_position > _bufferPosition + _buffer.size() || // seek'd beyond buffer pos
_position >= _bufferPosition + (soff_t) BufferSize) // seek'd, or exceeded buffer limit
{
FlushBuffer(_position);
}
size_t pos_in_buff = static_cast(_position - _bufferPosition);
size_t chunk_sz = std::min(size, BufferSize - pos_in_buff);
if (_buffer.size() < pos_in_buff + chunk_sz)
_buffer.resize(pos_in_buff + chunk_sz);
memcpy(_buffer.data() + pos_in_buff, from, chunk_sz);
_position += chunk_sz;
from += chunk_sz;
size -= chunk_sz;
}
_end = std::max(_end, _position);
return from - static_cast(buffer);
}
int32_t BufferedStream::WriteByte(uint8_t val) {
auto sz = Write(&val, 1);
if (sz != 1) {
return -1;
}
return val;
}
soff_t BufferedStream::Seek(soff_t offset, StreamSeek origin) {
soff_t want_pos = -1;
switch (origin) {
case StreamSeek::kSeekCurrent: want_pos = _position + offset; break;
case StreamSeek::kSeekBegin: want_pos = _start + offset; break;
case StreamSeek::kSeekEnd: want_pos = _end + offset; break;
default: return -1;
}
// clamp to the valid range
_position = MIN(MAX(want_pos, _start), _end);
return _position - _start; // convert to a stream section pos
}
//-----------------------------------------------------------------------------
// BufferedSectionStream
//-----------------------------------------------------------------------------
BufferedSectionStream::BufferedSectionStream(const String &file_name, soff_t start_pos, soff_t end_pos,
FileOpenMode open_mode, FileWorkMode work_mode, DataEndianess stream_endianess)
: BufferedStream(file_name, open_mode, work_mode, stream_endianess) {
assert(start_pos <= end_pos);
start_pos = MIN(start_pos, end_pos);
_start = MIN(start_pos, _end);
_end = MIN(end_pos, _end);
BufferedStream::Seek(0, kSeekBegin);
}
} // namespace Shared
} // namespace AGS
} // namespace AGS3