Files
scummvm/video/4xm_decoder.cpp
T
2026-02-09 23:30:31 +01:00

667 lines
21 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/4xm_decoder.h"
#include "audio/audiostream.h"
#include "audio/decoders/adpcm.h"
#include "audio/decoders/raw.h"
#include "common/debug.h"
#include "common/endian.h"
#include "common/memstream.h"
#include "common/stream.h"
#include "common/textconsole.h"
#include "graphics/managed_surface.h"
#include "video/4xm_utils.h"
namespace Video {
Common::Rational floatToRational(float value) {
int num = static_cast<int>(1000 * value);
int denom = 1000;
return {num, denom};
}
namespace {
Common::String tagName(uint32 tag) {
char name[5] = {char(tag >> 24), char(tag >> 16), char(tag >> 8), char(tag), 0};
return {name};
}
static const int8_t mv[256][2] = {
{0, 0}, {0, -1}, {-1, 0}, {1, 0}, {0, 1}, {-1, -1}, {1, -1}, {-1, 1}, {1, 1}, {0, -2}, {-2, 0}, {2, 0}, {0, 2}, {-1, -2}, {1, -2}, {-2, -1}, {2, -1}, {-2, 1}, {2, 1}, {-1, 2}, {1, 2}, {-2, -2}, {2, -2}, {-2, 2}, {2, 2}, {0, -3}, {-3, 0}, {3, 0}, {0, 3}, {-1, -3}, {1, -3}, {-3, -1}, {3, -1}, {-3, 1}, {3, 1}, {-1, 3}, {1, 3}, {-2, -3}, {2, -3}, {-3, -2}, {3, -2}, {-3, 2}, {3, 2}, {-2, 3}, {2, 3}, {0, -4}, {-4, 0}, {4, 0}, {0, 4}, {-1, -4}, {1, -4}, {-4, -1}, {4, -1}, {4, 1}, {-1, 4}, {1, 4}, {-3, -3}, {-3, 3}, {3, 3}, {-2, -4}, {-4, -2}, {4, -2}, {-4, 2}, {-2, 4}, {2, 4}, {-3, -4}, {3, -4}, {4, -3}, {-5, 0}, {-4, 3}, {-3, 4}, {3, 4}, {-1, -5}, {-5, -1}, {-5, 1}, {-1, 5}, {-2, -5}, {2, -5}, {5, -2}, {5, 2}, {-4, -4}, {-4, 4}, {-3, -5}, {-5, -3}, {-5, 3}, {3, 5}, {-6, 0}, {0, 6}, {-6, -1}, {-6, 1}, {1, 6}, {2, -6}, {-6, 2}, {2, 6}, {-5, -4}, {5, 4}, {4, 5}, {-6, -3}, {6, 3}, {-7, 0}, {-1, -7}, {5, -5}, {-7, 1}, {-1, 7}, {4, -6}, {6, 4}, {-2, -7}, {-7, 2}, {-3, -7}, {7, -3}, {3, 7}, {6, -5}, {0, -8}, {-1, -8}, {-7, -4}, {-8, 1}, {4, 7}, {2, -8}, {-2, 8}, {6, 6}, {-8, 3}, {5, -7}, {-5, 7}, {8, -4}, {0, -9}, {-9, -1}, {1, 9}, {7, -6}, {-7, 6}, {-5, -8}, {-5, 8}, {-9, 3}, {9, -4}, {7, -7}, {8, -6}, {6, 8}, {10, 1}, {-10, 2}, {9, -5}, {10, -3}, {-8, -7}, {-10, -4}, {6, -9}, {-11, 0}, {11, 1}, {-11, -2}, {-2, 11}, {7, -9}, {-7, 9}, {10, 6}, {-4, 11}, {8, -9}, {8, 9}, {5, 11}, {7, -10}, {12, -3}, {11, 6}, {-9, -9}, {8, 10}, {5, 12}, {-11, 7}, {13, 2}, {6, -12}, {10, 9}, {-11, 8}, {-7, 12}, {0, 14}, {14, -2}, {-9, 11}, {-6, 13}, {-14, -4}, {-5, -14}, {5, 14}, {-15, -1}, {-14, -6}, {3, -15}, {11, -11}, {-7, 14}, {-5, 15}, {8, -14}, {15, 6}, {3, 16}, {7, -15}, {-16, 5}, {0, 17}, {-16, -6}, {-10, 14}, {-16, 7}, {12, 13}, {-16, 8}, {-17, 6}, {-18, 3}, {-7, 17}, {15, 11}, {16, 10}, {2, -19}, {3, -19}, {-11, -16}, {-18, 8}, {-19, -6}, {2, -20}, {-17, -11}, {-10, -18}, {8, 19}, {-21, -1}, {-20, 7}, {-4, 21}, {21, 5}, {15, 16}, {2, -22}, {-10, -20}, {-22, 5}, {20, -11}, {-7, -22}, {-12, 20}, {23, -5}, {13, -20}, {24, -2}, {-15, 19}, {-11, 22}, {16, 19}, {23, -10}, {-18, -18}, {-9, -24}, {24, -10}, {-3, 26}, {-23, 13}, {-18, -20}, {17, 21}, {-4, 27}, {27, 6}, {1, -28}, {-11, 26}, {-17, -23}, {7, 28}, {11, -27}, {29, 5}, {-23, -19}, {-28, -11}, {-21, 22}, {-30, 7}, {-17, 26}, {-27, 16}, {13, 29}, {19, -26}, {10, -31}, {-14, -30}, {20, -27}, {-29, 18}, {-16, -31}, {-28, -22}, {21, -30}, {-25, 28}, {26, -29}, {25, -32}, {-32, -32}};
} // namespace
class FourXMDecoder::FourXMAudioTrack : public AudioTrack {
uint _audioType;
uint _audioChannels;
uint _sampleRate;
Common::ScopedPtr<Audio::QueuingAudioStream> _output;
public:
FourXMAudioTrack(FourXMDecoder *dec, uint trackIdx, uint audioType, uint audioChannels, uint sampleRate) : AudioTrack(Audio::Mixer::SoundType::kPlainSoundType), _audioType(audioType), _audioChannels(audioChannels), _sampleRate(sampleRate),
_output(Audio::makeQueuingAudioStream(sampleRate, audioChannels > 1)) {
}
byte getAudioType() const { return _audioType; }
void decode(byte *buf, uint size) {
if (_audioType == 0) {
// Raw PCM data
byte flags = Audio::FLAG_16BITS;
#ifdef SCUMM_LITTLE_ENDIAN
flags |= Audio::FLAG_LITTLE_ENDIAN;
#endif
if (_audioChannels > 1)
flags |= Audio::FLAG_STEREO;
_output->queueBuffer(buf, size, DisposeAfterUse::YES, flags);
} else if (_audioType == 1) {
auto *input = new Common::MemoryReadStream(buf, size, DisposeAfterUse::YES);
_output->queueAudioStream(Audio::makeADPCMStream(input, DisposeAfterUse::YES, size, Audio::ADPCMType::kADPCM4XM, _sampleRate, _audioChannels));
} else {
free(buf);
warning("unsupported audio type %u", _audioType);
}
}
private:
Audio::AudioStream *getAudioStream() const override { return _output.get(); }
};
class FourXMDecoder::FourXMVideoTrack : public FixedRateVideoTrack {
FourXMDecoder *_dec;
Common::Rational _frameRate;
uint _w, _h;
uint16 _version = 0;
Common::ScopedPtr<Graphics::ManagedSurface> _framePtr, _lastFramePtr;
Graphics::Surface *_frame = nullptr, *_lastFrame = nullptr;
FourXM::HuffmanDecoder _blockType[4] = {};
int _mv[256];
Common::HashMap<byte, Common::Array<byte>> _cframes;
public:
FourXMVideoTrack(FourXMDecoder *dec, const Common::Rational &frameRate, uint w, uint h, uint16 version) : _dec(dec), _frameRate(frameRate), _w(w), _h(h), _version(version) {
if (_version <= 1)
error("versions 0 and 1 are not supported");
_blockType[0].initStatistics({16, 8, 4, 2, 1, 1});
_blockType[1].initStatistics({8, 0, 4, 2, 1, 1});
_blockType[2].initStatistics({8, 4, 0, 2, 1, 1});
_blockType[3].initStatistics({8, 0, 0, 4, 2, 1, 1});
}
~FourXMVideoTrack();
uint16 getWidth() const override { return _w; }
uint16 getHeight() const override { return _h; }
Graphics::PixelFormat getPixelFormat() const override {
return Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0); // RGB565
}
int getCurFrame() const override;
int getFrameCount() const override;
const Graphics::Surface *decodeNextFrame() override;
void decode(uint32 tag, byte *buf, uint size);
void decode_ifrm(Common::SeekableReadStream *stream);
void decode_pfrm(Common::SeekableReadStream *stream);
void decode_cfrm(Common::SeekableReadStream *stream);
private:
void decode_pfrm_block(uint16 *dst, const uint16 *src, int stride, int log2w, int log2h, FourXM::BEWordBitStream &bitStream, Common::MemoryReadStream &wordStream, Common::MemoryReadStream &byteStream);
Common::Rational getFrameRate() const override { return _frameRate; }
};
const Graphics::Surface *FourXMDecoder::FourXMVideoTrack::decodeNextFrame() {
if (!_framePtr) {
_framePtr.reset(new Graphics::ManagedSurface());
_framePtr->create(_w, _h, getPixelFormat());
_frame = _framePtr->surfacePtr();
_lastFramePtr.reset(new Graphics::ManagedSurface());
_lastFramePtr->create(_w, _h, getPixelFormat());
_lastFrame = _lastFramePtr->surfacePtr();
assert(_framePtr->pitch == _lastFramePtr->pitch);
auto pitch = _framePtr->pitch / _framePtr->format.bytesPerPixel;
for (uint i = 0; i < 256; i++)
_mv[i] = mv[i][0] + mv[i][1] * pitch;
}
_dec->decodeNextFrameImpl();
return _lastFrame;
}
int FourXMDecoder::FourXMVideoTrack::getCurFrame() const {
return _dec->_curFrame;
}
int FourXMDecoder::FourXMVideoTrack::getFrameCount() const {
return _dec->_frames.size();
}
FourXMDecoder::FourXMVideoTrack::~FourXMVideoTrack() = default;
namespace {
static const uint8_t iquant[64] = {
16,
15,
13,
19,
24,
31,
28,
17,
17,
23,
25,
31,
36,
63,
45,
21,
18,
24,
27,
37,
52,
59,
49,
20,
16,
28,
34,
40,
60,
80,
51,
20,
18,
31,
48,
66,
68,
86,
56,
21,
19,
38,
56,
59,
64,
64,
48,
20,
27,
48,
55,
55,
56,
51,
35,
15,
20,
35,
34,
32,
31,
22,
15,
8,
};
const byte zigzag[] = {
0, 1, 8, 16, 9, 2, 3, 10, 17, 24, 32, 25, 18, 11, 4,
5, 12, 19, 26, 33, 40, 48, 41, 34, 27, 20, 13, 6, 7,
14, 21, 28, 35, 42, 49, 56, 57, 50, 43, 36, 29, 22,
15, 23, 30, 37, 44, 51, 58, 59, 52, 45, 38, 31, 39,
46, 53, 60, 61, 54, 47, 55, 62, 63};
static const int8_t size2index[4][4] = {
{-1, 3, 1, 1},
{3, 0, 0, 0},
{2, 0, 0, 0},
{2, 0, 0, 0},
};
} // namespace
void FourXMDecoder::FourXMVideoTrack::decode_ifrm(Common::SeekableReadStream *stream) {
auto bitstreamSize = stream->readUint32LE();
Common::Array<byte> bitstreamData(bitstreamSize);
stream->read(bitstreamData.data(), bitstreamData.size());
auto prefixSize = stream->readUint32LE();
/* auto tokenCount = */ stream->readUint32LE();
Common::Array<byte> prefixStream(prefixSize * 4);
stream->read(prefixStream.data(), prefixStream.size());
assert(stream->pos() == stream->size());
auto prefixData = FourXM::HuffmanDecoder::unpack(prefixStream.data(), prefixStream.size(), 4);
FourXM::BEByteBitStream bitstream(bitstreamData.data(), bitstreamData.size(), 0);
uint prefixOffset = 0;
int lastDC = 0;
auto &format = _frame->format;
const auto dstPitch = _frame->pitch / format.bytesPerPixel - 16;
for (int mbY = 0; mbY < _frame->h; mbY += 16) {
for (int mbX = 0; mbX < _frame->w; mbX += 16) {
int16_t block[6][64] = {};
auto readBlock = [&](byte blockIdx, int16_t *ac) {
int dc = prefixData[prefixOffset++];
if (dc >> 4)
error("dc run code");
dc = bitstream.readInt(dc);
dc = lastDC + dc * iquant[0];
lastDC = dc;
ac[0] = dc;
for (uint idx = 1; idx < 64;) {
auto b = prefixData[prefixOffset++];
if (b == 0x00) {
break;
} else if (b == 0xf0) {
idx += 16;
} else {
auto h = b >> 4;
auto l = b & 0x0f;
idx += h;
if (l && idx < 64) {
auto ac_idx = zigzag[idx];
ac[ac_idx] = iquant[ac_idx] * bitstream.readInt(l);
++idx;
}
}
}
if (blockIdx < 4)
ac[0] += 0x80 * 8 * 8;
FourXM::idct(ac);
};
for (int b = 0; b != 6; ++b)
readBlock(b, block[b]);
auto acIdx = [](byte y, byte x) {
assert(y < 8 && x < 8);
return y << 3 | x;
};
auto blockCB = block[4];
auto blockCR = block[5];
auto *dst = static_cast<uint16 *>(_frame->getBasePtr(mbX, mbY));
for (byte y = 0; y != 16; ++y, dst += dstPitch) {
for (byte x = 0; x != 16; ++x) {
auto yb = y & 7;
auto xb = x & 7;
auto mbBlockIdx = ((y >> 3) << 1) | (x >> 3);
auto Y = block[mbBlockIdx][acIdx(yb, xb)];
auto cblockIdx = acIdx(y >> 1, x >> 1);
auto CB = blockCB[cblockIdx];
auto CR = blockCR[cblockIdx];
int CG = (CB + CR) >> 1;
CB *= 2;
auto color = format.RGBToColor(Y + CR, Y - CG, Y + CB);
*dst++ = color;
}
}
}
}
assert(prefixOffset == prefixData.size());
SWAP(_frame, _lastFrame);
}
namespace {
template<bool Scale>
void mcdc(uint16_t *dst, const uint16_t *src, int log2w,
int log2h, int stride, uint dc) {
dc |= dc << 16;
int h = 1 << log2h;
if (Scale) {
for (int i = 0; i < h; ++i) {
auto *dst32 = reinterpret_cast<uint32_t *>(dst);
auto *src32 = reinterpret_cast<const uint32_t *>(src);
switch (log2w) {
case 3:
dst32[2] = src32[2] + dc;
dst32[3] = src32[3] + dc;
// fall through
case 2:
dst32[1] = src32[1] + dc;
// fall through
case 1:
dst32[0] = src32[0] + dc;
break;
case 0:
*dst = *src + dc;
}
src += stride;
dst += stride;
}
} else {
for (int i = 0; i < h; ++i) {
auto *dst32 = reinterpret_cast<uint32_t *>(dst);
switch (log2w) {
case 3:
dst32[2] = dc;
dst32[3] = dc;
// fall through
case 2:
dst32[1] = dc;
// fall through
case 1:
dst32[0] = dc;
break;
case 0:
*dst = dc;
}
dst += stride;
}
}
}
} // namespace
void FourXMDecoder::FourXMVideoTrack::decode_pfrm_block(uint16 *dst, const uint16 *src, int stride, int log2w, int log2h, FourXM::BEWordBitStream &bs, Common::MemoryReadStream &wordStream, Common::MemoryReadStream &byteStream) {
assert(log2w >= 0 && log2h >= 0);
auto index = size2index[log2h][log2w];
assert(index >= 0);
auto &huff = _blockType[index];
auto code = huff.next(bs);
if (code == 1) {
--log2h;
decode_pfrm_block(dst, src, stride, log2w, log2h, bs, wordStream, byteStream);
auto offset = stride << log2h;
decode_pfrm_block(dst + offset, src + offset, stride, log2w, log2h, bs, wordStream, byteStream);
return;
} else if (code == 2) {
--log2w;
decode_pfrm_block(dst, src, stride, log2w, log2h, bs, wordStream, byteStream);
auto offset = 1 << log2w;
decode_pfrm_block(dst + offset, src + offset, stride, log2w, log2h, bs, wordStream, byteStream);
return;
} else if (code == 6) {
assert(wordStream.pos() + 4 <= wordStream.size());
if (log2w) {
dst[0] = wordStream.readUint16LE();
dst[1] = wordStream.readUint16LE();
} else {
dst[0] = wordStream.readUint16LE();
dst[stride] = wordStream.readUint16LE();
}
return;
}
if (code == 3 && _version >= 2)
return;
if (code == 0) {
assert(byteStream.pos() < byteStream.size());
src += _mv[byteStream.readByte()];
mcdc<true>(dst, src, log2w, log2h, stride, 0);
} else if (code == 4) {
assert(byteStream.pos() < byteStream.size());
assert(wordStream.pos() + 2 <= wordStream.size());
src += _mv[byteStream.readByte()];
auto dc = wordStream.readUint16LE();
mcdc<true>(dst, src, log2w, log2h, stride, dc);
} else if (code == 5) {
assert(wordStream.pos() + 2 <= wordStream.size());
auto dc = wordStream.readUint16LE();
mcdc<false>(dst, src, log2w, log2h, stride, dc);
} else {
error("invalid code %d (steps %u,%u)", code, log2w, log2h);
}
}
void FourXMDecoder::FourXMVideoTrack::decode_pfrm(Common::SeekableReadStream *stream) {
stream->skip(8);
auto bitStreamSize = stream->readUint32LE();
auto wordStreamSize = stream->readUint32LE();
auto byteStreamSize = stream->readUint32LE();
Common::Array<byte> bitStreamData(bitStreamSize);
stream->read(bitStreamData.data(), bitStreamData.size());
Common::Array<byte> wordStreamData(wordStreamSize);
stream->read(wordStreamData.data(), wordStreamData.size());
Common::Array<byte> byteStreamData(byteStreamSize);
stream->read(byteStreamData.data(), byteStreamData.size());
FourXM::BEWordBitStream bitStream(reinterpret_cast<const uint32 *>(bitStreamData.data()), bitStreamData.size() / 4, 0);
Common::MemoryReadStream wordStream(wordStreamData.data(), wordStreamData.size());
Common::MemoryReadStream byteStream(byteStreamData.data(), byteStreamData.size());
uint16 *dst = static_cast<uint16 *>(_frame->getPixels());
const uint16 *src = static_cast<const uint16 *>(_lastFrame->getPixels());
assert(_frame->format.bytesPerPixel == 2);
auto stride = _frame->pitch / _frame->format.bytesPerPixel;
assert(stride == _frame->pitch / _frame->format.bytesPerPixel);
for (int y = 0, h = _frame->h; y < h; y += 8) {
for (int x = 0, w = _frame->w; x < w; x += 8) {
decode_pfrm_block(dst + x, src + x, stride, 3, 3, bitStream, wordStream, byteStream);
}
dst += stride << 3;
src += stride << 3;
}
SWAP(_frame, _lastFrame);
}
void FourXMDecoder::FourXMVideoTrack::decode_cfrm(Common::SeekableReadStream *stream) {
auto frameIdx = stream->readUint32LE();
auto frameSize = stream->readUint32LE();
auto streamSize = stream->size() - stream->pos();
auto &cframe = _cframes[frameIdx];
auto dstOffset = cframe.size();
cframe.resize(cframe.size() + streamSize);
stream->read(cframe.data() + dstOffset, streamSize);
if (cframe.size() >= frameSize) {
assert(_dec->_curFrame == frameIdx);
Common::MemoryReadStream ms(cframe.data(), cframe.size());
decode_pfrm(&ms);
_cframes.erase(frameIdx);
}
}
void FourXMDecoder::FourXMVideoTrack::decode(uint32 tag, byte *buf, uint size) {
Common::MemoryReadStream ms(buf, size, DisposeAfterUse::YES);
switch (tag) {
case MKTAG('i', 'f', 'r', 'm'):
decode_ifrm(&ms);
break;
case MKTAG('p', 'f', 'r', 'm'):
decode_pfrm(&ms);
break;
case MKTAG('c', 'f', 'r', 'm'):
decode_cfrm(&ms);
break;
default:
warning("uknown video frame %s", tagName(tag).c_str());
break;
}
}
void FourXMDecoder::decodeNextFrameImpl() {
if (_curFrame >= _frames.size())
return;
auto &frame = _frames[_curFrame];
_stream->seek(frame.offset);
uint32 listType = _stream->readUint32BE();
if (listType != MKTAG('F', 'R', 'A', 'M')) {
error("invalid FRAM offset for frame %u", _curFrame);
}
while (_stream->pos() < frame.end) {
uint32 tag = _stream->readUint32BE();
uint32 size = _stream->readUint32LE();
auto pos = _stream->pos();
auto loadBuf = [this](uint bufSize) {
byte *buf = static_cast<byte *>(malloc(bufSize));
if (!buf)
error("failed to allocate %u bytes", bufSize);
if (_stream->read(buf, bufSize) != bufSize)
error("loadBuf: short read");
return buf;
};
switch (tag) {
case MKTAG('s', 'n', 'd', '_'): {
if (!_audio)
error("no audio track but got sound frame");
switch (_audio->getAudioType()) {
case 0: {
uint offset = 0;
while (offset < size) {
auto trackIdx = _stream->readUint32LE();
auto packetSize = _stream->readUint32LE();
if (trackIdx == 0 && _audio) {
debug("audio data %u %u", trackIdx, packetSize);
_audio->decode(loadBuf(packetSize), packetSize);
} else {
_stream->skip(packetSize);
}
offset += packetSize + 8;
}
} break;
case 1: {
auto trackIdx = _stream->readUint32LE();
_stream->skip(4);
if (trackIdx == 0 && _audio) {
_audio->decode(loadBuf(size - 8), size - 8);
}
} break;
default:
warning("unknown audio type");
}
} break;
case MKTAG('i', 'f', 'r', 'm'):
case MKTAG('p', 'f', 'r', 'm'):
case MKTAG('c', 'f', 'r', 'm'): {
auto trackIdx = _stream->readUint32LE();
if (trackIdx == 0)
_video->decode(tag, loadBuf(size - 4), size - 4);
} break;
default:
warning("unknown frame type %s", tagName(tag).c_str());
_stream->skip(size);
}
_stream->seek(pos + size + (size & 1));
}
++_curFrame;
}
void FourXMDecoder::readList(uint32 listEnd) {
assert(_stream);
uint32 listType = _stream->readUint32BE();
if (listType == MKTAG('F', 'R', 'A', 'M')) {
_frames.push_back({_stream->pos() - 4, listEnd});
return;
}
while (_stream->pos() < listEnd) {
uint32 tag = _stream->readUint32BE();
uint32 size = _stream->readUint32LE();
auto pos = _stream->pos();
if (tag == MKTAG('L', 'I', 'S', 'T')) {
readList(pos + size);
}
switch (listType) {
case MKTAG('H', 'N', 'F', 'O'):
switch (tag) {
case MKTAG('s', 't', 'd', '_'):
_dataRate = _stream->readUint32LE();
_frameRate = floatToRational(_stream->readFloatLE());
debug("data rate: %u, frame rate: %d/%d", _dataRate, _frameRate.getNumerator(), _frameRate.getDenominator());
break;
default:
break;
}
break;
case MKTAG('V', 'T', 'R', 'K'):
switch (tag) {
case MKTAG('v', 't', 'r', 'k'): {
_stream->skip(8);
auto extra = _stream->readUint32LE();
_stream->skip(16);
auto w = _stream->readUint32LE();
auto h = _stream->readUint32LE();
debug("video %ux%u, version: %u", w, h, extra >> 16);
addTrack(_video = new FourXMVideoTrack(this, _frameRate, w, h, extra >> 16));
} break;
default:
break;
}
break;
case MKTAG('S', 'T', 'R', 'K'):
switch (tag) {
case MKTAG('s', 't', 'r', 'k'): {
auto trackIdx = _stream->readUint32LE();
auto audioType = _stream->readUint32LE();
_stream->skip(20);
auto audioChannels = _stream->readUint32LE();
auto sampleRate = _stream->readUint32LE();
auto sampleResolution = _stream->readUint32LE();
debug("audio track idx: %u type: %u channels: %u sample rate: %u bits: %u", trackIdx, audioType, audioChannels, sampleRate, sampleResolution);
if (sampleResolution != 16)
error("only 16 bit audio is supported");
addTrack(_audio = new FourXMAudioTrack(this, trackIdx, audioType, audioChannels, sampleRate));
} break;
default:
break;
}
break;
default:
break;
}
_stream->seek(pos + size + (size & 1));
}
}
bool FourXMDecoder::loadStream(Common::SeekableReadStream *stream) {
if (!stream->size()) {
return false;
}
uint32 riffTag = stream->readUint32BE();
if (riffTag != MKTAG('R', 'I', 'F', 'F')) {
warning("Failed to find RIFF header");
return false;
}
uint32 fileSize = stream->readUint32LE();
uint32 riffType = stream->readUint32BE();
if (riffType != MKTAG('4', 'X', 'M', 'V')) {
warning("RIFF not an 4XM file, got: %08x", riffType);
return false;
}
_stream = stream;
while (stream->pos() < fileSize) {
uint32 tag = stream->readUint32BE();
uint32 size = stream->readUint32LE();
auto pos = stream->pos();
if (tag == MKTAG('L', 'I', 'S', 'T')) {
readList(pos + size);
}
stream->seek(pos + size + (size & 1));
}
return getNumTracks() != 0;
}
} // namespace Video