scummvm/image/codecs/msrle4.cpp
2016-05-25 05:37:07 +02:00

141 lines
3.3 KiB
C++

/* 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 off ffmpeg's msrledec.c
#include "common/debug.h"
#include "image/codecs/msrle4.h"
#include "common/stream.h"
#include "common/textconsole.h"
#include "common/util.h"
namespace Image {
MSRLE4Decoder::MSRLE4Decoder(uint16 width, uint16 height, byte bitsPerPixel) {
_surface = new Graphics::Surface();
_surface->create(width, height, Graphics::PixelFormat::createFormatCLUT8());
_bitsPerPixel = bitsPerPixel;
}
MSRLE4Decoder::~MSRLE4Decoder() {
_surface->free();
delete _surface;
}
const Graphics::Surface *MSRLE4Decoder::decodeFrame(Common::SeekableReadStream &stream) {
if (_bitsPerPixel == 4) {
decode4(stream);
} else
error("Unhandled %d bit Microsoft RLE encoding", _bitsPerPixel);
return _surface;
}
void MSRLE4Decoder::decode4(Common::SeekableReadStream &stream) {
int x = 0;
int y = _surface->h - 1;
byte *output = (byte *)_surface->getBasePtr(x, y);
byte *output_end = (byte *)_surface->getBasePtr(_surface->w, y);
while (!stream.eos()) {
byte count = stream.readByte();
if (count == 0) {
byte value = stream.readByte();
if (value == 0) {
// End of line
x = 0;
y--;
output = (byte *)_surface->getBasePtr(x, y);
} else if (value == 1) {
// End of image
return;
} else if (value == 2) {
// Skip
count = stream.readByte();
value = stream.readByte();
x += count;
y -= value;
if (y < 0) {
warning("MS RLE Codec: Skip beyond picture bounds");
return;
}
output = (byte *)_surface->getBasePtr(x, y);
} else {
// Copy data
int odd_pixel = value & 1;
int rle_code = (value + 1) / 2;
int extra_byte = rle_code & 0x01;
if (output + value > output_end) {
stream.skip(rle_code + extra_byte);
continue;
}
for (int i = 0; i < rle_code; i++) {
byte color = stream.readByte();
*output++ = (color & 0xf0) >> 4;
if (i + 1 == rle_code && odd_pixel) {
break;
}
*output++ = color & 0x0f;
}
if (extra_byte)
stream.skip(1);
x += value;
}
} else {
// Run data
if (output + count > output_end)
continue;
byte color = stream.readByte();
for (int i = 0; i < count; i++, x++) {
*output++ = (color & 0xf0) >> 4;
i++;
x++;
if (i == count)
break;
*output++ = color & 0x0f;
}
}
}
warning("MS RLE Codec: No end-of-picture code");
}
} // End of namespace Image