mirror of
https://github.com/id-Software/DOOM-3-BFG.git
synced 2024-12-14 22:41:49 +00:00
2518 lines
58 KiB
C++
2518 lines
58 KiB
C++
/*
|
|
===========================================================================
|
|
|
|
Doom 3 BFG Edition GPL Source Code
|
|
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
|
|
Copyright (C) 2013-2014 Robert Beckebans
|
|
Copyright (C) 2014 Carl Kenner
|
|
|
|
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
|
|
|
|
Doom 3 BFG Edition Source Code 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.
|
|
|
|
Doom 3 BFG Edition Source Code 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 Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
===========================================================================
|
|
*/
|
|
|
|
#pragma hdrstop
|
|
#include "precompiled.h"
|
|
|
|
|
|
extern idCVar s_noSound;
|
|
|
|
#define JPEG_INTERNALS
|
|
//extern "C" {
|
|
#include "../libs/jpeg-6/jpeglib.h"
|
|
//}
|
|
|
|
#include "tr_local.h"
|
|
|
|
#define CIN_system 1
|
|
#define CIN_loop 2
|
|
#define CIN_hold 4
|
|
#define CIN_silent 8
|
|
#define CIN_shader 16
|
|
|
|
#if defined(USE_FFMPEG)
|
|
// Carl: ffmpg for bink video files
|
|
extern "C"
|
|
{
|
|
|
|
//#ifdef WIN32
|
|
#ifndef INT64_C
|
|
#define INT64_C(c) (c ## LL)
|
|
#define UINT64_C(c) (c ## ULL)
|
|
#endif
|
|
#include <inttypes.h>
|
|
//#endif
|
|
|
|
#include <libavcodec/avcodec.h>
|
|
#include <libavformat/avformat.h>
|
|
#include <libswscale/swscale.h>
|
|
}
|
|
#endif
|
|
|
|
class idCinematicLocal : public idCinematic
|
|
{
|
|
public:
|
|
idCinematicLocal();
|
|
virtual ~idCinematicLocal();
|
|
|
|
virtual bool InitFromFile( const char* qpath, bool looping );
|
|
virtual cinData_t ImageForTime( int milliseconds );
|
|
virtual int AnimationLength();
|
|
// RB begin
|
|
bool IsPlaying() const;
|
|
// RB end
|
|
virtual void Close();
|
|
virtual void ResetTime( int time );
|
|
|
|
private:
|
|
|
|
#if defined(USE_FFMPEG)
|
|
int video_stream_index;
|
|
AVFormatContext* fmt_ctx;
|
|
AVFrame* frame;
|
|
AVFrame* frame2;
|
|
AVCodec* dec;
|
|
AVCodecContext* dec_ctx;
|
|
SwsContext* img_convert_ctx;
|
|
bool hasFrame;
|
|
long framePos;
|
|
|
|
cinData_t ImageForTimeFFMPEG( int milliseconds );
|
|
bool InitFromFFMPEGFile( const char* qpath, bool looping );
|
|
void FFMPEGReset();
|
|
#endif
|
|
idImage* img;
|
|
bool isRoQ;
|
|
|
|
// RB: 64 bit fixes, changed long to int
|
|
size_t mcomp[256];
|
|
// RB end
|
|
byte** qStatus[2];
|
|
idStr fileName;
|
|
int CIN_WIDTH, CIN_HEIGHT;
|
|
idFile* iFile;
|
|
cinStatus_t status;
|
|
// RB: 64 bit fixes, changed long to int
|
|
int tfps;
|
|
int RoQPlayed;
|
|
int ROQSize;
|
|
unsigned int RoQFrameSize;
|
|
int onQuad;
|
|
int numQuads;
|
|
int samplesPerLine;
|
|
unsigned int roq_id;
|
|
int screenDelta;
|
|
byte* buf;
|
|
int samplesPerPixel; // defaults to 2
|
|
unsigned int xsize, ysize, maxsize, minsize;
|
|
int normalBuffer0;
|
|
int roq_flags;
|
|
int roqF0;
|
|
int roqF1;
|
|
int t[2];
|
|
int roqFPS;
|
|
int drawX, drawY;
|
|
// RB end
|
|
|
|
int animationLength;
|
|
int startTime;
|
|
float frameRate;
|
|
|
|
byte* image;
|
|
|
|
bool looping;
|
|
bool dirty;
|
|
bool half;
|
|
bool smootheddouble;
|
|
bool inMemory;
|
|
|
|
void RoQ_init();
|
|
void blitVQQuad32fs( byte** status, unsigned char* data );
|
|
void RoQShutdown();
|
|
void RoQInterrupt();
|
|
|
|
void move8_32( byte* src, byte* dst, int spl );
|
|
void move4_32( byte* src, byte* dst, int spl );
|
|
void blit8_32( byte* src, byte* dst, int spl );
|
|
void blit4_32( byte* src, byte* dst, int spl );
|
|
void blit2_32( byte* src, byte* dst, int spl );
|
|
|
|
// RB: 64 bit fixes, changed long to int
|
|
unsigned short yuv_to_rgb( int y, int u, int v );
|
|
unsigned int yuv_to_rgb24( int y, int u, int v );
|
|
|
|
void decodeCodeBook( byte* input, unsigned short roq_flags );
|
|
void recurseQuad( int startX, int startY, int quadSize, int xOff, int yOff );
|
|
void setupQuad( int xOff, int yOff );
|
|
void readQuadInfo( byte* qData );
|
|
void RoQPrepMcomp( int xoff, int yoff );
|
|
void RoQReset();
|
|
// RB end
|
|
};
|
|
|
|
// Carl: ROQ files from original Doom 3
|
|
const int DEFAULT_CIN_WIDTH = 512;
|
|
const int DEFAULT_CIN_HEIGHT = 512;
|
|
const int MAXSIZE = 8;
|
|
const int MINSIZE = 4;
|
|
|
|
const int ROQ_FILE = 0x1084;
|
|
const int ROQ_QUAD = 0x1000;
|
|
const int ROQ_QUAD_INFO = 0x1001;
|
|
const int ROQ_CODEBOOK = 0x1002;
|
|
const int ROQ_QUAD_VQ = 0x1011;
|
|
const int ROQ_QUAD_JPEG = 0x1012;
|
|
const int ROQ_QUAD_HANG = 0x1013;
|
|
const int ROQ_PACKET = 0x1030;
|
|
const int ZA_SOUND_MONO = 0x1020;
|
|
const int ZA_SOUND_STEREO = 0x1021;
|
|
|
|
// temporary buffers used by all cinematics
|
|
// RB: 64 bit fixes, changed long to int
|
|
static int ROQ_YY_tab[256];
|
|
static int ROQ_UB_tab[256];
|
|
static int ROQ_UG_tab[256];
|
|
static int ROQ_VG_tab[256];
|
|
static int ROQ_VR_tab[256];
|
|
// RB end
|
|
static byte* file = NULL;
|
|
static unsigned short* vq2 = NULL;
|
|
static unsigned short* vq4 = NULL;
|
|
static unsigned short* vq8 = NULL;
|
|
|
|
|
|
|
|
//===========================================
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::InitCinematic
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
void idCinematic::InitCinematic()
|
|
{
|
|
#if defined(USE_FFMPEG)
|
|
// Carl: ffmpeg for Bink and regular video files
|
|
//common->Warning( "Loading FFMPEG...\n" );
|
|
avcodec_register_all();
|
|
av_register_all();
|
|
#endif
|
|
|
|
// Carl: Doom 3 ROQ:
|
|
float t_ub, t_vr, t_ug, t_vg;
|
|
int i;
|
|
|
|
// generate YUV tables
|
|
t_ub = ( 1.77200f / 2.0f ) * ( float )( 1 << 6 ) + 0.5f;
|
|
t_vr = ( 1.40200f / 2.0f ) * ( float )( 1 << 6 ) + 0.5f;
|
|
t_ug = ( 0.34414f / 2.0f ) * ( float )( 1 << 6 ) + 0.5f;
|
|
t_vg = ( 0.71414f / 2.0f ) * ( float )( 1 << 6 ) + 0.5f;
|
|
for( i = 0; i < 256; i++ )
|
|
{
|
|
float x = ( float )( 2 * i - 255 );
|
|
|
|
ROQ_UB_tab[i] = ( int )( ( t_ub * x ) + ( 1 << 5 ) );
|
|
ROQ_VR_tab[i] = ( int )( ( t_vr * x ) + ( 1 << 5 ) );
|
|
ROQ_UG_tab[i] = ( int )( ( -t_ug * x ) );
|
|
ROQ_VG_tab[i] = ( int )( ( -t_vg * x ) + ( 1 << 5 ) );
|
|
ROQ_YY_tab[i] = ( int )( ( i << 6 ) | ( i >> 2 ) );
|
|
}
|
|
|
|
file = ( byte* )Mem_Alloc( 65536, TAG_CINEMATIC );
|
|
vq2 = ( word* )Mem_Alloc( 256 * 16 * 4 * sizeof( word ), TAG_CINEMATIC );
|
|
vq4 = ( word* )Mem_Alloc( 256 * 64 * 4 * sizeof( word ), TAG_CINEMATIC );
|
|
vq8 = ( word* )Mem_Alloc( 256 * 256 * 4 * sizeof( word ), TAG_CINEMATIC );
|
|
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::ShutdownCinematic
|
|
==============
|
|
*/
|
|
void idCinematic::ShutdownCinematic()
|
|
{
|
|
// Carl: Original Doom 3 RoQ files:
|
|
Mem_Free( file );
|
|
file = NULL;
|
|
Mem_Free( vq2 );
|
|
vq2 = NULL;
|
|
Mem_Free( vq4 );
|
|
vq4 = NULL;
|
|
Mem_Free( vq8 );
|
|
vq8 = NULL;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::Alloc
|
|
==============
|
|
*/
|
|
idCinematic* idCinematic::Alloc()
|
|
{
|
|
return new idCinematicLocal; //Carl: Use the proper class like in Doom 3, not just the unimplemented abstract one.
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::~idCinematic
|
|
==============
|
|
*/
|
|
idCinematic::~idCinematic( )
|
|
{
|
|
Close();
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::InitFromFile
|
|
==============
|
|
*/
|
|
bool idCinematic::InitFromFile( const char* qpath, bool looping )
|
|
{
|
|
return false; //Carl: this is just the abstract virtual method
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::AnimationLength
|
|
==============
|
|
*/
|
|
int idCinematic::AnimationLength()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::GetStartTime
|
|
==============
|
|
*/
|
|
int idCinematic::GetStartTime()
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::ResetTime
|
|
==============
|
|
*/
|
|
void idCinematic::ResetTime( int milliseconds )
|
|
{
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::ImageForTime
|
|
==============
|
|
*/
|
|
cinData_t idCinematic::ImageForTime( int milliseconds )
|
|
{
|
|
cinData_t c;
|
|
memset( &c, 0, sizeof( c ) );
|
|
declManager->FindMaterial( "doom64.tga" );
|
|
c.image = globalImages->GetImage( "doom64.tga" );
|
|
return c;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::ExportToTGA
|
|
==============
|
|
*/
|
|
void idCinematic::ExportToTGA( bool skipExisting )
|
|
{
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::GetFrameRate
|
|
==============
|
|
*/
|
|
float idCinematic::GetFrameRate() const
|
|
{
|
|
return 30.0f;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematic::Close
|
|
==============
|
|
*/
|
|
void idCinematic::Close()
|
|
{
|
|
}
|
|
|
|
// RB begin
|
|
bool idCinematic::IsPlaying() const
|
|
{
|
|
return false;
|
|
}
|
|
// RB end
|
|
|
|
|
|
//===========================================
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::idCinematicLocal
|
|
==============
|
|
*/
|
|
idCinematicLocal::idCinematicLocal()
|
|
{
|
|
qStatus[0] = ( byte** )Mem_Alloc( 32768 * sizeof( byte* ), TAG_CINEMATIC );
|
|
qStatus[1] = ( byte** )Mem_Alloc( 32768 * sizeof( byte* ), TAG_CINEMATIC );
|
|
|
|
#if defined(USE_FFMPEG)
|
|
// Carl: ffmpeg stuff, for bink and normal video files:
|
|
isRoQ = false;
|
|
// fmt_ctx = avformat_alloc_context();
|
|
frame = avcodec_alloc_frame();
|
|
frame2 = avcodec_alloc_frame();
|
|
dec_ctx = NULL;
|
|
fmt_ctx = NULL;
|
|
video_stream_index = -1;
|
|
img_convert_ctx = NULL;
|
|
hasFrame = false;
|
|
#endif
|
|
|
|
// Carl: Original Doom 3 RoQ files:
|
|
image = NULL;
|
|
status = FMV_EOF;
|
|
buf = NULL;
|
|
iFile = NULL;
|
|
img = globalImages->AllocStandaloneImage( "_cinematic" );
|
|
if( img != NULL )
|
|
{
|
|
idImageOpts opts;
|
|
opts.format = FMT_RGBA8;
|
|
opts.colorFormat = CFM_DEFAULT;
|
|
opts.width = 32;
|
|
opts.height = 32;
|
|
opts.numLevels = 1;
|
|
img->AllocImage( opts, TF_LINEAR, TR_REPEAT );
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::~idCinematicLocal
|
|
==============
|
|
*/
|
|
idCinematicLocal::~idCinematicLocal()
|
|
{
|
|
Close();
|
|
|
|
// Carl: Original Doom 3 RoQ files:
|
|
Mem_Free( qStatus[0] );
|
|
qStatus[0] = NULL;
|
|
Mem_Free( qStatus[1] );
|
|
qStatus[1] = NULL;
|
|
|
|
#if defined(USE_FFMPEG)
|
|
// Carl: ffmpeg for bink and other video files:
|
|
|
|
// RB: TODO double check this. It seems we have different versions of ffmpeg on Kubuntu 13.10 and the win32 development files
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
avcodec_free_frame( &frame );
|
|
avcodec_free_frame( &frame2 );
|
|
#else
|
|
av_freep( &frame );
|
|
av_freep( &frame2 );
|
|
#endif
|
|
|
|
if( fmt_ctx )
|
|
{
|
|
avformat_free_context( fmt_ctx );
|
|
}
|
|
|
|
if( img_convert_ctx )
|
|
{
|
|
sws_freeContext( img_convert_ctx );
|
|
}
|
|
#endif
|
|
|
|
delete img;
|
|
img = NULL;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::InitFromFFMPEGFile
|
|
==============
|
|
*/
|
|
#if defined(USE_FFMPEG)
|
|
bool idCinematicLocal::InitFromFFMPEGFile( const char* qpath, bool amilooping )
|
|
{
|
|
int ret;
|
|
looping = amilooping;
|
|
startTime = 0;
|
|
isRoQ = false;
|
|
CIN_HEIGHT = DEFAULT_CIN_HEIGHT;
|
|
CIN_WIDTH = DEFAULT_CIN_WIDTH;
|
|
|
|
idStr fullpath;
|
|
idFile* testFile = fileSystem->OpenFileRead( qpath );
|
|
if( testFile )
|
|
{
|
|
fullpath = testFile->GetFullPath();
|
|
fileSystem->CloseFile( testFile );
|
|
}
|
|
// RB: case sensitivity HACK for Linux
|
|
else if( idStr::Cmpn( qpath, "sound/vo", 8 ) == 0 )
|
|
{
|
|
idStr newPath( qpath );
|
|
newPath.Replace( "sound/vo", "sound/VO" );
|
|
|
|
testFile = fileSystem->OpenFileRead( qpath );
|
|
if( testFile )
|
|
{
|
|
fullpath = testFile->GetFullPath();
|
|
fileSystem->CloseFile( testFile );
|
|
}
|
|
else
|
|
{
|
|
common->Warning( "idCinematic: Cannot open FFMPEG video file: '%s', %d\n", qpath, looping );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
//idStr fullpath = fileSystem->RelativePathToOSPath( qpath, "fs_basepath" );
|
|
|
|
if( ( ret = avformat_open_input( &fmt_ctx, fullpath, NULL, NULL ) ) < 0 )
|
|
{
|
|
common->Warning( "idCinematic: Cannot open FFMPEG video file: '%s', %d\n", qpath, looping );
|
|
return false;
|
|
}
|
|
if( ( ret = avformat_find_stream_info( fmt_ctx, NULL ) ) < 0 )
|
|
{
|
|
common->Warning( "idCinematic: Cannot find stream info: '%s', %d\n", qpath, looping );
|
|
return false;
|
|
}
|
|
/* select the video stream */
|
|
ret = av_find_best_stream( fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, &dec, 0 );
|
|
if( ret < 0 )
|
|
{
|
|
common->Warning( "idCinematic: Cannot find a video stream in: '%s', %d\n", qpath, looping );
|
|
return false;
|
|
}
|
|
video_stream_index = ret;
|
|
dec_ctx = fmt_ctx->streams[video_stream_index]->codec;
|
|
/* init the video decoder */
|
|
if( ( ret = avcodec_open2( dec_ctx, dec, NULL ) ) < 0 )
|
|
{
|
|
common->Warning( "idCinematic: Cannot open video decoder for: '%s', %d\n", qpath, looping );
|
|
return false;
|
|
}
|
|
|
|
CIN_WIDTH = dec_ctx->width;
|
|
CIN_HEIGHT = dec_ctx->height;
|
|
/** Calculate Duration in seconds
|
|
* This is the fundamental unit of time (in seconds) in terms
|
|
* of which frame timestamps are represented. For fixed-fps content,
|
|
* timebase should be 1/framerate and timestamp increments should be
|
|
* identically 1.
|
|
* - encoding: MUST be set by user.
|
|
* - decoding: Set by libavcodec.
|
|
*/
|
|
AVRational avr = dec_ctx->time_base;
|
|
/**
|
|
* For some codecs, the time base is closer to the field rate than the frame rate.
|
|
* Most notably, H.264 and MPEG-2 specify time_base as half of frame duration
|
|
* if no telecine is used ...
|
|
*
|
|
* Set to time_base ticks per frame. Default 1, e.g., H.264/MPEG-2 set it to 2.
|
|
*/
|
|
int ticksPerFrame = dec_ctx->ticks_per_frame;
|
|
float durationSec = static_cast<double>( fmt_ctx->streams[video_stream_index]->duration ) * static_cast<double>( ticksPerFrame ) / static_cast<double>( avr.den );
|
|
animationLength = durationSec * 1000;
|
|
frameRate = av_q2d( fmt_ctx->streams[video_stream_index]->r_frame_rate );
|
|
buf = NULL;
|
|
hasFrame = false;
|
|
framePos = -1;
|
|
common->Printf( "Loaded FFMPEG file: '%s', looping=%d%dx%d, %f FPS, %f sec\n", qpath, looping, CIN_WIDTH, CIN_HEIGHT, frameRate, durationSec );
|
|
image = ( byte* )Mem_Alloc( CIN_WIDTH * CIN_HEIGHT * 4 * 2, TAG_CINEMATIC );
|
|
avpicture_fill( ( AVPicture* )frame2, image, PIX_FMT_BGR32, CIN_WIDTH, CIN_HEIGHT );
|
|
if( img_convert_ctx )
|
|
{
|
|
sws_freeContext( img_convert_ctx );
|
|
}
|
|
img_convert_ctx = sws_getContext( dec_ctx->width, dec_ctx->height, dec_ctx->pix_fmt, CIN_WIDTH, CIN_HEIGHT, PIX_FMT_BGR32, SWS_BICUBIC, NULL, NULL, NULL );
|
|
status = FMV_PLAY;
|
|
|
|
startTime = 0;
|
|
ImageForTime( 0 );
|
|
status = ( looping ) ? FMV_PLAY : FMV_IDLE;
|
|
|
|
//startTime = Sys_Milliseconds();
|
|
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::FFMPEGReset
|
|
==============
|
|
*/
|
|
#if defined(USE_FFMPEG)
|
|
void idCinematicLocal::FFMPEGReset()
|
|
{
|
|
// RB: don't reset startTime here because that breaks video replays in the PDAs
|
|
//startTime = 0;
|
|
|
|
framePos = -1;
|
|
|
|
if( av_seek_frame( fmt_ctx, video_stream_index, 0, 0 ) >= 0 )
|
|
{
|
|
status = FMV_LOOPED;
|
|
}
|
|
else
|
|
{
|
|
status = FMV_EOF;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::InitFromFile
|
|
==============
|
|
*/
|
|
bool idCinematicLocal::InitFromFile( const char* qpath, bool amilooping )
|
|
{
|
|
unsigned short RoQID;
|
|
|
|
Close();
|
|
|
|
inMemory = 0;
|
|
animationLength = 100000;
|
|
|
|
// Carl: if no folder is specified, look in the video folder
|
|
if( strstr( qpath, "/" ) == NULL && strstr( qpath, "\\" ) == NULL )
|
|
{
|
|
sprintf( fileName, "video/%s", qpath );
|
|
}
|
|
else
|
|
{
|
|
sprintf( fileName, "%s", qpath );
|
|
}
|
|
// Carl: Look for original Doom 3 RoQ files first:
|
|
idStr ext;
|
|
fileName.ExtractFileExtension( ext );
|
|
fileName = fileName.StripFileExtension();
|
|
fileName = fileName + ".roq";
|
|
//if (fileName == "video\\loadvideo.roq") {
|
|
// fileName = "video\\idlogo.roq";
|
|
//}
|
|
|
|
iFile = fileSystem->OpenFileRead( fileName );
|
|
|
|
// Carl: If the RoQ file doesn't exist, try using ffmpeg instead:
|
|
if( !iFile )
|
|
{
|
|
#if defined(USE_FFMPEG)
|
|
//idLib::Warning( "Original Doom 3 RoQ Cinematic not found: '%s'\n", fileName.c_str() );
|
|
idStr temp = fileName.StripFileExtension() + ".bik";
|
|
animationLength = 0;
|
|
hasFrame = false;
|
|
RoQShutdown();
|
|
fileName = temp;
|
|
//idLib::Warning( "New filename: '%s'\n", fileName.c_str() );
|
|
return InitFromFFMPEGFile( fileName.c_str(), amilooping );
|
|
#else
|
|
animationLength = 0;
|
|
return false;
|
|
#endif
|
|
}
|
|
// Carl: The rest of this function is for original Doom 3 RoQ files:
|
|
isRoQ = true;
|
|
ROQSize = iFile->Length();
|
|
|
|
looping = amilooping;
|
|
|
|
CIN_HEIGHT = DEFAULT_CIN_HEIGHT;
|
|
CIN_WIDTH = DEFAULT_CIN_WIDTH;
|
|
samplesPerPixel = 4;
|
|
startTime = 0; //Sys_Milliseconds();
|
|
buf = NULL;
|
|
|
|
iFile->Read( file, 16 );
|
|
|
|
RoQID = ( unsigned short )( file[0] ) + ( unsigned short )( file[1] ) * 256;
|
|
|
|
frameRate = file[6];
|
|
if( frameRate == 32.0f )
|
|
{
|
|
frameRate = 1000.0f / 32.0f;
|
|
}
|
|
|
|
if( RoQID == ROQ_FILE )
|
|
{
|
|
RoQ_init();
|
|
status = FMV_PLAY;
|
|
ImageForTime( 0 );
|
|
status = ( looping ) ? FMV_PLAY : FMV_IDLE;
|
|
return true;
|
|
}
|
|
|
|
RoQShutdown();
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::Close
|
|
==============
|
|
*/
|
|
void idCinematicLocal::Close()
|
|
{
|
|
if( image )
|
|
{
|
|
Mem_Free( ( void* )image );
|
|
image = NULL;
|
|
buf = NULL;
|
|
status = FMV_EOF;
|
|
}
|
|
|
|
RoQShutdown();
|
|
|
|
#if defined(USE_FFMPEG)
|
|
hasFrame = false;
|
|
|
|
if( !isRoQ )
|
|
{
|
|
if( img_convert_ctx )
|
|
{
|
|
sws_freeContext( img_convert_ctx );
|
|
}
|
|
|
|
img_convert_ctx = NULL;
|
|
|
|
if( dec_ctx )
|
|
{
|
|
avcodec_close( dec_ctx );
|
|
}
|
|
|
|
if( fmt_ctx )
|
|
{
|
|
avformat_close_input( &fmt_ctx );
|
|
}
|
|
status = FMV_EOF;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::AnimationLength
|
|
==============
|
|
*/
|
|
int idCinematicLocal::AnimationLength()
|
|
{
|
|
return animationLength;
|
|
}
|
|
|
|
// RB begin
|
|
bool idCinematicLocal::IsPlaying() const
|
|
{
|
|
return ( status == FMV_PLAY );
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::ResetTime
|
|
==============
|
|
*/
|
|
void idCinematicLocal::ResetTime( int time )
|
|
{
|
|
startTime = time; //originally this was: ( backEnd.viewDef ) ? 1000 * backEnd.viewDef->floatTime : -1;
|
|
status = FMV_PLAY;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::ImageForTime
|
|
==============
|
|
*/
|
|
cinData_t idCinematicLocal::ImageForTime( int thisTime )
|
|
{
|
|
#if defined(USE_FFMPEG)
|
|
// Carl: Handle BFG format BINK videos separately
|
|
if( !isRoQ )
|
|
return ImageForTimeFFMPEG( thisTime );
|
|
#endif
|
|
|
|
// Carl: Handle original Doom 3 RoQ video files
|
|
cinData_t cinData;
|
|
|
|
if( thisTime == 0 )
|
|
{
|
|
thisTime = Sys_Milliseconds();
|
|
}
|
|
|
|
if( thisTime < 0 )
|
|
{
|
|
thisTime = 0;
|
|
}
|
|
|
|
memset( &cinData, 0, sizeof( cinData ) );
|
|
|
|
// if ( r_skipROQ.GetBool() ) {
|
|
if( r_skipDynamicTextures.GetBool() )
|
|
{
|
|
return cinData;
|
|
}
|
|
|
|
if( !iFile )
|
|
{
|
|
// RB: neither .bik or .roq found
|
|
return cinData;
|
|
}
|
|
|
|
if( status == FMV_EOF || status == FMV_IDLE )
|
|
{
|
|
return cinData;
|
|
}
|
|
|
|
if( buf == NULL || startTime == -1 )
|
|
{
|
|
if( startTime == -1 )
|
|
{
|
|
RoQReset();
|
|
}
|
|
startTime = thisTime;
|
|
}
|
|
|
|
tfps = ( ( thisTime - startTime ) * frameRate ) / 1000;
|
|
|
|
if( tfps < 0 )
|
|
{
|
|
tfps = 0;
|
|
}
|
|
|
|
if( tfps < numQuads )
|
|
{
|
|
RoQReset();
|
|
buf = NULL;
|
|
status = FMV_PLAY;
|
|
}
|
|
|
|
if( buf == NULL )
|
|
{
|
|
while( buf == NULL )
|
|
{
|
|
RoQInterrupt();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while( ( tfps != numQuads && status == FMV_PLAY ) )
|
|
{
|
|
RoQInterrupt();
|
|
}
|
|
}
|
|
|
|
if( status == FMV_LOOPED )
|
|
{
|
|
status = FMV_PLAY;
|
|
while( buf == NULL && status == FMV_PLAY )
|
|
{
|
|
RoQInterrupt();
|
|
}
|
|
startTime = thisTime;
|
|
}
|
|
|
|
if( status == FMV_EOF )
|
|
{
|
|
if( looping )
|
|
{
|
|
RoQReset();
|
|
buf = NULL;
|
|
if( status == FMV_LOOPED )
|
|
{
|
|
status = FMV_PLAY;
|
|
}
|
|
while( buf == NULL && status == FMV_PLAY )
|
|
{
|
|
RoQInterrupt();
|
|
}
|
|
startTime = thisTime;
|
|
}
|
|
else
|
|
{
|
|
status = FMV_IDLE;
|
|
RoQShutdown();
|
|
}
|
|
}
|
|
|
|
cinData.imageWidth = CIN_WIDTH;
|
|
cinData.imageHeight = CIN_HEIGHT;
|
|
cinData.status = status;
|
|
img->UploadScratch( image, CIN_WIDTH, CIN_HEIGHT );
|
|
cinData.image = img;
|
|
|
|
return cinData;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::ImageForTimeFFMPEG
|
|
==============
|
|
*/
|
|
#if defined(USE_FFMPEG)
|
|
cinData_t idCinematicLocal::ImageForTimeFFMPEG( int thisTime )
|
|
{
|
|
cinData_t cinData;
|
|
|
|
if( thisTime <= 0 )
|
|
{
|
|
thisTime = Sys_Milliseconds();
|
|
}
|
|
|
|
memset( &cinData, 0, sizeof( cinData ) );
|
|
if( r_skipDynamicTextures.GetBool() || status == FMV_EOF || status == FMV_IDLE )
|
|
{
|
|
return cinData;
|
|
}
|
|
|
|
if( !fmt_ctx )
|
|
{
|
|
// RB: .bik requested but not found
|
|
return cinData;
|
|
}
|
|
|
|
if( ( !hasFrame ) || startTime == -1 )
|
|
{
|
|
if( startTime == -1 )
|
|
{
|
|
FFMPEGReset();
|
|
}
|
|
startTime = thisTime;
|
|
}
|
|
|
|
long desiredFrame = ( ( thisTime - startTime ) * frameRate ) / 1000;
|
|
if( desiredFrame < 0 )
|
|
{
|
|
desiredFrame = 0;
|
|
}
|
|
|
|
if( desiredFrame < framePos )
|
|
{
|
|
FFMPEGReset();
|
|
}
|
|
|
|
if( hasFrame && desiredFrame == framePos )
|
|
{
|
|
cinData.imageWidth = CIN_WIDTH;
|
|
cinData.imageHeight = CIN_HEIGHT;
|
|
cinData.status = status;
|
|
cinData.image = img;
|
|
return cinData;
|
|
}
|
|
|
|
AVPacket packet;
|
|
while( framePos < desiredFrame )
|
|
{
|
|
int frameFinished = 0;
|
|
|
|
// Do a single frame by getting packets until we have a full frame
|
|
while( !frameFinished )
|
|
{
|
|
// if we got to the end or failed
|
|
if( av_read_frame( fmt_ctx, &packet ) < 0 )
|
|
{
|
|
// can't read any more, set to EOF
|
|
status = FMV_EOF;
|
|
if( looping )
|
|
{
|
|
desiredFrame = 0;
|
|
FFMPEGReset();
|
|
framePos = -1;
|
|
startTime = thisTime;
|
|
if( av_read_frame( fmt_ctx, &packet ) < 0 )
|
|
{
|
|
status = FMV_IDLE;
|
|
return cinData;
|
|
}
|
|
status = FMV_PLAY;
|
|
}
|
|
else
|
|
{
|
|
status = FMV_IDLE;
|
|
return cinData;
|
|
}
|
|
}
|
|
// Is this a packet from the video stream?
|
|
if( packet.stream_index == video_stream_index )
|
|
{
|
|
// Decode video frame
|
|
avcodec_decode_video2( dec_ctx, frame, &frameFinished, &packet );
|
|
}
|
|
// Free the packet that was allocated by av_read_frame
|
|
av_free_packet( &packet );
|
|
}
|
|
|
|
framePos++;
|
|
}
|
|
|
|
// We have reached the desired frame
|
|
// Convert the image from its native format to RGB
|
|
sws_scale( img_convert_ctx, frame->data, frame->linesize, 0, dec_ctx->height, frame2->data, frame2->linesize );
|
|
cinData.imageWidth = CIN_WIDTH;
|
|
cinData.imageHeight = CIN_HEIGHT;
|
|
cinData.status = status;
|
|
img->UploadScratch( image, CIN_WIDTH, CIN_HEIGHT );
|
|
hasFrame = true;
|
|
cinData.image = img;
|
|
|
|
return cinData;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::move8_32
|
|
==============
|
|
*/
|
|
void idCinematicLocal::move8_32( byte* src, byte* dst, int spl )
|
|
{
|
|
#if 1
|
|
int* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( int* )src;
|
|
ddst = ( int* )dst;
|
|
dspl = spl >> 2;
|
|
|
|
ddst[0 * dspl + 0] = dsrc[0 * dspl + 0];
|
|
ddst[0 * dspl + 1] = dsrc[0 * dspl + 1];
|
|
ddst[0 * dspl + 2] = dsrc[0 * dspl + 2];
|
|
ddst[0 * dspl + 3] = dsrc[0 * dspl + 3];
|
|
ddst[0 * dspl + 4] = dsrc[0 * dspl + 4];
|
|
ddst[0 * dspl + 5] = dsrc[0 * dspl + 5];
|
|
ddst[0 * dspl + 6] = dsrc[0 * dspl + 6];
|
|
ddst[0 * dspl + 7] = dsrc[0 * dspl + 7];
|
|
|
|
ddst[1 * dspl + 0] = dsrc[1 * dspl + 0];
|
|
ddst[1 * dspl + 1] = dsrc[1 * dspl + 1];
|
|
ddst[1 * dspl + 2] = dsrc[1 * dspl + 2];
|
|
ddst[1 * dspl + 3] = dsrc[1 * dspl + 3];
|
|
ddst[1 * dspl + 4] = dsrc[1 * dspl + 4];
|
|
ddst[1 * dspl + 5] = dsrc[1 * dspl + 5];
|
|
ddst[1 * dspl + 6] = dsrc[1 * dspl + 6];
|
|
ddst[1 * dspl + 7] = dsrc[1 * dspl + 7];
|
|
|
|
ddst[2 * dspl + 0] = dsrc[2 * dspl + 0];
|
|
ddst[2 * dspl + 1] = dsrc[2 * dspl + 1];
|
|
ddst[2 * dspl + 2] = dsrc[2 * dspl + 2];
|
|
ddst[2 * dspl + 3] = dsrc[2 * dspl + 3];
|
|
ddst[2 * dspl + 4] = dsrc[2 * dspl + 4];
|
|
ddst[2 * dspl + 5] = dsrc[2 * dspl + 5];
|
|
ddst[2 * dspl + 6] = dsrc[2 * dspl + 6];
|
|
ddst[2 * dspl + 7] = dsrc[2 * dspl + 7];
|
|
|
|
ddst[3 * dspl + 0] = dsrc[3 * dspl + 0];
|
|
ddst[3 * dspl + 1] = dsrc[3 * dspl + 1];
|
|
ddst[3 * dspl + 2] = dsrc[3 * dspl + 2];
|
|
ddst[3 * dspl + 3] = dsrc[3 * dspl + 3];
|
|
ddst[3 * dspl + 4] = dsrc[3 * dspl + 4];
|
|
ddst[3 * dspl + 5] = dsrc[3 * dspl + 5];
|
|
ddst[3 * dspl + 6] = dsrc[3 * dspl + 6];
|
|
ddst[3 * dspl + 7] = dsrc[3 * dspl + 7];
|
|
|
|
ddst[4 * dspl + 0] = dsrc[4 * dspl + 0];
|
|
ddst[4 * dspl + 1] = dsrc[4 * dspl + 1];
|
|
ddst[4 * dspl + 2] = dsrc[4 * dspl + 2];
|
|
ddst[4 * dspl + 3] = dsrc[4 * dspl + 3];
|
|
ddst[4 * dspl + 4] = dsrc[4 * dspl + 4];
|
|
ddst[4 * dspl + 5] = dsrc[4 * dspl + 5];
|
|
ddst[4 * dspl + 6] = dsrc[4 * dspl + 6];
|
|
ddst[4 * dspl + 7] = dsrc[4 * dspl + 7];
|
|
|
|
ddst[5 * dspl + 0] = dsrc[5 * dspl + 0];
|
|
ddst[5 * dspl + 1] = dsrc[5 * dspl + 1];
|
|
ddst[5 * dspl + 2] = dsrc[5 * dspl + 2];
|
|
ddst[5 * dspl + 3] = dsrc[5 * dspl + 3];
|
|
ddst[5 * dspl + 4] = dsrc[5 * dspl + 4];
|
|
ddst[5 * dspl + 5] = dsrc[5 * dspl + 5];
|
|
ddst[5 * dspl + 6] = dsrc[5 * dspl + 6];
|
|
ddst[5 * dspl + 7] = dsrc[5 * dspl + 7];
|
|
|
|
ddst[6 * dspl + 0] = dsrc[6 * dspl + 0];
|
|
ddst[6 * dspl + 1] = dsrc[6 * dspl + 1];
|
|
ddst[6 * dspl + 2] = dsrc[6 * dspl + 2];
|
|
ddst[6 * dspl + 3] = dsrc[6 * dspl + 3];
|
|
ddst[6 * dspl + 4] = dsrc[6 * dspl + 4];
|
|
ddst[6 * dspl + 5] = dsrc[6 * dspl + 5];
|
|
ddst[6 * dspl + 6] = dsrc[6 * dspl + 6];
|
|
ddst[6 * dspl + 7] = dsrc[6 * dspl + 7];
|
|
|
|
ddst[7 * dspl + 0] = dsrc[7 * dspl + 0];
|
|
ddst[7 * dspl + 1] = dsrc[7 * dspl + 1];
|
|
ddst[7 * dspl + 2] = dsrc[7 * dspl + 2];
|
|
ddst[7 * dspl + 3] = dsrc[7 * dspl + 3];
|
|
ddst[7 * dspl + 4] = dsrc[7 * dspl + 4];
|
|
ddst[7 * dspl + 5] = dsrc[7 * dspl + 5];
|
|
ddst[7 * dspl + 6] = dsrc[7 * dspl + 6];
|
|
ddst[7 * dspl + 7] = dsrc[7 * dspl + 7];
|
|
#else
|
|
double* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( double* )src;
|
|
ddst = ( double* )dst;
|
|
dspl = spl >> 3;
|
|
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::move4_32
|
|
==============
|
|
*/
|
|
void idCinematicLocal::move4_32( byte* src, byte* dst, int spl )
|
|
{
|
|
#if 1
|
|
int* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( int* )src;
|
|
ddst = ( int* )dst;
|
|
dspl = spl >> 2;
|
|
|
|
ddst[0 * dspl + 0] = dsrc[0 * dspl + 0];
|
|
ddst[0 * dspl + 1] = dsrc[0 * dspl + 1];
|
|
ddst[0 * dspl + 2] = dsrc[0 * dspl + 2];
|
|
ddst[0 * dspl + 3] = dsrc[0 * dspl + 3];
|
|
|
|
ddst[1 * dspl + 0] = dsrc[1 * dspl + 0];
|
|
ddst[1 * dspl + 1] = dsrc[1 * dspl + 1];
|
|
ddst[1 * dspl + 2] = dsrc[1 * dspl + 2];
|
|
ddst[1 * dspl + 3] = dsrc[1 * dspl + 3];
|
|
|
|
ddst[2 * dspl + 0] = dsrc[2 * dspl + 0];
|
|
ddst[2 * dspl + 1] = dsrc[2 * dspl + 1];
|
|
ddst[2 * dspl + 2] = dsrc[2 * dspl + 2];
|
|
ddst[2 * dspl + 3] = dsrc[2 * dspl + 3];
|
|
|
|
ddst[3 * dspl + 0] = dsrc[3 * dspl + 0];
|
|
ddst[3 * dspl + 1] = dsrc[3 * dspl + 1];
|
|
ddst[3 * dspl + 2] = dsrc[3 * dspl + 2];
|
|
ddst[3 * dspl + 3] = dsrc[3 * dspl + 3];
|
|
#else
|
|
double* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( double* )src;
|
|
ddst = ( double* )dst;
|
|
dspl = spl >> 3;
|
|
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += dspl;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::blit8_32
|
|
==============
|
|
*/
|
|
void idCinematicLocal::blit8_32( byte* src, byte* dst, int spl )
|
|
{
|
|
#if 1
|
|
int* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( int* )src;
|
|
ddst = ( int* )dst;
|
|
dspl = spl >> 2;
|
|
|
|
ddst[0 * dspl + 0] = dsrc[ 0];
|
|
ddst[0 * dspl + 1] = dsrc[ 1];
|
|
ddst[0 * dspl + 2] = dsrc[ 2];
|
|
ddst[0 * dspl + 3] = dsrc[ 3];
|
|
ddst[0 * dspl + 4] = dsrc[ 4];
|
|
ddst[0 * dspl + 5] = dsrc[ 5];
|
|
ddst[0 * dspl + 6] = dsrc[ 6];
|
|
ddst[0 * dspl + 7] = dsrc[ 7];
|
|
|
|
ddst[1 * dspl + 0] = dsrc[ 8];
|
|
ddst[1 * dspl + 1] = dsrc[ 9];
|
|
ddst[1 * dspl + 2] = dsrc[10];
|
|
ddst[1 * dspl + 3] = dsrc[11];
|
|
ddst[1 * dspl + 4] = dsrc[12];
|
|
ddst[1 * dspl + 5] = dsrc[13];
|
|
ddst[1 * dspl + 6] = dsrc[14];
|
|
ddst[1 * dspl + 7] = dsrc[15];
|
|
|
|
ddst[2 * dspl + 0] = dsrc[16];
|
|
ddst[2 * dspl + 1] = dsrc[17];
|
|
ddst[2 * dspl + 2] = dsrc[18];
|
|
ddst[2 * dspl + 3] = dsrc[19];
|
|
ddst[2 * dspl + 4] = dsrc[20];
|
|
ddst[2 * dspl + 5] = dsrc[21];
|
|
ddst[2 * dspl + 6] = dsrc[22];
|
|
ddst[2 * dspl + 7] = dsrc[23];
|
|
|
|
ddst[3 * dspl + 0] = dsrc[24];
|
|
ddst[3 * dspl + 1] = dsrc[25];
|
|
ddst[3 * dspl + 2] = dsrc[26];
|
|
ddst[3 * dspl + 3] = dsrc[27];
|
|
ddst[3 * dspl + 4] = dsrc[28];
|
|
ddst[3 * dspl + 5] = dsrc[29];
|
|
ddst[3 * dspl + 6] = dsrc[30];
|
|
ddst[3 * dspl + 7] = dsrc[31];
|
|
|
|
ddst[4 * dspl + 0] = dsrc[32];
|
|
ddst[4 * dspl + 1] = dsrc[33];
|
|
ddst[4 * dspl + 2] = dsrc[34];
|
|
ddst[4 * dspl + 3] = dsrc[35];
|
|
ddst[4 * dspl + 4] = dsrc[36];
|
|
ddst[4 * dspl + 5] = dsrc[37];
|
|
ddst[4 * dspl + 6] = dsrc[38];
|
|
ddst[4 * dspl + 7] = dsrc[39];
|
|
|
|
ddst[5 * dspl + 0] = dsrc[40];
|
|
ddst[5 * dspl + 1] = dsrc[41];
|
|
ddst[5 * dspl + 2] = dsrc[42];
|
|
ddst[5 * dspl + 3] = dsrc[43];
|
|
ddst[5 * dspl + 4] = dsrc[44];
|
|
ddst[5 * dspl + 5] = dsrc[45];
|
|
ddst[5 * dspl + 6] = dsrc[46];
|
|
ddst[5 * dspl + 7] = dsrc[47];
|
|
|
|
ddst[6 * dspl + 0] = dsrc[48];
|
|
ddst[6 * dspl + 1] = dsrc[49];
|
|
ddst[6 * dspl + 2] = dsrc[50];
|
|
ddst[6 * dspl + 3] = dsrc[51];
|
|
ddst[6 * dspl + 4] = dsrc[52];
|
|
ddst[6 * dspl + 5] = dsrc[53];
|
|
ddst[6 * dspl + 6] = dsrc[54];
|
|
ddst[6 * dspl + 7] = dsrc[55];
|
|
|
|
ddst[7 * dspl + 0] = dsrc[56];
|
|
ddst[7 * dspl + 1] = dsrc[57];
|
|
ddst[7 * dspl + 2] = dsrc[58];
|
|
ddst[7 * dspl + 3] = dsrc[59];
|
|
ddst[7 * dspl + 4] = dsrc[60];
|
|
ddst[7 * dspl + 5] = dsrc[61];
|
|
ddst[7 * dspl + 6] = dsrc[62];
|
|
ddst[7 * dspl + 7] = dsrc[63];
|
|
#else
|
|
double* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( double* )src;
|
|
ddst = ( double* )dst;
|
|
dspl = spl >> 3;
|
|
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
dsrc += 4;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
ddst[2] = dsrc[2];
|
|
ddst[3] = dsrc[3];
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::blit4_32
|
|
==============
|
|
*/
|
|
void idCinematicLocal::blit4_32( byte* src, byte* dst, int spl )
|
|
{
|
|
#if 1
|
|
int* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( int* )src;
|
|
ddst = ( int* )dst;
|
|
dspl = spl >> 2;
|
|
|
|
ddst[0 * dspl + 0] = dsrc[ 0];
|
|
ddst[0 * dspl + 1] = dsrc[ 1];
|
|
ddst[0 * dspl + 2] = dsrc[ 2];
|
|
ddst[0 * dspl + 3] = dsrc[ 3];
|
|
ddst[1 * dspl + 0] = dsrc[ 4];
|
|
ddst[1 * dspl + 1] = dsrc[ 5];
|
|
ddst[1 * dspl + 2] = dsrc[ 6];
|
|
ddst[1 * dspl + 3] = dsrc[ 7];
|
|
ddst[2 * dspl + 0] = dsrc[ 8];
|
|
ddst[2 * dspl + 1] = dsrc[ 9];
|
|
ddst[2 * dspl + 2] = dsrc[10];
|
|
ddst[2 * dspl + 3] = dsrc[11];
|
|
ddst[3 * dspl + 0] = dsrc[12];
|
|
ddst[3 * dspl + 1] = dsrc[13];
|
|
ddst[3 * dspl + 2] = dsrc[14];
|
|
ddst[3 * dspl + 3] = dsrc[15];
|
|
#else
|
|
double* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( double* )src;
|
|
ddst = ( double* )dst;
|
|
dspl = spl >> 3;
|
|
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += 2;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += 2;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
dsrc += 2;
|
|
ddst += dspl;
|
|
ddst[0] = dsrc[0];
|
|
ddst[1] = dsrc[1];
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::blit2_32
|
|
==============
|
|
*/
|
|
void idCinematicLocal::blit2_32( byte* src, byte* dst, int spl )
|
|
{
|
|
#if 1
|
|
int* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( int* )src;
|
|
ddst = ( int* )dst;
|
|
dspl = spl >> 2;
|
|
|
|
ddst[0 * dspl + 0] = dsrc[0];
|
|
ddst[0 * dspl + 1] = dsrc[1];
|
|
ddst[1 * dspl + 0] = dsrc[2];
|
|
ddst[1 * dspl + 1] = dsrc[3];
|
|
#else
|
|
double* dsrc, *ddst;
|
|
int dspl;
|
|
|
|
dsrc = ( double* )src;
|
|
ddst = ( double* )dst;
|
|
dspl = spl >> 3;
|
|
|
|
ddst[0] = dsrc[0];
|
|
ddst[dspl] = dsrc[1];
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::blitVQQuad32fs
|
|
==============
|
|
*/
|
|
void idCinematicLocal::blitVQQuad32fs( byte** status, unsigned char* data )
|
|
{
|
|
unsigned short newd, celdata, code;
|
|
unsigned int index, i;
|
|
|
|
newd = 0;
|
|
celdata = 0;
|
|
index = 0;
|
|
|
|
do
|
|
{
|
|
if( !newd )
|
|
{
|
|
newd = 7;
|
|
celdata = data[0] + data[1] * 256;
|
|
data += 2;
|
|
}
|
|
else
|
|
{
|
|
newd--;
|
|
}
|
|
|
|
code = ( unsigned short )( celdata & 0xc000 );
|
|
celdata <<= 2;
|
|
|
|
switch( code )
|
|
{
|
|
case 0x8000: // vq code
|
|
blit8_32( ( byte* )&vq8[( *data ) * 128], status[index], samplesPerLine );
|
|
data++;
|
|
index += 5;
|
|
break;
|
|
case 0xc000: // drop
|
|
index++; // skip 8x8
|
|
for( i = 0; i < 4; i++ )
|
|
{
|
|
if( !newd )
|
|
{
|
|
newd = 7;
|
|
celdata = data[0] + data[1] * 256;
|
|
data += 2;
|
|
}
|
|
else
|
|
{
|
|
newd--;
|
|
}
|
|
|
|
code = ( unsigned short )( celdata & 0xc000 );
|
|
celdata <<= 2;
|
|
|
|
switch( code ) // code in top two bits of code
|
|
{
|
|
case 0x8000: // 4x4 vq code
|
|
blit4_32( ( byte* )&vq4[( *data ) * 32], status[index], samplesPerLine );
|
|
data++;
|
|
break;
|
|
case 0xc000: // 2x2 vq code
|
|
blit2_32( ( byte* )&vq2[( *data ) * 8], status[index], samplesPerLine );
|
|
data++;
|
|
blit2_32( ( byte* )&vq2[( *data ) * 8], status[index] + 8, samplesPerLine );
|
|
data++;
|
|
blit2_32( ( byte* )&vq2[( *data ) * 8], status[index] + samplesPerLine * 2, samplesPerLine );
|
|
data++;
|
|
blit2_32( ( byte* )&vq2[( *data ) * 8], status[index] + samplesPerLine * 2 + 8, samplesPerLine );
|
|
data++;
|
|
break;
|
|
case 0x4000: // motion compensation
|
|
move4_32( status[index] + mcomp[( *data )], status[index], samplesPerLine );
|
|
data++;
|
|
break;
|
|
}
|
|
index++;
|
|
}
|
|
break;
|
|
case 0x4000: // motion compensation
|
|
move8_32( status[index] + mcomp[( *data )], status[index], samplesPerLine );
|
|
data++;
|
|
index += 5;
|
|
break;
|
|
case 0x0000:
|
|
index += 5;
|
|
break;
|
|
}
|
|
}
|
|
while( status[index] != NULL );
|
|
}
|
|
|
|
#define VQ2TO4(a,b,c,d) { \
|
|
*c++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*c++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*c++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*c++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[0]; \
|
|
*d++ = a[1]; \
|
|
*d++ = a[1]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[0]; \
|
|
*d++ = b[1]; \
|
|
*d++ = b[1]; \
|
|
a += 2; b += 2; }
|
|
|
|
#define VQ2TO2(a,b,c,d) { \
|
|
*c++ = *a; \
|
|
*d++ = *a; \
|
|
*d++ = *a; \
|
|
*c++ = *b; \
|
|
*d++ = *b; \
|
|
*d++ = *b; \
|
|
*d++ = *a; \
|
|
*d++ = *a; \
|
|
*d++ = *b; \
|
|
*d++ = *b; \
|
|
a++; b++; }
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::yuv_to_rgb
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
unsigned short idCinematicLocal::yuv_to_rgb( int y, int u, int v )
|
|
{
|
|
int r, g, b, YY = ( int )( ROQ_YY_tab[( y )] );
|
|
|
|
r = ( YY + ROQ_VR_tab[v] ) >> 9;
|
|
g = ( YY + ROQ_UG_tab[u] + ROQ_VG_tab[v] ) >> 8;
|
|
b = ( YY + ROQ_UB_tab[u] ) >> 9;
|
|
|
|
if( r < 0 ) r = 0;
|
|
if( g < 0 ) g = 0;
|
|
if( b < 0 ) b = 0;
|
|
if( r > 31 ) r = 31;
|
|
if( g > 63 ) g = 63;
|
|
if( b > 31 ) b = 31;
|
|
|
|
return ( unsigned short )( ( r << 11 ) + ( g << 5 ) + ( b ) );
|
|
}
|
|
// RB end
|
|
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::yuv_to_rgb24
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
unsigned int idCinematicLocal::yuv_to_rgb24( int y, int u, int v )
|
|
{
|
|
int r, g, b, YY = ( int )( ROQ_YY_tab[( y )] );
|
|
|
|
r = ( YY + ROQ_VR_tab[v] ) >> 6;
|
|
g = ( YY + ROQ_UG_tab[u] + ROQ_VG_tab[v] ) >> 6;
|
|
b = ( YY + ROQ_UB_tab[u] ) >> 6;
|
|
|
|
if( r < 0 ) r = 0;
|
|
if( g < 0 ) g = 0;
|
|
if( b < 0 ) b = 0;
|
|
if( r > 255 ) r = 255;
|
|
if( g > 255 ) g = 255;
|
|
if( b > 255 ) b = 255;
|
|
|
|
return LittleLong( ( r ) + ( g << 8 ) + ( b << 16 ) );
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::decodeCodeBook
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
void idCinematicLocal::decodeCodeBook( byte* input, unsigned short roq_flags )
|
|
{
|
|
int i, j, two, four;
|
|
unsigned short* aptr, *bptr, *cptr, *dptr;
|
|
int y0, y1, y2, y3, cr, cb;
|
|
unsigned int* iaptr, *ibptr, *icptr, *idptr;
|
|
|
|
if( !roq_flags )
|
|
{
|
|
two = four = 256;
|
|
}
|
|
else
|
|
{
|
|
two = roq_flags >> 8;
|
|
if( !two ) two = 256;
|
|
four = roq_flags & 0xff;
|
|
}
|
|
|
|
four *= 2;
|
|
|
|
bptr = ( unsigned short* )vq2;
|
|
|
|
if( !half )
|
|
{
|
|
if( !smootheddouble )
|
|
{
|
|
//
|
|
// normal height
|
|
//
|
|
if( samplesPerPixel == 2 )
|
|
{
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input++;
|
|
y1 = ( int ) * input++;
|
|
y2 = ( int ) * input++;
|
|
y3 = ( int ) * input++;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y1, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y3, cr, cb );
|
|
}
|
|
|
|
cptr = ( unsigned short* )vq4;
|
|
dptr = ( unsigned short* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
aptr = ( unsigned short* )vq2 + ( *input++ ) * 4;
|
|
bptr = ( unsigned short* )vq2 + ( *input++ ) * 4;
|
|
for( j = 0; j < 2; j++ )
|
|
VQ2TO4( aptr, bptr, cptr, dptr );
|
|
}
|
|
}
|
|
else if( samplesPerPixel == 4 )
|
|
{
|
|
ibptr = ( unsigned int* )bptr;
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input++;
|
|
y1 = ( int ) * input++;
|
|
y2 = ( int ) * input++;
|
|
y3 = ( int ) * input++;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*ibptr++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y1, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y2, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y3, cr, cb );
|
|
}
|
|
|
|
icptr = ( unsigned int* )vq4;
|
|
idptr = ( unsigned int* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
iaptr = ( unsigned int* )vq2 + ( *input++ ) * 4;
|
|
ibptr = ( unsigned int* )vq2 + ( *input++ ) * 4;
|
|
for( j = 0; j < 2; j++ )
|
|
VQ2TO4( iaptr, ibptr, icptr, idptr );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// double height, smoothed
|
|
//
|
|
if( samplesPerPixel == 2 )
|
|
{
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input++;
|
|
y1 = ( int ) * input++;
|
|
y2 = ( int ) * input++;
|
|
y3 = ( int ) * input++;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y1, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ( ( y0 * 3 ) + y2 ) / 4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ( ( y1 * 3 ) + y3 ) / 4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ( y0 + ( y2 * 3 ) ) / 4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( ( y1 + ( y3 * 3 ) ) / 4, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y3, cr, cb );
|
|
}
|
|
|
|
cptr = ( unsigned short* )vq4;
|
|
dptr = ( unsigned short* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
aptr = ( unsigned short* )vq2 + ( *input++ ) * 8;
|
|
bptr = ( unsigned short* )vq2 + ( *input++ ) * 8;
|
|
for( j = 0; j < 2; j++ )
|
|
{
|
|
VQ2TO4( aptr, bptr, cptr, dptr );
|
|
VQ2TO4( aptr, bptr, cptr, dptr );
|
|
}
|
|
}
|
|
}
|
|
else if( samplesPerPixel == 4 )
|
|
{
|
|
ibptr = ( unsigned int* )bptr;
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input++;
|
|
y1 = ( int ) * input++;
|
|
y2 = ( int ) * input++;
|
|
y3 = ( int ) * input++;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*ibptr++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y1, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( ( ( y0 * 3 ) + y2 ) / 4, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( ( ( y1 * 3 ) + y3 ) / 4, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( ( y0 + ( y2 * 3 ) ) / 4, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( ( y1 + ( y3 * 3 ) ) / 4, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y2, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y3, cr, cb );
|
|
}
|
|
|
|
icptr = ( unsigned int* )vq4;
|
|
idptr = ( unsigned int* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
iaptr = ( unsigned int* )vq2 + ( *input++ ) * 8;
|
|
ibptr = ( unsigned int* )vq2 + ( *input++ ) * 8;
|
|
for( j = 0; j < 2; j++ )
|
|
{
|
|
VQ2TO4( iaptr, ibptr, icptr, idptr );
|
|
VQ2TO4( iaptr, ibptr, icptr, idptr );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// 1/4 screen
|
|
//
|
|
if( samplesPerPixel == 2 )
|
|
{
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input;
|
|
input += 2;
|
|
y2 = ( int ) * input;
|
|
input += 2;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*bptr++ = yuv_to_rgb( y0, cr, cb );
|
|
*bptr++ = yuv_to_rgb( y2, cr, cb );
|
|
}
|
|
|
|
cptr = ( unsigned short* )vq4;
|
|
dptr = ( unsigned short* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
aptr = ( unsigned short* )vq2 + ( *input++ ) * 2;
|
|
bptr = ( unsigned short* )vq2 + ( *input++ ) * 2;
|
|
for( j = 0; j < 2; j++ )
|
|
{
|
|
VQ2TO2( aptr, bptr, cptr, dptr );
|
|
}
|
|
}
|
|
}
|
|
else if( samplesPerPixel == 4 )
|
|
{
|
|
ibptr = ( unsigned int* ) bptr;
|
|
for( i = 0; i < two; i++ )
|
|
{
|
|
y0 = ( int ) * input;
|
|
input += 2;
|
|
y2 = ( int ) * input;
|
|
input += 2;
|
|
cr = ( int ) * input++;
|
|
cb = ( int ) * input++;
|
|
*ibptr++ = yuv_to_rgb24( y0, cr, cb );
|
|
*ibptr++ = yuv_to_rgb24( y2, cr, cb );
|
|
}
|
|
|
|
icptr = ( unsigned int* )vq4;
|
|
idptr = ( unsigned int* )vq8;
|
|
|
|
for( i = 0; i < four; i++ )
|
|
{
|
|
iaptr = ( unsigned int* )vq2 + ( *input++ ) * 2;
|
|
ibptr = ( unsigned int* )vq2 + ( *input++ ) * 2;
|
|
for( j = 0; j < 2; j++ )
|
|
{
|
|
VQ2TO2( iaptr, ibptr, icptr, idptr );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::recurseQuad
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
void idCinematicLocal::recurseQuad( int startX, int startY, int quadSize, int xOff, int yOff )
|
|
{
|
|
byte* scroff;
|
|
int bigx, bigy, lowx, lowy, useY;
|
|
int offset;
|
|
|
|
offset = screenDelta;
|
|
|
|
lowx = lowy = 0;
|
|
bigx = xsize;
|
|
bigy = ysize;
|
|
|
|
if( bigx > CIN_WIDTH ) bigx = CIN_WIDTH;
|
|
if( bigy > CIN_HEIGHT ) bigy = CIN_HEIGHT;
|
|
|
|
if( ( startX >= lowx ) && ( startX + quadSize ) <= ( bigx ) && ( startY + quadSize ) <= ( bigy ) && ( startY >= lowy ) && quadSize <= MAXSIZE )
|
|
{
|
|
useY = startY;
|
|
scroff = image + ( useY + ( ( CIN_HEIGHT - bigy ) >> 1 ) + yOff ) * ( samplesPerLine ) + ( ( ( startX + xOff ) ) * samplesPerPixel );
|
|
|
|
qStatus[0][onQuad ] = scroff;
|
|
qStatus[1][onQuad++] = scroff + offset;
|
|
}
|
|
|
|
if( quadSize != MINSIZE )
|
|
{
|
|
quadSize >>= 1;
|
|
recurseQuad( startX, startY , quadSize, xOff, yOff );
|
|
recurseQuad( startX + quadSize, startY , quadSize, xOff, yOff );
|
|
recurseQuad( startX, startY + quadSize , quadSize, xOff, yOff );
|
|
recurseQuad( startX + quadSize, startY + quadSize , quadSize, xOff, yOff );
|
|
}
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::setupQuad
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
void idCinematicLocal::setupQuad( int xOff, int yOff )
|
|
{
|
|
int numQuadCels, i, x, y;
|
|
byte* temp;
|
|
|
|
numQuadCels = ( CIN_WIDTH * CIN_HEIGHT ) / ( 16 );
|
|
numQuadCels += numQuadCels / 4 + numQuadCels / 16;
|
|
numQuadCels += 64; // for overflow
|
|
|
|
numQuadCels = ( xsize * ysize ) / ( 16 );
|
|
numQuadCels += numQuadCels / 4;
|
|
numQuadCels += 64; // for overflow
|
|
|
|
onQuad = 0;
|
|
|
|
for( y = 0; y < ( int )ysize; y += 16 )
|
|
for( x = 0; x < ( int )xsize; x += 16 )
|
|
recurseQuad( x, y, 16, xOff, yOff );
|
|
|
|
temp = NULL;
|
|
|
|
for( i = ( numQuadCels - 64 ); i < numQuadCels; i++ )
|
|
{
|
|
qStatus[0][i] = temp; // eoq
|
|
qStatus[1][i] = temp; // eoq
|
|
}
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::readQuadInfo
|
|
==============
|
|
*/
|
|
void idCinematicLocal::readQuadInfo( byte* qData )
|
|
{
|
|
xsize = qData[0] + qData[1] * 256;
|
|
ysize = qData[2] + qData[3] * 256;
|
|
maxsize = qData[4] + qData[5] * 256;
|
|
minsize = qData[6] + qData[7] * 256;
|
|
|
|
CIN_HEIGHT = ysize;
|
|
CIN_WIDTH = xsize;
|
|
|
|
samplesPerLine = CIN_WIDTH * samplesPerPixel;
|
|
screenDelta = CIN_HEIGHT * samplesPerLine;
|
|
|
|
if( !image )
|
|
{
|
|
image = ( byte* )Mem_Alloc( CIN_WIDTH * CIN_HEIGHT * samplesPerPixel * 2, TAG_CINEMATIC );
|
|
}
|
|
|
|
half = false;
|
|
smootheddouble = false;
|
|
|
|
// RB: 64 bit fixes, changed unsigned int to ptrdiff_t
|
|
t[0] = ( 0 - ( ptrdiff_t )image ) + ( ptrdiff_t )image + screenDelta;
|
|
t[1] = ( 0 - ( ( ptrdiff_t )image + screenDelta ) ) + ( ptrdiff_t )image;
|
|
// RB end
|
|
|
|
drawX = CIN_WIDTH;
|
|
drawY = CIN_HEIGHT;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::RoQPrepMcomp
|
|
==============
|
|
*/
|
|
// RB: 64 bit fixes, changed long to int
|
|
void idCinematicLocal::RoQPrepMcomp( int xoff, int yoff )
|
|
{
|
|
int i, j, x, y, temp, temp2;
|
|
|
|
i = samplesPerLine;
|
|
j = samplesPerPixel;
|
|
if( xsize == ( ysize * 4 ) && !half )
|
|
{
|
|
j = j + j;
|
|
i = i + i;
|
|
}
|
|
|
|
for( y = 0; y < 16; y++ )
|
|
{
|
|
temp2 = ( y + yoff - 8 ) * i;
|
|
for( x = 0; x < 16; x++ )
|
|
{
|
|
temp = ( x + xoff - 8 ) * j;
|
|
mcomp[( x * 16 ) + y] = normalBuffer0 - ( temp2 + temp );
|
|
}
|
|
}
|
|
}
|
|
// RB end
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::RoQReset
|
|
==============
|
|
*/
|
|
void idCinematicLocal::RoQReset()
|
|
{
|
|
|
|
iFile->Seek( 0, FS_SEEK_SET );
|
|
iFile->Read( file, 16 );
|
|
RoQ_init();
|
|
status = FMV_LOOPED;
|
|
}
|
|
|
|
|
|
typedef struct
|
|
{
|
|
struct jpeg_source_mgr pub; /* public fields */
|
|
|
|
byte* infile; /* source stream */
|
|
JOCTET* buffer; /* start of buffer */
|
|
boolean start_of_file; /* have we gotten any data yet? */
|
|
int memsize;
|
|
} my_source_mgr;
|
|
|
|
typedef my_source_mgr* my_src_ptr;
|
|
|
|
#define INPUT_BUF_SIZE 32768 /* choose an efficiently fread'able size */
|
|
|
|
/* jpeg error handling */
|
|
struct jpeg_error_mgr jerr;
|
|
|
|
/*
|
|
* Fill the input buffer --- called whenever buffer is emptied.
|
|
*
|
|
* In typical applications, this should read fresh data into the buffer
|
|
* (ignoring the current state of next_input_byte & bytes_in_buffer),
|
|
* reset the pointer & count to the start of the buffer, and return TRUE
|
|
* indicating that the buffer has been reloaded. It is not necessary to
|
|
* fill the buffer entirely, only to obtain at least one more byte.
|
|
*
|
|
* There is no such thing as an EOF return. If the end of the file has been
|
|
* reached, the routine has a choice of ERREXIT() or inserting fake data into
|
|
* the buffer. In most cases, generating a warning message and inserting a
|
|
* fake EOI marker is the best course of action --- this will allow the
|
|
* decompressor to output however much of the image is there. However,
|
|
* the resulting error message is misleading if the real problem is an empty
|
|
* input file, so we handle that case specially.
|
|
*
|
|
* In applications that need to be able to suspend compression due to input
|
|
* not being available yet, a FALSE return indicates that no more data can be
|
|
* obtained right now, but more may be forthcoming later. In this situation,
|
|
* the decompressor will return to its caller (with an indication of the
|
|
* number of scanlines it has read, if any). The application should resume
|
|
* decompression after it has loaded more data into the input buffer. Note
|
|
* that there are substantial restrictions on the use of suspension --- see
|
|
* the documentation.
|
|
*
|
|
* When suspending, the decompressor will back up to a convenient restart point
|
|
* (typically the start of the current MCU). next_input_byte & bytes_in_buffer
|
|
* indicate where the restart point will be if the current call returns FALSE.
|
|
* Data beyond this point must be rescanned after resumption, so move it to
|
|
* the front of the buffer rather than discarding it.
|
|
*/
|
|
|
|
|
|
METHODDEF boolean fill_input_buffer( j_decompress_ptr cinfo )
|
|
{
|
|
my_src_ptr src = ( my_src_ptr ) cinfo->src;
|
|
int nbytes;
|
|
|
|
nbytes = INPUT_BUF_SIZE;
|
|
if( nbytes > src->memsize ) nbytes = src->memsize;
|
|
if( nbytes == 0 )
|
|
{
|
|
/* Insert a fake EOI marker */
|
|
src->buffer[0] = ( JOCTET ) 0xFF;
|
|
src->buffer[1] = ( JOCTET ) JPEG_EOI;
|
|
nbytes = 2;
|
|
}
|
|
else
|
|
{
|
|
memcpy( src->buffer, src->infile, INPUT_BUF_SIZE );
|
|
src->infile = src->infile + nbytes;
|
|
src->memsize = src->memsize - INPUT_BUF_SIZE;
|
|
}
|
|
src->pub.next_input_byte = src->buffer;
|
|
src->pub.bytes_in_buffer = nbytes;
|
|
src->start_of_file = FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
/*
|
|
* Initialize source --- called by jpeg_read_header
|
|
* before any data is actually read.
|
|
*/
|
|
|
|
|
|
METHODDEF void init_source( j_decompress_ptr cinfo )
|
|
{
|
|
my_src_ptr src = ( my_src_ptr ) cinfo->src;
|
|
|
|
/* We reset the empty-input-file flag for each image,
|
|
* but we don't clear the input buffer.
|
|
* This is correct behavior for reading a series of images from one source.
|
|
*/
|
|
src->start_of_file = TRUE;
|
|
}
|
|
|
|
/*
|
|
* Skip data --- used to skip over a potentially large amount of
|
|
* uninteresting data (such as an APPn marker).
|
|
*
|
|
* Writers of suspendable-input applications must note that skip_input_data
|
|
* is not granted the right to give a suspension return. If the skip extends
|
|
* beyond the data currently in the buffer, the buffer can be marked empty so
|
|
* that the next read will cause a fill_input_buffer call that can suspend.
|
|
* Arranging for additional bytes to be discarded before reloading the input
|
|
* buffer is the application writer's problem.
|
|
*/
|
|
|
|
METHODDEF void
|
|
skip_input_data( j_decompress_ptr cinfo, long num_bytes )
|
|
{
|
|
my_src_ptr src = ( my_src_ptr ) cinfo->src;
|
|
|
|
/* Just a dumb implementation for now. Could use fseek() except
|
|
* it doesn't work on pipes. Not clear that being smart is worth
|
|
* any trouble anyway --- large skips are infrequent.
|
|
*/
|
|
if( num_bytes > 0 )
|
|
{
|
|
src->infile = src->infile + num_bytes;
|
|
src->pub.next_input_byte += ( size_t ) num_bytes;
|
|
src->pub.bytes_in_buffer -= ( size_t ) num_bytes;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* An additional method that can be provided by data source modules is the
|
|
* resync_to_restart method for error recovery in the presence of RST markers.
|
|
* For the moment, this source module just uses the default resync method
|
|
* provided by the JPEG library. That method assumes that no backtracking
|
|
* is possible.
|
|
*/
|
|
|
|
|
|
/*
|
|
* Terminate source --- called by jpeg_finish_decompress
|
|
* after all data has been read. Often a no-op.
|
|
*
|
|
* NB: *not* called by jpeg_abort or jpeg_destroy; surrounding
|
|
* application must deal with any cleanup that should happen even
|
|
* for error exit.
|
|
*/
|
|
|
|
METHODDEF void
|
|
term_source( j_decompress_ptr cinfo )
|
|
{
|
|
cinfo = cinfo;
|
|
/* no work necessary here */
|
|
}
|
|
|
|
GLOBAL void
|
|
jpeg_memory_src( j_decompress_ptr cinfo, byte* infile, int size )
|
|
{
|
|
my_src_ptr src;
|
|
|
|
/* The source object and input buffer are made permanent so that a series
|
|
* of JPEG images can be read from the same file by calling jpeg_stdio_src
|
|
* only before the first one. (If we discarded the buffer at the end of
|
|
* one image, we'd likely lose the start of the next one.)
|
|
* This makes it unsafe to use this manager and a different source
|
|
* manager serially with the same JPEG object. Caveat programmer.
|
|
*/
|
|
if( cinfo->src == NULL ) /* first time for this JPEG object? */
|
|
{
|
|
cinfo->src = ( struct jpeg_source_mgr* )
|
|
( *cinfo->mem->alloc_small )( ( j_common_ptr ) cinfo, JPOOL_PERMANENT,
|
|
sizeof( my_source_mgr ) );
|
|
src = ( my_src_ptr ) cinfo->src;
|
|
src->buffer = ( JOCTET* )
|
|
( *cinfo->mem->alloc_small )( ( j_common_ptr ) cinfo, JPOOL_PERMANENT,
|
|
INPUT_BUF_SIZE * sizeof( JOCTET ) );
|
|
}
|
|
|
|
src = ( my_src_ptr ) cinfo->src;
|
|
src->pub.init_source = init_source;
|
|
src->pub.fill_input_buffer = fill_input_buffer;
|
|
src->pub.skip_input_data = skip_input_data;
|
|
src->pub.resync_to_restart = jpeg_resync_to_restart; /* use default method */
|
|
src->pub.term_source = term_source;
|
|
src->infile = infile;
|
|
src->memsize = size;
|
|
src->pub.bytes_in_buffer = 0; /* forces fill_input_buffer on first read */
|
|
src->pub.next_input_byte = NULL; /* until buffer loaded */
|
|
}
|
|
|
|
int JPEGBlit( byte* wStatus, byte* data, int datasize )
|
|
{
|
|
/* This struct contains the JPEG decompression parameters and pointers to
|
|
* working space (which is allocated as needed by the JPEG library).
|
|
*/
|
|
struct jpeg_decompress_struct cinfo;
|
|
/* We use our private extension JPEG error handler.
|
|
* Note that this struct must live as long as the main JPEG parameter
|
|
* struct, to avoid dangling-pointer problems.
|
|
*/
|
|
/* More stuff */
|
|
JSAMPARRAY buffer; /* Output row buffer */
|
|
int row_stride; /* physical row width in output buffer */
|
|
|
|
/* Step 1: allocate and initialize JPEG decompression object */
|
|
|
|
/* We set up the normal JPEG error routines, then override error_exit. */
|
|
cinfo.err = jpeg_std_error( &jerr );
|
|
|
|
/* Now we can initialize the JPEG decompression object. */
|
|
jpeg_create_decompress( &cinfo );
|
|
|
|
/* Step 2: specify data source (eg, a file) */
|
|
|
|
jpeg_memory_src( &cinfo, data, datasize );
|
|
|
|
/* Step 3: read file parameters with jpeg_read_header() */
|
|
|
|
jpeg_read_header( &cinfo, TRUE );
|
|
/* We can ignore the return value from jpeg_read_header since
|
|
* (a) suspension is not possible with the stdio data source, and
|
|
* (b) we passed TRUE to reject a tables-only JPEG file as an error.
|
|
* See libjpeg.doc for more info.
|
|
*/
|
|
|
|
/* Step 4: set parameters for decompression */
|
|
|
|
/* In this example, we don't need to change any of the defaults set by
|
|
* jpeg_read_header(), so we do nothing here.
|
|
*/
|
|
|
|
/* Step 5: Start decompressor */
|
|
|
|
cinfo.dct_method = JDCT_IFAST;
|
|
cinfo.dct_method = JDCT_FASTEST;
|
|
cinfo.dither_mode = JDITHER_NONE;
|
|
cinfo.do_fancy_upsampling = FALSE;
|
|
// cinfo.out_color_space = JCS_GRAYSCALE;
|
|
|
|
jpeg_start_decompress( &cinfo );
|
|
/* We can ignore the return value since suspension is not possible
|
|
* with the stdio data source.
|
|
*/
|
|
|
|
/* We may need to do some setup of our own at this point before reading
|
|
* the data. After jpeg_start_decompress() we have the correct scaled
|
|
* output image dimensions available, as well as the output colormap
|
|
* if we asked for color quantization.
|
|
* In this example, we need to make an output work buffer of the right size.
|
|
*/
|
|
/* JSAMPLEs per row in output buffer */
|
|
row_stride = cinfo.output_width * cinfo.output_components;
|
|
|
|
/* Make a one-row-high sample array that will go away when done with image */
|
|
buffer = ( *cinfo.mem->alloc_sarray )
|
|
( ( j_common_ptr ) &cinfo, JPOOL_IMAGE, row_stride, 1 );
|
|
|
|
/* Step 6: while (scan lines remain to be read) */
|
|
/* jpeg_read_scanlines(...); */
|
|
|
|
/* Here we use the library's state variable cinfo.output_scanline as the
|
|
* loop counter, so that we don't have to keep track ourselves.
|
|
*/
|
|
|
|
wStatus += ( cinfo.output_height - 1 ) * row_stride;
|
|
while( cinfo.output_scanline < cinfo.output_height )
|
|
{
|
|
/* jpeg_read_scanlines expects an array of pointers to scanlines.
|
|
* Here the array is only one element long, but you could ask for
|
|
* more than one scanline at a time if that's more convenient.
|
|
*/
|
|
jpeg_read_scanlines( &cinfo, &buffer[0], 1 );
|
|
|
|
/* Assume put_scanline_someplace wants a pointer and sample count. */
|
|
memcpy( wStatus, &buffer[0][0], row_stride );
|
|
/*
|
|
int x;
|
|
unsigned int *buf = (unsigned int *)&buffer[0][0];
|
|
unsigned int *out = (unsigned int *)wStatus;
|
|
for(x=0;x<cinfo.output_width;x++) {
|
|
unsigned int pixel = buf[x];
|
|
byte *roof = (byte *)&pixel;
|
|
byte temp = roof[0];
|
|
roof[0] = roof[2];
|
|
roof[2] = temp;
|
|
out[x] = pixel;
|
|
}
|
|
*/
|
|
wStatus -= row_stride;
|
|
}
|
|
|
|
/* Step 7: Finish decompression */
|
|
|
|
jpeg_finish_decompress( &cinfo );
|
|
/* We can ignore the return value since suspension is not possible
|
|
* with the stdio data source.
|
|
*/
|
|
|
|
/* Step 8: Release JPEG decompression object */
|
|
|
|
/* This is an important step since it will release a good deal of memory. */
|
|
jpeg_destroy_decompress( &cinfo );
|
|
|
|
/* At this point you may want to check to see whether any corrupt-data
|
|
* warnings occurred (test whether jerr.pub.num_warnings is nonzero).
|
|
*/
|
|
|
|
/* And we're done! */
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::RoQInterrupt
|
|
==============
|
|
*/
|
|
void idCinematicLocal::RoQInterrupt()
|
|
{
|
|
byte* framedata;
|
|
|
|
iFile->Read( file, RoQFrameSize + 8 );
|
|
if( RoQPlayed >= ROQSize )
|
|
{
|
|
if( looping )
|
|
{
|
|
RoQReset();
|
|
}
|
|
else
|
|
{
|
|
status = FMV_EOF;
|
|
}
|
|
return;
|
|
}
|
|
|
|
framedata = file;
|
|
//
|
|
// new frame is ready
|
|
//
|
|
redump:
|
|
switch( roq_id )
|
|
{
|
|
case ROQ_QUAD_VQ:
|
|
if( ( numQuads & 1 ) )
|
|
{
|
|
normalBuffer0 = t[1];
|
|
RoQPrepMcomp( roqF0, roqF1 );
|
|
blitVQQuad32fs( qStatus[1], framedata );
|
|
buf = image + screenDelta;
|
|
}
|
|
else
|
|
{
|
|
normalBuffer0 = t[0];
|
|
RoQPrepMcomp( roqF0, roqF1 );
|
|
blitVQQuad32fs( qStatus[0], framedata );
|
|
buf = image;
|
|
}
|
|
if( numQuads == 0 ) // first frame
|
|
{
|
|
memcpy( image + screenDelta, image, samplesPerLine * ysize );
|
|
}
|
|
numQuads++;
|
|
dirty = true;
|
|
break;
|
|
case ROQ_CODEBOOK:
|
|
decodeCodeBook( framedata, ( unsigned short )roq_flags );
|
|
break;
|
|
case ZA_SOUND_MONO:
|
|
break;
|
|
case ZA_SOUND_STEREO:
|
|
break;
|
|
case ROQ_QUAD_INFO:
|
|
if( numQuads == -1 )
|
|
{
|
|
readQuadInfo( framedata );
|
|
setupQuad( 0, 0 );
|
|
}
|
|
if( numQuads != 1 ) numQuads = 0;
|
|
break;
|
|
case ROQ_PACKET:
|
|
inMemory = ( roq_flags != 0 );
|
|
RoQFrameSize = 0; // for header
|
|
break;
|
|
case ROQ_QUAD_HANG:
|
|
RoQFrameSize = 0;
|
|
break;
|
|
case ROQ_QUAD_JPEG:
|
|
if( !numQuads )
|
|
{
|
|
normalBuffer0 = t[0];
|
|
JPEGBlit( image, framedata, RoQFrameSize );
|
|
memcpy( image + screenDelta, image, samplesPerLine * ysize );
|
|
numQuads++;
|
|
}
|
|
break;
|
|
default:
|
|
status = FMV_EOF;
|
|
break;
|
|
}
|
|
//
|
|
// read in next frame data
|
|
//
|
|
if( RoQPlayed >= ROQSize )
|
|
{
|
|
if( looping )
|
|
{
|
|
RoQReset();
|
|
}
|
|
else
|
|
{
|
|
status = FMV_EOF;
|
|
}
|
|
return;
|
|
}
|
|
|
|
framedata += RoQFrameSize;
|
|
roq_id = framedata[0] + framedata[1] * 256;
|
|
RoQFrameSize = framedata[2] + framedata[3] * 256 + framedata[4] * 65536;
|
|
roq_flags = framedata[6] + framedata[7] * 256;
|
|
roqF0 = ( char )framedata[7];
|
|
roqF1 = ( char )framedata[6];
|
|
|
|
if( RoQFrameSize > 65536 || roq_id == 0x1084 )
|
|
{
|
|
common->DPrintf( "roq_size>65536||roq_id==0x1084\n" );
|
|
status = FMV_EOF;
|
|
if( looping )
|
|
{
|
|
RoQReset();
|
|
}
|
|
return;
|
|
}
|
|
if( inMemory && ( status != FMV_EOF ) )
|
|
{
|
|
inMemory = false;
|
|
framedata += 8;
|
|
goto redump;
|
|
}
|
|
//
|
|
// one more frame hits the dust
|
|
//
|
|
// assert(RoQFrameSize <= 65536);
|
|
// r = Sys_StreamedRead( file, RoQFrameSize+8, 1, iFile );
|
|
RoQPlayed += RoQFrameSize + 8;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::RoQ_init
|
|
==============
|
|
*/
|
|
void idCinematicLocal::RoQ_init()
|
|
{
|
|
|
|
RoQPlayed = 24;
|
|
|
|
/* get frame rate */
|
|
roqFPS = file[ 6] + file[ 7] * 256;
|
|
|
|
if( !roqFPS ) roqFPS = 30;
|
|
|
|
numQuads = -1;
|
|
|
|
roq_id = file[ 8] + file[ 9] * 256;
|
|
RoQFrameSize = file[10] + file[11] * 256 + file[12] * 65536;
|
|
roq_flags = file[14] + file[15] * 256;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idCinematicLocal::RoQShutdown
|
|
==============
|
|
*/
|
|
void idCinematicLocal::RoQShutdown()
|
|
{
|
|
if( status == FMV_IDLE )
|
|
{
|
|
return;
|
|
}
|
|
status = FMV_IDLE;
|
|
|
|
if( iFile )
|
|
{
|
|
fileSystem->CloseFile( iFile );
|
|
iFile = NULL;
|
|
}
|
|
|
|
fileName = "";
|
|
}
|
|
|
|
//===========================================
|
|
|
|
/*
|
|
==============
|
|
idSndWindow::InitFromFile
|
|
==============
|
|
*/
|
|
bool idSndWindow::InitFromFile( const char* qpath, bool looping )
|
|
{
|
|
idStr fname = qpath;
|
|
|
|
fname.ToLower();
|
|
if( !fname.Icmp( "waveform" ) )
|
|
{
|
|
showWaveform = true;
|
|
}
|
|
else
|
|
{
|
|
showWaveform = false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idSndWindow::ImageForTime
|
|
==============
|
|
*/
|
|
cinData_t idSndWindow::ImageForTime( int milliseconds )
|
|
{
|
|
return soundSystem->ImageForTime( milliseconds, showWaveform );
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idSndWindow::AnimationLength
|
|
==============
|
|
*/
|
|
int idSndWindow::AnimationLength()
|
|
{
|
|
return -1;
|
|
}
|