aboutsummaryrefslogtreecommitdiff
path: root/engines/sci/gfx/seq_decoder.cpp
diff options
context:
space:
mode:
authorWalter van Niftrik2009-05-21 22:03:23 +0000
committerWalter van Niftrik2009-05-21 22:03:23 +0000
commit36fe37443d53c64d0172791a5bc6226c7fe199c0 (patch)
tree55e21c6a9b38d19437410156f9470c68fe3e6389 /engines/sci/gfx/seq_decoder.cpp
parent7ed8d7f573da40d19299f14d304491184fe221a8 (diff)
downloadscummvm-rg350-36fe37443d53c64d0172791a5bc6226c7fe199c0.tar.gz
scummvm-rg350-36fe37443d53c64d0172791a5bc6226c7fe199c0.tar.bz2
scummvm-rg350-36fe37443d53c64d0172791a5bc6226c7fe199c0.zip
SCI: Added support for KQ6 movies.
svn-id: r40774
Diffstat (limited to 'engines/sci/gfx/seq_decoder.cpp')
-rw-r--r--engines/sci/gfx/seq_decoder.cpp197
1 files changed, 197 insertions, 0 deletions
diff --git a/engines/sci/gfx/seq_decoder.cpp b/engines/sci/gfx/seq_decoder.cpp
new file mode 100644
index 0000000000..ac12678b09
--- /dev/null
+++ b/engines/sci/gfx/seq_decoder.cpp
@@ -0,0 +1,197 @@
+/* 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.
+
+ * 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
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ * $URL$
+ * $Id$
+ *
+ */
+
+#include "common/archive.h"
+#include "sci/gfx/seq_decoder.h"
+#include "sci/gfx/gfx_resource.h"
+#include "sci/gfx/gfx_tools.h"
+
+namespace Sci {
+
+SeqDecoder::~SeqDecoder() {
+ closeFile();
+}
+
+bool SeqDecoder::loadFile(const char *fileName) {
+ closeFile();
+
+ _fileStream = SearchMan.createReadStreamForMember(fileName);
+ if (!_fileStream)
+ return false;
+
+ _frameCount = _fileStream->readUint16LE();
+ int paletteSize = _fileStream->readUint32LE();
+
+ byte *paletteData = new byte[paletteSize];
+ _fileStream->read(paletteData, paletteSize);
+ _palette = gfxr_read_pal11(-1, paletteData, paletteSize);
+ delete[] paletteData;
+
+ _currentFrame = 0;
+
+ return true;
+}
+
+void SeqDecoder::closeFile() {
+ if (!_fileStream)
+ return;
+
+ delete _fileStream;
+ _fileStream = 0;
+
+ delete _palette;
+ _palette = 0;
+}
+
+#define WRITE_TO_BUFFER(n) \
+ if (writeRow * width + writeCol + (n) > width * height) { \
+ warning("SEQ player: writing out of bounds, aborting"); \
+ return false; \
+ } \
+ if (litPos + (n) > litSize) { \
+ warning("SEQ player: reading out of bounds, aborting"); \
+ } \
+ memcpy(dest + writeRow * width + writeCol, litData + litPos, n);
+
+bool SeqDecoder::decodeFrame(byte *rleData, int rleSize, byte *litData, int litSize, byte *dest, int width, int height, int colorKey) {
+ int writeRow = 0;
+ int writeCol = 0;
+ int litPos = 0;
+ int rlePos = 0;
+
+ memset(dest, colorKey, width * height);
+
+ while (rlePos < rleSize) {
+ int op = rleData[rlePos++];
+
+ if ((op & 0xc0) == 0xc0) {
+ op &= 0x3f;
+ if (op == 0) {
+ // Go to next line in target buffer
+ writeRow++;
+ writeCol = 0;
+ } else {
+ // Skip bytes on current line
+ writeCol += op;
+ }
+ } else if (op & 0x80) {
+ op &= 0x3f;
+ if (op == 0) {
+ // Copy remainder of current line
+ int rem = width - writeCol;
+
+ WRITE_TO_BUFFER(rem);
+ writeRow++;
+ writeCol = 0;
+ litPos += rem;
+ } else {
+ // Copy bytes
+ WRITE_TO_BUFFER(op);
+ writeCol += op;
+ litPos += op;
+ }
+ } else {
+ uint16 count = ((op & 7) << 8) | rleData[rlePos++];
+
+ switch (op >> 3) {
+ case 2:
+ // Skip bytes
+ writeCol += count;
+ break;
+ case 3:
+ // Copy bytes
+ WRITE_TO_BUFFER(count);
+ writeCol += count;
+ litPos += count;
+ break;
+ case 6: {
+ // Copy rows
+ if (count == 0)
+ count = height - writeRow;
+
+ for (int i = 0; i < count; i++) {
+ WRITE_TO_BUFFER(width);
+ litPos += width;
+ writeRow++;
+ }
+ break;
+ }
+ case 7:
+ // Skip rows
+ if (count == 0)
+ count = height - writeRow;
+
+ writeRow += count;
+ break;
+ default:
+ warning("Unsupported operation %i encountered", op >> 3);
+ return false;
+ }
+ }
+ }
+
+ return true;
+}
+
+gfx_pixmap_t *SeqDecoder::getFrame(bool &hasNext) {
+ int frameWidth = _fileStream->readUint16LE();
+ int frameHeight = _fileStream->readUint16LE();
+ int frameLeft = _fileStream->readUint16LE();
+ int frameTop = _fileStream->readUint16LE();
+ int colorKey = _fileStream->readByte();
+ int type = _fileStream->readByte();
+ _fileStream->seek(2, SEEK_CUR);
+ uint16 bytes = _fileStream->readUint16LE();
+ _fileStream->seek(2, SEEK_CUR);
+ uint16 rle_bytes = _fileStream->readUint16LE();
+ _fileStream->seek(6, SEEK_CUR);
+ uint32 offset = _fileStream->readUint32LE();
+
+ _fileStream->seek(offset);
+ gfx_pixmap_t *pixmap = gfx_new_pixmap(frameWidth, frameHeight, 0, 0, 0);
+
+ assert(pixmap);
+
+ gfx_pixmap_alloc_index_data(pixmap);
+
+ if (type == 0)
+ _fileStream->read(pixmap->index_data, bytes);
+ else {
+ byte *buf = new byte[bytes];
+ _fileStream->read(buf, bytes);
+ decodeFrame(buf, rle_bytes, buf + rle_bytes, bytes - rle_bytes, pixmap->index_data, frameWidth, frameHeight, colorKey);
+ }
+
+ pixmap->xoffset = frameLeft;
+ pixmap->yoffset = frameTop;
+ pixmap->color_key = colorKey;
+ pixmap->palette = _palette->getref();
+
+ hasNext = ++_currentFrame < _frameCount;
+
+ return pixmap;
+}
+
+} // End of namespace Sci