2007-12-28 08:43:31 +00:00
|
|
|
/* 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 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2014-02-18 01:34:18 +00:00
|
|
|
*
|
2007-12-28 08:43:31 +00:00
|
|
|
* 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.
|
2014-02-18 01:34:18 +00:00
|
|
|
*
|
2007-12-28 08:43:31 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2011-05-02 15:19:35 +00:00
|
|
|
// Disable symbol overrides so that we can use zlib.h
|
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
|
|
|
|
2007-12-28 08:43:31 +00:00
|
|
|
#include "common/zlib.h"
|
2011-08-06 07:00:47 +00:00
|
|
|
#include "common/ptr.h"
|
2008-09-16 11:50:10 +00:00
|
|
|
#include "common/util.h"
|
2010-11-19 00:10:09 +00:00
|
|
|
#include "common/stream.h"
|
2014-03-27 23:30:38 +00:00
|
|
|
#include "common/debug.h"
|
2013-11-02 17:19:00 +00:00
|
|
|
#include "common/textconsole.h"
|
2007-12-28 08:43:31 +00:00
|
|
|
|
|
|
|
#if defined(USE_ZLIB)
|
2008-09-16 11:50:10 +00:00
|
|
|
#ifdef __SYMBIAN32__
|
|
|
|
#include <zlib\zlib.h>
|
|
|
|
#else
|
|
|
|
#include <zlib.h>
|
|
|
|
#endif
|
2008-01-03 08:39:49 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
#if ZLIB_VERNUM < 0x1204
|
|
|
|
#error Version 1.2.0.4 or newer of zlib is required for this code
|
|
|
|
#endif
|
2008-01-03 08:39:49 +00:00
|
|
|
#endif
|
2007-12-28 08:43:31 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
|
2007-12-28 08:43:31 +00:00
|
|
|
namespace Common {
|
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
#if defined(USE_ZLIB)
|
|
|
|
|
2008-09-16 11:42:21 +00:00
|
|
|
bool uncompress(byte *dst, unsigned long *dstLen, const byte *src, unsigned long srcLen) {
|
|
|
|
return Z_OK == ::uncompress(dst, dstLen, src, srcLen);
|
2007-12-28 08:43:31 +00:00
|
|
|
}
|
|
|
|
|
2012-01-09 14:47:59 +00:00
|
|
|
bool inflateZlibHeaderless(byte *dst, uint dstLen, const byte *src, uint srcLen, const byte *dict, uint dictLen) {
|
2011-08-20 20:41:10 +00:00
|
|
|
if (!dst || !dstLen || !src || !srcLen)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Initialize zlib
|
|
|
|
z_stream stream;
|
|
|
|
stream.next_in = const_cast<byte *>(src);
|
|
|
|
stream.avail_in = srcLen;
|
|
|
|
stream.next_out = dst;
|
|
|
|
stream.avail_out = dstLen;
|
|
|
|
stream.zalloc = Z_NULL;
|
|
|
|
stream.zfree = Z_NULL;
|
|
|
|
stream.opaque = Z_NULL;
|
|
|
|
|
|
|
|
// Negative MAX_WBITS tells zlib there's no zlib header
|
|
|
|
int err = inflateInit2(&stream, -MAX_WBITS);
|
|
|
|
if (err != Z_OK)
|
|
|
|
return false;
|
|
|
|
|
2012-01-09 14:47:59 +00:00
|
|
|
// Set the dictionary, if provided
|
|
|
|
if (dict != 0) {
|
|
|
|
err = inflateSetDictionary(&stream, const_cast<byte *>(dict), dictLen);
|
|
|
|
if (err != Z_OK)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-08-20 20:41:10 +00:00
|
|
|
err = inflate(&stream, Z_SYNC_FLUSH);
|
|
|
|
if (err != Z_OK && err != Z_STREAM_END) {
|
|
|
|
inflateEnd(&stream);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
inflateEnd(&stream);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-05-28 20:54:49 +00:00
|
|
|
enum {
|
|
|
|
kTempBufSize = 65536
|
|
|
|
};
|
|
|
|
|
|
|
|
bool inflateZlibInstallShield(byte *dst, uint dstLen, const byte *src, uint srcLen) {
|
|
|
|
if (!dst || !dstLen || !src || !srcLen)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// See if we have sync bytes. If so, just use our function for that.
|
|
|
|
if (srcLen >= 4 && READ_BE_UINT32(src + srcLen - 4) == 0xFFFF)
|
|
|
|
return inflateZlibHeaderless(dst, dstLen, src, srcLen);
|
|
|
|
|
|
|
|
// Otherwise, we have some custom code we get to use here.
|
|
|
|
|
|
|
|
byte *temp = (byte *)malloc(kTempBufSize);
|
|
|
|
|
|
|
|
uint32 bytesRead = 0, bytesProcessed = 0;
|
|
|
|
while (bytesRead < srcLen) {
|
|
|
|
uint16 chunkSize = READ_LE_UINT16(src + bytesRead);
|
|
|
|
bytesRead += 2;
|
|
|
|
|
|
|
|
// Initialize zlib
|
|
|
|
z_stream stream;
|
|
|
|
stream.next_in = const_cast<byte *>(src + bytesRead);
|
|
|
|
stream.avail_in = chunkSize;
|
|
|
|
stream.next_out = temp;
|
|
|
|
stream.avail_out = kTempBufSize;
|
|
|
|
stream.zalloc = Z_NULL;
|
|
|
|
stream.zfree = Z_NULL;
|
|
|
|
stream.opaque = Z_NULL;
|
|
|
|
|
|
|
|
// Negative MAX_WBITS tells zlib there's no zlib header
|
|
|
|
int err = inflateInit2(&stream, -MAX_WBITS);
|
|
|
|
if (err != Z_OK)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
err = inflate(&stream, Z_FINISH);
|
|
|
|
if (err != Z_OK && err != Z_STREAM_END) {
|
|
|
|
inflateEnd(&stream);
|
|
|
|
free(temp);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(dst + bytesProcessed, temp, stream.total_out);
|
|
|
|
bytesProcessed += stream.total_out;
|
|
|
|
|
|
|
|
inflateEnd(&stream);
|
|
|
|
bytesRead += chunkSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(temp);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-03-27 23:30:38 +00:00
|
|
|
#ifndef RELEASE_BUILD
|
|
|
|
static bool _shownBackwardSeekingWarning = false;
|
|
|
|
#endif
|
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
/**
|
|
|
|
* A simple wrapper class which can be used to wrap around an arbitrary
|
|
|
|
* other SeekableReadStream and will then provide on-the-fly decompression support.
|
|
|
|
* Assumes the compressed data to be in gzip format.
|
|
|
|
*/
|
2011-08-06 07:47:19 +00:00
|
|
|
class GZipReadStream : public SeekableReadStream {
|
2008-09-16 11:50:10 +00:00
|
|
|
protected:
|
|
|
|
enum {
|
|
|
|
BUFSIZE = 16384 // 1 << MAX_WBITS
|
|
|
|
};
|
|
|
|
|
|
|
|
byte _buf[BUFSIZE];
|
|
|
|
|
2011-08-06 07:00:47 +00:00
|
|
|
ScopedPtr<SeekableReadStream> _wrapped;
|
2008-09-16 11:50:10 +00:00
|
|
|
z_stream _stream;
|
|
|
|
int _zlibErr;
|
|
|
|
uint32 _pos;
|
|
|
|
uint32 _origSize;
|
|
|
|
bool _eos;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2014-03-27 23:30:38 +00:00
|
|
|
GZipReadStream(SeekableReadStream *w, uint32 knownSize = 0) : _wrapped(w), _stream() {
|
2008-09-16 11:50:10 +00:00
|
|
|
assert(w != 0);
|
|
|
|
|
|
|
|
// Verify file header is correct
|
|
|
|
w->seek(0, SEEK_SET);
|
|
|
|
uint16 header = w->readUint16BE();
|
|
|
|
assert(header == 0x1F8B ||
|
|
|
|
((header & 0x0F00) == 0x0800 && header % 31 == 0));
|
|
|
|
|
|
|
|
if (header == 0x1F8B) {
|
|
|
|
// Retrieve the original file size
|
|
|
|
w->seek(-4, SEEK_END);
|
|
|
|
_origSize = w->readUint32LE();
|
|
|
|
} else {
|
|
|
|
// Original size not available in zlib format
|
2012-07-18 11:54:15 +00:00
|
|
|
// use an otherwise known size if supplied.
|
|
|
|
_origSize = knownSize;
|
2008-09-16 11:50:10 +00:00
|
|
|
}
|
|
|
|
_pos = 0;
|
|
|
|
w->seek(0, SEEK_SET);
|
|
|
|
_eos = false;
|
|
|
|
|
|
|
|
// Adding 32 to windowBits indicates to zlib that it is supposed to
|
|
|
|
// automatically detect whether gzip or zlib headers are used for
|
|
|
|
// the compressed file. This feature was added in zlib 1.2.0.4,
|
|
|
|
// released 10 August 2003.
|
|
|
|
// Note: This is *crucial* for savegame compatibility, do *not* remove!
|
|
|
|
_zlibErr = inflateInit2(&_stream, MAX_WBITS + 32);
|
|
|
|
if (_zlibErr != Z_OK)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Setup input buffer
|
|
|
|
_stream.next_in = _buf;
|
|
|
|
_stream.avail_in = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
~GZipReadStream() {
|
|
|
|
inflateEnd(&_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool err() const { return (_zlibErr != Z_OK) && (_zlibErr != Z_STREAM_END); }
|
|
|
|
void clearErr() {
|
|
|
|
// only reset _eos; I/O errors are not recoverable
|
|
|
|
_eos = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 read(void *dataPtr, uint32 dataSize) {
|
|
|
|
_stream.next_out = (byte *)dataPtr;
|
|
|
|
_stream.avail_out = dataSize;
|
|
|
|
|
|
|
|
// Keep going while we get no error
|
|
|
|
while (_zlibErr == Z_OK && _stream.avail_out) {
|
|
|
|
if (_stream.avail_in == 0 && !_wrapped->eos()) {
|
|
|
|
// If we are out of input data: Read more data, if available.
|
|
|
|
_stream.next_in = _buf;
|
|
|
|
_stream.avail_in = _wrapped->read(_buf, BUFSIZE);
|
|
|
|
}
|
|
|
|
_zlibErr = inflate(&_stream, Z_NO_FLUSH);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the position counter
|
|
|
|
_pos += dataSize - _stream.avail_out;
|
|
|
|
|
|
|
|
if (_zlibErr == Z_STREAM_END && _stream.avail_out > 0)
|
|
|
|
_eos = true;
|
|
|
|
|
|
|
|
return dataSize - _stream.avail_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool eos() const {
|
|
|
|
return _eos;
|
|
|
|
}
|
|
|
|
int32 pos() const {
|
|
|
|
return _pos;
|
|
|
|
}
|
|
|
|
int32 size() const {
|
|
|
|
return _origSize;
|
|
|
|
}
|
|
|
|
bool seek(int32 offset, int whence = SEEK_SET) {
|
|
|
|
int32 newPos = 0;
|
2009-09-30 16:16:53 +00:00
|
|
|
switch (whence) {
|
2008-09-16 11:50:10 +00:00
|
|
|
case SEEK_SET:
|
|
|
|
newPos = offset;
|
|
|
|
break;
|
|
|
|
case SEEK_CUR:
|
|
|
|
newPos = _pos + offset;
|
2013-11-02 17:19:00 +00:00
|
|
|
break;
|
|
|
|
case SEEK_END:
|
|
|
|
// NOTE: This can be an expensive operation (see below).
|
2013-11-02 17:58:52 +00:00
|
|
|
newPos = size() + offset;
|
2013-11-02 17:19:00 +00:00
|
|
|
break;
|
2008-09-16 11:50:10 +00:00
|
|
|
}
|
2008-12-22 11:22:15 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
assert(newPos >= 0);
|
|
|
|
|
|
|
|
if ((uint32)newPos < _pos) {
|
|
|
|
// To search backward, we have to restart the whole decompression
|
|
|
|
// from the start of the file. A rather wasteful operation, best
|
|
|
|
// to avoid it. :/
|
2013-11-02 17:19:00 +00:00
|
|
|
|
2014-03-27 23:30:38 +00:00
|
|
|
#ifndef RELEASE_BUILD
|
2013-11-02 17:19:00 +00:00
|
|
|
if (!_shownBackwardSeekingWarning) {
|
|
|
|
// We only throw this warning once per stream, to avoid
|
|
|
|
// getting the console swarmed with warnings when consecutive
|
|
|
|
// seeks are made.
|
2014-03-27 23:30:38 +00:00
|
|
|
debug(1, "Backward seeking in GZipReadStream detected");
|
2013-11-02 17:19:00 +00:00
|
|
|
_shownBackwardSeekingWarning = true;
|
|
|
|
}
|
2014-03-27 23:30:38 +00:00
|
|
|
#endif
|
2013-11-02 17:19:00 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
_pos = 0;
|
|
|
|
_wrapped->seek(0, SEEK_SET);
|
|
|
|
_zlibErr = inflateReset(&_stream);
|
|
|
|
if (_zlibErr != Z_OK)
|
|
|
|
return false; // FIXME: STREAM REWRITE
|
|
|
|
_stream.next_in = _buf;
|
|
|
|
_stream.avail_in = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
offset = newPos - _pos;
|
|
|
|
|
|
|
|
// Skip the given amount of data (very inefficient if one tries to skip
|
|
|
|
// huge amounts of data, but usually client code will only skip a few
|
|
|
|
// bytes, so this should be fine.
|
|
|
|
byte tmpBuf[1024];
|
|
|
|
while (!err() && offset > 0) {
|
|
|
|
offset -= read(tmpBuf, MIN((int32)sizeof(tmpBuf), offset));
|
|
|
|
}
|
2008-12-22 11:22:15 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
_eos = false;
|
|
|
|
return true; // FIXME: STREAM REWRITE
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A simple wrapper class which can be used to wrap around an arbitrary
|
|
|
|
* other WriteStream and will then provide on-the-fly compression support.
|
|
|
|
* The compressed data is written in the gzip format.
|
|
|
|
*/
|
2011-08-06 07:47:19 +00:00
|
|
|
class GZipWriteStream : public WriteStream {
|
2008-09-16 11:50:10 +00:00
|
|
|
protected:
|
|
|
|
enum {
|
|
|
|
BUFSIZE = 16384 // 1 << MAX_WBITS
|
|
|
|
};
|
|
|
|
|
|
|
|
byte _buf[BUFSIZE];
|
2011-08-06 07:00:47 +00:00
|
|
|
ScopedPtr<WriteStream> _wrapped;
|
2008-09-16 11:50:10 +00:00
|
|
|
z_stream _stream;
|
|
|
|
int _zlibErr;
|
2016-08-04 09:14:06 +00:00
|
|
|
uint32 _pos;
|
2008-09-16 11:50:10 +00:00
|
|
|
|
|
|
|
void processData(int flushType) {
|
|
|
|
// This function is called by both write() and finalize().
|
|
|
|
while (_zlibErr == Z_OK && (_stream.avail_in || flushType == Z_FINISH)) {
|
|
|
|
if (_stream.avail_out == 0) {
|
|
|
|
if (_wrapped->write(_buf, BUFSIZE) != BUFSIZE) {
|
|
|
|
_zlibErr = Z_ERRNO;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
_stream.next_out = _buf;
|
|
|
|
_stream.avail_out = BUFSIZE;
|
|
|
|
}
|
|
|
|
_zlibErr = deflate(&_stream, flushType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2016-08-04 09:14:06 +00:00
|
|
|
GZipWriteStream(WriteStream *w) : _wrapped(w), _stream(), _pos(0) {
|
2008-09-16 11:50:10 +00:00
|
|
|
assert(w != 0);
|
|
|
|
|
|
|
|
// Adding 16 to windowBits indicates to zlib that it is supposed to
|
|
|
|
// write gzip headers. This feature was added in zlib 1.2.0.4,
|
|
|
|
// released 10 August 2003.
|
|
|
|
// Note: This is *crucial* for savegame compatibility, do *not* remove!
|
|
|
|
_zlibErr = deflateInit2(&_stream,
|
|
|
|
Z_DEFAULT_COMPRESSION,
|
|
|
|
Z_DEFLATED,
|
|
|
|
MAX_WBITS + 16,
|
|
|
|
8,
|
2008-12-22 11:22:15 +00:00
|
|
|
Z_DEFAULT_STRATEGY);
|
2008-09-16 11:50:10 +00:00
|
|
|
assert(_zlibErr == Z_OK);
|
|
|
|
|
|
|
|
_stream.next_out = _buf;
|
|
|
|
_stream.avail_out = BUFSIZE;
|
|
|
|
_stream.avail_in = 0;
|
|
|
|
_stream.next_in = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
~GZipWriteStream() {
|
|
|
|
finalize();
|
|
|
|
deflateEnd(&_stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool err() const {
|
|
|
|
// CHECKME: does Z_STREAM_END make sense here?
|
|
|
|
return (_zlibErr != Z_OK && _zlibErr != Z_STREAM_END) || _wrapped->err();
|
|
|
|
}
|
|
|
|
|
|
|
|
void clearErr() {
|
|
|
|
// Note: we don't reset the _zlibErr here, as it is not
|
|
|
|
// clear in general how
|
|
|
|
_wrapped->clearErr();
|
|
|
|
}
|
|
|
|
|
|
|
|
void finalize() {
|
|
|
|
if (_zlibErr != Z_OK)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Process whatever remaining data there is.
|
|
|
|
processData(Z_FINISH);
|
2007-12-28 08:43:31 +00:00
|
|
|
|
2008-09-16 11:50:10 +00:00
|
|
|
// Since processData only writes out blocks of size BUFSIZE,
|
|
|
|
// we may have to flush some stragglers.
|
|
|
|
uint remainder = BUFSIZE - _stream.avail_out;
|
|
|
|
if (remainder > 0) {
|
|
|
|
if (_wrapped->write(_buf, remainder) != remainder) {
|
|
|
|
_zlibErr = Z_ERRNO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finalize the wrapped savefile, too
|
|
|
|
_wrapped->finalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 write(const void *dataPtr, uint32 dataSize) {
|
|
|
|
if (err())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Hook in the new data ...
|
|
|
|
// Note: We need to make a const_cast here, as zlib is not aware
|
|
|
|
// of the const keyword.
|
|
|
|
_stream.next_in = const_cast<byte *>((const byte *)dataPtr);
|
|
|
|
_stream.avail_in = dataSize;
|
|
|
|
|
|
|
|
// ... and flush it to disk
|
|
|
|
processData(Z_NO_FLUSH);
|
|
|
|
|
2016-08-04 09:14:06 +00:00
|
|
|
_pos += dataSize - _stream.avail_in;
|
2008-09-16 11:50:10 +00:00
|
|
|
return dataSize - _stream.avail_in;
|
|
|
|
}
|
2016-08-04 08:36:21 +00:00
|
|
|
|
2016-08-04 09:14:06 +00:00
|
|
|
virtual int32 pos() const { return _pos; }
|
2008-09-16 11:50:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // USE_ZLIB
|
|
|
|
|
2012-07-18 11:54:15 +00:00
|
|
|
SeekableReadStream *wrapCompressedReadStream(SeekableReadStream *toBeWrapped, uint32 knownSize) {
|
2008-09-16 11:50:10 +00:00
|
|
|
if (toBeWrapped) {
|
|
|
|
uint16 header = toBeWrapped->readUint16BE();
|
|
|
|
bool isCompressed = (header == 0x1F8B ||
|
|
|
|
((header & 0x0F00) == 0x0800 &&
|
|
|
|
header % 31 == 0));
|
|
|
|
toBeWrapped->seek(-2, SEEK_CUR);
|
2012-07-15 00:36:47 +00:00
|
|
|
if (isCompressed) {
|
|
|
|
#if defined(USE_ZLIB)
|
2012-07-18 11:54:15 +00:00
|
|
|
return new GZipReadStream(toBeWrapped, knownSize);
|
2012-07-15 00:36:47 +00:00
|
|
|
#else
|
|
|
|
delete toBeWrapped;
|
|
|
|
return NULL;
|
2007-12-28 08:43:31 +00:00
|
|
|
#endif
|
2012-07-15 00:36:47 +00:00
|
|
|
}
|
|
|
|
}
|
2008-09-16 11:50:10 +00:00
|
|
|
return toBeWrapped;
|
|
|
|
}
|
|
|
|
|
2011-08-06 07:47:19 +00:00
|
|
|
WriteStream *wrapCompressedWriteStream(WriteStream *toBeWrapped) {
|
2008-09-16 11:50:10 +00:00
|
|
|
#if defined(USE_ZLIB)
|
|
|
|
if (toBeWrapped)
|
|
|
|
return new GZipWriteStream(toBeWrapped);
|
|
|
|
#endif
|
|
|
|
return toBeWrapped;
|
|
|
|
}
|
|
|
|
|
2007-12-28 08:43:31 +00:00
|
|
|
|
2013-01-26 18:33:27 +00:00
|
|
|
} // End of namespace Common
|