163 lines
5.9 KiB
C++
163 lines
5.9 KiB
C++
/* 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
//=============================================================================
|
|
//
|
|
// Standard AGS stream implementation for reading raw data with support for
|
|
// converting to opposite endianess. Most I/O devices should inherit this
|
|
// class and provide implementation for basic reading and writing only.
|
|
//
|
|
//=============================================================================
|
|
|
|
#ifndef AGS_SHARED_UTIL_DATASTREAM_H
|
|
#define AGS_SHARED_UTIL_DATASTREAM_H
|
|
|
|
#include "ags/shared/util/bbop.h"
|
|
#include "ags/shared/util/stream.h"
|
|
|
|
namespace AGS3 {
|
|
namespace AGS {
|
|
namespace Shared {
|
|
|
|
class DataStream : public Stream {
|
|
public:
|
|
DataStream(DataEndianess stream_endianess = kLittleEndian);
|
|
~DataStream() override;
|
|
|
|
int16_t ReadInt16() override;
|
|
int32_t ReadInt32() override;
|
|
int64_t ReadInt64() override;
|
|
|
|
//
|
|
// Read- and WriteArray methods return number of full elements (NOT bytes)
|
|
// read or written, or -1 if end of stream is reached
|
|
//
|
|
// Note that ReadArray and WriteArray do NOT convert byte order even when
|
|
// work with data of different endianess; they are meant for optimal
|
|
// reading and writing blocks of raw bytes
|
|
inline size_t ReadArray(void *buffer, size_t elem_size, size_t count) override {
|
|
return Read(buffer, elem_size * count) / elem_size;
|
|
}
|
|
|
|
inline size_t ReadArrayOfInt16(int16_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
ReadAndConvertArrayOfInt16(buffer, count) : ReadArray(buffer, sizeof(int16_t), count);
|
|
}
|
|
|
|
inline size_t ReadArrayOfInt32(int32_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
ReadAndConvertArrayOfInt32(buffer, count) : ReadArray(buffer, sizeof(int32_t), count);
|
|
}
|
|
|
|
inline size_t ReadArrayOfInt64(int64_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
ReadAndConvertArrayOfInt64(buffer, count) : ReadArray(buffer, sizeof(int64_t), count);
|
|
}
|
|
|
|
size_t WriteInt16(int16_t val) override;
|
|
size_t WriteInt32(int32_t val) override;
|
|
size_t WriteInt64(int64_t val) override;
|
|
|
|
inline size_t WriteArray(const void *buffer, size_t elem_size, size_t count) override {
|
|
return Write(buffer, elem_size * count) / elem_size;
|
|
}
|
|
|
|
inline size_t WriteArrayOfInt16(const int16_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
WriteAndConvertArrayOfInt16(buffer, count) : WriteArray(buffer, sizeof(int16_t), count);
|
|
}
|
|
|
|
inline size_t WriteArrayOfInt32(const int32_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
WriteAndConvertArrayOfInt32(buffer, count) : WriteArray(buffer, sizeof(int32_t), count);
|
|
}
|
|
|
|
inline size_t WriteArrayOfInt64(const int64_t *buffer, size_t count) override {
|
|
return MustSwapBytes() ?
|
|
WriteAndConvertArrayOfInt64(buffer, count) : WriteArray(buffer, sizeof(int64_t), count);
|
|
}
|
|
|
|
protected:
|
|
DataEndianess _streamEndianess;
|
|
|
|
// Helper methods for reading/writing arrays of basic types and
|
|
// converting their elements to opposite endianess (swapping bytes).
|
|
size_t ReadAndConvertArrayOfInt16(int16_t *buffer, size_t count);
|
|
size_t ReadAndConvertArrayOfInt32(int32_t *buffer, size_t count);
|
|
size_t ReadAndConvertArrayOfInt64(int64_t *buffer, size_t count);
|
|
size_t WriteAndConvertArrayOfInt16(const int16_t *buffer, size_t count);
|
|
size_t WriteAndConvertArrayOfInt32(const int32_t *buffer, size_t count);
|
|
size_t WriteAndConvertArrayOfInt64(const int64_t *buffer, size_t count);
|
|
|
|
inline bool MustSwapBytes() {
|
|
return kDefaultSystemEndianess != _streamEndianess;
|
|
}
|
|
|
|
inline void ConvertInt16(int16_t &val) {
|
|
if (MustSwapBytes()) val = BBOp::SwapBytesInt16(val);
|
|
}
|
|
inline void ConvertInt32(int32_t &val) {
|
|
if (MustSwapBytes()) val = BBOp::SwapBytesInt32(val);
|
|
}
|
|
inline void ConvertInt64(int64_t &val) {
|
|
if (MustSwapBytes()) val = BBOp::SwapBytesInt64(val);
|
|
}
|
|
};
|
|
|
|
//
|
|
// DataStreamSection wraps another stream and restricts its access
|
|
// to a particular range of offsets of the base stream.
|
|
// StreamSection does NOT own the base stream, and closing
|
|
// a "stream section" does NOT close the base stream.
|
|
// Base stream must stay in memory for as long as there are
|
|
// "stream sections" referring it.
|
|
class DataStreamSection : public DataStream {
|
|
public:
|
|
// Constructs a StreamSection over a base stream,
|
|
// restricting working range to [start, end), i.e. end offset is
|
|
// +1 past allowed position.
|
|
DataStreamSection(Stream *base, soff_t start, soff_t end);
|
|
const char *GetPath() const { return _base->GetPath().GetCStr(); }
|
|
bool EOS() const override { return _position >= _end; }
|
|
bool GetError() const override { return _base->GetError(); }
|
|
soff_t GetLength() const override { return _end - _start; }
|
|
soff_t GetPosition() const override { return _position - _start; }
|
|
size_t Read(void *buffer, size_t len) override;
|
|
int32_t ReadByte() override;
|
|
size_t Write(const void *buffer, size_t len) override;
|
|
int32_t WriteByte(uint8_t b) override;
|
|
soff_t Seek(soff_t offset, StreamSeek origin = kSeekCurrent) override;
|
|
bool Flush() override { return _base->Flush(); }
|
|
void Close() override;
|
|
|
|
private:
|
|
Stream *_base = nullptr;
|
|
soff_t _start = 0;
|
|
soff_t _end = 0;
|
|
soff_t _position = 0;
|
|
};
|
|
|
|
} // namespace Shared
|
|
} // namespace AGS
|
|
} // namespace AGS3
|
|
|
|
#endif
|