1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317
|
/* 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/>.
*
*/
//=============================================================================
//
// Base stream class.
//
// Provides default implementation for a few helper methods.
//
// Only streams with uncommon behavior should be derived directly from Stream.
// Most I/O devices should inherit DataStream instead.
//
//=============================================================================
#ifndef AGS_SHARED_UTIL_STREAM_H
#define AGS_SHARED_UTIL_STREAM_H
#include "ags/shared/util/iags_stream.h"
#include "ags/shared/util/string.h"
#include "ags/lib/allegro/file.h"
#include "common/stream.h"
#include "common/types.h"
namespace AGS3 {
namespace AGS {
namespace Shared {
// TODO: merge with FileWorkMode (historical mistake)
enum StreamWorkMode {
kStream_Read,
kStream_Write
};
class Stream : public IAGSStream {
public:
Stream() = default;
Stream(const String &path)
: _path(path) {}
virtual ~Stream() {}
// Returns an optional path of a stream's source, such as a filepath;
// primarily for diagnostic purposes
const String &GetPath() const { return _path; }
// Tells if there were errors during previous io operation(s);
// the call to GetError() *resets* the error record.
virtual bool GetError() const { return false; }
// Flush stream buffer to the underlying device
virtual bool Flush() = 0;
//-----------------------------------------------------
// Helper methods
//-----------------------------------------------------
int8_t ReadInt8() override {
return ReadByte();
}
size_t WriteInt8(int8_t val) override {
int32_t ival = WriteByte(val);
return ival >= 0 ? ival : 0;
}
bool ReadBool() override {
return ReadInt8() != 0;
}
size_t WriteBool(bool val) override {
return WriteInt8(val ? 1 : 0);
}
// Practically identical to Read() and Write(), these two helpers' only
// meaning is to underline the purpose of data being (de)serialized
size_t ReadArrayOfInt8(int8_t *buffer, size_t count) override {
return Read(buffer, count);
}
size_t WriteArrayOfInt8(const int8_t *buffer, size_t count) override {
return Write(buffer, count);
}
// Fill the requested number of bytes with particular value
size_t WriteByteCount(uint8_t b, size_t count);
protected:
String _path; // optional name of the stream's source (e.g. filepath)
};
class ScummVMReadStream : public Common::SeekableReadStream {
private:
IAGSStream *_stream;
DisposeAfterUse::Flag _disposeAfterUse;
public:
ScummVMReadStream(IAGSStream *src, DisposeAfterUse::Flag disposeAfterUse =
DisposeAfterUse::YES) : _stream(src), _disposeAfterUse(disposeAfterUse) {
}
~ScummVMReadStream() override {
if (_disposeAfterUse == DisposeAfterUse::YES)
delete _stream;
}
bool eos() const override {
return _stream->EOS();
}
uint32 read(void *dataPtr, uint32 dataSize) override {
return _stream->Read(dataPtr, dataSize);
}
int64 pos() const override {
return _stream->GetPosition();
}
int64 size() const override {
return _stream->GetLength();
}
bool seek(int64 offset, int whence = SEEK_SET) override {
StreamSeek origin = kSeekBegin;
if (whence == SEEK_CUR)
origin = kSeekCurrent;
if (whence == SEEK_END)
origin = kSeekEnd;
return _stream->Seek(offset, origin);
}
};
class ScummVMPackReadStream : public Common::SeekableReadStream {
private:
PACKFILE *_file;
DisposeAfterUse::Flag _disposeAfterUse;
public:
ScummVMPackReadStream(PACKFILE *src, DisposeAfterUse::Flag disposeAfterUse =
DisposeAfterUse::YES) : _file(src), _disposeAfterUse(disposeAfterUse) {
}
~ScummVMPackReadStream() override {
if (_disposeAfterUse == DisposeAfterUse::YES)
delete _file;
}
bool eos() const override {
return _file->pack_feof();
}
uint32 read(void *dataPtr, uint32 dataSize) override {
return _file->pack_fread(dataPtr, dataSize);
}
int64 pos() const override {
error("Unsupported");
}
int64 size() const override {
error("Unsupported");
}
bool seek(int64 offset, int whence = SEEK_SET) override {
error("Unsupported");
}
};
class StreamScummVMFile : public Stream {
private:
Common::SeekableReadStream *_stream;
DisposeAfterUse::Flag _disposeAfterUse;
public:
StreamScummVMFile(Common::SeekableReadStream *stream,
DisposeAfterUse::Flag disposeAfterUse = DisposeAfterUse::NO) :
_stream(stream), _disposeAfterUse(disposeAfterUse) {
}
~StreamScummVMFile() override {
Close();
}
void Close() override {
if (_disposeAfterUse == DisposeAfterUse::YES)
delete _stream;
_stream = nullptr;
}
bool IsValid() const override {
return _stream != nullptr;
}
bool EOS() const override {
return _stream->eos();
}
soff_t GetLength() const override {
return _stream->size();
}
soff_t GetPosition() const override {
return _stream->pos();
}
bool CanRead() const override {
return true;
}
bool CanWrite() const override {
return false;
}
bool CanSeek() const override {
return true;
}
size_t Read(void *buffer, size_t size) override {
return _stream->read(buffer, size);
}
int32_t ReadByte() override {
return _stream->readByte();
}
size_t Write(const void *buffer, size_t size) override {
return 0;
}
int32_t WriteByte(uint8_t b) override {
return 0;
}
int8_t ReadInt8() override {
return (int8)_stream->readByte();
}
int16_t ReadInt16() override {
return _stream->readSint16LE();
}
int32_t ReadInt32() override {
return _stream->readSint32LE();
}
int64_t ReadInt64() override {
return _stream->readSint64LE();
}
bool ReadBool() override {
return _stream->readByte() != 0;
}
size_t ReadArray(void *buffer, size_t elem_size, size_t count) override {
return _stream->read(buffer, elem_size * count) / elem_size;
}
size_t ReadArrayOfInt8(int8_t *buffer, size_t count) override {
return _stream->read(buffer, count);
}
size_t ReadArrayOfInt16(int16_t *buffer, size_t count) override {
for (size_t i = 0; i < count; ++i)
*buffer++ = _stream->readSint16LE();
return count;
}
size_t ReadArrayOfInt32(int32_t *buffer, size_t count) override {
for (size_t i = 0; i < count; ++i)
*buffer++ = _stream->readSint32LE();
return count;
}
size_t ReadArrayOfInt64(int64_t *buffer, size_t count) override {
for (size_t i = 0; i < count; ++i)
*buffer++ = _stream->readSint64LE();
return count;
}
size_t WriteInt8(int8_t val) override {
return 0;
}
size_t WriteInt16(int16_t val) override {
return 0;
}
size_t WriteInt32(int32_t val) override {
return 0;
}
size_t WriteInt64(int64_t val) override {
return 0;
}
size_t WriteBool(bool val) override {
return 0;
}
size_t WriteArray(const void *buffer, size_t elem_size, size_t count) override {
return 0;
}
size_t WriteArrayOfInt8(const int8_t *buffer, size_t count) override {
return 0;
}
size_t WriteArrayOfInt16(const int16_t *buffer, size_t count) override {
return 0;
}
size_t WriteArrayOfInt32(const int32_t *buffer, size_t count) override {
return 0;
}
size_t WriteArrayOfInt64(const int64_t *buffer, size_t count) override {
return 0;
}
soff_t Seek(soff_t offset, StreamSeek origin = kSeekCurrent) override {
return _stream->seek(offset, origin);
}
bool GetError() const override {
return _stream->err();
}
bool Flush() override {
return true;
}
};
} // namespace Shared
} // namespace AGS
} // namespace AGS3
#endif
|