scummvm/image/codecs/msrle.cpp

139 lines
3.2 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 "image/codecs/msrle.h"
#include "common/stream.h"
#include "common/textconsole.h"
namespace Image {
MSRLEDecoder::MSRLEDecoder(uint16 width, uint16 height, byte bitsPerPixel) {
_surface = new Graphics::Surface();
_surface->create(width, height, Graphics::PixelFormat::createFormatCLUT8());
_bitsPerPixel = bitsPerPixel;
}
MSRLEDecoder::~MSRLEDecoder() {
_surface->free();
delete _surface;
}
const Graphics::Surface *MSRLEDecoder::decodeFrame(Common::SeekableReadStream &stream) {
if (_bitsPerPixel == 8) {
decode8(stream);
} else
error("Unhandled %d bit Microsoft RLE encoding", _bitsPerPixel);
return _surface;
}
void MSRLEDecoder::decode8(Common::SeekableReadStream &stream) {
int x = 0;
int y = _surface->h - 1;
byte *data = (byte *) _surface->getPixels();
uint16 width = _surface->w;
uint16 height = _surface->h;
byte *output = data + ((height - 1) * width);
byte *output_end = data + ((height) * width);
while (!stream.eos()) {
byte count = stream.readByte();
byte value = stream.readByte();
if (count == 0) {
if (value == 0) {
// End of line
x = 0;
y--;
output = data + (y * width);
} else if (value == 1) {
// End of image
return;
} else if (value == 2) {
// Skip
count = stream.readByte();
value = stream.readByte();
y -= value;
x += count;
if (y < 0) {
warning("MS RLE Codec: Skip beyond picture bounds");
return;
}
output = data + ((y * width) + x);
} else {
// Copy data
if (y < 0) {
warning("MS RLE Codec: Copy data is beyond picture bounds");
return;
}
if (output + value > output_end) {
if (stream.pos() + value >= stream.size())
break;
else
stream.skip(value);
continue;
}
for (int i = 0; i < value; i++)
*output++ = stream.readByte();
if (value & 1)
stream.skip(1);
x += value;
}
} else {
// Run data
if (y < 0) {
warning("MS RLE Codec: Run data is beyond picture bounds");
return;
}
if (output + count > output_end)
continue;
for (int i = 0; i < count; i++, x++)
*output++ = value;
}
}
warning("MS RLE Codec: No end-of-picture code");
}
} // End of namespace Image