/* * 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 */ #pragma once #include "files.h" #include "animtexture.h" #include "s_music.h" #ifndef playmve_h_ #define playmve_h_ class InterplayDecoder { public: enum { CHUNK_PREAMBLE_SIZE = 4, OPCODE_PREAMBLE_SIZE = 4, CHUNK_INIT_AUDIO = 0x0000, CHUNK_AUDIO_ONLY = 0x0001, CHUNK_INIT_VIDEO = 0x0002, CHUNK_VIDEO = 0x0003, CHUNK_SHUTDOWN = 0x0004, CHUNK_END = 0x0005, /* these last types are used internally */ CHUNK_DONE = 0xFFFC, CHUNK_NOMEM = 0xFFFD, CHUNK_EOF = 0xFFFE, CHUNK_BAD = 0xFFFF, OPCODE_END_OF_STREAM = 0x00, OPCODE_END_OF_CHUNK = 0x01, OPCODE_CREATE_TIMER = 0x02, OPCODE_INIT_AUDIO_BUFFERS = 0x03, OPCODE_START_STOP_AUDIO = 0x04, OPCODE_INIT_VIDEO_BUFFERS = 0x05, OPCODE_UNKNOWN_06 = 0x06, OPCODE_SEND_BUFFER = 0x07, OPCODE_AUDIO_FRAME = 0x08, OPCODE_SILENCE_FRAME = 0x09, OPCODE_INIT_VIDEO_MODE = 0x0A, OPCODE_CREATE_GRADIENT = 0x0B, OPCODE_SET_PALETTE = 0x0C, OPCODE_SET_PALETTE_COMPRESSED = 0x0D, OPCODE_UNKNOWN_0E = 0x0E, OPCODE_SET_DECODING_MAP = 0x0F, OPCODE_UNKNOWN_10 = 0x10, OPCODE_VIDEO_DATA = 0x11, OPCODE_UNKNOWN_12 = 0x12, OPCODE_UNKNOWN_13 = 0x13, OPCODE_UNKNOWN_14 = 0x14, OPCODE_UNKNOWN_15 = 0x15, PALETTE_COUNT = 256, kAudioBlocks = 20 // alloc a lot of blocks - need to store lots of audio data before video frames start. }; InterplayDecoder(); ~InterplayDecoder(); bool Open(FileReader &fr); void Close(); bool RunFrame(uint64_t clock); struct AudioData { int hFx; int nChannels; uint16_t nSampleRate; uint8_t nBitDepth; int16_t samples[6000 * kAudioBlocks]; // must be a multiple of the stream buffer size int nWrite; int nRead; }; AudioData audio; AnimTextures animtex; AnimTextures& animTex() { return animtex; } private: struct DecodeMap { uint8_t* pData; uint32_t nSize; }; struct Palette { uint8_t r; uint8_t g; uint8_t b; }; uint8_t* GetCurrentFrame(); uint8_t* GetPreviousFrame(); void SwapFrames(); void CopyBlock(uint8_t* pDest, uint8_t* pSrc); void DecodeBlock0(int32_t offset); void DecodeBlock1(int32_t offset); void DecodeBlock2(int32_t offset); void DecodeBlock3(int32_t offset); void DecodeBlock4(int32_t offset); void DecodeBlock5(int32_t offset); void DecodeBlock7(int32_t offset); void DecodeBlock8(int32_t offset); void DecodeBlock9(int32_t offset); void DecodeBlock10(int32_t offset); void DecodeBlock11(int32_t offset); void DecodeBlock12(int32_t offset); void DecodeBlock13(int32_t offset); void DecodeBlock14(int32_t offset); void DecodeBlock15(int32_t offset); FileReader fr; bool bIsPlaying, bAudioStarted; uint32_t nTimerRate, nTimerDiv; uint32_t nWidth, nHeight, nFrame; double nFps; uint64_t nFrameDuration; uint8_t* pVideoBuffers[2]; uint32_t nCurrentVideoBuffer, nPreviousVideoBuffer; int32_t videoStride; DecodeMap decodeMap; Palette palette[256]; uint64_t nNextFrameTime = 0; SoundStream* stream = nullptr; }; #endif