ScummVM API documentation
stream.h
1 /* ScummVM - Graphic Adventure Engine
2  *
3  * ScummVM is the legal property of its developers, whose names
4  * are too numerous to list here. Please refer to the COPYRIGHT
5  * file distributed with this source distribution.
6  *
7  * This program is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see <http://www.gnu.org/licenses/>.
19  *
20  */
21 
22 //=============================================================================
23 //
24 // Base stream class.
25 //
26 // Provides default implementation for a few helper methods.
27 //
28 // Only streams with uncommon behavior should be derived directly from Stream.
29 // Most I/O devices should inherit DataStream instead.
30 // Streams that wrap other streams should inherit ProxyStream.
31 //
32 //=============================================================================
33 
34 #ifndef AGS_SHARED_UTIL_STREAM_H
35 #define AGS_SHARED_UTIL_STREAM_H
36 
37 #include "ags/shared/util/iags_stream.h"
38 #include "ags/lib/allegro/file.h"
39 #include "common/stream.h"
40 #include "common/types.h"
41 
42 namespace AGS3 {
43 namespace AGS {
44 namespace Shared {
45 
46 // TODO: merge with FileWorkMode (historical mistake)
47 enum StreamWorkMode {
48  kStream_Read,
49  kStream_Write
50 };
51 
52 class Stream : public IAGSStream {
53 public:
54  virtual ~Stream() {}
55 
56  // Tells if the stream has errors
57  virtual bool HasErrors() const {
58  return false;
59  }
60  // Flush stream buffer to the underlying device
61  virtual bool Flush() = 0;
62 
63  //-----------------------------------------------------
64  // Helper methods
65  //-----------------------------------------------------
66  int8_t ReadInt8() override {
67  return ReadByte();
68  }
69 
70  size_t WriteInt8(int8_t val) override {
71  int32_t ival = WriteByte(val);
72  return ival >= 0 ? ival : 0;
73  }
74 
75  bool ReadBool() override {
76  return ReadInt8() != 0;
77  }
78 
79  size_t WriteBool(bool val) override {
80  return WriteInt8(val ? 1 : 0);
81  }
82 
83  // Practically identical to Read() and Write(), these two helpers' only
84  // meaning is to underline the purpose of data being (de)serialized
85  size_t ReadArrayOfInt8(int8_t *buffer, size_t count) override {
86  return Read(buffer, count);
87  }
88  size_t WriteArrayOfInt8(const int8_t *buffer, size_t count) override {
89  return Write(buffer, count);
90  }
91 
92  // Fill the requested number of bytes with particular value
93  size_t WriteByteCount(uint8_t b, size_t count);
94 };
95 
97 private:
98  IAGSStream *_stream;
99  DisposeAfterUse::Flag _disposeAfterUse;
100 public:
101  ScummVMReadStream(IAGSStream *src, DisposeAfterUse::Flag disposeAfterUse =
102  DisposeAfterUse::YES) : _stream(src), _disposeAfterUse(disposeAfterUse) {
103  }
104  ~ScummVMReadStream() override {
105  if (_disposeAfterUse == DisposeAfterUse::YES)
106  delete _stream;
107  }
108 
109  bool eos() const override {
110  return _stream->EOS();
111  }
112 
113  uint32 read(void *dataPtr, uint32 dataSize) override {
114  return _stream->Read(dataPtr, dataSize);
115  }
116 
117  int64 pos() const override {
118  return _stream->GetPosition();
119  }
120 
121  int64 size() const override {
122  return _stream->GetLength();
123  }
124 
125  bool seek(int64 offset, int whence = SEEK_SET) override {
126  StreamSeek origin = kSeekBegin;
127  if (whence == SEEK_CUR)
128  origin = kSeekCurrent;
129  if (whence == SEEK_END)
130  origin = kSeekEnd;
131 
132  return _stream->Seek(offset, origin);
133  }
134 };
135 
136 
138 private:
139  PACKFILE *_file;
140  DisposeAfterUse::Flag _disposeAfterUse;
141 public:
142  ScummVMPackReadStream(PACKFILE *src, DisposeAfterUse::Flag disposeAfterUse =
143  DisposeAfterUse::YES) : _file(src), _disposeAfterUse(disposeAfterUse) {
144  }
145  ~ScummVMPackReadStream() override {
146  if (_disposeAfterUse == DisposeAfterUse::YES)
147  delete _file;
148  }
149 
150  bool eos() const override {
151  return _file->pack_feof();
152  }
153 
154  uint32 read(void *dataPtr, uint32 dataSize) override {
155  return _file->pack_fread(dataPtr, dataSize);
156  }
157 
158  int64 pos() const override {
159  error("Unsupported");
160  }
161 
162  int64 size() const override {
163  error("Unsupported");
164  }
165 
166  bool seek(int64 offset, int whence = SEEK_SET) override {
167  error("Unsupported");
168  }
169 };
170 
171 class StreamScummVMFile : public Stream {
172 private:
174  DisposeAfterUse::Flag _disposeAfterUse;
175 public:
177  DisposeAfterUse::Flag disposeAfterUse = DisposeAfterUse::NO) :
178  _stream(stream), _disposeAfterUse(disposeAfterUse) {
179  }
180  ~StreamScummVMFile() override {
181  Close();
182  }
183 
184  void Close() override {
185  if (_disposeAfterUse == DisposeAfterUse::YES)
186  delete _stream;
187  _stream = nullptr;
188  }
189 
190  bool IsValid() const override {
191  return _stream != nullptr;
192  }
193  bool EOS() const override {
194  return _stream->eos();
195  }
196  soff_t GetLength() const override {
197  return _stream->size();
198  }
199  soff_t GetPosition() const override {
200  return _stream->pos();
201  }
202  bool CanRead() const override {
203  return true;
204  }
205  bool CanWrite() const override {
206  return false;
207  }
208  bool CanSeek() const override {
209  return true;
210  }
211 
212  size_t Read(void *buffer, size_t size) override {
213  return _stream->read(buffer, size);
214  }
215  int32_t ReadByte() override {
216  return _stream->readByte();
217  }
218  size_t Write(const void *buffer, size_t size) override {
219  return 0;
220  }
221  int32_t WriteByte(uint8_t b) override {
222  return 0;
223  }
224 
225  int8_t ReadInt8() override {
226  return (int8)_stream->readByte();
227  }
228  int16_t ReadInt16() override {
229  return _stream->readSint16LE();
230  }
231  int32_t ReadInt32() override {
232  return _stream->readSint32LE();
233  }
234  int64_t ReadInt64() override {
235  return _stream->readSint64LE();
236  }
237  bool ReadBool() override {
238  return _stream->readByte() != 0;
239  }
240  size_t ReadArray(void *buffer, size_t elem_size, size_t count) override {
241  return _stream->read(buffer, elem_size * count) / elem_size;
242  }
243  size_t ReadArrayOfInt8(int8_t *buffer, size_t count) override {
244  return _stream->read(buffer, count);
245  }
246  size_t ReadArrayOfInt16(int16_t *buffer, size_t count) override {
247  for (size_t i = 0; i < count; ++i)
248  *buffer++ = _stream->readSint16LE();
249  return count;
250  }
251  size_t ReadArrayOfInt32(int32_t *buffer, size_t count) override {
252  for (size_t i = 0; i < count; ++i)
253  *buffer++ = _stream->readSint32LE();
254  return count;
255  }
256  size_t ReadArrayOfInt64(int64_t *buffer, size_t count) override {
257  for (size_t i = 0; i < count; ++i)
258  *buffer++ = _stream->readSint64LE();
259  return count;
260  }
261 
262  size_t WriteInt8(int8_t val) override {
263  return 0;
264  }
265  size_t WriteInt16(int16_t val) override {
266  return 0;
267  }
268  size_t WriteInt32(int32_t val) override {
269  return 0;
270  }
271  size_t WriteInt64(int64_t val) override {
272  return 0;
273  }
274  size_t WriteBool(bool val) override {
275  return 0;
276  }
277  size_t WriteArray(const void *buffer, size_t elem_size, size_t count) override {
278  return 0;
279  }
280  size_t WriteArrayOfInt8(const int8_t *buffer, size_t count) override {
281  return 0;
282  }
283  size_t WriteArrayOfInt16(const int16_t *buffer, size_t count) override {
284  return 0;
285  }
286  size_t WriteArrayOfInt32(const int32_t *buffer, size_t count) override {
287  return 0;
288  }
289  size_t WriteArrayOfInt64(const int64_t *buffer, size_t count) override {
290  return 0;
291  }
292 
293  bool Seek(soff_t offset, StreamSeek origin = kSeekCurrent) override {
294  return _stream->seek(offset, origin);
295  }
296 
297  bool HasErrors() const override {
298  return _stream->err();
299  }
300  bool Flush() override {
301  return true;
302  }
303 };
304 
305 } // namespace Shared
306 } // namespace AGS
307 } // namespace AGS3
308 
309 #endif
Definition: achievements_tables.h:27
uint32 read(void *dataPtr, uint32 dataSize) override
Definition: stream.h:113
virtual int64 size() const =0
uint32 read(void *dataPtr, uint32 dataSize) override
Definition: stream.h:154
virtual int64 pos() const =0
virtual bool err() const
Definition: stream.h:61
FORCEINLINE int32 readSint32LE()
Definition: stream.h:555
int64 size() const override
Definition: stream.h:121
Definition: file.h:62
Definition: iags_stream.h:53
virtual bool seek(int64 offset, int whence=SEEK_SET)=0
virtual bool eos() const =0
bool eos() const override
Definition: stream.h:150
Definition: stream.h:745
bool eos() const override
Definition: stream.h:109
byte readByte()
Definition: stream.h:434
bool seek(int64 offset, int whence=SEEK_SET) override
Definition: stream.h:125
int64 size() const override
Definition: stream.h:162
FORCEINLINE int64 readSint64LE()
Definition: stream.h:567
bool seek(int64 offset, int whence=SEEK_SET) override
Definition: stream.h:166
int64 pos() const override
Definition: stream.h:158
Definition: stream.h:171
void NORETURN_PRE error(MSVC_PRINTF const char *s,...) GCC_PRINTF(1
FORCEINLINE int16 readSint16LE()
Definition: stream.h:543
int64 pos() const override
Definition: stream.h:117
Definition: stream.h:52
Definition: ags.h:40
virtual uint32 read(void *dataPtr, uint32 dataSize)=0