aboutsummaryrefslogtreecommitdiff
path: root/graphics/decoders
diff options
context:
space:
mode:
authorMarisa-Chan2014-06-13 21:43:04 +0700
committerMarisa-Chan2014-06-13 21:43:04 +0700
commit45589950c0fb1a449351e6a00ef10d42290d8bae (patch)
tree44e4eedcb7e69d5fc386155b000ed038af07251d /graphics/decoders
parent48360645dcd5f8fddb135b6e31ae5cae4be8d77f (diff)
parent5c005ad3a3f1df0bc968c85c1cf0fc48e36ab0b2 (diff)
downloadscummvm-rg350-45589950c0fb1a449351e6a00ef10d42290d8bae.tar.gz
scummvm-rg350-45589950c0fb1a449351e6a00ef10d42290d8bae.tar.bz2
scummvm-rg350-45589950c0fb1a449351e6a00ef10d42290d8bae.zip
Merge remote-tracking branch 'upstream/master' into zvision
Conflicts: engines/zvision/animation/rlf_animation.cpp engines/zvision/animation_control.h engines/zvision/core/console.cpp engines/zvision/core/events.cpp engines/zvision/cursors/cursor.cpp engines/zvision/cursors/cursor_manager.cpp engines/zvision/cursors/cursor_manager.h engines/zvision/fonts/truetype_font.cpp engines/zvision/graphics/render_manager.cpp engines/zvision/graphics/render_manager.h engines/zvision/inventory/inventory_manager.h engines/zvision/inventory_manager.h engines/zvision/meta_animation.h engines/zvision/module.mk engines/zvision/scripting/actions.cpp engines/zvision/scripting/control.h engines/zvision/scripting/controls/animation_control.cpp engines/zvision/scripting/controls/animation_control.h engines/zvision/scripting/controls/input_control.cpp engines/zvision/scripting/controls/lever_control.cpp engines/zvision/scripting/controls/timer_node.cpp engines/zvision/scripting/controls/timer_node.h engines/zvision/scripting/puzzle.h engines/zvision/scripting/scr_file_handling.cpp engines/zvision/scripting/script_manager.cpp engines/zvision/scripting/script_manager.h engines/zvision/sidefx.cpp engines/zvision/sound/zork_raw.cpp engines/zvision/sound/zork_raw.h engines/zvision/video/video.cpp engines/zvision/video/zork_avi_decoder.h engines/zvision/zvision.cpp engines/zvision/zvision.h
Diffstat (limited to 'graphics/decoders')
-rw-r--r--graphics/decoders/bmp.cpp182
-rw-r--r--graphics/decoders/bmp.h66
-rw-r--r--graphics/decoders/iff.cpp241
-rw-r--r--graphics/decoders/iff.h119
-rw-r--r--graphics/decoders/image_decoder.h98
-rw-r--r--graphics/decoders/jpeg.cpp266
-rw-r--r--graphics/decoders/jpeg.h95
-rw-r--r--graphics/decoders/pcx.cpp213
-rw-r--r--graphics/decoders/pcx.h68
-rw-r--r--graphics/decoders/pict.cpp580
-rw-r--r--graphics/decoders/pict.h140
-rw-r--r--graphics/decoders/png.cpp241
-rw-r--r--graphics/decoders/png.h67
-rw-r--r--graphics/decoders/tga.cpp427
-rw-r--r--graphics/decoders/tga.h98
15 files changed, 0 insertions, 2901 deletions
diff --git a/graphics/decoders/bmp.cpp b/graphics/decoders/bmp.cpp
deleted file mode 100644
index 2eabbb7631..0000000000
--- a/graphics/decoders/bmp.cpp
+++ /dev/null
@@ -1,182 +0,0 @@
-/* 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.
- */
-
-#include "common/stream.h"
-#include "common/textconsole.h"
-
-#include "graphics/pixelformat.h"
-#include "graphics/surface.h"
-#include "graphics/decoders/bmp.h"
-
-namespace Graphics {
-
-BitmapDecoder::BitmapDecoder() {
- _surface = 0;
- _palette = 0;
- _paletteColorCount = 0;
-}
-
-BitmapDecoder::~BitmapDecoder() {
- destroy();
-}
-
-void BitmapDecoder::destroy() {
- if (_surface) {
- _surface->free();
- delete _surface; _surface = 0;
- }
-
- delete[] _palette; _palette = 0;
- _paletteColorCount = 0;
-}
-
-bool BitmapDecoder::loadStream(Common::SeekableReadStream &stream) {
- destroy();
-
- if (stream.readByte() != 'B')
- return false;
-
- if (stream.readByte() != 'M')
- return false;
-
- /* uint32 fileSize = */ stream.readUint32LE();
- /* uint16 res1 = */ stream.readUint16LE();
- /* uint16 res2 = */ stream.readUint16LE();
- uint32 imageOffset = stream.readUint32LE();
-
- uint32 infoSize = stream.readUint32LE();
- if (infoSize != 40) {
- warning("Only Windows v3 bitmaps are supported");
- return false;
- }
-
- uint32 width = stream.readUint32LE();
- int32 height = stream.readSint32LE();
-
- if (width == 0 || height == 0)
- return false;
-
- if (height < 0) {
- warning("Right-side up bitmaps not supported");
- return false;
- }
-
- /* uint16 planes = */ stream.readUint16LE();
- uint16 bitsPerPixel = stream.readUint16LE();
-
- if (bitsPerPixel != 8 && bitsPerPixel != 24 && bitsPerPixel != 32) {
- warning("%dbpp bitmaps not supported", bitsPerPixel);
- return false;
- }
-
- uint32 compression = stream.readUint32LE();
-
- if (compression != 0) {
- warning("Compressed bitmaps not supported");
- return false;
- }
-
- /* uint32 imageSize = */ stream.readUint32LE();
- /* uint32 pixelsPerMeterX = */ stream.readUint32LE();
- /* uint32 pixelsPerMeterY = */ stream.readUint32LE();
- _paletteColorCount = stream.readUint32LE();
- /* uint32 colorsImportant = */ stream.readUint32LE();
-
- if (bitsPerPixel == 8) {
- if (_paletteColorCount == 0)
- _paletteColorCount = 256;
-
- // Read the palette
- _palette = new byte[_paletteColorCount * 3];
- for (uint16 i = 0; i < _paletteColorCount; i++) {
- _palette[i * 3 + 2] = stream.readByte();
- _palette[i * 3 + 1] = stream.readByte();
- _palette[i * 3 + 0] = stream.readByte();
- stream.readByte();
- }
- }
-
- // Start us at the beginning of the image
- stream.seek(imageOffset);
-
- Graphics::PixelFormat format = Graphics::PixelFormat::createFormatCLUT8();
-
- // BGRA for 24bpp and 32 bpp
- if (bitsPerPixel == 24 || bitsPerPixel == 32)
- format = Graphics::PixelFormat(4, 8, 8, 8, 8, 8, 16, 24, 0);
-
- _surface = new Graphics::Surface();
- _surface->create(width, height, format);
-
- int srcPitch = width * (bitsPerPixel >> 3);
- const int extraDataLength = (srcPitch % 4) ? 4 - (srcPitch % 4) : 0;
-
- if (bitsPerPixel == 8) {
- byte *dst = (byte *)_surface->getPixels();
-
- for (int32 i = 0; i < height; i++) {
- stream.read(dst + (height - i - 1) * width, width);
- stream.skip(extraDataLength);
- }
- } else if (bitsPerPixel == 24) {
- byte *dst = (byte *)_surface->getBasePtr(0, height - 1);
-
- for (int32 i = 0; i < height; i++) {
- for (uint32 j = 0; j < width; j++) {
- byte b = stream.readByte();
- byte g = stream.readByte();
- byte r = stream.readByte();
- uint32 color = format.RGBToColor(r, g, b);
-
- *((uint32 *)dst) = color;
- dst += format.bytesPerPixel;
- }
-
- stream.skip(extraDataLength);
- dst -= _surface->pitch * 2;
- }
- } else { // 32 bpp
- byte *dst = (byte *)_surface->getBasePtr(0, height - 1);
-
- for (int32 i = 0; i < height; i++) {
- for (uint32 j = 0; j < width; j++) {
- byte b = stream.readByte();
- byte g = stream.readByte();
- byte r = stream.readByte();
- // Ignore the last byte, as in v3 it is unused
- // and should thus NOT be used as alpha.
- // ref: http://msdn.microsoft.com/en-us/library/windows/desktop/dd183376%28v=vs.85%29.aspx
- stream.readByte();
- uint32 color = format.RGBToColor(r, g, b);
-
- *((uint32 *)dst) = color;
- dst += format.bytesPerPixel;
- }
-
- stream.skip(extraDataLength);
- dst -= _surface->pitch * 2;
- }
- }
-
- return true;
-}
-
-} // End of namespace Graphics
diff --git a/graphics/decoders/bmp.h b/graphics/decoders/bmp.h
deleted file mode 100644
index 779da352be..0000000000
--- a/graphics/decoders/bmp.h
+++ /dev/null
@@ -1,66 +0,0 @@
-/* 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.
- */
-
-/**
- * @file
- * Image decoder used in engines:
- * - hugo
- * - mohawk
- * - wintermute
- */
-
-#ifndef GRAPHICS_DECODERS_BMP_H
-#define GRAPHICS_DECODERS_BMP_H
-
-#include "common/scummsys.h"
-#include "common/str.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common{
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct PixelFormat;
-struct Surface;
-
-class BitmapDecoder : public ImageDecoder {
-public:
- BitmapDecoder();
- virtual ~BitmapDecoder();
-
- // ImageDecoder API
- void destroy();
- virtual bool loadStream(Common::SeekableReadStream &stream);
- virtual const Surface *getSurface() const { return _surface; }
- const byte *getPalette() const { return _palette; }
- uint16 getPaletteColorCount() const { return _paletteColorCount; }
-
-private:
- Surface *_surface;
- byte *_palette;
- uint16 _paletteColorCount;
-};
-
-} // End of namespace Graphics
-
-#endif
diff --git a/graphics/decoders/iff.cpp b/graphics/decoders/iff.cpp
deleted file mode 100644
index 7b37969fc1..0000000000
--- a/graphics/decoders/iff.cpp
+++ /dev/null
@@ -1,241 +0,0 @@
-/* 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.
- */
-
-#include "common/iff_container.h"
-#include "common/stream.h"
-#include "common/util.h"
-
-#include "graphics/decoders/iff.h"
-
-namespace Graphics {
-
-IFFDecoder::IFFDecoder() {
- _surface = 0;
- _palette = 0;
-
- destroy();
-}
-
-IFFDecoder::~IFFDecoder() {
- destroy();
-}
-
-void IFFDecoder::destroy() {
- if (_surface) {
- _surface->free();
- delete _surface;
- _surface = 0;
- }
-
- if (_palette) {
- delete[] _palette;
- _palette = 0;
- }
-
- memset(&_header, 0, sizeof(Header));
- _paletteRanges.clear();
- _type = TYPE_UNKNOWN;
- _paletteColorCount = 0;
- _numRelevantPlanes = 8;
- _pixelPacking = false;
-}
-
-bool IFFDecoder::loadStream(Common::SeekableReadStream &stream) {
- destroy();
-
- const uint32 form = stream.readUint32BE();
-
- if (form != ID_FORM) {
- warning("Failed reading IFF-file");
- return false;
- }
-
- stream.skip(4);
-
- const uint32 type = stream.readUint32BE();
-
- switch (type) {
- case ID_ILBM:
- _type = TYPE_ILBM;
- break;
- case ID_PBM:
- _type = TYPE_PBM;
- break;
- }
-
- if (type == TYPE_UNKNOWN) {
- warning("Failed reading IFF-file");
- return false;
- }
-
- while (1) {
- const uint32 chunkType = stream.readUint32BE();
- const uint32 chunkSize = stream.readUint32BE();
-
- if (stream.eos())
- break;
-
- switch (chunkType) {
- case ID_BMHD:
- loadHeader(stream);
- break;
- case ID_CMAP:
- loadPalette(stream, chunkSize);
- break;
- case ID_CRNG:
- loadPaletteRange(stream, chunkSize);
- break;
- case ID_BODY:
- loadBitmap(stream);
- break;
- default:
- stream.skip(chunkSize);
- }
- }
-
- return true;
-}
-
-void IFFDecoder::loadHeader(Common::SeekableReadStream &stream) {
- _header.width = stream.readUint16BE();
- _header.height = stream.readUint16BE();
- _header.x = stream.readUint16BE();
- _header.y = stream.readUint16BE();
- _header.numPlanes = stream.readByte();
- _header.masking = stream.readByte();
- _header.compression = stream.readByte();
- _header.flags = stream.readByte();
- _header.transparentColor = stream.readUint16BE();
- _header.xAspect = stream.readByte();
- _header.yAspect = stream.readByte();
- _header.pageWidth = stream.readUint16BE();
- _header.pageHeight = stream.readUint16BE();
-
- assert(_header.width >= 1);
- assert(_header.height >= 1);
- assert(_header.numPlanes >= 1 && _header.numPlanes <= 8 && _header.numPlanes != 7);
-}
-
-void IFFDecoder::loadPalette(Common::SeekableReadStream &stream, const uint32 size) {
- _palette = new byte[size];
- stream.read(_palette, size);
- _paletteColorCount = size / 3;
-}
-
-void IFFDecoder::loadPaletteRange(Common::SeekableReadStream &stream, const uint32 size) {
- PaletteRange range;
-
- range.timer = stream.readSint16BE();
- range.step = stream.readSint16BE();
- range.flags = stream.readSint16BE();
- range.first = stream.readByte();
- range.last = stream.readByte();
-
- _paletteRanges.push_back(range);
-}
-
-void IFFDecoder::loadBitmap(Common::SeekableReadStream &stream) {
- _numRelevantPlanes = MIN(_numRelevantPlanes, _header.numPlanes);
-
- if (_numRelevantPlanes != 1 && _numRelevantPlanes != 2 && _numRelevantPlanes != 4)
- _pixelPacking = false;
-
- uint16 outPitch = _header.width;
-
- if (_pixelPacking)
- outPitch /= (8 / _numRelevantPlanes);
-
- // FIXME: CLUT8 is not a proper format for packed bitmaps but there is no way to tell it to use 1, 2 or 4 bits per pixel
- _surface = new Graphics::Surface();
- _surface->create(outPitch, _header.height, Graphics::PixelFormat::createFormatCLUT8());
-
- if (_type == TYPE_ILBM) {
- uint32 scanlinePitch = ((_header.width + 15) >> 4) << 1;
- byte *scanlines = new byte[scanlinePitch * _header.numPlanes];
- byte *data = (byte *)_surface->getPixels();
-
- for (uint16 i = 0; i < _header.height; ++i) {
- byte *scanline = scanlines;
-
- for (uint16 j = 0; j < _header.numPlanes; ++j) {
- uint16 outSize = scanlinePitch;
-
- if (_header.compression) {
- Common::PackBitsReadStream packStream(stream);
- packStream.read(scanline, outSize);
- } else {
- stream.read(scanline, outSize);
- }
-
- scanline += outSize;
- }
-
- packPixels(scanlines, data, scanlinePitch, outPitch);
- data += outPitch;
- }
-
- delete[] scanlines;
- } else if (_type == TYPE_PBM) {
- byte *data = (byte *)_surface->getPixels();
- uint32 outSize = _header.width * _header.height;
-
- if (_header.compression) {
- Common::PackBitsReadStream packStream(stream);
- packStream.read(data, outSize);
- } else {
- stream.read(data, outSize);
- }
- }
-}
-
-void IFFDecoder::packPixels(byte *scanlines, byte *data, const uint16 scanlinePitch, const uint16 outPitch) {
- uint32 numPixels = _header.width;
-
- if (_pixelPacking)
- numPixels = outPitch * (8 / _numRelevantPlanes);
-
- for (uint32 x = 0; x < numPixels; ++x) {
- byte *scanline = scanlines;
- byte pixel = 0;
- byte offset = x >> 3;
- byte bit = 0x80 >> (x & 7);
-
- // first build a pixel by scanning all the usable planes in the input
- for (uint32 plane = 0; plane < _numRelevantPlanes; ++plane) {
- if (scanline[offset] & bit)
- pixel |= (1 << plane);
-
- scanline += scanlinePitch;
- }
-
- // then output the pixel according to the requested packing
- if (!_pixelPacking)
- data[x] = pixel;
- else if (_numRelevantPlanes == 1)
- data[x / 8] |= (pixel << (x & 7));
- else if (_numRelevantPlanes == 2)
- data[x / 4] |= (pixel << ((x & 3) << 1));
- else if (_numRelevantPlanes == 4)
- data[x / 2] |= (pixel << ((x & 1) << 2));
- }
-}
-
-} // End of namespace Graphics
diff --git a/graphics/decoders/iff.h b/graphics/decoders/iff.h
deleted file mode 100644
index beac62e519..0000000000
--- a/graphics/decoders/iff.h
+++ /dev/null
@@ -1,119 +0,0 @@
-/* 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.
- */
-
-/**
- * @file
- * Image decoder used in engines:
- * - gob
- * - parallaction
- * - queen
- * - saga
- */
-
-#ifndef GRAPHICS_DECODERS_IFF_H
-#define GRAPHICS_DECODERS_IFF_H
-
-#include "common/array.h"
-#include "common/endian.h"
-
-#include "graphics/surface.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common {
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct Surface;
-
-class IFFDecoder : public ImageDecoder {
-public:
- struct Header {
- uint16 width, height;
- uint16 x, y;
- byte numPlanes;
- byte masking;
- byte compression;
- byte flags;
- uint16 transparentColor;
- byte xAspect, yAspect;
- uint16 pageWidth, pageHeight;
- };
-
- struct PaletteRange {
- int16 timer, step, flags;
- byte first, last;
- };
-
- enum Type {
- TYPE_UNKNOWN = 0,
- TYPE_ILBM,
- TYPE_PBM
- };
-
- IFFDecoder();
- virtual ~IFFDecoder();
-
- // ImageDecoder API
- void destroy();
- bool loadStream(Common::SeekableReadStream &stream);
- const Header *getHeader() const { return &_header; }
- const Surface *getSurface() const { return _surface; }
- const byte *getPalette() const { return _palette; }
- const Common::Array<PaletteRange> &getPaletteRanges() const { return _paletteRanges; }
- uint16 getPaletteColorCount() const { return _paletteColorCount; }
-
- /**
- * The number of planes to decode, also determines the pixel packing if _packPixels is true.
- * 8 == decode all planes, map 1 pixel in 1 byte. (default, no packing even if _packPixels is true)
- */
- void setNumRelevantPlanes(const uint8 numRelevantPlanes) { _numRelevantPlanes = numRelevantPlanes; }
-
- /**
- * Enables pixel packing, the amount of packing is determined by _numRelevantPlanes
- * 1 == decode first plane, pack 8 pixels in 1 byte. This makes _surface->w 1/8th of _header.width
- * 2 == decode first 2 planes, pack 4 pixels in 1 byte. This makes _surface->w 1/4th of _header.width
- * 4 == decode first 4 planes, pack 2 pixels in 1 byte. This makes _surface->w half of _header.width
- * Packed bitmaps won't have a proper surface format since there is no way to tell it to use 1, 2 or 4 bits per pixel
- */
- void setPixelPacking(const bool pixelPacking) { _pixelPacking = pixelPacking; }
-private:
-
- Header _header;
- Surface *_surface;
- byte *_palette;
- Common::Array<PaletteRange> _paletteRanges;
- Type _type;
- uint16 _paletteColorCount;
- uint8 _numRelevantPlanes;
- bool _pixelPacking;
-
- void loadHeader(Common::SeekableReadStream &stream);
- void loadPalette(Common::SeekableReadStream &stream, const uint32 size);
- void loadPaletteRange(Common::SeekableReadStream &stream, const uint32 size);
- void loadBitmap(Common::SeekableReadStream &stream);
- void packPixels(byte *scanlines, byte *data, const uint16 scanlinePitch, const uint16 outPitch);
-};
-
-} // End of namespace Graphics
-
-#endif // GRAPHICS_DECODERS_IFF_H
diff --git a/graphics/decoders/image_decoder.h b/graphics/decoders/image_decoder.h
deleted file mode 100644
index 49e31c6e3a..0000000000
--- a/graphics/decoders/image_decoder.h
+++ /dev/null
@@ -1,98 +0,0 @@
-/* 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.
- */
-
-#ifndef GRAPHICS_DECODERS_IMAGEDECODER_H
-#define GRAPHICS_DECODERS_IMAGEDECODER_H
-
-#include "common/scummsys.h"
-#include "common/str.h"
-
-namespace Common{
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct PixelFormat;
-struct Surface;
-
-/**
- * A representation of an image decoder that maintains ownership of the surface
- * and palette it decodes to.
- */
-class ImageDecoder {
-public:
- virtual ~ImageDecoder() {}
-
- /**
- * Load an image from the specified stream
- *
- * @param stream the input stream
- * @return whether loading the file succeeded
- * @see getSurface
- * @see getPalette
- */
- virtual bool loadStream(Common::SeekableReadStream &stream) = 0;
-
- /**
- * Destroy this decoder's surface and palette
- */
- virtual void destroy() = 0;
-
- /**
- * Get the decoded surface
- *
- * This surface is owned by this ImageDecoder and will remain valid
- * until destroy() or loadStream() is called, or until this ImageDecoder's
- * destructor is called.
- *
- * @return the decoded surface, or 0 if no surface is present
- */
- virtual const Surface *getSurface() const = 0;
-
- /**
- * Get the decoded palette
- *
- * This palette is owned by this ImageDecoder and will remain valid
- * until destroy() or loadStream() is called, or until this ImageDecoder's
- * destructor is called.
- *
- * The palette's format is the same as PaletteManager's palette
- * (interleaved RGB values).
- *
- * @return the decoded palette, or undefined if no palette is present
- */
- virtual const byte *getPalette() const { return 0; }
-
- /**
- * Query if the decoded image has a palette.
- */
- virtual bool hasPalette() const { return getPaletteColorCount() != 0; }
-
- /** Return the starting index of the palette. */
- virtual byte getPaletteStartIndex() const { return 0; }
- /** Return the number of colors in the palette. */
- virtual uint16 getPaletteColorCount() const { return 0; }
-};
-
-} // End of namespace Graphics
-
-#endif
diff --git a/graphics/decoders/jpeg.cpp b/graphics/decoders/jpeg.cpp
deleted file mode 100644
index c858884095..0000000000
--- a/graphics/decoders/jpeg.cpp
+++ /dev/null
@@ -1,266 +0,0 @@
-/* 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.
- *
- */
-
-// libjpeg uses forbidden symbols in its header. Thus, we need to allow them
-// here.
-#define FORBIDDEN_SYMBOL_ALLOW_ALL
-
-#include "graphics/pixelformat.h"
-#include "graphics/decoders/jpeg.h"
-
-#include "common/debug.h"
-#include "common/endian.h"
-#include "common/stream.h"
-#include "common/textconsole.h"
-
-#ifdef USE_JPEG
-// The original release of libjpeg v6b did not contain any extern "C" in case
-// its header files are included in a C++ environment. To avoid any linking
-// issues we need to add it on our own.
-extern "C" {
-#include <jpeglib.h>
-#include <jerror.h>
-}
-#endif
-
-namespace Graphics {
-
-JPEGDecoder::JPEGDecoder() : ImageDecoder(), _surface(), _colorSpace(kColorSpaceRGBA) {
-}
-
-JPEGDecoder::~JPEGDecoder() {
- destroy();
-}
-
-const Surface *JPEGDecoder::getSurface() const {
- return &_surface;
-}
-
-void JPEGDecoder::destroy() {
- _surface.free();
-}
-
-#ifdef USE_JPEG
-namespace {
-
-#define JPEG_BUFFER_SIZE 4096
-
-struct StreamSource : public jpeg_source_mgr {
- Common::SeekableReadStream *stream;
- bool startOfFile;
- JOCTET buffer[JPEG_BUFFER_SIZE];
-};
-
-void initSource(j_decompress_ptr cinfo) {
- StreamSource *source = (StreamSource *)cinfo->src;
- source->startOfFile = true;
-}
-
-boolean fillInputBuffer(j_decompress_ptr cinfo) {
- StreamSource *source = (StreamSource *)cinfo->src;
-
- uint32 bufferSize = source->stream->read((byte *)source->buffer, sizeof(source->buffer));
- if (bufferSize == 0) {
- if (source->startOfFile) {
- // An empty file is a fatal error
- ERREXIT(cinfo, JERR_INPUT_EMPTY);
- } else {
- // Otherwise we insert an EOF marker
- WARNMS(cinfo, JWRN_JPEG_EOF);
- source->buffer[0] = (JOCTET)0xFF;
- source->buffer[1] = (JOCTET)JPEG_EOI;
- bufferSize = 2;
- }
- }
-
- source->next_input_byte = source->buffer;
- source->bytes_in_buffer = bufferSize;
- source->startOfFile = false;
-
- return TRUE;
-}
-
-void skipInputData(j_decompress_ptr cinfo, long numBytes) {
- StreamSource *source = (StreamSource *)cinfo->src;
-
- if (numBytes > 0) {
- if (numBytes > (long)source->bytes_in_buffer) {
- // In case we need to skip more bytes than there are in the buffer
- // we will skip the remaining data and fill the buffer again
- numBytes -= (long)source->bytes_in_buffer;
-
- // Skip the remaining bytes
- source->stream->skip(numBytes);
-
- // Fill up the buffer again
- (*source->fill_input_buffer)(cinfo);
- } else {
- source->next_input_byte += (size_t)numBytes;
- source->bytes_in_buffer -= (size_t)numBytes;
- }
-
- }
-}
-
-void termSource(j_decompress_ptr cinfo) {
-}
-
-void jpeg_scummvm_src(j_decompress_ptr cinfo, Common::SeekableReadStream *stream) {
- StreamSource *source;
-
- // Initialize the source in case it has not been done yet.
- if (cinfo->src == NULL) {
- cinfo->src = (jpeg_source_mgr *)(*cinfo->mem->alloc_small)((j_common_ptr)cinfo, JPOOL_PERMANENT, sizeof(StreamSource));
- }
-
- source = (StreamSource *)cinfo->src;
- source->init_source = &initSource;
- source->fill_input_buffer = &fillInputBuffer;
- source->skip_input_data = &skipInputData;
- source->resync_to_restart = &jpeg_resync_to_restart;
- source->term_source = &termSource;
- source->bytes_in_buffer = 0;
- source->next_input_byte = NULL;
-
- source->stream = stream;
-}
-
-void errorExit(j_common_ptr cinfo) {
- char buffer[JMSG_LENGTH_MAX];
- (*cinfo->err->format_message)(cinfo, buffer);
- // This function is not allowed to return to the caller, thus we simply
- // error out with our error handling here.
- error("%s", buffer);
-}
-
-void outputMessage(j_common_ptr cinfo) {
- char buffer[JMSG_LENGTH_MAX];
- (*cinfo->err->format_message)(cinfo, buffer);
- // Is using debug here a good idea? Or do we want to ignore all libjpeg
- // messages?
- debug(3, "libjpeg: %s", buffer);
-}
-
-} // End of anonymous namespace
-#endif
-
-bool JPEGDecoder::loadStream(Common::SeekableReadStream &stream) {
-#ifdef USE_JPEG
- // Reset member variables from previous decodings
- destroy();
-
- jpeg_decompress_struct cinfo;
- jpeg_error_mgr jerr;
-
- // Initialize error handling callbacks
- cinfo.err = jpeg_std_error(&jerr);
- cinfo.err->error_exit = &errorExit;
- cinfo.err->output_message = &outputMessage;
-
- // Initialize the decompression structure
- jpeg_create_decompress(&cinfo);
-
- // Initialize our buffer handling
- jpeg_scummvm_src(&cinfo, &stream);
-
- // Read the file header
- jpeg_read_header(&cinfo, TRUE);
-
- // We can request YUV output because Groovie requires it
- switch (_colorSpace) {
- case kColorSpaceRGBA:
- cinfo.out_color_space = JCS_RGB;
- break;
-
- case kColorSpaceYUV:
- cinfo.out_color_space = JCS_YCbCr;
- break;
- }
-
- // Actually start decompressing the image
- jpeg_start_decompress(&cinfo);
-
- // Allocate buffers for the output data
- switch (_colorSpace) {
- case kColorSpaceRGBA:
- // We use RGBA8888 in this scenario
- _surface.create(cinfo.output_width, cinfo.output_height, Graphics::PixelFormat(4, 8, 8, 8, 0, 24, 16, 8, 0));
- break;
-
- case kColorSpaceYUV:
- // We use YUV with 3 bytes per pixel otherwise.
- // This is pretty ugly since our PixelFormat cannot express YUV...
- _surface.create(cinfo.output_width, cinfo.output_height, Graphics::PixelFormat(3, 0, 0, 0, 0, 0, 0, 0, 0));
- break;
- }
-
- // Allocate buffer for one scanline
- assert(cinfo.output_components == 3);
- JDIMENSION pitch = cinfo.output_width * cinfo.output_components;
- assert(_surface.pitch >= pitch);
- JSAMPARRAY buffer = (*cinfo.mem->alloc_sarray)((j_common_ptr)&cinfo, JPOOL_IMAGE, pitch, 1);
-
- // Go through the image data scanline by scanline
- while (cinfo.output_scanline < cinfo.output_height) {
- byte *dst = (byte *)_surface.getBasePtr(0, cinfo.output_scanline);
-
- jpeg_read_scanlines(&cinfo, buffer, 1);
-
- const byte *src = buffer[0];
- switch (_colorSpace) {
- case kColorSpaceRGBA: {
- for (int remaining = cinfo.output_width; remaining > 0; --remaining) {
- byte r = *src++;
- byte g = *src++;
- byte b = *src++;
- // We need to insert a alpha value of 255 (opaque) here.
-#ifdef SCUMM_BIG_ENDIAN
- *dst++ = r;
- *dst++ = g;
- *dst++ = b;
- *dst++ = 0xFF;
-#else
- *dst++ = 0xFF;
- *dst++ = b;
- *dst++ = g;
- *dst++ = r;
-#endif
- }
- } break;
-
- case kColorSpaceYUV:
- memcpy(dst, src, pitch);
- break;
- }
- }
-
- // We are done with decompressing, thus free all the data
- jpeg_finish_decompress(&cinfo);
- jpeg_destroy_decompress(&cinfo);
-
- return true;
-#else
- return false;
-#endif
-}
-
-} // End of Graphics namespace
diff --git a/graphics/decoders/jpeg.h b/graphics/decoders/jpeg.h
deleted file mode 100644
index 8460bc2698..0000000000
--- a/graphics/decoders/jpeg.h
+++ /dev/null
@@ -1,95 +0,0 @@
-/* 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.
- *
- */
-
-/**
- * @file
- * Image decoder used in engines:
- * - groovie
- * - mohawk
- * - wintermute
- */
-
-#ifndef GRAPHICS_JPEG_H
-#define GRAPHICS_JPEG_H
-
-#include "graphics/surface.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common {
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-class JPEGDecoder : public ImageDecoder {
-public:
- JPEGDecoder();
- ~JPEGDecoder();
-
- // ImageDecoder API
- virtual void destroy();
- virtual bool loadStream(Common::SeekableReadStream &str);
- virtual const Surface *getSurface() const;
-
- // Special API for JPEG
- enum ColorSpace {
- /**
- * Output 32bit RGBA data.
- *
- * This is the default output.
- */
- kColorSpaceRGBA,
-
- /**
- * Output (interleaved) YUV data.
- *
- * Be aware that some images cannot be output in YUV mode.
- * These are (non-standard) JPEG images which are in RGB colorspace.
- *
- * The resulting Surface will have a PixelFormat with 3 bytes per
- * pixel and the remaining entries are completely zeroed. This works
- * around the fact that PixelFormat can only describe RGB formats.
- *
- * You should only use this when you are really aware of what you are
- * doing!
- */
- kColorSpaceYUV
- };
-
- /**
- * Request the output color space. This can be used to obtain raw YUV
- * data from the JPEG file. But this might not work for all files!
- *
- * The decoder itself defaults to RGBA.
- *
- * @param outSpace The color space to output.
- */
- void setOutputColorSpace(ColorSpace outSpace) { _colorSpace = outSpace; }
-
-private:
- Graphics::Surface _surface;
- ColorSpace _colorSpace;
-};
-
-} // End of Graphics namespace
-
-#endif // GRAPHICS_JPEG_H
diff --git a/graphics/decoders/pcx.cpp b/graphics/decoders/pcx.cpp
deleted file mode 100644
index eb9b4c997d..0000000000
--- a/graphics/decoders/pcx.cpp
+++ /dev/null
@@ -1,213 +0,0 @@
-/* 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.
- */
-
-#include "common/stream.h"
-#include "common/textconsole.h"
-
-#include "graphics/pixelformat.h"
-#include "graphics/surface.h"
-#include "graphics/decoders/pcx.h"
-
-/**
- * Based on the PCX specs:
- * http://www.fileformat.info/format/pcx/spec/a10e75307b3a4cc49c3bbe6db4c41fa2/view.htm
- * and the PCX decoder of FFmpeg (libavcodec/pcx.c):
- * http://git.videolan.org/?p=ffmpeg.git;a=blob;f=libavcodec/pcx.c
- */
-
-namespace Graphics {
-
-PCXDecoder::PCXDecoder() {
- _surface = 0;
- _palette = 0;
- _paletteColorCount = 0;
-}
-
-PCXDecoder::~PCXDecoder() {
- destroy();
-}
-
-void PCXDecoder::destroy() {
- if (_surface) {
- _surface->free();
- delete _surface;
- _surface = 0;
- }
-
- delete[] _palette;
- _palette = 0;
- _paletteColorCount = 0;
-}
-
-bool PCXDecoder::loadStream(Common::SeekableReadStream &stream) {
- destroy();
-
- if (stream.readByte() != 0x0a) // ZSoft PCX
- return false;
-
- byte version = stream.readByte(); // 0 - 5
- if (version > 5)
- return false;
-
- bool compressed = stream.readByte(); // encoding, 1 = run length encoding
- byte bitsPerPixel = stream.readByte(); // 1, 2, 4 or 8
-
- // Window
- uint16 xMin = stream.readUint16LE();
- uint16 yMin = stream.readUint16LE();
- uint16 xMax = stream.readUint16LE();
- uint16 yMax = stream.readUint16LE();
-
- uint16 width = xMax - xMin + 1;
- uint16 height = yMax - yMin + 1;
-
- if (xMax < xMin || yMax < yMin) {
- warning("Invalid PCX image dimensions");
- return false;
- }
-
- stream.skip(4); // HDpi, VDpi
-
- // Read the EGA palette (colormap)
- _palette = new byte[16 * 3];
- for (uint16 i = 0; i < 16; i++) {
- _palette[i * 3 + 0] = stream.readByte();
- _palette[i * 3 + 1] = stream.readByte();
- _palette[i * 3 + 2] = stream.readByte();
- }
-
- if (stream.readByte() != 0) // reserved, should be set to 0
- return false;
-
- byte nPlanes = stream.readByte();
- uint16 bytesPerLine = stream.readUint16LE();
- uint16 bytesPerscanLine = nPlanes * bytesPerLine;
-
- if (bytesPerscanLine < width * bitsPerPixel * nPlanes / 8) {
- warning("PCX data is corrupted");
- return false;
- }
-
- stream.skip(60); // PaletteInfo, HscreenSize, VscreenSize, Filler
-
- _surface = new Graphics::Surface();
-
- byte *scanLine = new byte[bytesPerscanLine];
- byte *dst;
- int x, y;
-
- if (nPlanes == 3 && bitsPerPixel == 8) { // 24bpp
- Graphics::PixelFormat format = Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0);
- _surface->create(width, height, format);
- dst = (byte *)_surface->getPixels();
- _paletteColorCount = 0;
-
- for (y = 0; y < height; y++) {
- decodeRLE(stream, scanLine, bytesPerscanLine, compressed);
-
- for (x = 0; x < width; x++) {
- byte b = scanLine[x];
- byte g = scanLine[x + bytesPerLine];
- byte r = scanLine[x + (bytesPerLine << 1)];
- uint32 color = format.RGBToColor(r, g, b);
-
- *((uint32 *)dst) = color;
- dst += format.bytesPerPixel;
- }
- }
- } else if (nPlanes == 1 && bitsPerPixel == 8) { // 8bpp indexed
- _surface->create(width, height, Graphics::PixelFormat::createFormatCLUT8());
- dst = (byte *)_surface->getPixels();
- _paletteColorCount = 16;
-
- for (y = 0; y < height; y++, dst += _surface->pitch) {
- decodeRLE(stream, scanLine, bytesPerscanLine, compressed);
- memcpy(dst, scanLine, width);
- }
-
- if (version == 5) {
- if (stream.readByte() != 12) {
- warning("Expected a palette after the PCX image data");
- delete[] scanLine;
- return false;
- }
-
- // Read the VGA palette
- delete[] _palette;
- _palette = new byte[256 * 3];
- for (uint16 i = 0; i < 256; i++) {
- _palette[i * 3 + 0] = stream.readByte();
- _palette[i * 3 + 1] = stream.readByte();
- _palette[i * 3 + 2] = stream.readByte();
- }
-
- _paletteColorCount = 256;
- }
- } else if ((nPlanes == 2 || nPlanes == 3 || nPlanes == 4) && bitsPerPixel == 1) { // planar, 4, 8 or 16 colors
- _surface->create(width, height, Graphics::PixelFormat::createFormatCLUT8());
- dst = (byte *)_surface->getPixels();
- _paletteColorCount = 16;
-
- for (y = 0; y < height; y++, dst += _surface->pitch) {
- decodeRLE(stream, scanLine, bytesPerscanLine, compressed);
-
- for (x = 0; x < width; x++) {
- int m = 0x80 >> (x & 7), v = 0;
- for (int i = nPlanes - 1; i >= 0; i--) {
- v <<= 1;
- v += (scanLine[i * bytesPerLine + (x >> 3)] & m) == 0 ? 0 : 1;
- }
- dst[x] = v;
- }
- }
- } else {
- // Known unsupported case: 1 plane and bpp < 8 (1, 2 or 4)
- warning("Invalid PCX file (%d planes, %d bpp)", nPlanes, bitsPerPixel);
- delete[] scanLine;
- return false;
- }
-
- delete[] scanLine;
-
- return true;
-}
-
-void PCXDecoder::decodeRLE(Common::SeekableReadStream &stream, byte *dst, uint32 bytesPerscanLine, bool compressed) {
- uint32 i = 0;
- byte run, value;
-
- if (compressed) {
- while (i < bytesPerscanLine) {
- run = 1;
- value = stream.readByte();
- if (value >= 0xc0) {
- run = value & 0x3f;
- value = stream.readByte();
- }
- while (i < bytesPerscanLine && run--)
- dst[i++] = value;
- }
- } else {
- stream.read(dst, bytesPerscanLine);
- }
-}
-
-} // End of namespace Graphics
diff --git a/graphics/decoders/pcx.h b/graphics/decoders/pcx.h
deleted file mode 100644
index b25166b3d9..0000000000
--- a/graphics/decoders/pcx.h
+++ /dev/null
@@ -1,68 +0,0 @@
-/* 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.
- */
-
-/**
- * PCX decoder used in engines:
- * - dreamweb
- * - hugo
- * - queen
- * - tucker
- */
-
-#ifndef GRAPHICS_DECODERS_PCX_H
-#define GRAPHICS_DECODERS_PCX_H
-
-#include "common/scummsys.h"
-#include "common/str.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common{
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct PixelFormat;
-struct Surface;
-
-class PCXDecoder : public ImageDecoder {
-public:
- PCXDecoder();
- virtual ~PCXDecoder();
-
- // ImageDecoder API
- void destroy();
- virtual bool loadStream(Common::SeekableReadStream &stream);
- virtual const Surface *getSurface() const { return _surface; }
- const byte *getPalette() const { return _palette; }
- uint16 getPaletteColorCount() const { return _paletteColorCount; }
-
-private:
- void decodeRLE(Common::SeekableReadStream &stream, byte *dst, uint32 bytesPerScanline, bool compressed);
-
- Surface *_surface;
- byte *_palette;
- uint16 _paletteColorCount;
-};
-
-} // End of namespace Graphics
-
-#endif
diff --git a/graphics/decoders/pict.cpp b/graphics/decoders/pict.cpp
deleted file mode 100644
index f3e17b33e2..0000000000
--- a/graphics/decoders/pict.cpp
+++ /dev/null
@@ -1,580 +0,0 @@
-/* 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.
- *
- */
-
-#include "common/debug.h"
-#include "common/endian.h"
-#include "common/stream.h"
-#include "common/substream.h"
-#include "common/textconsole.h"
-
-#include "graphics/surface.h"
-#include "graphics/decoders/jpeg.h"
-#include "graphics/decoders/pict.h"
-
-namespace Graphics {
-
-// The PICT code is based off of the QuickDraw specs:
-// http://developer.apple.com/legacy/mac/library/documentation/mac/QuickDraw/QuickDraw-461.html
-// http://developer.apple.com/legacy/mac/library/documentation/mac/QuickDraw/QuickDraw-269.html
-
-PICTDecoder::PICTDecoder() {
- _outputSurface = 0;
- _paletteColorCount = 0;
-}
-
-PICTDecoder::~PICTDecoder() {
- destroy();
-}
-
-void PICTDecoder::destroy() {
- if (_outputSurface) {
- _outputSurface->free();
- delete _outputSurface;
- _outputSurface = 0;
- }
-
- _paletteColorCount = 0;
-}
-
-#define OPCODE(a, b, c) _opcodes.push_back(PICTOpcode(a, &PICTDecoder::b, c))
-
-void PICTDecoder::setupOpcodesCommon() {
- OPCODE(0x0000, o_nop, "NOP");
- OPCODE(0x0001, o_clip, "Clip");
- OPCODE(0x0003, o_txFont, "TxFont");
- OPCODE(0x0004, o_txFace, "TxFace");
- OPCODE(0x0007, o_pnSize, "PnSize");
- OPCODE(0x000D, o_txSize, "TxSize");
- OPCODE(0x0010, o_txRatio, "TxRatio");
- OPCODE(0x0011, o_versionOp, "VersionOp");
- OPCODE(0x001E, o_nop, "DefHilite");
- OPCODE(0x0028, o_longText, "LongText");
- OPCODE(0x00A1, o_longComment, "LongComment");
- OPCODE(0x00FF, o_opEndPic, "OpEndPic");
- OPCODE(0x0C00, o_headerOp, "HeaderOp");
-}
-
-void PICTDecoder::setupOpcodesNormal() {
- setupOpcodesCommon();
- OPCODE(0x0098, on_packBitsRect, "PackBitsRect");
- OPCODE(0x009A, on_directBitsRect, "DirectBitsRect");
- OPCODE(0x8200, on_compressedQuickTime, "CompressedQuickTime");
-}
-
-void PICTDecoder::setupOpcodesQuickTime() {
- setupOpcodesCommon();
- OPCODE(0x0098, oq_packBitsRect, "PackBitsRect");
- OPCODE(0x009A, oq_directBitsRect, "DirectBitsRect");
- OPCODE(0x8200, oq_compressedQuickTime, "CompressedQuickTime");
-}
-
-#undef OPCODE
-
-void PICTDecoder::o_nop(Common::SeekableReadStream &) {
- // Nothing to do
-}
-
-void PICTDecoder::o_clip(Common::SeekableReadStream &stream) {
- // Ignore
- stream.skip(stream.readUint16BE() - 2);
-}
-
-void PICTDecoder::o_txFont(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
-}
-
-void PICTDecoder::o_txFace(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readByte();
-}
-
-void PICTDecoder::o_pnSize(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
- stream.readUint16BE();
-}
-
-void PICTDecoder::o_txSize(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
-}
-
-void PICTDecoder::o_txRatio(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
- stream.readUint16BE();
- stream.readUint16BE();
- stream.readUint16BE();
-}
-
-void PICTDecoder::o_versionOp(Common::SeekableReadStream &stream) {
- // We only support v2 extended
- if (stream.readUint16BE() != 0x02FF)
- error("Unknown PICT version");
-}
-
-void PICTDecoder::o_longText(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
- stream.readUint16BE();
- stream.skip(stream.readByte());
-}
-
-void PICTDecoder::o_longComment(Common::SeekableReadStream &stream) {
- // Ignore
- stream.readUint16BE();
- stream.skip(stream.readUint16BE());
-}
-
-void PICTDecoder::o_opEndPic(Common::SeekableReadStream &stream) {
- // We've reached the end of the picture
- _continueParsing = false;
-}
-
-void PICTDecoder::o_headerOp(Common::SeekableReadStream &stream) {
- // Read the basic header, but we don't really have to do anything with it
- /* uint16 version = */ stream.readUint16BE();
- stream.readUint16BE(); // Reserved
- /* uint32 hRes = */ stream.readUint32BE();
- /* uint32 vRes = */ stream.readUint32BE();
- Common::Rect origResRect;
- origResRect.top = stream.readUint16BE();
- origResRect.left = stream.readUint16BE();
- origResRect.bottom = stream.readUint16BE();
- origResRect.right = stream.readUint16BE();
- stream.readUint32BE(); // Reserved
-}
-
-void PICTDecoder::on_packBitsRect(Common::SeekableReadStream &stream) {
- // Unpack data (8bpp or lower)
- unpackBitsRect(stream, true);
-}
-
-void PICTDecoder::on_directBitsRect(Common::SeekableReadStream &stream) {
- // Unpack data (16bpp or higher)
- unpackBitsRect(stream, false);
-}
-
-void PICTDecoder::on_compressedQuickTime(Common::SeekableReadStream &stream) {
- // OK, here's the fun. We get to completely change how QuickDraw draws
- // the data in PICT files.
-
- // Swap out the opcodes to the new ones
- _opcodes.clear();
- setupOpcodesQuickTime();
-
- // We'll decode the first QuickTime data from here, but the QuickTime-specific
- // opcodes will take over from here on out. Normal opcodes, signing off.
- decodeCompressedQuickTime(stream);
-}
-
-void PICTDecoder::oq_packBitsRect(Common::SeekableReadStream &stream) {
- // Skip any data here (8bpp or lower)
- skipBitsRect(stream, true);
-}
-
-void PICTDecoder::oq_directBitsRect(Common::SeekableReadStream &stream) {
- // Skip any data here (16bpp or higher)
- skipBitsRect(stream, false);
-}
-
-void PICTDecoder::oq_compressedQuickTime(Common::SeekableReadStream &stream) {
- // Just pass the data along
- decodeCompressedQuickTime(stream);
-}
-
-bool PICTDecoder::loadStream(Common::SeekableReadStream &stream) {
- destroy();
-
- // Initialize opcodes to their normal state
- _opcodes.clear();
- setupOpcodesNormal();
-
- _continueParsing = true;
- memset(_palette, 0, sizeof(_palette));
-
- uint16 fileSize = stream.readUint16BE();
-
- // If we have no file size here, we probably have a PICT from a file
- // and not a resource. The other two bytes are the fileSize which we
- // don't actually need (and already read if from a resource).
- if (!fileSize)
- stream.seek(512 + 2);
-
- _imageRect.top = stream.readUint16BE();
- _imageRect.left = stream.readUint16BE();
- _imageRect.bottom = stream.readUint16BE();
- _imageRect.right = stream.readUint16BE();
- _imageRect.debugPrint(0, "PICT Rect:");
-
- // NOTE: This is only a subset of the full PICT format.
- // - Only V2 (Extended) Images Supported
- // - CompressedQuickTime (JPEG) compressed data is supported
- // - DirectBitsRect/PackBitsRect compressed data is supported
- for (uint32 opNum = 0; !stream.eos() && !stream.err() && stream.pos() < stream.size() && _continueParsing; opNum++) {
- // PICT v2 opcodes are two bytes
- uint16 opcode = stream.readUint16BE();
-
- if (opNum == 0 && opcode != 0x0011) {
- warning("Cannot find PICT version opcode");
- return false;
- } else if (opNum == 1 && opcode != 0x0C00) {
- warning("Cannot find PICT header opcode");
- return false;
- }
-
- // Since opcodes are word-aligned, we need to mark our starting
- // position here.
- uint32 startPos = stream.pos();
-
- for (uint32 i = 0; i < _opcodes.size(); i++) {
- if (_opcodes[i].op == opcode) {
- debug(4, "Running PICT opcode %04x '%s'", opcode, _opcodes[i].desc);
- (this->*(_opcodes[i].proc))(stream);
- break;
- } else if (i == _opcodes.size() - 1) {
- // Unknown opcode; attempt to continue forward
- warning("Unknown PICT opcode %04x", opcode);
- }
- }
-
- // Align
- stream.skip((stream.pos() - startPos) & 1);
- }
-
- return _outputSurface;
-}
-
-PICTDecoder::PixMap PICTDecoder::readPixMap(Common::SeekableReadStream &stream, bool hasBaseAddr) {
- PixMap pixMap;
- pixMap.baseAddr = hasBaseAddr ? stream.readUint32BE() : 0;
- pixMap.rowBytes = stream.readUint16BE() & 0x3fff;
- pixMap.bounds.top = stream.readUint16BE();
- pixMap.bounds.left = stream.readUint16BE();
- pixMap.bounds.bottom = stream.readUint16BE();
- pixMap.bounds.right = stream.readUint16BE();
- pixMap.pmVersion = stream.readUint16BE();
- pixMap.packType = stream.readUint16BE();
- pixMap.packSize = stream.readUint32BE();
- pixMap.hRes = stream.readUint32BE();
- pixMap.vRes = stream.readUint32BE();
- pixMap.pixelType = stream.readUint16BE();
- pixMap.pixelSize = stream.readUint16BE();
- pixMap.cmpCount = stream.readUint16BE();
- pixMap.cmpSize = stream.readUint16BE();
- pixMap.planeBytes = stream.readUint32BE();
- pixMap.pmTable = stream.readUint32BE();
- pixMap.pmReserved = stream.readUint32BE();
- return pixMap;
-}
-
-struct PackBitsRectData {
- PICTDecoder::PixMap pixMap;
- Common::Rect srcRect;
- Common::Rect dstRect;
- uint16 mode;
-};
-
-void PICTDecoder::unpackBitsRect(Common::SeekableReadStream &stream, bool withPalette) {
- PackBitsRectData packBitsData;
- packBitsData.pixMap = readPixMap(stream, !withPalette);
-
- // Read in the palette if there is one present
- if (withPalette) {
- // See http://developer.apple.com/legacy/mac/library/documentation/mac/QuickDraw/QuickDraw-267.html
- stream.readUint32BE(); // seed
- stream.readUint16BE(); // flags
- _paletteColorCount = stream.readUint16BE() + 1;
-
- for (uint32 i = 0; i < _paletteColorCount; i++) {
- stream.readUint16BE();
- _palette[i * 3] = stream.readUint16BE() >> 8;
- _palette[i * 3 + 1] = stream.readUint16BE() >> 8;
- _palette[i * 3 + 2] = stream.readUint16BE() >> 8;
- }
- }
-
- packBitsData.srcRect.top = stream.readUint16BE();
- packBitsData.srcRect.left = stream.readUint16BE();
- packBitsData.srcRect.bottom = stream.readUint16BE();
- packBitsData.srcRect.right = stream.readUint16BE();
- packBitsData.dstRect.top = stream.readUint16BE();
- packBitsData.dstRect.left = stream.readUint16BE();
- packBitsData.dstRect.bottom = stream.readUint16BE();
- packBitsData.dstRect.right = stream.readUint16BE();
- packBitsData.mode = stream.readUint16BE();
-
- uint16 width = packBitsData.srcRect.width();
- uint16 height = packBitsData.srcRect.height();
-
- byte bytesPerPixel = 0;
-
- if (packBitsData.pixMap.pixelSize <= 8)
- bytesPerPixel = 1;
- else if (packBitsData.pixMap.pixelSize == 32)
- bytesPerPixel = packBitsData.pixMap.cmpCount;
- else
- bytesPerPixel = packBitsData.pixMap.pixelSize / 8;
-
- // Ensure we have enough space in the buffer to hold an entire line's worth of pixels
- uint32 lineSize = MAX<int>(width * bytesPerPixel + (8 * 2 / packBitsData.pixMap.pixelSize), packBitsData.pixMap.rowBytes);
- byte *buffer = new byte[lineSize * height];
-
- // Read in amount of data per row
- for (uint16 i = 0; i < packBitsData.pixMap.bounds.height(); i++) {
- // NOTE: Compression 0 is "default". The format in SCI games is packed when 0.
- // In the future, we may need to have something to set the "default" packing
- // format, but this is good for now.
-
- if (packBitsData.pixMap.packType == 1 || packBitsData.pixMap.rowBytes < 8) { // Unpacked, Pad-Byte (on 24-bit)
- // TODO: Finish this. Hasn't been needed (yet).
- error("Unpacked DirectBitsRect data (padded)");
- } else if (packBitsData.pixMap.packType == 2) { // Unpacked, No Pad-Byte (on 24-bit)
- // TODO: Finish this. Hasn't been needed (yet).
- error("Unpacked DirectBitsRect data (not padded)");
- } else if (packBitsData.pixMap.packType == 0 || packBitsData.pixMap.packType > 2) { // Packed
- uint16 byteCount = (packBitsData.pixMap.rowBytes > 250) ? stream.readUint16BE() : stream.readByte();
- unpackBitsLine(buffer + i * width * bytesPerPixel, packBitsData.pixMap.rowBytes, stream.readStream(byteCount), packBitsData.pixMap.pixelSize, bytesPerPixel);
- }
- }
-
- _outputSurface = new Graphics::Surface();
-
- switch (bytesPerPixel) {
- case 1:
- // Just copy to the image
- _outputSurface->create(width, height, PixelFormat::createFormatCLUT8());
- memcpy(_outputSurface->getPixels(), buffer, _outputSurface->w * _outputSurface->h);
- break;
- case 2:
- // We have a 16-bit surface
- _outputSurface->create(width, height, PixelFormat(2, 5, 5, 5, 0, 10, 5, 0, 0));
- for (uint16 y = 0; y < _outputSurface->h; y++)
- for (uint16 x = 0; x < _outputSurface->w; x++)
- WRITE_UINT16(_outputSurface->getBasePtr(x, y), READ_UINT16(buffer + (y * _outputSurface->w + x) * 2));
- break;
- case 3:
- // We have a planar 24-bit surface
- _outputSurface->create(width, height, Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0));
- for (uint16 y = 0; y < _outputSurface->h; y++) {
- for (uint16 x = 0; x < _outputSurface->w; x++) {
- byte r = *(buffer + y * _outputSurface->w * 3 + x);
- byte g = *(buffer + y * _outputSurface->w * 3 + _outputSurface->w + x);
- byte b = *(buffer + y * _outputSurface->w * 3 + _outputSurface->w * 2 + x);
- *((uint32 *)_outputSurface->getBasePtr(x, y)) = _outputSurface->format.RGBToColor(r, g, b);
- }
- }
- break;
- case 4:
- // We have a planar 32-bit surface
- // Note that we ignore the alpha channel since it seems to not be correct
- // Mac OS X does not ignore it, but then displays it incorrectly. Photoshop
- // does ignore it and displays it correctly.
- _outputSurface->create(width, height, Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0));
- for (uint16 y = 0; y < _outputSurface->h; y++) {
- for (uint16 x = 0; x < _outputSurface->w; x++) {
- byte a = 0xFF;
- byte r = *(buffer + y * _outputSurface->w * 4 + _outputSurface->w + x);
- byte g = *(buffer + y * _outputSurface->w * 4 + _outputSurface->w * 2 + x);
- byte b = *(buffer + y * _outputSurface->w * 4 + _outputSurface->w * 3 + x);
- *((uint32 *)_outputSurface->getBasePtr(x, y)) = _outputSurface->format.ARGBToColor(a, r, g, b);
- }
- }
- break;
- }
-
- delete[] buffer;
-}
-
-void PICTDecoder::unpackBitsLine(byte *out, uint32 length, Common::SeekableReadStream *data, byte bitsPerPixel, byte bytesPerPixel) {
- uint32 dataDecoded = 0;
- byte bytesPerDecode = (bytesPerPixel == 2) ? 2 : 1;
-
- while (data->pos() < data->size() && dataDecoded < length) {
- byte op = data->readByte();
-
- if (op & 0x80) {
- uint32 runSize = (op ^ 255) + 2;
- uint16 value = (bytesPerDecode == 2) ? data->readUint16BE() : data->readByte();
-
- for (uint32 i = 0; i < runSize; i++) {
- if (bytesPerDecode == 2) {
- WRITE_UINT16(out, value);
- out += 2;
- } else {
- outputPixelBuffer(out, value, bitsPerPixel);
- }
- }
- dataDecoded += runSize * bytesPerDecode;
- } else {
- uint32 runSize = op + 1;
-
- if (bytesPerDecode == 1) {
- for (uint32 i = 0; i < runSize; i++)
- outputPixelBuffer(out, data->readByte(), bitsPerPixel);
- } else {
- for (uint32 i = 0; i < runSize; i++) {
- WRITE_UINT16(out, data->readUint16BE());
- out += 2;
- }
- }
-
- dataDecoded += runSize * bytesPerDecode;
- }
- }
-
- // HACK: Even if the data is 24-bit, rowBytes is still 32-bit
- if (bytesPerPixel == 3)
- dataDecoded += length / 4;
-
- if (length != dataDecoded)
- warning("Mismatched PackBits read (%d/%d)", dataDecoded, length);
-
- delete data;
-}
-
-void PICTDecoder::outputPixelBuffer(byte *&out, byte value, byte bitsPerPixel) {
- switch (bitsPerPixel) {
- case 1:
- for (int i = 7; i >= 0; i--)
- *out++ = (value >> i) & 1;
- break;
- case 2:
- for (int i = 6; i >= 0; i -= 2)
- *out++ = (value >> i) & 3;
- break;
- case 4:
- *out++ = (value >> 4) & 0xf;
- *out++ = value & 0xf;
- break;
- default:
- *out++ = value;
- }
-}
-
-void PICTDecoder::skipBitsRect(Common::SeekableReadStream &stream, bool withPalette) {
- // Step through a PackBitsRect/DirectBitsRect function
-
- if (!withPalette)
- stream.readUint32BE();
-
- uint16 rowBytes = stream.readUint16BE();
- uint16 height = stream.readUint16BE();
- stream.readUint16BE();
- height = stream.readUint16BE() - height;
- stream.readUint16BE();
-
- uint16 packType;
-
- // Top two bits signify PixMap vs BitMap
- if (rowBytes & 0xC000) {
- // PixMap
- stream.readUint16BE();
- packType = stream.readUint16BE();
- stream.skip(14);
- stream.readUint16BE(); // pixelSize
- stream.skip(16);
-
- if (withPalette) {
- stream.readUint32BE();
- stream.readUint16BE();
- stream.skip((stream.readUint16BE() + 1) * 8);
- }
-
- rowBytes &= 0x3FFF;
- } else {
- // BitMap
- packType = 0;
- }
-
- stream.skip(18);
-
- for (uint16 i = 0; i < height; i++) {
- if (packType == 1 || packType == 2 || rowBytes < 8)
- error("Unpacked PackBitsRect data");
- else if (packType == 0 || packType > 2)
- stream.skip((rowBytes > 250) ? stream.readUint16BE() : stream.readByte());
- }
-}
-
-// Compressed QuickTime details can be found here:
-// http://developer.apple.com/legacy/mac/library/#documentation/QuickTime/Rm/CompressDecompress/ImageComprMgr/B-Chapter/2TheImageCompression.html
-// http://developer.apple.com/legacy/mac/library/#documentation/QuickTime/Rm/CompressDecompress/ImageComprMgr/F-Chapter/6WorkingwiththeImage.html
-void PICTDecoder::decodeCompressedQuickTime(Common::SeekableReadStream &stream) {
- // First, read all the fields from the opcode
- uint32 dataSize = stream.readUint32BE();
- uint32 startPos = stream.pos();
-
- /* uint16 version = */ stream.readUint16BE();
-
- // Read in the display matrix
- uint32 matrix[3][3];
- for (uint32 i = 0; i < 3; i++)
- for (uint32 j = 0; j < 3; j++)
- matrix[i][j] = stream.readUint32BE();
-
- // We currently only support offseting images vertically from the matrix
- uint16 xOffset = 0;
- uint16 yOffset = matrix[2][1] >> 16;
-
- uint32 matteSize = stream.readUint32BE();
- stream.skip(8); // matte rect
- /* uint16 transferMode = */ stream.readUint16BE();
- stream.skip(8); // src rect
- /* uint32 accuracy = */ stream.readUint32BE();
- uint32 maskSize = stream.readUint32BE();
-
- // Skip the matte and mask
- stream.skip(matteSize + maskSize);
-
- // Now we've reached the image descriptor, so read the relevant data from that
- uint32 idStart = stream.pos();
- uint32 idSize = stream.readUint32BE();
- uint32 codec = stream.readUint32BE();
- stream.skip(36); // miscellaneous stuff
- uint32 jpegSize = stream.readUint32BE();
- stream.skip(idSize - (stream.pos() - idStart)); // more useless stuff
-
- if (codec != MKTAG('j', 'p', 'e', 'g'))
- error("Unhandled CompressedQuickTime format '%s'", tag2str(codec));
-
- Common::SeekableSubReadStream jpegStream(&stream, stream.pos(), stream.pos() + jpegSize);
-
- JPEGDecoder jpeg;
- if (!jpeg.loadStream(jpegStream))
- error("PICTDecoder::decodeCompressedQuickTime(): Could not decode JPEG data");
-
- const Graphics::Surface *jpegSurface = jpeg.getSurface();
-
- if (!_outputSurface) {
- _outputSurface = new Graphics::Surface();
- _outputSurface->create(_imageRect.width(), _imageRect.height(), jpegSurface->format);
- }
-
- for (uint16 y = 0; y < jpegSurface->h; y++)
- memcpy(_outputSurface->getBasePtr(0 + xOffset, y + yOffset), jpegSurface->getBasePtr(0, y), jpegSurface->w * jpegSurface->format.bytesPerPixel);
-
- stream.seek(startPos + dataSize);
-}
-
-} // End of namespace Graphics
diff --git a/graphics/decoders/pict.h b/graphics/decoders/pict.h
deleted file mode 100644
index 6f0d86c7a1..0000000000
--- a/graphics/decoders/pict.h
+++ /dev/null
@@ -1,140 +0,0 @@
-/* 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.
- *
- */
-
-/**
- * @file
- * Image decoder used in engines:
- * - mohawk
- * - pegasus
- * - sci
- */
-
-#ifndef GRAPHICS_PICT_H
-#define GRAPHICS_PICT_H
-
-#include "common/array.h"
-#include "common/rect.h"
-#include "common/scummsys.h"
-
-#include "graphics/decoders/image_decoder.h"
-#include "graphics/pixelformat.h"
-
-namespace Common {
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct Surface;
-
-#define DECLARE_OPCODE(x) void x(Common::SeekableReadStream &stream)
-
-class PICTDecoder : public ImageDecoder {
-public:
- PICTDecoder();
- ~PICTDecoder();
-
- // ImageDecoder API
- bool loadStream(Common::SeekableReadStream &stream);
- void destroy();
- const Surface *getSurface() const { return _outputSurface; }
- const byte *getPalette() const { return _palette; }
- uint16 getPaletteColorCount() const { return _paletteColorCount; }
-
- struct PixMap {
- uint32 baseAddr;
- uint16 rowBytes;
- Common::Rect bounds;
- uint16 pmVersion;
- uint16 packType;
- uint32 packSize;
- uint32 hRes;
- uint32 vRes;
- uint16 pixelType;
- uint16 pixelSize;
- uint16 cmpCount;
- uint16 cmpSize;
- uint32 planeBytes;
- uint32 pmTable;
- uint32 pmReserved;
- };
-
- static PixMap readPixMap(Common::SeekableReadStream &stream, bool hasBaseAddr = true);
-
-private:
- Common::Rect _imageRect;
- byte _palette[256 * 3];
- uint16 _paletteColorCount;
- Graphics::Surface *_outputSurface;
- bool _continueParsing;
-
- // Utility Functions
- void unpackBitsRect(Common::SeekableReadStream &stream, bool withPalette);
- void unpackBitsLine(byte *out, uint32 length, Common::SeekableReadStream *stream, byte bitsPerPixel, byte bytesPerPixel);
- void skipBitsRect(Common::SeekableReadStream &stream, bool withPalette);
- void decodeCompressedQuickTime(Common::SeekableReadStream &stream);
- void outputPixelBuffer(byte *&out, byte value, byte bitsPerPixel);
-
- // Opcodes
- typedef void (PICTDecoder::*OpcodeProcPICT)(Common::SeekableReadStream &stream);
- struct PICTOpcode {
- PICTOpcode() { op = 0; proc = 0; desc = 0; }
- PICTOpcode(uint16 o, OpcodeProcPICT p, const char *d) { op = o; proc = p; desc = d; }
- uint16 op;
- OpcodeProcPICT proc;
- const char *desc;
- };
- Common::Array<PICTOpcode> _opcodes;
-
- // Common Opcodes
- void setupOpcodesCommon();
- DECLARE_OPCODE(o_nop);
- DECLARE_OPCODE(o_clip);
- DECLARE_OPCODE(o_txFont);
- DECLARE_OPCODE(o_txFace);
- DECLARE_OPCODE(o_pnSize);
- DECLARE_OPCODE(o_txSize);
- DECLARE_OPCODE(o_txRatio);
- DECLARE_OPCODE(o_versionOp);
- DECLARE_OPCODE(o_longText);
- DECLARE_OPCODE(o_longComment);
- DECLARE_OPCODE(o_opEndPic);
- DECLARE_OPCODE(o_headerOp);
-
- // Regular-mode Opcodes
- void setupOpcodesNormal();
- DECLARE_OPCODE(on_packBitsRect);
- DECLARE_OPCODE(on_directBitsRect);
- DECLARE_OPCODE(on_compressedQuickTime);
-
- // QuickTime-mode Opcodes
- void setupOpcodesQuickTime();
- DECLARE_OPCODE(oq_packBitsRect);
- DECLARE_OPCODE(oq_directBitsRect);
- DECLARE_OPCODE(oq_compressedQuickTime);
-};
-
-#undef DECLARE_OPCODE
-
-} // End of namespace Graphics
-
-#endif
diff --git a/graphics/decoders/png.cpp b/graphics/decoders/png.cpp
deleted file mode 100644
index 5acb7b36f7..0000000000
--- a/graphics/decoders/png.cpp
+++ /dev/null
@@ -1,241 +0,0 @@
-/* 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.
- *
- */
-
-// Since we need to work with libpng here, we need to allow all symbols
-// to avoid compilation issues.
-#define FORBIDDEN_SYMBOL_ALLOW_ALL
-#include "common/scummsys.h"
-
-#ifdef USE_PNG
-#include <png.h>
-#endif
-
-#include "graphics/decoders/png.h"
-
-#include "graphics/pixelformat.h"
-#include "graphics/surface.h"
-
-#include "common/stream.h"
-
-namespace Graphics {
-
-PNGDecoder::PNGDecoder() : _outputSurface(0), _palette(0), _paletteColorCount(0), _stream(0) {
-}
-
-PNGDecoder::~PNGDecoder() {
- destroy();
-}
-
-void PNGDecoder::destroy() {
- if (_outputSurface) {
- _outputSurface->free();
- delete _outputSurface;
- _outputSurface = 0;
- }
- delete[] _palette;
- _palette = NULL;
-}
-
-#ifdef USE_PNG
-// libpng-error-handling:
-void pngError(png_structp pngptr, png_const_charp errorMsg) {
- error("%s", errorMsg);
-}
-
-void pngWarning(png_structp pngptr, png_const_charp warningMsg) {
- warning("%s", warningMsg);
-}
-
-// libpng-I/O-helper:
-void pngReadFromStream(png_structp pngPtr, png_bytep data, png_size_t length) {
- void *readIOptr = png_get_io_ptr(pngPtr);
- Common::SeekableReadStream *stream = (Common::SeekableReadStream *)readIOptr;
- stream->read(data, length);
-}
-#endif
-
-/*
- * This code is based on Broken Sword 2.5 engine
- *
- * Copyright (c) Malte Thiesen, Daniel Queteschiner and Michael Elsdoerfer
- *
- * Licensed under GNU GPL v2
- *
- */
-
-bool PNGDecoder::loadStream(Common::SeekableReadStream &stream) {
-#ifdef USE_PNG
- destroy();
-
- _stream = &stream;
-
- // First, check the PNG signature
- if (_stream->readUint32BE() != MKTAG(0x89, 'P', 'N', 'G')) {
- delete _stream;
- return false;
- }
- if (_stream->readUint32BE() != MKTAG(0x0d, 0x0a, 0x1a, 0x0a)) {
- delete _stream;
- return false;
- }
-
- // The following is based on the guide provided in:
- //http://www.libpng.org/pub/png/libpng-1.2.5-manual.html#section-3
- //http://www.libpng.org/pub/png/libpng-1.4.0-manual.pdf
- // along with the png-loading code used in the sword25-engine.
- png_structp pngPtr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
- if (!pngPtr) {
- delete _stream;
- return false;
- }
- png_infop infoPtr = png_create_info_struct(pngPtr);
- if (!infoPtr) {
- png_destroy_read_struct(&pngPtr, NULL, NULL);
- delete _stream;
- return false;
- }
- png_infop endInfo = png_create_info_struct(pngPtr);
- if (!endInfo) {
- png_destroy_read_struct(&pngPtr, &infoPtr, NULL);
- delete _stream;
- return false;
- }
-
- png_set_error_fn(pngPtr, NULL, pngError, pngWarning);
- // TODO: The manual says errors should be handled via setjmp
-
- png_set_read_fn(pngPtr, _stream, pngReadFromStream);
- png_set_crc_action(pngPtr, PNG_CRC_DEFAULT, PNG_CRC_WARN_USE);
- // We already verified the PNG-header
- png_set_sig_bytes(pngPtr, 8);
-
- // Read PNG header
- png_read_info(pngPtr, infoPtr);
-
- // No handling for unknown chunks yet.
- int bitDepth, colorType, width, height, interlaceType;
- png_uint_32 w, h;
- png_get_IHDR(pngPtr, infoPtr, &w, &h, &bitDepth, &colorType, &interlaceType, NULL, NULL);
- width = w;
- height = h;
-
- // Allocate memory for the final image data.
- // To keep memory framentation low this happens before allocating memory for temporary image data.
- _outputSurface = new Graphics::Surface();
-
- // Images of all color formats except PNG_COLOR_TYPE_PALETTE
- // will be transformed into ARGB images
- if (colorType == PNG_COLOR_TYPE_PALETTE && !png_get_valid(pngPtr, infoPtr, PNG_INFO_tRNS)) {
- int numPalette = 0;
- png_colorp palette = NULL;
- uint32 success = png_get_PLTE(pngPtr, infoPtr, &palette, &numPalette);
- if (success != PNG_INFO_PLTE) {
- png_destroy_read_struct(&pngPtr, &infoPtr, NULL);
- return false;
- }
- _paletteColorCount = numPalette;
- _palette = new byte[_paletteColorCount * 3];
- for (int i = 0; i < _paletteColorCount; i++) {
- _palette[(i * 3)] = palette[i].red;
- _palette[(i * 3) + 1] = palette[i].green;
- _palette[(i * 3) + 2] = palette[i].blue;
-
- }
- _outputSurface->create(width, height, Graphics::PixelFormat::createFormatCLUT8());
- png_set_packing(pngPtr);
- } else {
- _outputSurface->create(width, height, Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0));
- if (!_outputSurface->getPixels()) {
- error("Could not allocate memory for output image.");
- }
- if (bitDepth == 16)
- png_set_strip_16(pngPtr);
- if (bitDepth < 8)
- png_set_expand(pngPtr);
- if (png_get_valid(pngPtr, infoPtr, PNG_INFO_tRNS))
- png_set_expand(pngPtr);
- if (colorType == PNG_COLOR_TYPE_GRAY ||
- colorType == PNG_COLOR_TYPE_GRAY_ALPHA)
- png_set_gray_to_rgb(pngPtr);
-
- // PNGs are Big-Endian:
-#ifdef SCUMM_LITTLE_ENDIAN
- png_set_bgr(pngPtr);
- png_set_swap_alpha(pngPtr);
- if (colorType != PNG_COLOR_TYPE_RGB_ALPHA)
- png_set_filler(pngPtr, 0xff, PNG_FILLER_BEFORE);
-#else
- if (colorType != PNG_COLOR_TYPE_RGB_ALPHA)
- png_set_filler(pngPtr, 0xff, PNG_FILLER_AFTER);
-#endif
-
- }
-
- // After the transformations have been registered, the image data is read again.
- png_set_interlace_handling(pngPtr);
- png_read_update_info(pngPtr, infoPtr);
- png_get_IHDR(pngPtr, infoPtr, &w, &h, &bitDepth, &colorType, NULL, NULL, NULL);
- width = w;
- height = h;
-
- if (interlaceType == PNG_INTERLACE_NONE) {
- // PNGs without interlacing can simply be read row by row.
- for (int i = 0; i < height; i++) {
- png_read_row(pngPtr, (png_bytep)_outputSurface->getBasePtr(0, i), NULL);
- }
- } else {
- // PNGs with interlacing require us to allocate an auxillary
- // buffer with pointers to all row starts.
-
- // Allocate row pointer buffer
- png_bytep *rowPtr = new png_bytep[height];
- if (!rowPtr) {
- error("Could not allocate memory for row pointers.");
- }
-
- // Initialize row pointers
- for (int i = 0; i < height; i++)
- rowPtr[i] = (png_bytep)_outputSurface->getBasePtr(0, i);
-
- // Read image data
- png_read_image(pngPtr, rowPtr);
-
- // Free row pointer buffer
- delete[] rowPtr;
- }
-
- // Read additional data at the end.
- png_read_end(pngPtr, NULL);
-
- // Destroy libpng structures
- png_destroy_read_struct(&pngPtr, &infoPtr, &endInfo);
-
- // We no longer need the file stream, thus close it here
- _stream = 0;
-
- return true;
-#else
- return false;
-#endif
-}
-
-} // End of Graphics namespace
diff --git a/graphics/decoders/png.h b/graphics/decoders/png.h
deleted file mode 100644
index 5e608eb7b1..0000000000
--- a/graphics/decoders/png.h
+++ /dev/null
@@ -1,67 +0,0 @@
-/* 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.
- *
- */
-
-/*
- * PNG decoder used in engines:
- * - sword25
- * - wintermute
- * Dependencies:
- * - libpng
- */
-
-#ifndef GRAPHICS_PNG_H
-#define GRAPHICS_PNG_H
-
-#include "common/scummsys.h"
-#include "common/textconsole.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common {
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-struct Surface;
-struct PixelFormat;
-
-class PNGDecoder : public ImageDecoder {
-public:
- PNGDecoder();
- ~PNGDecoder();
-
- bool loadStream(Common::SeekableReadStream &stream);
- void destroy();
- const Graphics::Surface *getSurface() const { return _outputSurface; }
- const byte *getPalette() const { return _palette; }
- uint16 getPaletteColorCount() const { return _paletteColorCount; }
-private:
- Common::SeekableReadStream *_stream;
- byte *_palette;
- uint16 _paletteColorCount;
-
- Graphics::Surface *_outputSurface;
-};
-
-} // End of namespace Graphics
-
-#endif // GRAPHICS_PNG_H
diff --git a/graphics/decoders/tga.cpp b/graphics/decoders/tga.cpp
deleted file mode 100644
index a9f136d238..0000000000
--- a/graphics/decoders/tga.cpp
+++ /dev/null
@@ -1,427 +0,0 @@
-/* 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.
- */
-
-/* Based on code from xoreos https://github.com/DrMcCoy/xoreos/
- * relicensed under GPLv2+ with permission from DrMcCoy and clone2727
- */
-
-#include "common/util.h"
-#include "common/stream.h"
-#include "common/textconsole.h"
-#include "common/error.h"
-
-#include "graphics/decoders/tga.h"
-
-namespace Graphics {
-
-TGADecoder::TGADecoder() {
- _colorMapSize = 0;
- _colorMapOrigin = 0;
- _colorMapLength = 0;
- _colorMapEntryLength = 0;
- _colorMap = NULL;
-}
-
-TGADecoder::~TGADecoder() {
- destroy();
-}
-
-void TGADecoder::destroy() {
- _surface.free();
- delete[] _colorMap;
-}
-
-bool TGADecoder::loadStream(Common::SeekableReadStream &tga) {
- byte imageType, pixelDepth;
- bool success;
- success = readHeader(tga, imageType, pixelDepth);
- if (success) {
- switch (imageType) {
- case TYPE_BW:
- case TYPE_TRUECOLOR:
- success = readData(tga, imageType, pixelDepth);
- break;
- case TYPE_RLE_BW:
- case TYPE_RLE_TRUECOLOR:
- case TYPE_RLE_CMAP:
- success = readDataRLE(tga, imageType, pixelDepth);
- break;
- case TYPE_CMAP:
- success = readDataColorMapped(tga, imageType, pixelDepth);
- break;
- default:
- success = false;
- break;
- }
- }
- if (tga.err() || !success) {
- warning("Failed reading TGA-file");
- return false;
- }
- return success;
-}
-
-bool TGADecoder::readHeader(Common::SeekableReadStream &tga, byte &imageType, byte &pixelDepth) {
- if (!tga.seek(0)) {
- warning("Failed reading TGA-file");
- return false;
- }
-
- // TGAs have an optional "id" string in the header
- uint32 idLength = tga.readByte();
-
- // Number of colors in the color map / palette
- int hasColorMap = tga.readByte();
-
- // Image type. See header for numeric constants
- imageType = tga.readByte();
-
- switch (imageType) {
- case TYPE_CMAP:
- case TYPE_TRUECOLOR:
- case TYPE_BW:
- case TYPE_RLE_CMAP:
- case TYPE_RLE_TRUECOLOR:
- case TYPE_RLE_BW:
- break;
- default:
- warning("Unsupported image type: %d", imageType);
- return false;
- }
-
- // Color map specifications
- if (hasColorMap == 0) {
- tga.skip(5);
- } else {
- _colorMapOrigin = tga.readUint16LE();
- _colorMapLength = tga.readUint16LE();
- _colorMapEntryLength = tga.readByte();
- }
- // Origin-defintions
- tga.skip(2 + 2);
-
- // Image dimensions
- _surface.w = tga.readUint16LE();
- _surface.h = tga.readUint16LE();
-
- // Bits per pixel
- pixelDepth = tga.readByte();
- _surface.format.bytesPerPixel = pixelDepth / 8;
-
- // Image descriptor
- byte imgDesc = tga.readByte();
- int attributeBits = imgDesc & 0x0F;
- assert((imgDesc & 0x10) == 0);
- _originTop = (imgDesc & 0x20);
-
- // Interleaving is not handled at this point
- //int interleave = (imgDesc & 0xC);
- if (imageType == TYPE_CMAP || imageType == TYPE_RLE_CMAP) {
- if (pixelDepth == 8) {
- _format = PixelFormat::createFormatCLUT8();
- } else {
- warning("Unsupported index-depth: %d", pixelDepth);
- return false;
- }
- } else if (imageType == TYPE_TRUECOLOR || imageType == TYPE_RLE_TRUECOLOR) {
- if (pixelDepth == 24) {
- _format = PixelFormat(3, 8, 8, 8, 0, 16, 8, 0, 0);
- } else if (pixelDepth == 32) {
- // HACK: According to the spec, attributeBits should determine the amount
- // of alpha-bits, however, as the game files that use this decoder seems
- // to ignore that fact, we force the amount to 8 for 32bpp files for now.
- _format = PixelFormat(4, 8, 8, 8, /* attributeBits */ 8, 16, 8, 0, 24);
- } else if (pixelDepth == 16 && imageType == TYPE_TRUECOLOR) {
- // 16bpp TGA is ARGB1555
- _format = PixelFormat(2, 5, 5, 5, attributeBits, 10, 5, 0, 15);
- } else {
- warning("Unsupported pixel depth: %d, %d", imageType, pixelDepth);
- return false;
- }
- } else if (imageType == TYPE_BW || TYPE_RLE_BW) {
- if (pixelDepth == 8) {
- _format = PixelFormat(4, 8, 8, 8, 0, 16, 8, 0, 0);
- } else {
- warning("Unsupported pixel depth: %d, %d", imageType, pixelDepth);
- return false;
- }
-
- } else {
- warning("Unsupported image type: %d", imageType);
- return false;
- }
-
- // Skip the id string
- tga.skip(idLength);
-
- if (hasColorMap) {
- return readColorMap(tga, imageType, pixelDepth);
- }
- return true;
-}
-
-bool TGADecoder::readColorMap(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth) {
- _colorMap = new byte[3 * _colorMapLength];
- for (int i = 0; i < _colorMapLength * 3; i += 3) {
- byte r, g, b;
- if (_colorMapEntryLength == 32) {
- byte a;
- PixelFormat format(4, 8, 8, 8, 0, 16, 8, 0, 24);
- uint32 color = tga.readUint32LE();
- format.colorToARGB(color, a, r, g, b);
- } else if (_colorMapEntryLength == 24) {
- r = tga.readByte();
- g = tga.readByte();
- b = tga.readByte();
- } else if (_colorMapEntryLength == 16) {
- byte a;
- PixelFormat format(2, 5, 5, 5, 0, 10, 5, 0, 15);
- uint16 color = tga.readUint16LE();
- format.colorToARGB(color, a, r, g, b);
- } else {
- warning("Unsupported image type: %d", imageType);
- r = g = b = 0;
- }
-#ifdef SCUMM_LITTLE_ENDIAN
- _colorMap[i] = r;
- _colorMap[i + 1] = g;
- _colorMap[i + 2] = b;
-#else
- _colorMap[i] = b;
- _colorMap[i + 1] = g;
- _colorMap[i + 2] = r;
-#endif
- }
- return true;
-}
-
-// Additional information found from http://paulbourke.net/dataformats/tga/
-// With some details from the link referenced in the header.
-bool TGADecoder::readData(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth) {
- // TrueColor
- if (imageType == TYPE_TRUECOLOR) {
- _surface.create(_surface.w, _surface.h, _format);
-
- if (pixelDepth == 16) {
- for (int i = 0; i < _surface.h; i++) {
- uint16 *dst;
- if (!_originTop) {
- dst = (uint16 *)_surface.getBasePtr(0, _surface.h - i - 1);
- } else {
- dst = (uint16 *)_surface.getBasePtr(0, i);
- }
- for (int j = 0; j < _surface.w; j++) {
- *dst++ = tga.readUint16LE();
- }
- }
- } else if (pixelDepth == 32) {
- for (int i = 0; i < _surface.h; i++) {
- uint32 *dst;
- if (!_originTop) {
- dst = (uint32 *)_surface.getBasePtr(0, _surface.h - i - 1);
- } else {
- dst = (uint32 *)_surface.getBasePtr(0, i);
- }
- for (int j = 0; j < _surface.w; j++) {
- *dst++ = tga.readUint32LE();
- }
- }
- } else if (pixelDepth == 24) {
- for (int i = 0; i < _surface.h; i++) {
- byte *dst;
- if (!_originTop) {
- dst = (byte *)_surface.getBasePtr(0, _surface.h - i - 1);
- } else {
- dst = (byte *)_surface.getBasePtr(0, i);
- }
- for (int j = 0; j < _surface.w; j++) {
- byte r = tga.readByte();
- byte g = tga.readByte();
- byte b = tga.readByte();
-#ifdef SCUMM_LITTLE_ENDIAN
- *dst++ = r;
- *dst++ = g;
- *dst++ = b;
-#else
- *dst++ = b;
- *dst++ = g;
- *dst++ = r;
-#endif
- }
- }
- }
- // Black/White
- } else if (imageType == TYPE_BW) {
- _surface.create(_surface.w, _surface.h, _format);
-
- byte *data = (byte *)_surface.getPixels();
- uint32 count = _surface.w * _surface.h;
-
- while (count-- > 0) {
- byte g = tga.readByte();
- *data++ = g;
- *data++ = g;
- *data++ = g;
- *data++ = g;
- }
- }
- return true;
-}
-
-bool TGADecoder::readDataColorMapped(Common::SeekableReadStream &tga, byte imageType, byte indexDepth) {
- // Color-mapped
- if (imageType == TYPE_CMAP) {
- _surface.create(_surface.w, _surface.h, _format);
- if (indexDepth == 8) {
- for (int i = 0; i < _surface.h; i++) {
- byte *dst;
- if (!_originTop) {
- dst = (byte *)_surface.getBasePtr(0, _surface.h - i - 1);
- } else {
- dst = (byte *)_surface.getBasePtr(0, i);
- }
- for (int j = 0; j < _surface.w; j++) {
- byte index = tga.readByte();
- *dst++ = index;
- }
- }
- } else if (indexDepth == 16) {
- warning("16 bit indexes not supported");
- return false;
- }
- } else {
- return false;
- }
- return true;
-}
-
-bool TGADecoder::readDataRLE(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth) {
- // RLE-TrueColor / RLE-Black/White
- if (imageType == TYPE_RLE_TRUECOLOR || imageType == TYPE_RLE_BW || imageType == TYPE_RLE_CMAP) {
- _surface.create(_surface.w, _surface.h, _format);
- uint32 count = _surface.w * _surface.h;
- byte *data = (byte *)_surface.getPixels();
-
- while (count > 0) {
- uint32 header = tga.readByte();
- byte type = (header & 0x80) >> 7;
- uint32 rleCount = (header & 0x7F) + 1;
-
- // RLE-packet
- if (type == 1) {
- if (pixelDepth == 32 && imageType == TYPE_RLE_TRUECOLOR) {
- uint32 color = tga.readUint32LE();
- while (rleCount-- > 0) {
- *((uint32 *)data) = color;
- data += 4;
- count--;
- }
- } else if (pixelDepth == 24 && imageType == TYPE_RLE_TRUECOLOR) {
- byte r = tga.readByte();
- byte g = tga.readByte();
- byte b = tga.readByte();
- while (rleCount-- > 0) {
-#ifdef SCUMM_LITTLE_ENDIAN
- *data++ = r;
- *data++ = g;
- *data++ = b;
-#else
- *data++ = b;
- *data++ = g;
- *data++ = r;
-#endif
- count--;
- }
- } else if (pixelDepth == 8 && imageType == TYPE_RLE_BW) {
- byte color = tga.readByte();
- while (rleCount-- > 0) {
- *data++ = color;
- *data++ = color;
- *data++ = color;
- *data++ = color;
- count--;
- }
- } else if (pixelDepth == 8 && imageType == TYPE_RLE_CMAP) {
- byte index = tga.readByte();
- while (rleCount-- > 0) {
- *data++ = index;
- count--;
- }
- } else {
- warning("Unhandled pixel-depth for image-type 10");
- return false;
- }
- // Raw-packet
- } else if (type == 0) {
- if (pixelDepth == 32 && imageType == TYPE_RLE_TRUECOLOR) {
- while (rleCount-- > 0) {
- uint32 color = tga.readUint32LE();
- *((uint32 *)data) = color;
- data += 4;
- count--;
- }
- } else if (pixelDepth == 24 && imageType == TYPE_RLE_TRUECOLOR) {
- while (rleCount-- > 0) {
- byte r = tga.readByte();
- byte g = tga.readByte();
- byte b = tga.readByte();
-#ifdef SCUMM_LITTLE_ENDIAN
- *data++ = r;
- *data++ = g;
- *data++ = b;
-#else
- *data++ = b;
- *data++ = g;
- *data++ = r;
-#endif
- count--;
- }
- } else if (pixelDepth == 8 && imageType == TYPE_RLE_BW) {
- while (rleCount-- > 0) {
- byte color = tga.readByte();
- *data++ = color;
- *data++ = color;
- *data++ = color;
- *data++ = color;
- count--;
- }
- } else if (pixelDepth == 8 && imageType == TYPE_RLE_CMAP) {
- while (rleCount-- > 0) {
- byte index = tga.readByte();
- *data++ = index;
- count--;
- }
- } else {
- warning("Unhandled pixel-depth for image-type 10");
- return false;
- }
- } else {
- warning("Unknown header for RLE-packet %d", type);
- return false;
- }
- }
- } else {
- return false;
- }
- return true;
-}
-
-} // End of namespace Graphics
diff --git a/graphics/decoders/tga.h b/graphics/decoders/tga.h
deleted file mode 100644
index d8ccf8f766..0000000000
--- a/graphics/decoders/tga.h
+++ /dev/null
@@ -1,98 +0,0 @@
-/* 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.
- */
-
-/* Based on code from eos https://github.com/DrMcCoy/xoreos/
- * relicensed under GPLv2+ with permission from DrMcCoy and clone2727
- */
-
-/*
- * TGA decoder used in engines:
- * - wintermute
- */
-
-#ifndef GRAPHICS_DECODERS_TGA_H
-#define GRAPHICS_DECODERS_TGA_H
-
-#include "graphics/surface.h"
-#include "graphics/decoders/image_decoder.h"
-
-namespace Common {
-class SeekableReadStream;
-}
-
-namespace Graphics {
-
-/** TarGa image-decoder
- * The following variations of TGA are supported:
- * - Type 1 - Color-mapped images in 16/24/32 bpp with 8 bit indexes
- * - Type 2 - 16/24/32 bpp Top AND Bottom origined.
- * - Type 3 - Black/White images, 8bpp.
- * - Type 9 - RLE-encoded color-mapped images. (8 bit indexes only)
- * - Type 10 - RLE-encoded TrueColor, 24/32bpp.
- * - Type 11 - RLE-encoded Black/White, 8bpp.
- *
- * No images are returned with a palette, instead they are converted
- * to 16 bpp for Type 1, or 32 bpp for Black/White-images.
- */
-class TGADecoder : public ImageDecoder {
-public:
- TGADecoder();
- virtual ~TGADecoder();
- virtual void destroy();
- virtual const Surface *getSurface() const { return &_surface; }
- virtual const byte *getPalette() const { return _colorMap; }
- virtual uint16 getPaletteColorCount() const { return _colorMapLength; }
- virtual bool loadStream(Common::SeekableReadStream &stream);
-private:
- // Format-spec from:
- //http://www.ludorg.net/amnesia/TGA_File_Format_Spec.html
- enum {
- TYPE_CMAP = 1,
- TYPE_TRUECOLOR = 2,
- TYPE_BW = 3,
- TYPE_RLE_CMAP = 9,
- TYPE_RLE_TRUECOLOR = 10,
- TYPE_RLE_BW = 11
- };
-
- // Color-map:
- bool _colorMapSize;
- byte *_colorMap;
- int16 _colorMapOrigin;
- int16 _colorMapLength;
- byte _colorMapEntryLength;
-
- // Origin may be at the top, or bottom
- bool _originTop;
-
- PixelFormat _format;
- Surface _surface;
- // Loading helpers
- bool readHeader(Common::SeekableReadStream &tga, byte &imageType, byte &pixelDepth);
- bool readData(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth);
- bool readDataColorMapped(Common::SeekableReadStream &tga, byte imageType, byte indexDepth);
- bool readDataRLE(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth);
- bool readColorMap(Common::SeekableReadStream &tga, byte imageType, byte pixelDepth);
-};
-
-} // End of namespace Graphics
-
-#endif // GRAPHICS_DECODERS_TGA_H