2004-04-17 09:57:15 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2006-01-18 17:39:49 +00:00
|
|
|
* Copyright (C) 2004-2006 The ScummVM project
|
2004-04-17 09:57:15 +00:00
|
|
|
*
|
|
|
|
* 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 2
|
|
|
|
* 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, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2004-04-17 09:57:15 +00:00
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef COMMON_STREAM_H
|
|
|
|
#define COMMON_STREAM_H
|
|
|
|
|
2005-06-24 15:23:51 +00:00
|
|
|
#include "common/stdafx.h"
|
2004-04-17 09:57:15 +00:00
|
|
|
#include "common/scummsys.h"
|
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
class String;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Virtual base class for both ReadStream and WriteStream.
|
|
|
|
*/
|
|
|
|
class Stream {
|
|
|
|
public:
|
2005-06-20 17:59:00 +00:00
|
|
|
virtual ~Stream() {}
|
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
/**
|
|
|
|
* Returns true if any I/O failure occured.
|
|
|
|
* This flag is never cleared automatically. In order to clear it,
|
|
|
|
* client code has to call clearIOFailed() explicitly.
|
2005-07-30 21:11:48 +00:00
|
|
|
*
|
2005-04-22 17:40:09 +00:00
|
|
|
* @todo Instead of returning a plain bool, maybe we should define
|
|
|
|
* a list of error codes which can be returned here.
|
|
|
|
*/
|
|
|
|
virtual bool ioFailed() const { return false; }
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
/**
|
|
|
|
* Reset the I/O error status.
|
|
|
|
*/
|
|
|
|
virtual void clearIOFailed() {}
|
|
|
|
};
|
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
/**
|
|
|
|
* Generic interface for a writable data stream.
|
|
|
|
*/
|
2005-04-22 17:40:09 +00:00
|
|
|
class WriteStream : virtual public Stream {
|
2004-04-17 09:57:15 +00:00
|
|
|
public:
|
2004-12-11 23:16:36 +00:00
|
|
|
/**
|
|
|
|
* Write data into the stream. Subclasses must implement this
|
|
|
|
* method; all other write methods are implemented using it.
|
|
|
|
*
|
2005-05-05 15:59:24 +00:00
|
|
|
* @param dataPtr pointer to the data to be written
|
|
|
|
* @param dataSize number of bytes to be written
|
2004-12-11 23:16:36 +00:00
|
|
|
* @return the number of bytes which were actually written.
|
|
|
|
*/
|
2005-05-05 15:59:24 +00:00
|
|
|
virtual uint32 write(const void *dataPtr, uint32 dataSize) = 0;
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2005-10-13 18:50:53 +00:00
|
|
|
/**
|
|
|
|
* Commit any buffered data to the underlying channel or
|
|
|
|
* storage medium; unbuffered streams can use the default
|
|
|
|
* implementation.
|
|
|
|
*/
|
|
|
|
virtual void flush() {}
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
// The remaining methods all have default implementations; subclasses
|
|
|
|
// need not (and should not) overload them.
|
|
|
|
|
|
|
|
void writeByte(byte value) {
|
|
|
|
write(&value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeSByte(int8 value) {
|
|
|
|
write(&value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeUint16LE(uint16 value) {
|
|
|
|
writeByte((byte)(value & 0xff));
|
|
|
|
writeByte((byte)(value >> 8));
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeUint32LE(uint32 value) {
|
|
|
|
writeUint16LE((uint16)(value & 0xffff));
|
|
|
|
writeUint16LE((uint16)(value >> 16));
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeUint16BE(uint16 value) {
|
|
|
|
writeByte((byte)(value >> 8));
|
|
|
|
writeByte((byte)(value & 0xff));
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeUint32BE(uint32 value) {
|
|
|
|
writeUint16BE((uint16)(value >> 16));
|
|
|
|
writeUint16BE((uint16)(value & 0xffff));
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeSint16LE(int16 value) {
|
|
|
|
writeUint16LE((uint16)value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void writeSint32LE(int32 value) {
|
|
|
|
writeUint32LE((uint32)value);
|
|
|
|
}
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
void writeSint16BE(int16 value) {
|
|
|
|
writeUint16BE((uint16)value);
|
|
|
|
}
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
void writeSint32BE(int32 value) {
|
|
|
|
writeUint32BE((uint32)value);
|
|
|
|
}
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
void writeString(const String &str);
|
2004-04-17 09:57:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
/**
|
|
|
|
* Generic interface for a readable data stream.
|
|
|
|
*/
|
2005-04-22 17:40:09 +00:00
|
|
|
class ReadStream : virtual public Stream {
|
2004-04-17 09:57:15 +00:00
|
|
|
public:
|
2005-04-28 20:59:19 +00:00
|
|
|
/**
|
|
|
|
* Returns true if the end of the stream has been reached.
|
|
|
|
*/
|
|
|
|
virtual bool eos() const = 0;
|
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
/**
|
|
|
|
* Read data from the stream. Subclasses must implement this
|
|
|
|
* method; all other read methods are implemented using it.
|
|
|
|
*
|
2005-05-05 15:59:24 +00:00
|
|
|
* @param dataPtr pointer to a buffer into which the data is read
|
|
|
|
* @param dataSize number of bytes to be read
|
2004-12-11 23:16:36 +00:00
|
|
|
* @return the number of bytes which were actually read.
|
|
|
|
*/
|
2005-05-05 15:59:24 +00:00
|
|
|
virtual uint32 read(void *dataPtr, uint32 dataSize) = 0;
|
2004-04-17 09:57:15 +00:00
|
|
|
|
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
// The remaining methods all have default implementations; subclasses
|
|
|
|
// need not (and should not) overload them.
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
byte readByte() {
|
|
|
|
byte b = 0;
|
|
|
|
read(&b, 1);
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
int8 readSByte() {
|
|
|
|
int8 b = 0;
|
|
|
|
read(&b, 1);
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16 readUint16LE() {
|
|
|
|
uint16 a = readByte();
|
|
|
|
uint16 b = readByte();
|
|
|
|
return a | (b << 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 readUint32LE() {
|
|
|
|
uint32 a = readUint16LE();
|
|
|
|
uint32 b = readUint16LE();
|
|
|
|
return (b << 16) | a;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16 readUint16BE() {
|
|
|
|
uint16 b = readByte();
|
|
|
|
uint16 a = readByte();
|
|
|
|
return a | (b << 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 readUint32BE() {
|
|
|
|
uint32 b = readUint16BE();
|
|
|
|
uint32 a = readUint16BE();
|
|
|
|
return (b << 16) | a;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16 readSint16LE() {
|
|
|
|
return (int16)readUint16LE();
|
|
|
|
}
|
|
|
|
|
|
|
|
int32 readSint32LE() {
|
|
|
|
return (int32)readUint32LE();
|
|
|
|
}
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
int16 readSint16BE() {
|
|
|
|
return (int16)readUint16BE();
|
|
|
|
}
|
2004-04-17 09:57:15 +00:00
|
|
|
|
2004-12-11 23:16:36 +00:00
|
|
|
int32 readSint32BE() {
|
|
|
|
return (int32)readUint32BE();
|
|
|
|
}
|
2004-04-17 09:57:15 +00:00
|
|
|
};
|
|
|
|
|
2005-01-09 01:41:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Interface for a seekable & readable data stream.
|
|
|
|
*
|
|
|
|
* @todo We really need better error handling here!
|
|
|
|
* Like seek should somehow indicate whether it failed.
|
|
|
|
*/
|
|
|
|
class SeekableReadStream : public ReadStream {
|
|
|
|
public:
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-01-09 01:41:43 +00:00
|
|
|
virtual uint32 pos() const = 0;
|
|
|
|
virtual uint32 size() const = 0;
|
|
|
|
|
|
|
|
virtual void seek(int32 offs, int whence = SEEK_SET) = 0;
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
/**
|
|
|
|
* Read one line of text from a CR or CR/LF terminated plain text file.
|
2005-07-30 21:11:48 +00:00
|
|
|
* This method is a rough analog of the (f)gets function.
|
|
|
|
*
|
2005-04-22 17:40:09 +00:00
|
|
|
* @param buf the buffer to store into
|
2005-05-05 15:59:24 +00:00
|
|
|
* @param bufSize the size of the buffer
|
2005-04-22 17:40:09 +00:00
|
|
|
* @return a pointer to the read string, or NULL if an error occured
|
|
|
|
* @note The line terminator (CR or CR/LF) is stripped and not inserted
|
|
|
|
* into the buffer.
|
|
|
|
*/
|
|
|
|
virtual char *readLine(char *buf, size_t bufSize);
|
2005-01-09 01:41:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2004-04-17 09:57:15 +00:00
|
|
|
/**
|
|
|
|
* XORReadStream is a wrapper around an arbitrary other ReadStream,
|
|
|
|
* which 'decrypts' the data being read by XORing all data bytes with the given
|
|
|
|
* encryption 'key'.
|
2004-04-17 16:29:03 +00:00
|
|
|
*
|
|
|
|
* Currently, this is not used anywhere, it's just a demo of how one can chain
|
|
|
|
* streams if necessary.
|
2004-04-17 09:57:15 +00:00
|
|
|
*/
|
|
|
|
class XORReadStream : public ReadStream {
|
|
|
|
private:
|
|
|
|
byte _encbyte;
|
|
|
|
ReadStream *_realStream;
|
|
|
|
public:
|
2004-07-19 10:20:00 +00:00
|
|
|
XORReadStream(ReadStream *in = 0, byte enc = 0) : _encbyte(enc), _realStream(in) {}
|
2004-04-17 16:29:03 +00:00
|
|
|
void setStream(ReadStream *in) { _realStream = in; }
|
2004-04-17 09:57:15 +00:00
|
|
|
void setEnc(byte value) { _encbyte = value; }
|
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
virtual bool eos() const { return _realStream->eos(); }
|
|
|
|
virtual bool ioFailed() const { return _realStream->ioFailed(); }
|
|
|
|
virtual void clearIOFailed() { _realStream->clearIOFailed(); }
|
|
|
|
|
2005-05-05 15:59:24 +00:00
|
|
|
uint32 read(void *dataPtr, uint32 dataSize) {
|
2004-04-17 16:29:03 +00:00
|
|
|
assert(_realStream);
|
2005-05-05 15:59:24 +00:00
|
|
|
uint32 len = _realStream->read(dataPtr, dataSize);
|
2004-04-17 09:57:15 +00:00
|
|
|
if (_encbyte) {
|
2005-05-05 15:59:24 +00:00
|
|
|
byte *p = (byte *)dataPtr;
|
2004-04-17 09:57:15 +00:00
|
|
|
byte *end = p + len;
|
|
|
|
while (p < end)
|
|
|
|
*p++ ^= _encbyte;
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2004-04-17 16:29:03 +00:00
|
|
|
/**
|
|
|
|
* Simple memory based 'stream', which implements the ReadStream interface for
|
|
|
|
* a plain memory block.
|
|
|
|
*/
|
2005-01-09 01:41:43 +00:00
|
|
|
class MemoryReadStream : public SeekableReadStream {
|
2004-04-17 09:57:15 +00:00
|
|
|
private:
|
|
|
|
const byte *_ptr;
|
2004-06-27 21:30:33 +00:00
|
|
|
const byte * const _ptrOrig;
|
|
|
|
const uint32 _bufSize;
|
2004-05-04 03:27:00 +00:00
|
|
|
uint32 _pos;
|
2005-03-30 21:59:12 +00:00
|
|
|
byte _encbyte;
|
|
|
|
|
2004-04-17 09:57:15 +00:00
|
|
|
public:
|
2005-03-30 22:01:17 +00:00
|
|
|
MemoryReadStream(const byte *buf, uint32 len) : _ptr(buf), _ptrOrig(buf), _bufSize(len), _pos(0), _encbyte(0) {}
|
2004-06-27 21:30:33 +00:00
|
|
|
|
2005-03-30 21:59:12 +00:00
|
|
|
void setEnc(byte value) { _encbyte = value; }
|
|
|
|
|
2005-05-05 15:59:24 +00:00
|
|
|
uint32 read(void *dataPtr, uint32 dataSize) {
|
2004-06-27 21:30:33 +00:00
|
|
|
// Read at most as many bytes as are still available...
|
2005-05-05 15:59:24 +00:00
|
|
|
if (dataSize > _bufSize - _pos)
|
|
|
|
dataSize = _bufSize - _pos;
|
|
|
|
memcpy(dataPtr, _ptr, dataSize);
|
2005-03-30 21:59:12 +00:00
|
|
|
|
|
|
|
if (_encbyte) {
|
2005-05-05 15:59:24 +00:00
|
|
|
byte *p = (byte *)dataPtr;
|
|
|
|
byte *end = p + dataSize;
|
2005-03-30 21:59:12 +00:00
|
|
|
while (p < end)
|
|
|
|
*p++ ^= _encbyte;
|
|
|
|
}
|
|
|
|
|
2005-05-05 15:59:24 +00:00
|
|
|
_ptr += dataSize;
|
|
|
|
_pos += dataSize;
|
2005-03-30 21:59:12 +00:00
|
|
|
|
2005-05-05 15:59:24 +00:00
|
|
|
return dataSize;
|
2004-04-17 09:57:15 +00:00
|
|
|
}
|
2004-05-04 03:27:00 +00:00
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
bool eos() const { return _pos == _bufSize; }
|
2004-12-11 23:16:36 +00:00
|
|
|
uint32 pos() const { return _pos; }
|
|
|
|
uint32 size() const { return _bufSize; }
|
|
|
|
|
2005-01-09 01:41:43 +00:00
|
|
|
void seek(int32 offs, int whence = SEEK_SET);
|
2004-04-17 09:57:15 +00:00
|
|
|
};
|
|
|
|
|
2005-03-30 21:59:12 +00:00
|
|
|
/**
|
|
|
|
* Simple memory based 'stream', which implements the WriteStream interface for
|
|
|
|
* a plain memory block.
|
|
|
|
*/
|
|
|
|
class MemoryWriteStream : public WriteStream {
|
|
|
|
private:
|
|
|
|
byte *_ptr;
|
|
|
|
const byte * const _ptrOrig;
|
|
|
|
const uint32 _bufSize;
|
|
|
|
uint32 _pos;
|
|
|
|
public:
|
|
|
|
MemoryWriteStream(byte *buf, uint32 len) : _ptr(buf), _ptrOrig(buf), _bufSize(len), _pos(0) {}
|
|
|
|
|
2005-05-05 15:59:24 +00:00
|
|
|
uint32 write(const void *dataPtr, uint32 dataSize) {
|
2005-03-30 21:59:12 +00:00
|
|
|
// Write at most as many bytes as are still available...
|
2005-05-05 15:59:24 +00:00
|
|
|
if (dataSize > _bufSize - _pos)
|
|
|
|
dataSize = _bufSize - _pos;
|
|
|
|
memcpy(_ptr, dataPtr, dataSize);
|
|
|
|
_ptr += dataSize;
|
|
|
|
_pos += dataSize;
|
|
|
|
return dataSize;
|
2005-03-30 21:59:12 +00:00
|
|
|
}
|
|
|
|
|
2005-04-22 17:40:09 +00:00
|
|
|
bool eos() const { return _pos == _bufSize; }
|
2005-03-30 21:59:12 +00:00
|
|
|
uint32 pos() const { return _pos; }
|
|
|
|
uint32 size() const { return _bufSize; }
|
|
|
|
};
|
|
|
|
|
2004-04-17 09:57:15 +00:00
|
|
|
} // End of namespace Common
|
|
|
|
|
|
|
|
#endif
|