scummvm/video/mve_decoder.cpp

548 lines
13 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "video/mve_decoder.h"
#include "audio/decoders/raw.h"
#include "common/endian.h"
#include "common/rect.h"
#include "common/stream.h"
#include "common/memstream.h"
#include "common/system.h"
#include "common/textconsole.h"
#include "common/events.h"
#include "graphics/surface.h"
#include "graphics/paletteman.h"
namespace Video {
MveDecoder::MveDecoder()
: _done(false),
_s(nullptr),
_dirtyPalette(false),
_skipMapSize(0),
_skipMap(nullptr),
_decodingMapSize(0),
_decodingMap(nullptr),
_frameNumber(-1),
_frameSize(0),
_frameData(nullptr),
_audioTrack(0),
_audioStream(nullptr)
{
for (int i = 0; i < 0x300; ++i)
_palette[i] = 0;
}
MveDecoder::~MveDecoder() {
close();
delete _audioStream;
delete[] _frameData;
delete[] _decodingMap;
delete[] _skipMap;
}
static const char signature[] = "Interplay MVE File\x1A";
bool MveDecoder::loadStream(Common::SeekableReadStream *stream) {
close();
byte signature_buffer[sizeof(signature)];
stream->read(signature_buffer, sizeof(signature_buffer));
if (memcmp(signature_buffer, signature, sizeof(signature))) {
warning("MveDecoder::loadStream(): attempted to load non-MVE data");
return false;
}
_s = stream;
uint16 h1 = _s->readUint16LE();
uint16 h2 = _s->readUint16LE();
uint16 h3 = _s->readUint16LE();
assert(h1 == 0x001a);
assert(h2 == 0x0100);
assert(h3 == 0x1133);
readPacketHeader();
while (!_done && _packetKind < 3) {
readNextPacket();
}
return true;
}
void MveDecoder::setAudioTrack(int track) {
assert(track >= 0 && track < 16);
_audioTrack= track;
}
void MveDecoder::applyPalette(PaletteManager *paletteManager) {
paletteManager->setPalette(_palette + 3 * _palStart, _palStart, _palCount);
}
void MveDecoder::copyBlock_8bit(Graphics::Surface &dst, Common::MemoryReadStream &s, int block) {
int x = (block % _widthInBlocks) * 8;
int y = (block / _widthInBlocks) * 8;
byte *p = (byte*)dst.getBasePtr(x, y);
for (int i = 0; i != 8; ++i) {
s.read(p, 8);
p += dst.pitch;
}
}
void MveDecoder::copyBlock_16bit(Graphics::Surface &dst, Common::MemoryReadStream &s, int block) {
int x = (block % _widthInBlocks) * 8;
int y = (block / _widthInBlocks) * 8;
byte *p = (byte*)dst.getBasePtr(x, y);
for (int i = 0; i != 8; ++i) {
for (int j = 0; j != 8; ++j) {
WRITE_UINT16(p+2*j, s.readUint16LE());
}
p += dst.pitch;
}
}
void MveDecoder::copyBlock(Graphics::Surface &dst, Graphics::Surface &src, int block, int offset) {
int dx = (block % _widthInBlocks) * 8;
int dy = (block / _widthInBlocks) * 8;
int sx = dx + offset % _width;
int sy = dy + offset / _width;
byte *dp = (byte*)dst.getBasePtr(dx, dy);
byte *sp = (byte*)src.getBasePtr(sx, sy);
for (int i = 0; i != 8; ++i) {
memmove(dp, sp, !_trueColor ? 8 : 16);
dp += dst.pitch;
sp += src.pitch;
}
}
void MveDecoder::copyBlock(Graphics::Surface &dst, Graphics::Surface &src, int dx, int dy, int off_x, int off_y) {
int sx = dx + off_x;
int sy = dy + off_y;
byte *dp = (byte*)dst.getBasePtr(dx, dy);
byte *sp = (byte*)src.getBasePtr(sx, sy);
for (int i = 0; i != 8; ++i) {
memmove(dp, sp, !_trueColor ? 8 : 16);
dp += dst.pitch;
sp += src.pitch;
}
}
void MveDecoder::decodeFormat6() {
_decodingMapSize = _widthInBlocks * _heightInBlocks * 2;
_decodingMap = _frameData + 14;
Common::MemoryReadStream opStream = Common::MemoryReadStream(_decodingMap, _decodingMapSize);
Common::MemoryReadStream frameStream = Common::MemoryReadStream(_frameData + _decodingMapSize + 14, _frameSize);
// Pass 1
opStream.seek(0);
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
uint16 op = opStream.readUint16LE();
if (op == 0) {
if (!_trueColor) {
copyBlock_8bit(_decodeSurface0, frameStream, b);
} else {
copyBlock_16bit(_decodeSurface0, frameStream, b);
}
}
}
// Pass 2
opStream.seek(0);
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
uint16 op = opStream.readUint16LE();
if (op != 0) {
Graphics::Surface &src = (op & 0x8000) ? _decodeSurface1 : _decodeSurface0;
int offset = int(op & 0x7fff) - 0x4000;
copyBlock(_decodeSurface0, src, b, offset);
}
}
// Pass 3
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
copyBlock(_frameSurface, _decodeSurface0, b);
}
Graphics::Surface t = _decodeSurface0;
_decodeSurface0 = _decodeSurface1;
_decodeSurface1 = t;
_decodingMap = nullptr;
}
void MveDecoder::decodeFormat10() {
MveSkipStream skipStream = MveSkipStream(_skipMap, _skipMapSize);
Common::MemoryReadStream opStream = Common::MemoryReadStream(_decodingMap, _decodingMapSize);
Common::MemoryReadStream frameStream = Common::MemoryReadStream(_frameData + 14, _frameSize - 14);
// Pass 1
opStream.seek(0);
skipStream.reset();
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
if (skipStream.skip()) continue;
uint16 op = opStream.readUint16LE();
if (op == 0) {
if (!_trueColor) {
copyBlock_8bit(_decodeSurface0, frameStream, b);
} else {
copyBlock_16bit(_decodeSurface0, frameStream, b);
}
}
}
// Pass 2
opStream.seek(0);
skipStream.reset();
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
if (skipStream.skip()) continue;
uint16 op = opStream.readUint16LE();
if (op != 0) {
Graphics::Surface &src = (op & 0x8000) ? _decodeSurface1 : _decodeSurface0;
int offset = int(op & 0x7fff) - 0x4000;
copyBlock(_decodeSurface0, src, b, offset);
}
}
// Pass 3
skipStream.reset();
for (int b = 0; b != _widthInBlocks * _heightInBlocks; ++b) {
if (skipStream.skip()) continue;
copyBlock(_frameSurface, _decodeSurface0, b);
}
Graphics::Surface t = _decodeSurface0;
_decodeSurface0 = _decodeSurface1;
_decodeSurface1 = t;
}
void MveDecoder::readPacketHeader() {
_packetLen = _s->readUint16LE();
_packetKind = _s->readUint16LE();
/*
switch (_packetKind) {
case 0:
warning("initialize audio");
break;
case 1:
warning("audio");
break;
case 2:
warning("initialize video");
break;
case 3:
warning("video");
break;
case 4:
warning("shutdown");
break;
case 5:
warning("end chunk");
break;
}
*/
}
void MveDecoder::readNextPacket() {
bool frameDone = false;
while (!_done && !frameDone) {
uint16 opLen = _s->readUint16LE();
uint16 opKind = _s->readUint16BE();
switch (opKind) {
case 0x0000:
{
_done = true;
assert(opLen == 0);
break;
}
case 0x0100:
{
assert(opLen == 0);
readPacketHeader();
break;
}
case 0x0200: // create timer
{
assert(opLen == 6);
uint32 rate = _s->readUint32LE();
uint16 subdiv = _s->readUint16LE();
_frameRate = Common::Rational(1000000, rate * subdiv);
break;
}
case 0x0300: // init audio
{
assert(opLen == 8);
/*uint16 unk =*/ _s->readUint16LE();
uint16 flags = _s->readUint16LE();
uint16 sampleRate = _s->readUint16LE();
/*uint16 bufLen =*/ _s->readUint16LE();
/*
warning("\t\tAudio: %dHz %s %s",
sampleRate,
(flags & 1) == 0 ? "mono" : "stereo",
(flags & 2) == 0 ? "8-bit" : "16-bit"
);
*/
assert((flags & 1) == 0);
assert((flags & 2) == 0);
_audioStream = Audio::makeQueuingAudioStream(sampleRate, (flags & 2) != 0);
addTrack(new MveAudioTrack(this));
break;
}
case 0x0400: // send audio
{
assert(opLen == 0);
break;
}
case 0x0502: // init video buffers
{
assert(opLen == 8);
uint16 width = _s->readUint16LE();
uint16 height = _s->readUint16LE();
/*uint16 count =*/ _s->readUint16LE();
uint16 trueColor = _s->readUint16LE();
_widthInBlocks = width;
_heightInBlocks = height;
_width = 8 * width;
_height = 8 * height;
_trueColor = !!trueColor;
if (!_trueColor) {
_pixelFormat = Graphics::PixelFormat::createFormatCLUT8();
} else {
_pixelFormat = Graphics::PixelFormat(2, 5, 5, 5, 0, 10, 5, 0, 0);
}
_decodeSurface0.create(_width, _height, _pixelFormat);
_decodeSurface0.fillRect(Common::Rect(_width, _height), 0);
_decodeSurface1.create(_width, _height, _pixelFormat);
_decodeSurface1.fillRect(Common::Rect(_width, _height), 0);
_frameSurface.create(_width, _height, _pixelFormat);
_frameSurface.fillRect(Common::Rect(_width, _height), 0);
addTrack(new MveVideoTrack(this));
break;
}
case 0x0600:
{
delete[] _frameData;
_frameData = new byte[opLen];
_frameSize = opLen;
_s->read(_frameData, _frameSize);
decodeFormat6();
break;
}
case 0x0701: // send video
{
assert(opLen == 6);
uint16 palStart = _s->readUint16LE();
uint16 palCount = _s->readUint16LE();
uint16 unk = _s->readUint16LE();
(void)unk;
if (palStart || palCount) {
_palStart = palStart;
_palCount = palCount;
}
_frameNumber += 1;
frameDone = true;
break;
}
case 0x0800: // audio frame
{
/*uint16 seq =*/ _s->readUint16LE();
uint16 mask = _s->readUint16LE();
uint16 len = _s->readUint16LE();
assert(opLen == len + 6);
assert(_audioStream);
if (mask & (1 << _audioTrack)) {
byte *audioFrame = new byte[len];
_s->read(audioFrame, len);
_audioStream->queueBuffer(audioFrame, len, DisposeAfterUse::YES, Audio::FLAG_UNSIGNED);
} else {
_s->skip(len);
}
break;
}
case 0x0900: // audio frame (silent)
{
assert(opLen == 6);
/*uint16 seq =*/ _s->readUint16LE();
/*uint16 mask =*/ _s->readUint16LE();
/*uint16 len =*/ _s->readUint16LE();
break;
}
case 0x0a00: // set video mode
{
assert(opLen == 6);
/*uint16 width =*/ _s->readUint16LE();
/*uint16 height =*/ _s->readUint16LE();
/*uint16 flags =*/ _s->readUint16LE();
break;
}
case 0x0c00:
{
uint16 palStart = _s->readUint16LE();
uint16 palCount = _s->readUint16LE();
assert(opLen >= 3 * palCount + 2);
for (int i = palStart; i < palStart + palCount; ++i) {
byte r = _s->readByte();
byte g = _s->readByte();
byte b = _s->readByte();
_palette[3*i+0] = (r << 2) | (r >> 4);
_palette[3*i+1] = (g << 2) | (g >> 4);
_palette[3*i+2] = (b << 2) | (b >> 4);
}
if (palCount & 1) {
_s->skip(1);
}
_dirtyPalette = true;
_palStart = palStart;
_palCount = palCount;
break;
}
case 0x0e00:
{
// TODO: Preallocate or keep existing buffer
delete[] _skipMap;
_skipMap = new byte[opLen];
_skipMapSize = opLen;
_s->read(_skipMap, _skipMapSize);
break;
}
case 0x0f00:
{
// TODO: Preallocate or keep existing buffer
delete[] _decodingMap;
_decodingMap = new byte[opLen];
_decodingMapSize = opLen;
_s->read(_decodingMap, _decodingMapSize);
break;
}
case 0x1000:
{
// TODO: Preallocate or keep existing buffer
delete[] _frameData;
_frameData = new byte[opLen];
_frameSize = opLen;
_s->read(_frameData, _frameSize);
decodeFormat10();
break;
}
default:
_s->skip(opLen);
error("Unknown opcode %04x", opKind);
break;
}
}
}
MveDecoder::MveVideoTrack::MveVideoTrack(MveDecoder *decoder) : _decoder(decoder) {
}
bool MveDecoder::MveVideoTrack::endOfTrack() const {
return _decoder->_done;
}
uint16 MveDecoder::MveVideoTrack::getWidth() const {
return _decoder->getWidth();
}
uint16 MveDecoder::MveVideoTrack::getHeight() const {
return _decoder->getHeight();
}
Graphics::PixelFormat MveDecoder::MveVideoTrack::getPixelFormat() const {
return _decoder->getPixelFormat();
}
int MveDecoder::MveVideoTrack::getCurFrame() const {
return _decoder->_frameNumber;
}
const Graphics::Surface *MveDecoder::MveVideoTrack::decodeNextFrame() {
return &_decoder->_frameSurface;
}
const byte *MveDecoder::MveVideoTrack::getPalette() const {
return _decoder->_palette;
}
bool MveDecoder::MveVideoTrack::hasDirtyPalette() const {
return _decoder->_dirtyPalette;
}
Common::Rational MveDecoder::MveVideoTrack::getFrameRate() const {
return _decoder->getFrameRate();
}
MveDecoder::MveAudioTrack::MveAudioTrack(MveDecoder *decoder) :
AudioTrack(Audio::Mixer::kPlainSoundType),
_decoder(decoder)
{
}
Audio::AudioStream *MveDecoder::MveAudioTrack::getAudioStream() const {
return _decoder->_audioStream;
}
} // End of namespace Video