/* * InterplayDecoder * Copyright (C) 2020 sirlemonhead * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /* This code is based on interplayvideo.c, dpcm.c and ipmovie.c from the FFmpeg project which can be obtained * from http://www.ffmpeg.org/. Below is the license from interplayvideo.c * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ /* * Interplay MVE Video Decoder * Copyright (C) 2003 The FFmpeg project * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include "playmve.h" #include "printf.h" #include "v_draw.h" #include "s_music.h" #include "cmdlib.h" static const int16_t delta_table[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 47, 51, 56, 61, 66, 72, 79, 86, 94, 102, 112, 122, 133, 145, 158, 173, 189, 206, 225, 245, 267, 292, 318, 348, 379, 414, 452, 493, 538, 587, 640, 699, 763, 832, 908, 991, 1081, 1180, 1288, 1405, 1534, 1673, 1826, 1993, 2175, 2373, 2590, 2826, 3084, 3365, 3672, 4008, 4373, 4772, 5208, 5683, 6202, 6767, 7385, 8059, 8794, 9597, 10472, 11428, 12471, 13609, 14851, 16206, 17685, 19298, 21060, 22981, 25078, 27367, 29864, 32589, -29973, -26728, -23186, -19322, -15105, -10503, -5481, -1, 1, 1, 5481, 10503, 15105, 19322, 23186, 26728, 29973, -32589, -29864, -27367, -25078, -22981, -21060, -19298, -17685, -16206, -14851, -13609, -12471, -11428, -10472, -9597, -8794, -8059, -7385, -6767, -6202, -5683, -5208, -4772, -4373, -4008, -3672, -3365, -3084, -2826, -2590, -2373, -2175, -1993, -1826, -1673, -1534, -1405, -1288, -1180, -1081, -991, -908, -832, -763, -699, -640, -587, -538, -493, -452, -414, -379, -348, -318, -292, -267, -245, -225, -206, -189, -173, -158, -145, -133, -122, -112, -102, -94, -86, -79, -72, -66, -61, -56, -51, -47, -43, -42, -41, -40, -39, -38, -37, -36, -35, -34, -33, -32, -31, -30, -29, -28, -27, -26, -25, -24, -23, -22, -21, -20, -19, -18, -17, -16, -15, -14, -13, -12, -11, -10, -9, -8, -7, -6, -5, -4, -3, -2, -1 }; // macros to fetch little-endian words from a bytestream #define LE_16(x) ((uint16_t)((*(x)) | ((*((x)+1)) << 8))) #define LE_32(x) (LE_16(x) | ((uint32_t)LE_16(x+2) << 16)) #define LE_64(x) (LE_32(x) | ((uint64_t)LE_32(x+4) << 32)) bool InterplayDecoder::FillSamples(void *buff, int len) { for (int i = 0; i < len;) { if(audio.nRead < audio.nWrite) { int todo = std::min(audio.nWrite-audio.nRead, (len-i) / 2); memcpy((char*)buff+i, &audio.samples[audio.nRead], todo*2); audio.nRead += todo; if (audio.nRead == audio.nWrite) audio.nRead = audio.nWrite = 0; i += todo*2; continue; } std::unique_lock plock(PacketMutex); while (audio.Packets.empty()) { if (!bIsPlaying || ProcessNextChunk() >= CHUNK_SHUTDOWN) { bIsPlaying = false; if (i == 0) return false; memset((char*)buff+i, 0, len-i); return true; } } AudioPacket pkt = std::move(audio.Packets.front()); audio.Packets.pop_front(); plock.unlock(); int nSamples = (int)pkt.nSize; const uint8_t *samplePtr = pkt.pData.get(); if (audio.bCompressed) { int predictor[2]; for (int ch = 0; ch < audio.nChannels; ch++) { predictor[ch] = (int16_t)LE_16(samplePtr); samplePtr += 2; audio.samples[audio.nWrite++] = predictor[ch]; } bool stereo = audio.nChannels == 2; nSamples -= 2*audio.nChannels; nSamples &= ~(int)stereo; int ch = 0; for (int j = 0; j < nSamples; ++j) { predictor[ch] += delta_table[*samplePtr++]; predictor[ch] = clamp(predictor[ch], -32768, 32767); audio.samples[audio.nWrite++] = predictor[ch]; // toggle channel ch ^= stereo ? 1 : 0; } } else if (audio.nBitDepth == 8) { for (int j = 0; j < nSamples; ++j) audio.samples[audio.nWrite++] = ((*samplePtr++)-128) << 8; } else { nSamples /= 2; for (int j = 0; j < nSamples; ++j) { audio.samples[audio.nWrite++] = (int16_t)LE_16(samplePtr); samplePtr += 2; } } } return true; } void InterplayDecoder::DisableAudio() { if (bAudioEnabled) { std::unique_lock plock(PacketMutex); bAudioEnabled = false; audio.Packets.clear(); } } InterplayDecoder::InterplayDecoder(bool soundenabled) { bIsPlaying = false; bAudioEnabled = soundenabled; nWidth = 0; nHeight = 0; nFrame = 0; memset(palette, 0, sizeof(palette)); nFps = 0.0; nFrameDuration = 0; nTimerRate = 0; nTimerDiv = 0; pVideoBuffers[0] = nullptr; pVideoBuffers[1] = nullptr; decodeMap.pData = nullptr; decodeMap.nSize = 0; nCurrentVideoBuffer = 0; nPreviousVideoBuffer = 1; videoStride = 0; } InterplayDecoder::~InterplayDecoder() { Close(); } void InterplayDecoder::SwapFrames() { int t = nPreviousVideoBuffer; nPreviousVideoBuffer = nCurrentVideoBuffer; nCurrentVideoBuffer = t; } int InterplayDecoder::ProcessNextChunk() { uint8_t chunkPreamble[CHUNK_PREAMBLE_SIZE]; if (fr.Read(chunkPreamble, CHUNK_PREAMBLE_SIZE) != CHUNK_PREAMBLE_SIZE) { Printf(TEXTCOLOR_RED "InterplayDecoder: could not read from file (EOF?)\n"); return CHUNK_EOF; } int chunkSize = LE_16(&chunkPreamble[0]); int chunkType = LE_16(&chunkPreamble[2]); ChunkData.resize(chunkSize); if (fr.Read(ChunkData.data(), chunkSize) != chunkSize) { Printf(TEXTCOLOR_RED "InterplayDecoder: could not read from file (EOF?)\n"); return CHUNK_BAD; } const uint8_t *palPtr = nullptr; const uint8_t *mapPtr = nullptr; const uint8_t *vidPtr = nullptr; int palStart = 0, palCount = 0; int mapSize = 0, vidSize = 0; // iterate through individual opcodes const uint8_t *chunkPtr = ChunkData.data(); while (chunkSize > 0 && chunkType != CHUNK_BAD) { if (chunkSize < OPCODE_PREAMBLE_SIZE) { Printf(TEXTCOLOR_RED "InterplayDecoder: opcode size too small\n"); return CHUNK_BAD; } int opcodeSize = LE_16(chunkPtr); int opcodeType = chunkPtr[2]; int opcodeVersion = chunkPtr[3]; chunkPtr += OPCODE_PREAMBLE_SIZE; chunkSize -= OPCODE_PREAMBLE_SIZE; if (chunkSize < opcodeSize) { Printf(TEXTCOLOR_RED "InterplayDecoder: opcode size too large for chunk\n"); return CHUNK_BAD; } chunkSize -= opcodeSize; switch (opcodeType) { case OPCODE_END_OF_STREAM: chunkPtr += opcodeSize; break; case OPCODE_END_OF_CHUNK: chunkPtr += opcodeSize; break; case OPCODE_CREATE_TIMER: nTimerRate = LE_32(chunkPtr); nTimerDiv = LE_16(chunkPtr+4); chunkPtr += 6; nFrameDuration = ((uint64_t)nTimerRate * nTimerDiv) * 1000; break; case OPCODE_INIT_AUDIO_BUFFERS: { // Skip 2 bytes uint16_t flags = LE_16(chunkPtr+2); audio.nSampleRate = LE_16(chunkPtr+4); chunkPtr += 6; uint32_t nBufferBytes = (opcodeVersion == 0) ? LE_16(chunkPtr) : LE_32(chunkPtr); chunkPtr += (opcodeVersion == 0) ? 2 : 4; audio.nChannels = (flags & 0x1) ? 2 : 1; audio.nBitDepth = (flags & 0x2) ? 16 : 8; audio.bCompressed = (opcodeVersion > 0 && (flags & 0x4)); audio.samples = std::make_unique(nBufferBytes / (audio.nBitDepth/8)); audio.nRead = audio.nWrite = 0; break; } case OPCODE_START_STOP_AUDIO: chunkPtr += opcodeSize; break; case OPCODE_INIT_VIDEO_BUFFERS: { assert(((opcodeVersion == 0 && opcodeSize >= 4) || (opcodeVersion == 1 && opcodeSize >= 6) || (opcodeVersion == 2 && opcodeSize >= 8)) && opcodeSize <= 8 && ! videoStride); nWidth = LE_16(chunkPtr) * 8; nHeight = LE_16(chunkPtr+2) * 8; int count, truecolour; if (opcodeVersion > 0) { count = LE_16(chunkPtr+4); if (opcodeVersion > 1) { truecolour = LE_16(chunkPtr+6); assert(truecolour == 0); } } chunkPtr += opcodeSize; pVideoBuffers[0] = new uint8_t[nWidth * nHeight]; pVideoBuffers[1] = new uint8_t[nWidth * nHeight]; videoStride = nWidth; animtex.SetSize(AnimTexture::Paletted, nWidth, nHeight); break; } case OPCODE_UNKNOWN_06: case OPCODE_UNKNOWN_0E: case OPCODE_UNKNOWN_10: case OPCODE_UNKNOWN_12: case OPCODE_UNKNOWN_13: case OPCODE_UNKNOWN_14: case OPCODE_UNKNOWN_15: chunkPtr += opcodeSize; break; case OPCODE_SEND_BUFFER: //int nPalStart = LE_16(chunkPtr); //int nPalCount = LE_16(chunkPtr+2); { VideoPacket pkt; pkt.pData = std::make_unique(palCount*3 + mapSize + vidSize); pkt.nPalStart = palStart; pkt.nPalCount = palCount; pkt.nDecodeMapSize = mapSize; pkt.nVideoDataSize = vidSize; if (palPtr) memcpy(pkt.pData.get(), palPtr, palCount*3); if (mapPtr) memcpy(pkt.pData.get() + palCount*3, mapPtr, mapSize); if (vidPtr) memcpy(pkt.pData.get() + palCount*3 + mapSize, vidPtr, vidSize); pkt.bSendFlag = true; VideoPackets.emplace_back(std::move(pkt)); } palPtr = nullptr; palStart = palCount = 0; mapPtr = nullptr; mapSize = 0; vidPtr = nullptr; vidSize = 0; chunkPtr += opcodeSize; break; case OPCODE_AUDIO_FRAME: { uint16_t seqIndex = LE_16(chunkPtr); uint16_t streamMask = LE_16(chunkPtr+2); uint16_t nSamples = LE_16(chunkPtr+4); // number of samples this chunk(?) chunkPtr += 6; // We only bother with stream 0 if (!(streamMask & 1) || !bAudioEnabled) { chunkPtr += opcodeSize - 6; break; } AudioPacket pkt; pkt.nSize = opcodeSize - 6; pkt.pData = std::make_unique(pkt.nSize); memcpy(pkt.pData.get(), chunkPtr, pkt.nSize); audio.Packets.emplace_back(std::move(pkt)); chunkPtr += opcodeSize - 6; break; } case OPCODE_SILENCE_FRAME: chunkPtr += opcodeSize; break; case OPCODE_INIT_VIDEO_MODE: chunkPtr += opcodeSize; break; case OPCODE_CREATE_GRADIENT: chunkPtr += opcodeSize; Printf("InterplayDecoder: Create gradient not supported.\n"); break; case OPCODE_SET_PALETTE: if (opcodeSize > 0x304 || opcodeSize < 4) { Printf("set_palette opcode with invalid size\n"); chunkType = CHUNK_BAD; break; } palStart = LE_16(chunkPtr); palCount = LE_16(chunkPtr+2); palPtr = chunkPtr + 4; if (palStart > 255 || palStart+palCount > 256) { Printf("set_palette indices out of range (%d -> %d)\n", palStart, palStart+palCount-1); chunkType = CHUNK_BAD; break; } if (opcodeSize-4 < palCount*3) { Printf("set_palette opcode too small (%d < %d)\n", opcodeSize-4, palCount*3); chunkType = CHUNK_BAD; break; } chunkPtr += opcodeSize; break; case OPCODE_SET_PALETTE_COMPRESSED: chunkPtr += opcodeSize; Printf("InterplayDecoder: Set palette compressed not supported.\n"); break; case OPCODE_SET_DECODING_MAP: mapPtr = chunkPtr; mapSize = opcodeSize; chunkPtr += opcodeSize; break; case OPCODE_VIDEO_DATA: vidPtr = chunkPtr; vidSize = opcodeSize; chunkPtr += opcodeSize; break; default: Printf("InterplayDecoder: Unknown opcode (0x%x v%d, %d bytes).\n", opcodeType, opcodeVersion, opcodeSize); chunkPtr += opcodeSize; break; } } if (chunkType < CHUNK_SHUTDOWN && (palPtr || mapPtr || vidPtr)) { VideoPacket pkt; pkt.pData = std::make_unique(palCount*3 + mapSize + vidSize); pkt.nPalStart = palStart; pkt.nPalCount = palCount; pkt.nDecodeMapSize = mapSize; pkt.nVideoDataSize = vidSize; if (palPtr) memcpy(pkt.pData.get(), palPtr, palCount*3); if(mapPtr) memcpy(pkt.pData.get() + palCount*3, mapPtr, mapSize); if(vidPtr) memcpy(pkt.pData.get() + palCount*3 + mapSize, vidPtr, vidSize); pkt.bSendFlag = false; VideoPackets.emplace_back(std::move(pkt)); } return chunkType; } void InterplayDecoder::Close() { bIsPlaying = false; fr.Close(); if (decodeMap.pData) { delete[] decodeMap.pData; decodeMap.pData = nullptr; } if (pVideoBuffers[0]) { delete[] pVideoBuffers[0]; pVideoBuffers[0] = nullptr; } if (pVideoBuffers[1]) { delete[] pVideoBuffers[1]; pVideoBuffers[1] = nullptr; } } bool InterplayDecoder::Open(FileReader &fr_) { // open the file (read only) char lsig[20]; // check the file signature fr_.Read((uint8_t*)lsig, sizeof(lsig)); if (memcmp(lsig, "Interplay MVE File\x1A\0", sizeof(lsig)) != 0) { Printf(TEXTCOLOR_RED "InterplayDecoder: Unknown MVE signature\n "); return false; } // skip the next 6 bytes fr_.Seek(6, FileReader::SeekCur); fr = std::move(fr_); if (ProcessNextChunk() != CHUNK_INIT_VIDEO) { Printf(TEXTCOLOR_RED "InterplayDecoder: First chunk not CHUNK_INIT_VIDEO\n"); return false; } uint8_t chunkPreamble[CHUNK_PREAMBLE_SIZE]; if (fr.Read(chunkPreamble, CHUNK_PREAMBLE_SIZE) != CHUNK_PREAMBLE_SIZE) { Printf(TEXTCOLOR_RED "InterplayDecoder: could not read from file (EOF?)\n"); return false; } fr.Seek(-CHUNK_PREAMBLE_SIZE, FileReader::SeekCur); int chunkType = LE_16(&chunkPreamble[2]); if (chunkType == CHUNK_VIDEO) bAudioEnabled = false; else { if (ProcessNextChunk() != CHUNK_INIT_AUDIO) { Printf(TEXTCOLOR_RED "InterplayDecoder: Second non-video chunk not CHUNK_INIT_AUDIO\n"); return false; } bAudioEnabled = audio.nSampleRate > 0; } bIsPlaying = true; return true; } bool InterplayDecoder::RunFrame(uint64_t clock) { // handle timing - wait until we're ready to process the next frame. if (nNextFrameTime > clock) { return true; } nNextFrameTime += nFrameDuration; bool doFrame = false; do { std::unique_lock plock(PacketMutex); while (VideoPackets.empty()) { if (!bIsPlaying || ProcessNextChunk() >= CHUNK_SHUTDOWN) { bIsPlaying = false; return false; } } VideoPacket pkt = std::move(VideoPackets.front()); VideoPackets.pop_front(); plock.unlock(); const uint8_t *palData = pkt.pData.get(); const uint8_t *mapData = palData + pkt.nPalCount*3; const uint8_t *vidData = mapData + pkt.nDecodeMapSize; if (pkt.nPalCount > 0) { int nPalEnd = pkt.nPalStart + pkt.nPalCount; for (int i = pkt.nPalStart; i < nPalEnd; i++) { palette[i].r = (*palData++) << 2; palette[i].g = (*palData++) << 2; palette[i].b = (*palData++) << 2; palette[i].r |= palette[i].r >> 6; palette[i].g |= palette[i].g >> 6; palette[i].b |= palette[i].b >> 6; } } if (pkt.nDecodeMapSize > 0) { if (!decodeMap.pData) { decodeMap.pData = new uint8_t[pkt.nDecodeMapSize]; decodeMap.nSize = pkt.nDecodeMapSize; } else { if (pkt.nDecodeMapSize != decodeMap.nSize) { delete[] decodeMap.pData; decodeMap.pData = new uint8_t[pkt.nDecodeMapSize]; decodeMap.nSize = pkt.nDecodeMapSize; } } memcpy(decodeMap.pData, mapData, pkt.nDecodeMapSize); } if (pkt.nVideoDataSize > 0 && decodeMap.nSize > 0) { auto pStart = vidData; // need to skip 14 bytes ChunkPtr = pStart + 14; int i = 0; for (uint32_t y = 0; y < nHeight; y += 8) { for (uint32_t x = 0; x < nWidth; x += 8) { uint32_t opcode; // alternate between getting low and high 4 bits if (i & 1) { opcode = decodeMap.pData[i >> 1] >> 4; } else { opcode = decodeMap.pData[i >> 1] & 0x0F; } i++; int32_t offset = x + (y * videoStride); switch (opcode) { default: break; case 0: DecodeBlock0(offset); break; case 1: DecodeBlock1(offset); break; case 2: DecodeBlock2(offset); break; case 3: DecodeBlock3(offset); break; case 4: DecodeBlock4(offset); break; case 5: DecodeBlock5(offset); break; case 7: DecodeBlock7(offset); break; case 8: DecodeBlock8(offset); break; case 9: DecodeBlock9(offset); break; case 10: DecodeBlock10(offset); break; case 11: DecodeBlock11(offset); break; case 12: DecodeBlock12(offset); break; case 13: DecodeBlock13(offset); break; case 14: DecodeBlock14(offset); break; case 15: DecodeBlock15(offset); break; } } } auto pEnd = ChunkPtr; // we can end up with 1 byte left we need to skip int nSkipBytes = pkt.nVideoDataSize - (int)(pEnd - pStart); assert(nSkipBytes <= 1); } doFrame = pkt.bSendFlag; } while(!doFrame); animtex.SetFrame(&palette[0].r , GetCurrentFrame()); nFrame++; SwapFrames(); return true; } void InterplayDecoder::CopyBlock(uint8_t* pDest, uint8_t* pSrc) { for (int y = 0; y < 8; y++) { memcpy(pDest, pSrc, 8); pSrc += (intptr_t)videoStride; pDest += (intptr_t)videoStride; } } void InterplayDecoder::DecodeBlock0(int32_t offset) { // copy from the same offset but from the previous frame uint8_t* pDest = GetCurrentFrame() + (intptr_t)offset; uint8_t* pSrc = GetPreviousFrame() + (intptr_t)offset; CopyBlock(pDest, pSrc); } void InterplayDecoder::DecodeBlock1(int32_t offset) { // nothing to do for this. } void InterplayDecoder::DecodeBlock2(int32_t offset) { // copy block from 2 frames ago using a motion vector; need 1 more byte uint8_t B = *ChunkPtr++; int x, y; if (B < 56) { x = 8 + (B % 7); y = B / 7; } else { x = -14 + ((B - 56) % 29); y = 8 + ((B - 56) / 29); } uint8_t* pDest = GetCurrentFrame() + (intptr_t)offset; uint8_t* pSrc = GetCurrentFrame() + (intptr_t)(int64_t)offset + (int64_t)x + (int64_t(y) * (int64_t)videoStride); CopyBlock(pDest, pSrc); } void InterplayDecoder::DecodeBlock3(int32_t offset) { // copy 8x8 block from current frame from an up/left block uint8_t B = *ChunkPtr++; int x, y; // need 1 more byte for motion if (B < 56) { x = -(8 + (B % 7)); y = -(B / 7); } else { x = -(-14 + ((B - 56) % 29)); y = -(8 + ((B - 56) / 29)); } uint8_t* pDest = GetCurrentFrame() + (intptr_t)offset; uint8_t* pSrc = GetCurrentFrame() + (intptr_t)(int64_t)offset + (int64_t)x + (int64_t(y) * (int64_t)videoStride); CopyBlock(pDest, pSrc); } void InterplayDecoder::DecodeBlock4(int32_t offset) { // copy a block from the previous frame; need 1 more byte int x, y; uint8_t B, BL, BH; B = *ChunkPtr++; BL = B & 0x0F; BH = (B >> 4) & 0x0F; x = -8 + BL; y = -8 + BH; uint8_t* pDest = GetCurrentFrame() + (intptr_t)offset; uint8_t* pSrc = GetPreviousFrame() + (intptr_t)(int64_t)offset + (int64_t)x + (int64_t(y) * (int64_t)videoStride); CopyBlock(pDest, pSrc); } void InterplayDecoder::DecodeBlock5(int32_t offset) { // copy a block from the previous frame using an expanded range; need 2 more bytes int8_t x = *ChunkPtr++; int8_t y = *ChunkPtr++; uint8_t* pDest = GetCurrentFrame() + (intptr_t)offset; uint8_t* pSrc = GetPreviousFrame() + (intptr_t)(int64_t)offset + (int64_t)x + (int64_t(y) * (int64_t)videoStride); CopyBlock(pDest, pSrc); } // Block6 is unknown and skipped void InterplayDecoder::DecodeBlock7(int32_t offset) { uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint32_t flags = 0; uint8_t P[2]; P[0] = *ChunkPtr++; P[1] = *ChunkPtr++; // 2-color encoding if (P[0] <= P[1]) { // need 8 more bytes from the stream for (int y = 0; y < 8; y++) { flags = (*ChunkPtr++) | 0x100; for (; flags != 1; flags >>= 1) { *pBuffer++ = P[flags & 1]; } pBuffer += (videoStride - 8); } } else { // need 2 more bytes from the stream flags = LE_16(ChunkPtr); ChunkPtr += 2; for (int y = 0; y < 8; y += 2) { for (int x = 0; x < 8; x += 2, flags >>= 1) { pBuffer[x] = pBuffer[x + 1] = pBuffer[x + videoStride] = pBuffer[x + 1 + videoStride] = P[flags & 1]; } pBuffer += videoStride * 2; } } } void InterplayDecoder::DecodeBlock8(int32_t offset) { uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint32_t flags = 0; uint8_t P[4]; // 2-color encoding for each 4x4 quadrant, or 2-color encoding on either top and bottom or left and right halves P[0] = *ChunkPtr++; P[1] = *ChunkPtr++; if (P[0] <= P[1]) { for (int y = 0; y < 16; y++) { // new values for each 4x4 block if (!(y & 3)) { if (y) { P[0] = *ChunkPtr++; P[1] = *ChunkPtr++; } flags = LE_16(ChunkPtr); ChunkPtr += 2; } for (int x = 0; x < 4; x++, flags >>= 1) { *pBuffer++ = P[flags & 1]; } pBuffer += videoStride - 4; // switch to right half if (y == 7) pBuffer -= 8 * videoStride - 4; } } else { flags = LE_32(ChunkPtr); ChunkPtr += 4; P[2] = *ChunkPtr++; P[3] = *ChunkPtr++; if (P[2] <= P[3]) { // vertical split; left & right halves are 2-color encoded for (int y = 0; y < 16; y++) { for (int x = 0; x < 4; x++, flags >>= 1) { *pBuffer++ = P[flags & 1]; } pBuffer += videoStride - 4; // switch to right half if (y == 7) { pBuffer -= 8 * videoStride - 4; P[0] = P[2]; P[1] = P[3]; flags = LE_32(ChunkPtr); ChunkPtr += 4; } } } else { // horizontal split; top & bottom halves are 2-color encoded for (int y = 0; y < 8; y++) { if (y == 4) { P[0] = P[2]; P[1] = P[3]; flags = LE_32(ChunkPtr); ChunkPtr += 4; } for (int x = 0; x < 8; x++, flags >>= 1) *pBuffer++ = P[flags & 1]; pBuffer += (videoStride - 8); } } } } void InterplayDecoder::DecodeBlock9(int32_t offset) { uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint8_t P[4]; memcpy(P, ChunkPtr, 4); ChunkPtr += 4; // 4-color encoding if (P[0] <= P[1]) { if (P[2] <= P[3]) { // 1 of 4 colors for each pixel, need 16 more bytes for (int y = 0; y < 8; y++) { // get the next set of 8 2-bit flags int flags = LE_16(ChunkPtr); ChunkPtr += 2; for (int x = 0; x < 8; x++, flags >>= 2) { *pBuffer++ = P[flags & 0x03]; } pBuffer += (videoStride - 8); } } else { // 1 of 4 colors for each 2x2 block, need 4 more bytes uint32_t flags = LE_32(ChunkPtr); ChunkPtr += 4; for (int y = 0; y < 8; y += 2) { for (int x = 0; x < 8; x += 2, flags >>= 2) { pBuffer[x] = pBuffer[x + 1] = pBuffer[x + videoStride] = pBuffer[x + 1 + videoStride] = P[flags & 0x03]; } pBuffer += videoStride * 2; } } } else { // 1 of 4 colors for each 2x1 or 1x2 block, need 8 more bytes uint64_t flags = LE_64(ChunkPtr); ChunkPtr += 8; if (P[2] <= P[3]) { for (int y = 0; y < 8; y++) { for (int x = 0; x < 8; x += 2, flags >>= 2) { pBuffer[x] = pBuffer[x + 1] = P[flags & 0x03]; } pBuffer += videoStride; } } else { for (int y = 0; y < 8; y += 2) { for (int x = 0; x < 8; x++, flags >>= 2) { pBuffer[x] = pBuffer[x + videoStride] = P[flags & 0x03]; } pBuffer += videoStride * 2; } } } } void InterplayDecoder::DecodeBlock10(int32_t offset) { uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint8_t P[8]; memcpy(P, ChunkPtr, 4); ChunkPtr += 4; // 4-color encoding for each 4x4 quadrant, or 4-color encoding on either top and bottom or left and right halves if (P[0] <= P[1]) { int flags = 0; // 4-color encoding for each quadrant; need 32 bytes for (int y = 0; y < 16; y++) { // new values for each 4x4 block if (!(y & 3)) { if (y) { memcpy(P, ChunkPtr, 4); ChunkPtr += 4; } flags = LE_32(ChunkPtr); ChunkPtr += 4; } for (int x = 0; x < 4; x++, flags >>= 2) { *pBuffer++ = P[flags & 0x03]; } pBuffer += videoStride - 4; // switch to right half if (y == 7) pBuffer -= 8 * videoStride - 4; } } else { // vertical split? int vert; uint64_t flags = LE_64(ChunkPtr); ChunkPtr += 8; memcpy(P + 4, ChunkPtr, 4); ChunkPtr += 4; vert = P[4] <= P[5]; // 4-color encoding for either left and right or top and bottom halves for (int y = 0; y < 16; y++) { for (int x = 0; x < 4; x++, flags >>= 2) *pBuffer++ = P[flags & 0x03]; if (vert) { pBuffer += videoStride - 4; // switch to right half if (y == 7) pBuffer -= 8 * videoStride - 4; } else if (y & 1) pBuffer += (videoStride - 8); // load values for second half if (y == 7) { memcpy(P, P + 4, 4); flags = LE_64(ChunkPtr); ChunkPtr += 8; } } } } void InterplayDecoder::DecodeBlock11(int32_t offset) { // 64-color encoding (each pixel in block is a different color) uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; for (int y = 0; y < 8; y++) { memcpy(pBuffer, ChunkPtr, 8); ChunkPtr += 8; pBuffer += videoStride; } } void InterplayDecoder::DecodeBlock12(int32_t offset) { // 16-color block encoding: each 2x2 block is a different color uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; for (int y = 0; y < 8; y += 2) { for (int x = 0; x < 8; x += 2) { pBuffer[x] = pBuffer[x + 1] = pBuffer[x + videoStride] = pBuffer[x + 1 + videoStride] = *ChunkPtr++; } pBuffer += videoStride * 2; } } void InterplayDecoder::DecodeBlock13(int32_t offset) { // 4-color block encoding: each 4x4 block is a different color uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint8_t P[2] = {}; for (int y = 0; y < 8; y++) { if (!(y & 3)) { P[0] = *ChunkPtr++; P[1] = *ChunkPtr++; } memset(pBuffer, P[0], 4); memset(pBuffer + 4, P[1], 4); pBuffer += videoStride; } } void InterplayDecoder::DecodeBlock14(int32_t offset) { // 1-color encoding : the whole block is 1 solid color uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint8_t pix = *ChunkPtr++; for (int y = 0; y < 8; y++) { memset(pBuffer, pix, 8); pBuffer += videoStride; } } void InterplayDecoder::DecodeBlock15(int32_t offset) { // dithered encoding uint8_t* pBuffer = GetCurrentFrame() + (intptr_t)offset; uint8_t P[2]; P[0] = *ChunkPtr++; P[1] = *ChunkPtr++; for (int y = 0; y < 8; y++) { for (int x = 0; x < 8; x += 2) { *pBuffer++ = P[y & 1]; *pBuffer++ = P[!(y & 1)]; } pBuffer += (videoStride - 8); } } uint8_t* InterplayDecoder::GetCurrentFrame() { return pVideoBuffers[nCurrentVideoBuffer]; } uint8_t* InterplayDecoder::GetPreviousFrame() { return pVideoBuffers[nPreviousVideoBuffer]; }