/* Copyright (C) 1999-2007 id Software, Inc. and contributors. For a list of contributors, see the accompanying CONTRIBUTORS file. This file is part of GtkRadiant. GtkRadiant is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. GtkRadiant 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 GtkRadiant; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ // To do // Sound error handling (when sound too short) // rle b4 huffing // adpcm encoding of sound #if 0 #include "qdata.h" #include "flex.h" #include "fc.h" #include "adpcm.h" #define MIN_REPT 15 #define MAX_REPT 0 #define HUF_TOKENS ( 256 + MAX_REPT ) #define BLOCKSIZE 8 #define M_PI 3.14159265358979323846 // matches value in gcc v2 math.h #define SQRT2 1.414213562 typedef struct hnode_s { int count; qboolean used; int children[2]; } hnode_t; typedef struct { int rate; int width; int channels; int loopstart; int samples; int dataofs; // chunk starts this many bytes from file start } wavinfo_t; // These weren`t picked out my ass.... // They were defined at http://www.rahul.net/jfm/dct.html // However, I think he plucked them out of his ass..... float Quantise[BLOCKSIZE * BLOCKSIZE]; float LUT_Quantise[BLOCKSIZE * BLOCKSIZE] = { 16.0F / 16.0F, 11.0F / 16.0F, 10.0F / 16.0F, 16.0F / 16.0F, 24.0F / 16.0F, 40.0F / 16.0F, 51.0F / 16.0F, 61.0F / 16.0F, 12.0F / 16.0F, 13.0F / 16.0F, 14.0F / 16.0F, 19.0F / 16.0F, 26.0F / 16.0F, 58.0F / 16.0F, 60.0F / 16.0F, 55.0F / 16.0F, 14.0F / 16.0F, 13.0F / 16.0F, 16.0F / 16.0F, 24.0F / 16.0F, 40.0F / 16.0F, 57.0F / 16.0F, 69.0F / 16.0F, 56.0F / 16.0F, 14.0F / 16.0F, 17.0F / 16.0F, 22.0F / 16.0F, 29.0F / 16.0F, 51.0F / 16.0F, 87.0F / 16.0F, 80.0F / 16.0F, 62.0F / 16.0F, 18.0F / 16.0F, 22.0F / 16.0F, 37.0F / 16.0F, 56.0F / 16.0F, 68.0F / 16.0F,109.0F / 16.0F,103.0F / 16.0F, 77.0F / 16.0F, 24.0F / 16.0F, 35.0F / 16.0F, 55.0F / 16.0F, 64.0F / 16.0F, 81.0F / 16.0F,104.0F / 16.0F,113.0F / 16.0F, 92.0F / 16.0F, 49.0F / 16.0F, 64.0F / 16.0F, 78.0F / 16.0F, 87.0F / 16.0F,103.0F / 16.0F,121.0F / 16.0F,120.0F / 16.0F,101.0F / 16.0F, 72.0F / 16.0F, 92.0F / 16.0F, 95.0F / 16.0F, 98.0F / 16.0F,112.0F / 16.0F,100.0F / 16.0F,103.0F / 16.0F, 99.0F / 16.0F }; int LUT_ZZ[BLOCKSIZE * BLOCKSIZE] = { 0, 1, 8, 16, 9, 2, 3, 10, 17, 24, 32, 25, 18, 11, 4, 5, 12, 19, 26, 33, 40, 48, 41, 34, 27, 20, 13, 6, 7, 14, 21, 28, 35, 42, 49, 56, 57, 50, 43, 36, 29, 22, 15, 23, 30, 37, 44, 51, 58, 59, 52, 45, 38, 31, 39, 46, 53, 60, 61, 54, 47, 55, 62, 63 }; char base[32]; byte *soundtrack; byte scaled[256][HUF_TOKENS]; unsigned int charbits1[256][HUF_TOKENS]; int charbitscount1[256][HUF_TOKENS]; hnode_t hnodes1[256][HUF_TOKENS * 2]; int numhnodes1[256]; int order0counts[256]; int numhnodes; hnode_t hnodes[512]; unsigned charbits[256]; int charbitscount[256]; CineHead_t cinehead; byte *data_p; byte *iff_end; byte *last_chunk; byte *iff_data; int iff_chunk_len; float dctbase[BLOCKSIZE][BLOCKSIZE]; float red[BLOCKSIZE * BLOCKSIZE]; float green[BLOCKSIZE * BLOCKSIZE]; float blue[BLOCKSIZE * BLOCKSIZE]; float temp[BLOCKSIZE * BLOCKSIZE]; wavinfo_t wavinfo; adpcm_t adpcm; /* =============================================================================== WAV loading =============================================================================== */ /* Intel ADPCM step variation table */ static int indexTable[16] = { -1, -1, -1, -1, 2, 4, 6, 8, -1, -1, -1, -1, 2, 4, 6, 8, }; static int stepsizeTable[89] = { 7, 8, 9, 10, 11, 12, 13, 14, 16, 17, 19, 21, 23, 25, 28, 31, 34, 37, 41, 45, 50, 55, 60, 66, 73, 80, 88, 97, 107, 118, 130, 143, 157, 173, 190, 209, 230, 253, 279, 307, 337, 371, 408, 449, 494, 544, 598, 658, 724, 796, 876, 963, 1060, 1166, 1282, 1411, 1552, 1707, 1878, 2066, 2272, 2499, 2749, 3024, 3327, 3660, 4026, 4428, 4871, 5358, 5894, 6484, 7132, 7845, 8630, 9493, 10442, 11487, 12635, 13899, 15289, 16818, 18500, 20350, 22385, 24623, 27086, 29794, 32767 }; #if 0 static void adpcm_decoder( char *indata, short *outdata, int len, adpcm_state_t *state ){ signed char *inp; /* Input buffer pointer */ short *outp; /* output buffer pointer */ int sign; /* Current adpcm sign bit */ int delta; /* Current adpcm output value */ int step; /* Stepsize */ int valpred; /* Predicted value */ int vpdiff; /* Current change to valpred */ int index; /* Current step change index */ int inputbuffer; /* place to keep next 4-bit value */ int bufferstep; /* toggle between inputbuffer/input */ outp = outdata; inp = (signed char *)indata; valpred = state->valprev; index = state->index; step = stepsizeTable[index]; bufferstep = 0; for (; len > 0; len-- ) { /* Step 1 - get the delta value */ if ( bufferstep ) { delta = inputbuffer & 0xf; } else { inputbuffer = *inp++; delta = ( inputbuffer >> 4 ) & 0xf; } bufferstep = !bufferstep; /* Step 2 - Find new index value (for later) */ index += indexTable[delta]; if ( index < 0 ) { index = 0; } if ( index > 88 ) { index = 88; } /* Step 3 - Separate sign and magnitude */ sign = delta & 8; delta = delta & 7; /* Step 4 - Compute difference and new predicted value */ /* ** Computes 'vpdiff = (delta+0.5)*step/4', but see comment ** in adpcm_coder. */ vpdiff = step >> 3; if ( delta & 4 ) { vpdiff += step; } if ( delta & 2 ) { vpdiff += step >> 1; } if ( delta & 1 ) { vpdiff += step >> 2; } if ( sign ) { valpred -= vpdiff; } else{ valpred += vpdiff; } /* Step 5 - clamp output value */ if ( valpred > 32767 ) { valpred = 32767; } else if ( valpred < -32768 ) { valpred = -32768; } /* Step 6 - Update step value */ step = stepsizeTable[index]; /* Step 7 - Output value */ *outp++ = valpred; } state->valprev = valpred; state->index = index; } #endif void adpcm_coder( short *inp, adpcm_t *adpcm ){ int val; /* Current input sample value */ int sign; /* Current adpcm sign bit */ int delta; /* Current adpcm output value */ int diff; /* Difference between val and valprev */ int step; /* Stepsize */ int valpred; /* Predicted output value */ int vpdiff; /* Current change to valpred */ int index; /* Current step change index */ int outputbuffer; /* place to keep previous 4-bit value */ int bufferstep; /* toggle between outputbuffer/output */ adpcm_state_t *state; char *outp; int len; state = &adpcm->state; len = state->count; outp = adpcm->adpcm; valpred = state->in_valprev; index = state->in_index; step = stepsizeTable[index]; bufferstep = 1; while ( len-- ) { val = *inp++; /* Step 1 - compute difference with previous value */ diff = val - valpred; sign = ( diff < 0 ) ? 8 : 0; if ( sign ) { diff = -diff; } /* Step 2 - Divide and clamp */ /* Note: ** This code *approximately* computes: ** delta = diff*4/step; ** vpdiff = (delta+0.5)*step/4; ** but in shift step bits are dropped. The net result of this is ** that even if you have fast mul/div hardware you cannot put it to ** good use since the fixup would be too expensive. */ delta = 0; vpdiff = ( step >> 3 ); if ( diff >= step ) { delta = 4; diff -= step; vpdiff += step; } step >>= 1; if ( diff >= step ) { delta |= 2; diff -= step; vpdiff += step; } step >>= 1; if ( diff >= step ) { delta |= 1; vpdiff += step; } /* Step 3 - Update previous value */ if ( sign ) { valpred -= vpdiff; } else{ valpred += vpdiff; } /* Step 4 - Clamp previous value to 16 bits */ if ( valpred > 32767 ) { valpred = 32767; } else if ( valpred < -32768 ) { valpred = -32768; } /* Step 5 - Assemble value, update index and step values */ delta |= sign; index += indexTable[delta]; if ( index < 0 ) { index = 0; } if ( index > 88 ) { index = 88; } step = stepsizeTable[index]; /* Step 6 - Output value */ if ( bufferstep ) { outputbuffer = ( delta << 4 ) & 0xf0; } else{ *outp++ = ( delta & 0x0f ) | outputbuffer; } bufferstep = !bufferstep; } /* Output last step, if needed */ if ( !bufferstep ) { *outp++ = outputbuffer; } state->out_valprev = valpred; state->out_index = index; } void FindNextChunk( char *name ){ while ( 1 ) { data_p = last_chunk; if ( data_p >= iff_end ) { // didn't find the chunk data_p = NULL; return; } data_p += 4; iff_chunk_len = *(long *)data_p; data_p += 4; if ( iff_chunk_len < 0 ) { data_p = NULL; return; } data_p -= 8; last_chunk = data_p + 8 + ( ( iff_chunk_len + 1 ) & ~1 ); if ( !strncmp( data_p, name, 4 ) ) { return; } } } void FindChunk( char *name ){ last_chunk = iff_data; FindNextChunk( name ); } void DumpChunks( void ){ char str[5]; str[4] = 0; data_p = iff_data; do { memcpy( str, data_p, 4 ); data_p += 4; iff_chunk_len = *(long *)data_p; data_p += 4; printf( "0x%x : %s (%d)\n", (int)( data_p - 4 ), str, iff_chunk_len ); data_p += ( iff_chunk_len + 1 ) & ~1; } while ( data_p < iff_end ); } /* ============ GetWavinfo ============ */ wavinfo_t GetWavinfo( char *name, byte *wav, int wavlength ){ wavinfo_t info; int i; int format; int samples; memset( &info, 0, sizeof( info ) ); if ( !wav ) { return( info ); } iff_data = wav; iff_end = wav + wavlength; // find "RIFF" chunk FindChunk( "RIFF" ); if ( !( data_p && !strncmp( data_p + 8, "WAVE", 4 ) ) ) { printf( "Missing RIFF/WAVE chunks\n" ); return( info ); } // get "fmt " chunk iff_data = data_p + 12; FindChunk( "fmt " ); if ( !data_p ) { printf( "Missing fmt chunk\n" ); return( info ); } data_p += 8; format = *(short *)data_p; data_p += 2; if ( format != 1 ) { printf( "Microsoft PCM format only\n" ); return( info ); } info.channels = *(short *)data_p; data_p += 2; info.rate = *(long *)data_p; data_p += 4; data_p += 6; info.width = *(short *)data_p / 8; data_p += 2; // get cue chunk FindChunk( "cue " ); if ( data_p ) { data_p += 32; info.loopstart = *(long *)data_p; data_p += 4; // if the next chunk is a LIST chunk, look for a cue length marker FindNextChunk( "LIST" ); if ( data_p ) { // this is not a proper parse, but it works with cooledit... if ( !strncmp( data_p + 28, "mark", 4 ) ) { data_p += 24; i = *(long *)data_p; // samples in loop data_p += 4; info.samples = info.loopstart + i; } } } else{ info.loopstart = -1; } // find data chunk FindChunk( "data" ); if ( !data_p ) { printf( "Missing data chunk\n" ); return( info ); } data_p += 4; samples = *(long *)data_p; data_p += 4; if ( info.samples ) { if ( samples < info.samples ) { Error( "Sound %s has a bad loop length", name ); } } else{ info.samples = samples; } info.dataofs = data_p - wav; return( info ); } // ============== // LoadSoundtrack // ============== void LoadSoundtrack(){ char name[1024]; FILE *f; int len; soundtrack = NULL; sprintf( name, "%svideo/%s/%s.wav", gamedir, base, base ); printf( "\nLoading sound : %s\n", name ); f = fopen( name, "rb" ); if ( !f ) { printf( "\nNo soundtrack for %s\n", base ); return; } len = Q_filelength( f ); soundtrack = SafeMalloc( len, "LoadSoundtrack" ); fread( soundtrack, 1, len, f ); fclose( f ); wavinfo = GetWavinfo( name, soundtrack, len ); adpcm.state.out_valprev = 0; adpcm.state.out_index = 0; } // ================== // WriteSound // ================== int WriteSound( FILE *output, int frame, int numframes ){ int start, end; int count; int empty = 0; int width; char *work; width = wavinfo.width * wavinfo.channels; start = ( ( frame * wavinfo.rate / 14 ) + 31 ) & 0xffffffe0; // start sample end = ( ( ( frame + numframes ) * wavinfo.rate / 14 ) + 31 ) & 0xffffffe0; // end sample count = end - start; work = soundtrack + wavinfo.dataofs + ( start * width ); adpcm.state.count = count * wavinfo.channels; // Number of samples adpcm.state.in_valprev = adpcm.state.out_valprev; adpcm.state.in_index = adpcm.state.out_index; adpcm_coder( (short *)work, &adpcm ); WriteHeader( output, FC_SOUND_22KMADPCM, FC_ADPCM_VERSION, ( adpcm.state.count / 2 ) + sizeof( adpcm_state_t ), (char *)&adpcm ); return( count / 2 ); } // ============================== // Basic run length encoder // ============================== char *RLEZZ( char *in, char *out ){ int srun; char count; int idx = 0; while ( idx < 64 ) { srun = idx; // Start of run while ( idx < 63 ) { if ( in[LUT_ZZ[idx]] != in[LUT_ZZ[idx + 1]] ) { break; } idx++; } count = (char)( idx - srun ); // count of repeated bytes if ( !count ) { while ( idx < 63 ) { if ( in[LUT_ZZ[idx]] == in[LUT_ZZ[idx + 1]] ) { break; } idx++; } if ( idx == 63 ) { idx++; } count = (char)( idx - srun ); // count of unique bytes *out++ = count; while ( count-- ) *out++ = in[LUT_ZZ[srun++]]; } else { *out++ = -( count + 1 ); *out++ = in[LUT_ZZ[idx]]; idx++; } } return( out ); } // ============================== // Discrete Cosine Transformation // ============================== void init_base( float quant ){ int y, x; for ( y = 0; y < BLOCKSIZE; y++ ) for ( x = 0; x < BLOCKSIZE; x++ ) { if ( y == 0 ) { dctbase[y][x] = 1; } else{ dctbase[y][x] = SQRT2 * cos( ( ( x * 2 + 1 ) * y * M_PI ) / ( BLOCKSIZE * 2 ) ); } } for ( y = 0; y < BLOCKSIZE * BLOCKSIZE; y++ ) Quantise[y] = LUT_Quantise[y] / quant; } void SplitComponents( byte *src, int width, int height ){ int i, j; float *tr = red; float *tg = green; float *tb = blue; for ( i = 0; i < BLOCKSIZE; i++, src += ( width - BLOCKSIZE ) * 4 ) for ( j = 0; j < BLOCKSIZE; j++ ) { *tr++ = ( (float)*src++ ) - 128.0F; *tg++ = ( (float)*src++ ) - 128.0F; *tb++ = ( (float)*src++ ) - 128.0F; src++; } } void transferH( float *src, float *dst ){ int y, dx, dy; float sum; float *work; for ( y = 0; y < BLOCKSIZE; y++, src += BLOCKSIZE ) { for ( dy = 0; dy < BLOCKSIZE; dy++ ) { sum = 0; work = src; for ( dx = 0; dx < BLOCKSIZE; dx++, work++ ) sum += dctbase[dy][dx] * *work; *dst++ = sum / BLOCKSIZE; } } } void transferV( float *src, float *dst ){ int x, dy, fy; float sum; float *work; for ( x = 0; x < BLOCKSIZE; x++, src++, dst++ ) { for ( fy = 0; fy < BLOCKSIZE; fy++ ) { sum = 0; work = src; for ( dy = 0; dy < BLOCKSIZE; dy++, work += BLOCKSIZE ) sum += dctbase[fy][dy] * *work; dst[fy * BLOCKSIZE] = sum / BLOCKSIZE; } } } char *Combine( byte *dst, float *p, float *q ){ int i, j; byte rlesrc[BLOCKSIZE * BLOCKSIZE]; int c; byte *work; work = rlesrc; for ( j = 0; j < BLOCKSIZE; j++ ) for ( i = 0; i < BLOCKSIZE; i++ ) { c = (int)( ( *p++ / *q++ ) + 128.5F ); c -= 128; if ( c < -128 ) { c = -128; } if ( c > 127 ) { c = 127; } *work++ = (char)c; } dst = RLEZZ( rlesrc, dst ); return( dst ); } char *CombineComponents( char *dst, int width, int height ){ dst = Combine( dst, red, Quantise ); dst = Combine( dst, green, Quantise ); dst = Combine( dst, blue, Quantise ); return( dst ); } void DCT( cblock_t *out, cblock_t in, int width, int height ){ int x, y; char *cursrc; char *curdst; curdst = out->data; for ( y = 0; y < height; y += BLOCKSIZE ) for ( x = 0; x < width; x += BLOCKSIZE ) { cursrc = in.data + ( ( y * width ) + x ) * 4; SplitComponents( cursrc, width, height ); transferH( red, temp ); transferV( temp, red ); transferH( green, temp ); transferV( temp, green ); transferH( blue, temp ); transferV( temp, blue ); curdst = CombineComponents( curdst, width, height ); } out->count = curdst - out->data; } // ================== // BuildChars1 // ================== void BuildChars1( int prev, int nodenum, unsigned bits, int bitcount ){ hnode_t *node; if ( nodenum < HUF_TOKENS ) { if ( bitcount > 32 ) { Error( "bitcount > 32" ); } charbits1[prev][nodenum] = bits; charbitscount1[prev][nodenum] = bitcount; return; } node = &hnodes1[prev][nodenum]; bits <<= 1; BuildChars1( prev, node->children[0], bits, bitcount + 1 ); bits |= 1; BuildChars1( prev, node->children[1], bits, bitcount + 1 ); } // ================== // SmallestNode1 // ================== int SmallestNode1( hnode_t *hnodes, int numhnodes ){ int i; int best, bestnode; best = 99999999; bestnode = -1; for ( i = 0; i < numhnodes; i++ ) { if ( hnodes[i].used ) { continue; } if ( !hnodes[i].count ) { continue; } if ( hnodes[i].count < best ) { best = hnodes[i].count; bestnode = i; } } if ( bestnode == -1 ) { return( -1 ); } hnodes[bestnode].used = true; return( bestnode ); } // ================== // BuildTree1 // ================== void BuildTree1( int prev ){ hnode_t *node, *nodebase; int numhnodes; // build the nodes numhnodes = HUF_TOKENS; nodebase = hnodes1[prev]; while ( 1 ) { node = &nodebase[numhnodes]; // pick two lowest counts node->children[0] = SmallestNode1( nodebase, numhnodes ); if ( node->children[0] == -1 ) { break; // no more } node->children[1] = SmallestNode1( nodebase, numhnodes ); if ( node->children[1] == -1 ) { break; } node->count = nodebase[node->children[0]].count + nodebase[node->children[1]].count; numhnodes++; } numhnodes1[prev] = numhnodes - 1; BuildChars1( prev, numhnodes - 1, 0, 0 ); } // ================== // Huffman1_Count // ================== void Huffman1_Count( cblock_t in ){ int i; int prev; int v; int rept; prev = 0; for ( i = 0; i < in.count; i++ ) { v = in.data[i]; order0counts[v]++; hnodes1[prev][v].count++; prev = v; for ( rept = 1; ( i + rept < in.count ) && ( rept < MAX_REPT ); rept++ ) if ( in.data[i + rept] != v ) { break; } if ( rept > MIN_REPT ) { hnodes1[prev][255 + rept].count++; i += rept - 1; } } } // ================== // Huffman1_Build // ================== void Huffman1_Build(){ int i, j, v; int max; int total; for ( i = 0; i < 256; i++ ) { // normalize and save the counts max = 0; for ( j = 0; j < HUF_TOKENS; j++ ) { if ( hnodes1[i][j].count > max ) { max = hnodes1[i][j].count; } } if ( max == 0 ) { max = 1; } total = 0; // easy to overflow 32 bits here! for ( j = 0; j < HUF_TOKENS; j++ ) { v = ( hnodes1[i][j].count * (double) 255 + max - 1 ) / max; if ( v > 255 ) { Error( "v > 255" ); } scaled[i][j] = hnodes1[i][j].count = v; if ( v ) { total++; } } if ( total == 1 ) { // must have two tokens if ( !scaled[i][0] ) { scaled[i][0] = hnodes1[i][0].count = 1; } else{ scaled[i][1] = hnodes1[i][1].count = 1; } } BuildTree1( i ); } } // ================== // Huffman1 // Order 1 compression with pre-built table // ================== cblock_t Huffman1( cblock_t in ){ int i; int outbits, c; unsigned bits; byte *out_p; cblock_t out; int prev; int v; int rept; out_p = out.data = SafeMalloc( ( in.count * 2 ) + 1024 + 4, "Huffman" ); memset( out_p, 0, ( in.count * 2 ) + 1024 + 4 ); // leave space for compressed count out_p += 4; // write count *(long *)out_p = in.count; out_p += 4; // write bits outbits = 0; prev = 0; for ( i = 0; i < in.count; i++ ) { v = in.data[i]; c = charbitscount1[prev][v]; bits = charbits1[prev][v]; if ( !c ) { Error( "!bits" ); } while ( c ) { c--; if ( bits & ( 1 << c ) ) { out_p[outbits >> 3] |= 1 << ( outbits & 7 ); } outbits++; } prev = v; // check for repeat encodes for ( rept = 1; ( i + rept < in.count ) && ( rept < MAX_REPT ); rept++ ) if ( in.data[i + rept] != v ) { break; } if ( rept > MIN_REPT ) { c = charbitscount1[prev][255 + rept]; bits = charbits1[prev][255 + rept]; if ( !c ) { Error( "!bits" ); } while ( c ) { c--; if ( bits & ( 1 << c ) ) { out_p[outbits >> 3] |= 1 << ( outbits & 7 ); } outbits++; } i += rept - 1; } } out_p += ( outbits + 7 ) >> 3; out.count = out_p - out.data; out_p = out.data; *(long *)out_p = out.count; return( out ); } // =================== // LoadFrame // =================== void LoadFrame( cblock_t *out, char *base, int frame ){ cblock_t in; int width, height; char name[1024]; FILE *f; in.data = NULL; in.count = -1; sprintf( name, "%svideo/%s/%s%04i.tga", gamedir, base, base, frame ); f = fopen( name, "rb" ); if ( !f ) { out->data = NULL; return; } fclose( f ); LoadTGA( name, &in.data, &width, &height ); if ( ( width != cinehead.Width ) || ( height != cinehead.Height ) ) { free( in.data ); printf( "Invalid picture size\n" ); out->data = NULL; return; } out->data = SafeMalloc( width * height * 3, "LoadFrame" ); // rle could possibly expand file so this not 100% safe (however DCT should force a lot of compression) DCT( out, in, width, height ); free( in.data ); } // ================================== // Cmd_Video // // video // ================================== void Cmd_Video(){ char savename[256]; char name[256]; FILE *output; int frame; int width, height; cblock_t in, huffman; int size; float dctconst; int maxsize, ssize; int min_rle_size, warnings; int ave_image, ave_sound; GetScriptToken( false ); strcpy( base, token ); if ( g_release ) { return; } GetScriptToken( false ); dctconst = atof( token ); GetScriptToken( false ); maxsize = atoi( token ); sprintf( savename, "%svideo/%s.cin", gamedir, base ); // clear stuff memset( charbits1, 0, sizeof( charbits1 ) ); memset( charbitscount1, 0, sizeof( charbitscount1 ) ); memset( hnodes1, 0, sizeof( hnodes1 ) ); memset( numhnodes1, 0, sizeof( numhnodes1 ) ); memset( order0counts, 0, sizeof( order0counts ) ); // load the entire sound wav file if present LoadSoundtrack(); cinehead.SndRate = wavinfo.rate; cinehead.SndWidth = wavinfo.width; cinehead.SndChannels = wavinfo.channels; sprintf( name, "%svideo/%s/%s0000.tga", gamedir, base, base ); printf( "Loading sequence : %s\n", name ); printf( "DCT constant : %f\n", dctconst ); LoadTGA( name, NULL, &width, &height ); output = fopen( savename, "wb" ); if ( !output ) { Error( "Can't open %s", savename ); } if ( ( width % BLOCKSIZE ) || ( height % BLOCKSIZE ) ) { Error( "Width and height must be a multiple of %d", BLOCKSIZE ); } cinehead.Width = width; cinehead.Height = height; init_base( dctconst ); // build the dictionary printf( "Counting : " ); min_rle_size = 0; for ( frame = 0; ; frame++ ) { printf( "." ); LoadFrame( &in, base, frame ); if ( !in.data ) { break; } Huffman1_Count( in ); if ( in.count > min_rle_size ) { min_rle_size = in.count; } free( in.data ); } printf( "\n" ); cinehead.NumFrames = frame; printf( "Num Frames : %d\n", frame ); cinehead.MaxRleSize = ( min_rle_size + 0x1f ) & 0xfffffe0; cinehead.MaxSndSize = ( ( 4 * wavinfo.rate * wavinfo.channels / 14 ) + 0x1f ) & 0xffffffe0; WriteHeader( output, FC_HEADER_NAME, FC_HEADER_VERSION, sizeof( CineHead_t ), &cinehead ); // build nodes and write counts Huffman1_Build(); WriteHeader( output, FC_HUFFBITS_NAME, FC_HUFFBITS_VERSION, sizeof( scaled ), scaled ); WriteHeader( output, FC_QUANT_NAME, FC_QUANT_VERSION, sizeof( Quantise ), Quantise ); ave_image = 0; ave_sound = 0; warnings = 0; // compress it with the dictionary if ( soundtrack ) { ssize = WriteSound( output, frame, 4 ); ave_sound += ssize; } for ( frame = 0; frame < cinehead.NumFrames; frame++ ) { // save some sound samples printf( "Packing : ", frame ); LoadFrame( &in, base, frame ); // save the image huffman = Huffman1( in ); printf( "%d bytes rle, %d bytes huffman", in.count, huffman.count ); size = ( huffman.count + 3 ) & 0xfffffffc; // round up to longwords if ( size > maxsize ) { printf( " ** WARNING **" ); warnings++; } printf( "\n" ); ave_image += huffman.count; WriteHeader( output, FC_IMAGE_NAME, FC_IMAGE_VERSION, size, huffman.data ); if ( soundtrack ) { ssize = WriteSound( output, frame + 4, 1 ); ave_sound += ssize; } free( in.data ); free( huffman.data ); } printf( "\nTotal size: %d (headers + %d image + %d sound)\n", ftell( output ), ave_image, ave_sound ); printf( "Data rate : %d bytes per sec (image and sound)\n", ( ave_image + ave_sound ) / cinehead.NumFrames ); printf( "Cin created ok with %d warnings.\n", warnings ); fclose( output ); if ( soundtrack ) { free( soundtrack ); } } #endif void Cmd_Video(){ } // end