/* 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 "common/std/algorithm.h"
#include "ags/shared/util/data_stream.h"
namespace AGS3 {
namespace AGS {
namespace Shared {
DataStream::DataStream(DataEndianess stream_endianess)
: _streamEndianess(stream_endianess) {
}
DataStream::~DataStream() {}
int16_t DataStream::ReadInt16() {
int16_t val = 0;
Read(&val, sizeof(int16_t));
ConvertInt16(val);
return val;
}
int32_t DataStream::ReadInt32() {
int32_t val = 0;
Read(&val, sizeof(int32_t));
ConvertInt32(val);
return val;
}
int64_t DataStream::ReadInt64() {
int64_t val = 0;
Read(&val, sizeof(int64_t));
ConvertInt64(val);
return val;
}
size_t DataStream::WriteInt16(int16_t val) {
ConvertInt16(val);
return Write(&val, sizeof(int16_t));
}
size_t DataStream::WriteInt32(int32_t val) {
ConvertInt32(val);
return Write(&val, sizeof(int32_t));
}
size_t DataStream::WriteInt64(int64_t val) {
ConvertInt64(val);
return Write(&val, sizeof(int64_t));
}
size_t DataStream::ReadAndConvertArrayOfInt16(int16_t *buffer, size_t count) {
count = ReadArray(buffer, sizeof(int16_t), count);
for (size_t i = 0; i < count; ++i, ++buffer) {
*buffer = BBOp::SwapBytesInt16(*buffer);
}
return count;
}
size_t DataStream::ReadAndConvertArrayOfInt32(int32_t *buffer, size_t count) {
count = ReadArray(buffer, sizeof(int32_t), count);
for (size_t i = 0; i < count; ++i, ++buffer) {
*buffer = BBOp::SwapBytesInt32(*buffer);
}
return count;
}
size_t DataStream::ReadAndConvertArrayOfInt64(int64_t *buffer, size_t count) {
count = ReadArray(buffer, sizeof(int64_t), count);
for (size_t i = 0; i < count; ++i, ++buffer) {
*buffer = BBOp::SwapBytesInt64(*buffer);
}
return count;
}
size_t DataStream::WriteAndConvertArrayOfInt16(const int16_t *buffer, size_t count) {
size_t elem;
for (elem = 0; elem < count; ++elem, ++buffer) {
int16_t val = *buffer;
ConvertInt16(val);
if (Write(&val, sizeof(int16_t)) < sizeof(int16_t)) {
break;
}
}
return elem;
}
size_t DataStream::WriteAndConvertArrayOfInt32(const int32_t *buffer, size_t count) {
size_t elem;
for (elem = 0; elem < count; ++elem, ++buffer) {
int32_t val = *buffer;
ConvertInt32(val);
if (Write(&val, sizeof(int32_t)) < sizeof(int32_t)) {
break;
}
}
return elem;
}
size_t DataStream::WriteAndConvertArrayOfInt64(const int64_t *buffer, size_t count) {
size_t elem;
for (elem = 0; elem < count; ++elem, ++buffer) {
int64_t val = *buffer;
ConvertInt64(val);
if (Write(&val, sizeof(int64_t)) < sizeof(int64_t)) {
break;
}
}
return elem;
}
DataStreamSection::DataStreamSection(Stream *base, soff_t start, soff_t end)
: _base(base) {
_start = std::max((soff_t)0, std::min(start, end));
_end = std::min(std::max((soff_t)0, end), base->GetLength());
soff_t pos = base->Seek(_start, kSeekBegin);
if (pos >= 0)
_position = pos;
else
_position = base->GetPosition();
}
size_t DataStreamSection::Read(void *buffer, size_t len) {
if (_position >= _end)
return 0;
len = std::min(len, static_cast(_end - _position));
_position += _base->Read(buffer, len);
return len;
}
int32_t DataStreamSection::ReadByte() {
if (_position >= _end)
return -1;
int32_t b = _base->ReadByte();
if (b >= 0)
_position++;
return b;
}
size_t DataStreamSection::Write(const void *buffer, size_t len) {
len = _base->Write(buffer, len);
_position += len;
_end = std::max(_end, _position); // we might be overwriting after seeking back
return len;
}
int32_t DataStreamSection::WriteByte(uint8_t b) {
int32_t rb = _base->WriteByte(b);
if (rb == b) {
_position++;
_end = std::max(_end, _position); // we might be overwriting after seeking back
}
return rb;
}
soff_t DataStreamSection::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;
}
want_pos = std::min(std::max(want_pos, _start), _end);
soff_t new_pos = _base->Seek(want_pos, kSeekBegin);
if (new_pos >= 0) // the position remains in case of seek error
_position = want_pos;
return _position - _start; // convert to a stream section pos
}
void DataStreamSection::Close() {
// We do not close nor delete the base stream, but release it from use
_base = nullptr;
}
} // namespace Shared
} // namespace AGS
} // namespace AGS3