/* 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 . * */ #include "video/4xm_decoder.h" #include "audio/audiostream.h" #include "audio/decoders/adpcm.h" #include "audio/decoders/raw.h" #include "common/bitstream.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 { FourXMDecoder::~FourXMDecoder() { close(); } Common::Rational floatToRational(float value) { int num = static_cast(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; Common::ScopedPtr _output; public: FourXMAudioTrack(FourXMDecoder *dec, uint trackIdx, uint audioType, uint audioChannels, uint sampleRate) : AudioTrack(Audio::Mixer::SoundType::kPlainSoundType), _audioType(audioType), _audioChannels(audioChannels) { switch (_audioType) { case 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.reset(Audio::makePacketizedRawStream(sampleRate, flags)); break; } case 1: _output.reset(Audio::makePacketizedADPCMStream(Audio::ADPCMType::kADPCM4XM, sampleRate, audioChannels)); break; default: error("FourXMAudioTrack: unknown audio type: %d", _audioType); } } byte getAudioType() const { return _audioType; } void decode(byte *buf, uint size) { auto *input = new Common::MemoryReadStream(buf, size, DisposeAfterUse::YES); _output->queuePacket(input); } 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 _framePtr, _lastFramePtr; Graphics::Surface *_frame = nullptr, *_lastFrame = nullptr; using HuffmanBitStream = Common::BitStreamMemory32LEMSB; using HuffmanType = Common::Huffman; Common::ScopedPtr _blockType[4]; int _mv[256]; Common::HashMap> _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].reset(new HuffmanType(HuffmanType::fromFrequencies({16, 8, 4, 2, 1, 1}))); _blockType[1].reset(new HuffmanType(HuffmanType::fromFrequencies({8, 0, 4, 2, 1, 1}))); _blockType[2].reset(new HuffmanType(HuffmanType::fromFrequencies({8, 4, 0, 2, 1, 1}))); _blockType[3].reset(new HuffmanType(HuffmanType::fromFrequencies({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, Common::BitStreamMemory32LEMSB &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 bitstreamData(bitstreamSize); stream->read(bitstreamData.data(), bitstreamData.size()); auto prefixSize = stream->readUint32LE(); /* auto tokenCount = */ stream->readUint32LE(); Common::Array prefixStream(prefixSize * 4); stream->read(prefixStream.data(), prefixStream.size()); assert(stream->pos() == stream->size()); const auto *huffPtr = prefixStream.data(); uint huffOffset = 0; auto prefixDecoder = FourXM::loadStatistics(huffPtr, huffOffset); huffOffset = (huffOffset + 3) & ~3u; Common::BitStreamMemoryStream huffMs(huffPtr + huffOffset, prefixStream.size() - huffOffset); Common::BitStreamMemory32LEMSB huffBs{&huffMs}; Common::BitStreamMemoryStream bitstreamInput(bitstreamData.data(), bitstreamData.size()); Common::BitStreamMemory8MSB bitstream(&bitstreamInput); 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 = prefixDecoder.getSymbol(huffBs); if (dc >> 4) error("dc run code"); dc = FourXM::readInt(bitstream, dc); dc = lastDC + dc * iquant[0]; lastDC = dc; ac[0] = dc; for (uint idx = 1; idx < 64;) { auto b = prefixDecoder.getSymbol(huffBs); 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] * FourXM::readInt(bitstream, 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(_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; } } } } SWAP(_frame, _lastFrame); } namespace { template 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) { switch (log2w) { case 3: WRITE_UINT32(dst + 4, READ_UINT32(src + 4) + dc); WRITE_UINT32(dst + 6, READ_UINT32(src + 6) + dc); // fall through case 2: WRITE_UINT32(dst + 2, READ_UINT32(src + 2) + dc); // fall through case 1: WRITE_UINT32(dst, READ_UINT32(src) + dc); break; case 0: *dst = *src + dc; } src += stride; dst += stride; } } else { for (int i = 0; i < h; ++i) { switch (log2w) { case 3: WRITE_UINT32(dst + 4, dc); WRITE_UINT32(dst + 6, dc); // fall through case 2: WRITE_UINT32(dst + 2, dc); // fall through case 1: WRITE_UINT32(dst, 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, Common::BitStreamMemory32LEMSB &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.getSymbol(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(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(dst, src, log2w, log2h, stride, dc); } else if (code == 5) { assert(wordStream.pos() + 2 <= wordStream.size()); auto dc = wordStream.readUint16LE(); mcdc(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 bitStreamData(bitStreamSize); stream->read(bitStreamData.data(), bitStreamData.size()); Common::Array wordStreamData(wordStreamSize); stream->read(wordStreamData.data(), wordStreamData.size()); Common::Array byteStreamData(byteStreamSize); stream->read(byteStreamData.data(), byteStreamData.size()); Common::BitStreamMemoryStream bitStreamInput(bitStreamData.data(), bitStreamData.size()); Common::BitStreamMemory32LEMSB bitStream(&bitStreamInput); Common::MemoryReadStream wordStream(wordStreamData.data(), wordStreamData.size()); Common::MemoryReadStream byteStream(byteStreamData.data(), byteStreamData.size()); uint16 *dst = static_cast(_frame->getPixels()); const uint16 *src = static_cast(_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(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) { _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) { _stream.reset(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; } 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