/* 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. * */ // ROQ video player based on this specification by Dr. Tim Ferguson: // http://www.csse.monash.edu.au/~timf/videocodec/idroq.txt #include "groovie/roq.h" #include "groovie/graphics.h" #include "groovie/groovie.h" #include "common/debug.h" #include "common/rect.h" #include "common/substream.h" #include "common/textconsole.h" #include "graphics/palette.h" #include "image/jpeg.h" #ifdef USE_RGB_COLOR // Required for the YUV to RGB conversion #include "graphics/conversion.h" #endif #include "audio/mixer.h" #include "audio/decoders/raw.h" namespace Groovie { ROQPlayer::ROQPlayer(GroovieEngine *vm) : VideoPlayer(vm), _codingTypeCount(0), _bg(&_vm->_graphicsMan->_background), _firstFrame(true) { // Create the work surfaces _currBuf = new Graphics::Surface(); _prevBuf = new Graphics::Surface(); } ROQPlayer::~ROQPlayer() { // Free the buffers _currBuf->free(); delete _currBuf; _prevBuf->free(); delete _prevBuf; } uint16 ROQPlayer::loadInternal() { // Begin reading the file debugC(1, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Loading video"); // Read the file header ROQBlockHeader blockHeader; if (!readBlockHeader(blockHeader)) { return 0; } // Verify the file signature if (blockHeader.type != 0x1084) { return 0; } // Clear the dirty flag _dirty = true; // Reset the codebooks _num2blocks = 0; _num4blocks = 0; // Reset the first frame flag _firstFrame = true; if ((blockHeader.size == 0) && (blockHeader.param == 0)) { // Set the offset scaling to 2 _offScale = 2; // Hardcoded FPS return 30; } else if (blockHeader.size == (uint32)-1) { // Set the offset scaling to 1 _offScale = 1; // In this case the block parameter is the framerate return blockHeader.param; } else { warning("Groovie::ROQ: Invalid header with size=%d and param=%d", blockHeader.size, blockHeader.param); return 0; } } void ROQPlayer::buildShowBuf() { for (int line = 0; line < _bg->h; line++) { uint32 *out = (uint32 *)_bg->getBasePtr(0, line); uint32 *in = (uint32 *)_currBuf->getBasePtr(0, line / _scaleY); for (int x = 0; x < _bg->w; x++) { // Copy a pixel *out++ = *in; // Skip to the next pixel if (!(x % _scaleX)) in++; } } // On the first frame, copy from the current buffer to the prev buffer if (_firstFrame) { _prevBuf->copyFrom(*_currBuf); _firstFrame = false; } // Swap buffers SWAP(_prevBuf, _currBuf); } bool ROQPlayer::playFrameInternal() { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Playing frame"); // Process the needed blocks until the next video frame bool endframe = false; while (!endframe && !_file->eos()) { endframe = processBlock(); } if (_dirty) { // Build the show buffer from the current buffer buildShowBuf(); } // Wait until the current frame can be shown waitFrame(); if (_dirty) { // Update the screen _syst->copyRectToScreen(_bg->getPixels(), _bg->pitch, 0, (_syst->getHeight() - _bg->h) / 2, _bg->w, _bg->h); _syst->updateScreen(); // Clear the dirty flag _dirty = false; } // Return whether the video has ended return _file->eos(); } bool ROQPlayer::readBlockHeader(ROQBlockHeader &blockHeader) { if (_file->eos()) { return false; } else { blockHeader.type = _file->readUint16LE(); blockHeader.size = _file->readUint32LE(); blockHeader.param = _file->readUint16LE(); debugC(10, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Block type = 0x%02X", blockHeader.type); debugC(10, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Block size = 0x%08X", blockHeader.size); debugC(10, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Block param = 0x%04X", blockHeader.param); return true; } } bool ROQPlayer::processBlock() { // Read the header of the block ROQBlockHeader blockHeader; if (!readBlockHeader(blockHeader)) { return true; } // Calculate where the block should end int32 endpos = _file->pos() + blockHeader.size; // Detect the end of the video if (_file->eos()) { return false; } bool ok = true; bool endframe = false; switch (blockHeader.type) { case 0x1001: // Video info ok = processBlockInfo(blockHeader); break; case 0x1002: // Quad codebook definition ok = processBlockQuadCodebook(blockHeader); break; case 0x1011: // Quad vector quantised video frame ok = processBlockQuadVector(blockHeader); _dirty = true; endframe = true; break; case 0x1012: // Still image (JPEG) ok = processBlockStill(blockHeader); _dirty = true; endframe = true; break; case 0x1013: // Hang assert(blockHeader.size == 0 && blockHeader.param == 0); endframe = true; break; case 0x1020: // Mono sound samples ok = processBlockSoundMono(blockHeader); break; case 0x1021: // Stereo sound samples ok = processBlockSoundStereo(blockHeader); break; case 0x1030: // Audio container endpos = _file->pos(); ok = processBlockAudioContainer(blockHeader); break; default: warning("Groovie::ROQ: Unknown block type: 0x%04X", blockHeader.type); ok = false; _file->skip(blockHeader.size); } if (endpos != _file->pos()) warning("Groovie::ROQ: BLOCK %04x Should have ended at %d, and has ended at %d", blockHeader.type, endpos, _file->pos()); // End the frame when the graphics have been modified or when there's an error return endframe || !ok; } bool ROQPlayer::processBlockInfo(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing info block"); // Verify the block header if (blockHeader.type != 0x1001 || blockHeader.size != 8 || (blockHeader.param != 0 && blockHeader.param != 1)) { warning("Groovie::ROQ: BlockInfo size=%d param=%d", blockHeader.size, blockHeader.param); return false; } // Reset the first frame flag _firstFrame = true; // Save the alpha channel size _alpha = blockHeader.param; // Read the information uint16 width = _file->readUint16LE(); uint16 height = _file->readUint16LE(); uint16 unk1 = _file->readUint16LE(); uint16 unk2 = _file->readUint16LE(); if (unk1 != 8 || unk2 != 4) { warning("Groovie::ROQ: unk1 = %d, unk2 = %d", unk1, unk2); return false; } // If the size of the image has changed, resize the buffers if ((width != _currBuf->w) || (height != _currBuf->h)) { // Calculate the maximum scale that fits the screen _scaleX = MIN(_syst->getWidth() / width, 2); _scaleY = MIN(_syst->getHeight() / height, 2); // Free the previous surfaces _currBuf->free(); _prevBuf->free(); // Allocate new buffers _currBuf->create(width, height, _vm->_pixelFormat); _prevBuf->create(width, height, _vm->_pixelFormat); } // Clear the buffers with black _currBuf->fillRect(Common::Rect(width, height), _vm->_pixelFormat.RGBToColor(0, 0, 0)); _prevBuf->fillRect(Common::Rect(width, height), _vm->_pixelFormat.RGBToColor(0, 0, 0)); return true; } bool ROQPlayer::processBlockQuadCodebook(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing quad codebook block"); // Get the number of 2x2 pixel blocks to read int newNum2blocks = blockHeader.param >> 8; if (newNum2blocks == 0) { newNum2blocks = 256; } if (newNum2blocks > _num2blocks) _num2blocks = newNum2blocks; // Get the number of 4x4 pixel blocks _num4blocks = blockHeader.param & 0xFF; if ((_num4blocks == 0) && (blockHeader.size > (uint32)_num2blocks * (6 + _alpha * 4))) { _num4blocks = 256; } // Read the 2x2 codebook uint32 *codebook = _codebook2; for (int i = 0; i < newNum2blocks; i++) { // Read the 4 Y components and their alpha channel byte y[4]; byte a[4]; for (int j = 0; j < 4; j++) { y[j] = _file->readByte(); a[j] = _alpha ? _file->readByte() : 255; } // Read the subsampled Cb and Cr byte u = _file->readByte(); byte v = _file->readByte(); // Convert the codebook to RGB right here for (int j = 0; j < 4; j++) { byte r, g, b; Graphics::YUV2RGB(y[j], u, v, r, g, b); *codebook++ = _vm->_pixelFormat.ARGBToColor(a[j], r, g, b); } } // Read the 4x4 codebook _file->read(_codebook4, _num4blocks * 4); return true; } bool ROQPlayer::processBlockQuadVector(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing quad vector block"); // Get the mean motion vectors int8 Mx = blockHeader.param >> 8; int8 My = blockHeader.param & 0xFF; // Calculate where the block should end int32 endpos =_file->pos() + blockHeader.size; // Reset the coding types _codingTypeCount = 0; // Traverse the image in 16x16 macroblocks for (int macroY = 0; macroY < _currBuf->h; macroY += 16) { for (int macroX = 0; macroX < _currBuf->w; macroX += 16) { // Traverse the macroblock in 8x8 blocks for (int blockY = 0; blockY < 16; blockY += 8) { for (int blockX = 0; blockX < 16; blockX += 8) { processBlockQuadVectorBlock(macroX + blockX, macroY + blockY, Mx, My); } } } } // HACK: Skip the remaining bytes int32 skipBytes = endpos -_file->pos(); if (skipBytes > 0) { _file->skip(skipBytes); if (skipBytes != 2) { warning("Groovie::ROQ: Skipped %d bytes", skipBytes); } } return true; } void ROQPlayer::processBlockQuadVectorBlock(int baseX, int baseY, int8 Mx, int8 My) { uint16 codingType = getCodingType(); switch (codingType) { case 0: // MOT: Skip block break; case 1: { // FCC: Copy an existing block byte argument = _file->readByte(); int16 DDx = 8 - (argument >> 4); int16 DDy = 8 - (argument & 0x0F); copy(8, baseX, baseY, DDx - Mx, DDy - My); break; } case 2: // SLD: Quad vector quantisation // Upsample the 4x4 pixel block paint8(_file->readByte(), baseX, baseY); break; case 3: // CCC: // Traverse the block in 4x4 sub-blocks for (int subBlockY = 0; subBlockY < 8; subBlockY += 4) { for (int subBlockX = 0; subBlockX < 8; subBlockX += 4) { processBlockQuadVectorBlockSub(baseX + subBlockX, baseY + subBlockY, Mx, My); } } break; } } void ROQPlayer::processBlockQuadVectorBlockSub(int baseX, int baseY, int8 Mx, int8 My) { debugC(6, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing quad vector sub block"); uint16 codingType = getCodingType(); switch (codingType) { case 0: // MOT: Skip block break; case 1: { // FCC: Copy an existing block byte argument = _file->readByte(); int16 DDx = 8 - (argument >> 4); int16 DDy = 8 - (argument & 0x0F); copy(4, baseX, baseY, DDx - Mx, DDy - My); break; } case 2: // SLD: Quad vector quantisation paint4(_file->readByte(), baseX, baseY); break; case 3: paint2(_file->readByte(), baseX , baseY); paint2(_file->readByte(), baseX + 2, baseY); paint2(_file->readByte(), baseX , baseY + 2); paint2(_file->readByte(), baseX + 2, baseY + 2); break; } } bool ROQPlayer::processBlockStill(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing still (JPEG) block"); Image::JPEGDecoder jpg; uint32 startPos = _file->pos(); Common::SeekableSubReadStream subStream(_file, startPos, startPos + blockHeader.size, DisposeAfterUse::NO); jpg.loadStream(subStream); const Graphics::Surface *srcSurf = jpg.getSurface(); _currBuf->free(); delete _currBuf; _currBuf = srcSurf->convertTo(_vm->_pixelFormat); _file->seek(startPos + blockHeader.size); return true; } bool ROQPlayer::processBlockSoundMono(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing mono sound block"); // Verify the block header if (blockHeader.type != 0x1020) { return false; } // Initialize the audio stream if needed if (!_audioStream) { _audioStream = Audio::makeQueuingAudioStream(22050, false); Audio::SoundHandle sound_handle; g_system->getMixer()->playStream(Audio::Mixer::kPlainSoundType, &sound_handle, _audioStream); } // Create the audio buffer int16 *buffer = (int16 *)malloc(blockHeader.size * 2); // Initialize the prediction with the block parameter int16 prediction = blockHeader.param ^ 0x8000; // Process the data for (uint16 i = 0; i < blockHeader.size; i++) { int16 data = _file->readByte(); if (data < 0x80) { prediction += data * data; } else { data -= 0x80; prediction -= data * data; } buffer[i] = prediction; } // Queue the read buffer byte flags = Audio::FLAG_16BITS; #ifdef SCUMM_LITTLE_ENDIAN flags |= Audio::FLAG_LITTLE_ENDIAN; #endif _audioStream->queueBuffer((byte *)buffer, blockHeader.size * 2, DisposeAfterUse::YES, flags); return true; } bool ROQPlayer::processBlockSoundStereo(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing stereo sound block"); // Verify the block header if (blockHeader.type != 0x1021) { return false; } // Initialize the audio stream if needed if (!_audioStream) { _audioStream = Audio::makeQueuingAudioStream(22050, true); Audio::SoundHandle sound_handle; g_system->getMixer()->playStream(Audio::Mixer::kPlainSoundType, &sound_handle, _audioStream); } // Create the audio buffer int16 *buffer = (int16 *)malloc(blockHeader.size * 2); // Initialize the prediction with the block parameter int16 predictionLeft = (blockHeader.param & 0xFF00) ^ 0x8000; int16 predictionRight = (blockHeader.param << 8) ^ 0x8000; bool left = true; // Process the data for (uint16 i = 0; i < blockHeader.size; i++) { int16 data = _file->readByte(); if (left) { if (data < 0x80) { predictionLeft += data * data; } else { data -= 0x80; predictionLeft -= data * data; } buffer[i] = predictionLeft; } else { if (data < 0x80) { predictionRight += data * data; } else { data -= 0x80; predictionRight -= data * data; } buffer[i] = predictionRight; } left = !left; } // Queue the read buffer byte flags = Audio::FLAG_16BITS | Audio::FLAG_STEREO; #ifdef SCUMM_LITTLE_ENDIAN flags |= Audio::FLAG_LITTLE_ENDIAN; #endif _audioStream->queueBuffer((byte *)buffer, blockHeader.size * 2, DisposeAfterUse::YES, flags); return true; } bool ROQPlayer::processBlockAudioContainer(ROQBlockHeader &blockHeader) { debugC(5, kGroovieDebugVideo | kGroovieDebugAll, "Groovie::ROQ: Processing audio container block: 0x%04X", blockHeader.param); return true; } byte ROQPlayer::getCodingType() { _codingType <<= 2; if (!_codingTypeCount) { _codingType = _file->readUint16LE(); _codingTypeCount = 8; } _codingTypeCount--; return (_codingType >> 14); } void ROQPlayer::paint2(byte i, int destx, int desty) { if (i > _num2blocks) { error("Groovie::ROQ: Invalid 2x2 block %d (%d available)", i, _num2blocks); } uint32 *block = _codebook2 + i * 4; uint32 *ptr = (uint32 *)_currBuf->getBasePtr(destx, desty); uint32 pitch = _currBuf->pitch / 4; ptr[0] = block[0]; ptr[1] = block[1]; ptr[pitch] = block[2]; ptr[pitch + 1] = block[3]; } void ROQPlayer::paint4(byte i, int destx, int desty) { if (i > _num4blocks) { error("Groovie::ROQ: Invalid 4x4 block %d (%d available)", i, _num4blocks); } byte *block4 = &_codebook4[i * 4]; for (int origy = 0; origy < 4; origy += 2) { for (int origx = 0; origx < 4; origx += 2) { paint2(*block4, destx + origx, desty + origy); block4++; } } } void ROQPlayer::paint8(byte i, int destx, int desty) { if (i > _num4blocks) { error("Groovie::ROQ: Invalid 4x4 block %d (%d available)", i, _num4blocks); } byte *block4 = &_codebook4[i * 4]; for (int y4 = 0; y4 < 2; y4++) { for (int x4 = 0; x4 < 2; x4++) { uint32 *block2 = _codebook2 + *block4++ * 4; for (int y2 = 0; y2 < 2; y2++) { for (int x2 = 0; x2 < 2; x2++) { uint32 *ptr = (uint32 *)_currBuf->getBasePtr(destx + x4 * 4 + x2 * 2, desty + y4 * 4 + y2 * 2); uint32 pitch = _currBuf->pitch / 4; uint32 color = *block2++; ptr[0] = ptr[1] = ptr[pitch] = ptr[pitch + 1] = color; } } } } } void ROQPlayer::copy(byte size, int destx, int desty, int offx, int offy) { offx *= _offScale / _scaleX; offy *= _offScale / _scaleY; // Get the beginning of the first line byte *dst = (byte *)_currBuf->getBasePtr(destx, desty); byte *src = (byte *)_prevBuf->getBasePtr(destx + offx, desty + offy); for (int i = 0; i < size; i++) { // Copy the current line memcpy(dst, src, size * _currBuf->format.bytesPerPixel); // Move to the beginning of the next line dst += _currBuf->pitch; src += _prevBuf->pitch; } } } // End of Groovie namespace