yquake2remaster/src/client/sound/snd_vorbis.c
2011-06-27 14:40:16 +00:00

789 lines
15 KiB
C

/*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
* USA.
*
* =======================================================================
*
* This file implements an interface to libvorbis for decoding
* OGG/Vorbis files. Strongly spoken this file isn't part of the sound
* system but part of the main client. It justs converts Vorbis streams
* into normal, raw Wave stream which are injected into snd_mem.c as if
* they were normal wave files. At this moment only background music
* playback and in theory .cin movie file playback is supported.
*
* =======================================================================
*/
#include <sys/time.h>
#include <errno.h>
#define OV_EXCLUDE_STATIC_CALLBACKS
#include <vorbis/vorbisfile.h>
#include "../header/client.h"
#include "header/local.h"
#include "header/vorbis.h"
extern int sound_started; /* Sound initialization flag. */
extern cvar_t *fs_basedir; /* Path to "music". */
qboolean ogg_first_init = true; /* First initialization flag. */
qboolean ogg_started = false; /* Initialization flag. */
int ogg_bigendian = 0;
byte *ogg_buffer; /* File buffer. */
char **ogg_filelist; /* List of Ogg Vorbis files. */
char ovBuf [ 4096 ]; /* Buffer for sound. */
int ogg_curfile; /* Index of currently played file. */
int ogg_numfiles; /* Number of Ogg Vorbis files. */
int ovSection; /* Position in Ogg Vorbis file. */
ogg_status_t ogg_status; /* Status indicator. */
cvar_t *ogg_autoplay; /* Play this song when started. */
cvar_t *ogg_check; /* Check Ogg files or not. */
cvar_t *ogg_playlist; /* Playlist. */
cvar_t *ogg_sequence; /* Sequence play indicator. */
cvar_t *ogg_volume; /* Music volume. */
OggVorbis_File ovFile; /* Ogg Vorbis file. */
vorbis_info *ogg_info; /* Ogg Vorbis file information */
/*
* Initialize the Ogg Vorbis subsystem.
*/
void
OGG_Init ( void )
{
cvar_t *cv; /* Cvar pointer. */
if ( ogg_started )
{
return;
}
Com_Printf( "Starting Ogg Vorbis.\n" );
/* Skip initialization if disabled. */
cv = Cvar_Get( "ogg_enable", "0", CVAR_ARCHIVE );
if ( cv->value != 1 )
{
Com_Printf( "Ogg Vorbis not initializing.\n" );
return;
}
if (bigendien == true)
{
ogg_bigendian = 1;
}
/* Cvars. */
ogg_autoplay = Cvar_Get( "ogg_autoplay", "?", CVAR_ARCHIVE );
ogg_check = Cvar_Get( "ogg_check", "0", CVAR_ARCHIVE );
ogg_playlist = Cvar_Get( "ogg_playlist", "playlist", CVAR_ARCHIVE );
ogg_sequence = Cvar_Get( "ogg_sequence", "loop", CVAR_ARCHIVE );
ogg_volume = Cvar_Get( "ogg_volume", "0.7", CVAR_ARCHIVE );
/* Console commands. */
Cmd_AddCommand( "ogg_list", OGG_ListCmd );
Cmd_AddCommand( "ogg_pause", OGG_PauseCmd );
Cmd_AddCommand( "ogg_play", OGG_PlayCmd );
Cmd_AddCommand( "ogg_reinit", OGG_Reinit );
Cmd_AddCommand( "ogg_resume", OGG_ResumeCmd );
Cmd_AddCommand( "ogg_seek", OGG_SeekCmd );
Cmd_AddCommand( "ogg_status", OGG_StatusCmd );
Cmd_AddCommand( "ogg_stop", OGG_Stop );
/* Build list of files. */
ogg_numfiles = 0;
if ( ogg_playlist->string [ 0 ] != '\0' )
{
OGG_LoadPlaylist( ogg_playlist->string );
}
if ( ogg_numfiles == 0 )
{
OGG_LoadFileList();
}
/* Check if we have Ogg Vorbis files. */
if ( ogg_numfiles <= 0 )
{
Com_Printf( "No Ogg Vorbis files found.\n" );
ogg_started = true; /* For OGG_Shutdown(). */
OGG_Shutdown();
return;
}
/* Initialize variables. */
if ( ogg_first_init )
{
srand( time( NULL ) );
ogg_buffer = NULL;
ogg_curfile = -1;
ogg_info = NULL;
ogg_status = STOP;
ogg_first_init = false;
}
ogg_started = true;
Com_Printf( "%d Ogg Vorbis files found.\n", ogg_numfiles );
/* Autoplay support. */
if ( ogg_autoplay->string [ 0 ] != '\0' )
{
OGG_ParseCmd( ogg_autoplay->string );
}
}
/*
* Shutdown the Ogg Vorbis subsystem.
*/
void
OGG_Shutdown ( void )
{
if ( !ogg_started )
{
return;
}
Com_Printf( "Shutting down Ogg Vorbis.\n" );
OGG_Stop();
/* Free the list of files. */
FS_FreeList( ogg_filelist, ogg_numfiles + 1 );
/* Remove console commands. */
Cmd_RemoveCommand( "ogg_list" );
Cmd_RemoveCommand( "ogg_pause" );
Cmd_RemoveCommand( "ogg_play" );
Cmd_RemoveCommand( "ogg_reinit" );
Cmd_RemoveCommand( "ogg_resume" );
Cmd_RemoveCommand( "ogg_seek" );
Cmd_RemoveCommand( "ogg_status" );
Cmd_RemoveCommand( "ogg_stop" );
ogg_started = false;
}
/*
* Reinitialize the Ogg Vorbis subsystem.
*/
void
OGG_Reinit ( void )
{
OGG_Shutdown();
OGG_Init();
}
/*
* Check if the file is a valid Ogg Vorbis file.
*/
qboolean
OGG_Check ( char *name )
{
qboolean res; /* Return value. */
byte *buffer; /* File buffer. */
int size; /* File size. */
OggVorbis_File ovf; /* Ogg Vorbis file. */
if ( ogg_check->value == 0 )
{
return ( true );
}
res = false;
if ( ( size = FS_LoadFile( name, (void **) &buffer ) ) > 0 )
{
if ( ov_test( NULL, &ovf, (char *) buffer, size ) == 0 )
{
res = true;
ov_clear( &ovf );
}
FS_FreeFile( buffer );
}
return ( res );
}
/*
* Change position in the file.
*/
void
OGG_Seek ( ogg_seek_t type, double offset )
{
double pos; /* Position in file (in seconds). */
double total; /* Length of file (in seconds). */
/* Check if the file is seekable. */
if ( ov_seekable( &ovFile ) == 0 )
{
Com_Printf( "OGG_Seek: file is not seekable.\n" );
return;
}
/* Get file information. */
pos = ov_time_tell( &ovFile );
total = ov_time_total( &ovFile, -1 );
switch ( type )
{
case ABS:
if ( ( offset >= 0 ) && ( offset <= total ) )
{
if ( ov_time_seek( &ovFile, offset ) != 0 )
{
Com_Printf( "OGG_Seek: could not seek.\n" );
}
else
{
Com_Printf( "%0.2f -> %0.2f of %0.2f.\n", pos, offset, total );
}
}
else
{
Com_Printf( "OGG_Seek: invalid offset.\n" );
}
break;
case REL:
if ( ( pos + offset >= 0 ) && ( pos + offset <= total ) )
{
if ( ov_time_seek( &ovFile, pos + offset ) != 0 )
{
Com_Printf( "OGG_Seek: could not seek.\n" );
}
else
{
Com_Printf( "%0.2f -> %0.2f of %0.2f.\n", pos, pos + offset, total );
}
}
else
{
Com_Printf( "OGG_Seek: invalid offset.\n" );
}
break;
}
}
/*
* Load list of Ogg Vorbis files in "music".
*/
void
OGG_LoadFileList ( void )
{
char **list; /* List of .ogg files. */
int i; /* Loop counter. */
int j; /* Real position in list. */
/* Get file list. */
list = FS_ListFiles2( va( "%s/*.ogg", OGG_DIR ), &ogg_numfiles, 0, SFF_SUBDIR | SFF_HIDDEN | SFF_SYSTEM );
ogg_numfiles--;
/* Check if there are posible Ogg files. */
if ( list == NULL )
{
return;
}
/* Allocate list of files. */
ogg_filelist = malloc( sizeof ( char * ) * ogg_numfiles );
/* Add valid Ogg Vorbis file to the list. */
for ( i = 0, j = 0; i < ogg_numfiles; i++ )
{
if ( !OGG_Check( list [ i ] ) )
{
free( list [ i ] );
continue;
}
ogg_filelist [ j++ ] = list [ i ];
}
/* Free the file list. */
free( list );
/* Adjust the list size (remove space for invalid music files). */
ogg_numfiles = j;
ogg_filelist = realloc( ogg_filelist, sizeof ( char * ) * ogg_numfiles );
}
/*
* Load playlist.
*/
void
OGG_LoadPlaylist ( char *playlist )
{
byte *buffer; /* Buffer to read the file. */
char *ptr; /* Pointer for parsing the file. */
int i; /* Loop counter. */
int size; /* Length of buffer and strings. */
/* Open playlist. */
if ( ( size = FS_LoadFile( va( "%s/%s.lst", OGG_DIR, ogg_playlist->string ), (void **) &buffer ) ) < 0 )
{
Com_Printf( "OGG_LoadPlaylist: could not open playlist: %s.\n", strerror( errno ) );
return;
}
/* Count the files in playlist. */
for ( ptr = strsep( (char **) &buffer, "\n" ); ptr != NULL; ptr = strsep( NULL, "\n" ) )
{
if ( (byte *) ptr != buffer )
{
ptr [ -1 ] = '\n';
}
if ( OGG_Check( va( "%s/%s", OGG_DIR, ptr ) ) )
{
ogg_numfiles++;
}
}
/* Allocate file list. */
ogg_filelist = malloc( sizeof ( char * ) * ogg_numfiles );
i = 0;
for ( ptr = strsep( (char **) &buffer, "\n" );
ptr != NULL;
ptr = strsep( NULL, "\n" ) )
{
if ( OGG_Check( va( "%s/%s", OGG_DIR, ptr ) ) )
{
ogg_filelist [ i++ ] = strdup( va( "%s/%s", OGG_DIR, ptr ) );
}
}
/* Free file buffer. */
FS_FreeFile( buffer );
}
/*
* Play Ogg Vorbis file (with absolute or relative index).
*/
qboolean
OGG_Open ( ogg_seek_t type, int offset )
{
int size; /* File size. */
int pos; /* Absolute position. */
int res; /* Error indicator. */
pos = -1;
switch ( type )
{
case ABS:
/* Absolute index. */
if ( ( offset < 0 ) || ( offset >= ogg_numfiles ) )
{
Com_Printf( "OGG_Open: %d out of range.\n", offset + 1 );
return ( false );
}
else
{
pos = offset;
}
break;
case REL:
/* Simulate a loopback. */
if ( ( ogg_curfile == -1 ) && ( offset < 0 ) )
{
offset++;
}
while ( ogg_curfile + offset < 0 )
{
offset += ogg_numfiles;
}
while ( ogg_curfile + offset >= ogg_numfiles )
{
offset -= ogg_numfiles;
}
pos = ogg_curfile + offset;
break;
}
/* Check running music. */
if ( ogg_status == PLAY )
{
if ( ogg_curfile == pos )
{
return ( true );
}
else
{
OGG_Stop();
}
}
/* Find file. */
if ( ( size = FS_LoadFile( ogg_filelist [ pos ], (void **) &ogg_buffer ) ) == -1 )
{
Com_Printf( "OGG_Open: could not open %d (%s): %s.\n", pos, ogg_filelist [ pos ], strerror( errno ) );
return ( false );
}
/* Open ogg vorbis file. */
if ( ( res = ov_open( NULL, &ovFile, (char *) ogg_buffer, size ) ) < 0 )
{
Com_Printf( "OGG_Open: '%s' is not a valid Ogg Vorbis file (error %i).\n", ogg_filelist [ pos ], res );
FS_FreeFile( ogg_buffer );
ogg_buffer = NULL;
return ( false );
}
ogg_info = ov_info(&ovFile, 0);
if (!ogg_info)
{
Com_Printf( "OGG_Open: Unable to get stream information for %s.\n", ogg_filelist [ pos ] );
ov_clear( &ovFile );
FS_FreeFile( ogg_buffer );
ogg_buffer = NULL;
return ( false );
}
/* Play file. */
ovSection = 0;
ogg_curfile = pos;
ogg_status = PLAY;
return ( true );
}
/*
* Play Ogg Vorbis file (with name only).
*/
qboolean
OGG_OpenName ( char *filename )
{
char *name; /* File name. */
int i; /* Loop counter. */
/* If the track name is '00' stop playback */
if (!strncmp(filename, "00", sizeof(char) * 3))
{
OGG_PauseCmd();
return ( false );
}
name = va( "%s/%s.ogg", OGG_DIR, filename );
for ( i = 0; i < ogg_numfiles; i++ )
{
if ( strcmp( name, ogg_filelist [ i ] ) == 0 )
{
break;
}
}
if ( i < ogg_numfiles )
{
return ( OGG_Open( ABS, i ) );
}
else
{
Com_Printf( "OGG_OpenName: '%s' not in the list.\n", filename );
return ( false );
}
}
/*
* Play a portion of the currently opened file.
*/
int
OGG_Read ( void )
{
int res; /* Number of bytes read. */
/* Read and resample. */
res = ov_read( &ovFile, ovBuf, sizeof ( ovBuf ), ogg_bigendian, OGG_SAMPLEWIDTH, 1, &ovSection );
S_RawSamples( res / (OGG_SAMPLEWIDTH * ogg_info->channels),
ogg_info->rate, OGG_SAMPLEWIDTH, ogg_info->channels, (byte *) ovBuf, ogg_volume->value );
/* Check for end of file. */
if ( res == 0 )
{
OGG_Stop();
OGG_Sequence();
}
return ( res );
}
/*
* Play files in sequence.
*/
void
OGG_Sequence ( void )
{
if ( strcmp( ogg_sequence->string, "next" ) == 0 )
{
OGG_Open( REL, 1 );
}
else if ( strcmp( ogg_sequence->string, "prev" ) == 0 )
{
OGG_Open( REL, -1 );
}
else if ( strcmp( ogg_sequence->string, "random" ) == 0 )
{
OGG_Open( ABS, rand() % ogg_numfiles );
}
else if ( strcmp( ogg_sequence->string, "loop" ) == 0 )
{
OGG_Open( REL, 0 );
}
else if ( strcmp( ogg_sequence->string, "none" ) != 0 )
{
Com_Printf( "Invalid value of ogg_sequence: %s\n", ogg_sequence->string );
Cvar_Set( "ogg_sequence", "none" );
}
}
/*
* Stop playing the current file.
*/
void
OGG_Stop ( void )
{
if ( ogg_status == STOP )
{
return;
}
ov_clear( &ovFile );
ogg_status = STOP;
ogg_info = NULL;
if ( ogg_buffer != NULL )
{
FS_FreeFile( ogg_buffer );
ogg_buffer = NULL;
}
}
/*
* Stream music.
*/
void
OGG_Stream ( void )
{
if ( !ogg_started )
{
return;
}
while ( ogg_status == PLAY && paintedtime + MAX_RAW_SAMPLES - 2048 > s_rawend )
{
OGG_Read();
}
}
/*
* List Ogg Vorbis files.
*/
void
OGG_ListCmd ( void )
{
int i;
for ( i = 0; i < ogg_numfiles; i++ )
{
Com_Printf( "%d %s\n", i + 1, ogg_filelist [ i ] );
}
Com_Printf( "%d Ogg Vorbis files.\n", ogg_numfiles );
}
/*
* Parse play controls.
*/
void
OGG_ParseCmd ( char *arg )
{
int n;
cvar_t *ogg_enable;
ogg_enable = Cvar_Get( "ogg_enable", "0", CVAR_ARCHIVE );
switch ( arg [ 0 ] )
{
case '#':
n = atoi( arg + 1 ) - 1;
OGG_Open( ABS, n );
break;
case '?':
OGG_Open( ABS, rand() % ogg_numfiles );
break;
case '>':
if ( strlen( arg ) > 1 )
{
OGG_Open( REL, atoi( arg + 1 ) );
}
else
{
OGG_Open( REL, 1 );
}
break;
case '<':
if ( strlen( arg ) > 1 )
{
OGG_Open( REL, -atoi( arg + 1 ) );
}
else
{
OGG_Open( REL, -1 );
}
break;
default:
if ( ogg_enable->value != 0 )
{
OGG_OpenName( arg );
}
break;
}
}
/*
* Pause current song.
*/
void
OGG_PauseCmd ( void )
{
if ( ogg_status == PLAY )
{
ogg_status = PAUSE;
}
}
/*
* Play control.
*/
void
OGG_PlayCmd ( void )
{
if ( Cmd_Argc() < 2 )
{
Com_Printf( "Usage: ogg_play {filename | #n | ? | >n | <n}\n" );
return;
}
OGG_ParseCmd( Cmd_Argv( 1 ) );
}
/*
* Resume current song.
*/
void
OGG_ResumeCmd ( void )
{
if ( ogg_status == PAUSE )
{
ogg_status = PLAY;
}
}
/*
* Change position in the file being played.
*/
void
OGG_SeekCmd ( void )
{
if ( ogg_status != STOP )
{
return;
}
if ( Cmd_Argc() < 2 )
{
Com_Printf( "Usage: ogg_seek {n | <n | >n}\n" );
return;
}
switch ( Cmd_Argv( 1 ) [ 0 ] )
{
case '>':
OGG_Seek( REL, atof( Cmd_Argv( 1 ) + 1 ) );
break;
case '<':
OGG_Seek( REL, -atof( Cmd_Argv( 1 ) + 1 ) );
break;
default:
OGG_Seek( ABS, atof( Cmd_Argv( 1 ) ) );
break;
}
}
/*
* Display status.
*/
void
OGG_StatusCmd ( void )
{
switch ( ogg_status )
{
case PLAY:
Com_Printf( "Playing file %d (%s) at %0.2f seconds.\n",
ogg_curfile + 1, ogg_filelist [ ogg_curfile ], ov_time_tell( &ovFile ) );
break;
case PAUSE:
Com_Printf( "Paused file %d (%s) at %0.2f seconds.\n",
ogg_curfile + 1, ogg_filelist [ ogg_curfile ], ov_time_tell( &ovFile ) );
break;
case STOP:
if ( ogg_curfile == -1 )
{
Com_Printf( "Stopped.\n" );
}
else
{
Com_Printf( "Stopped file %d (%s).\n",
ogg_curfile + 1, ogg_filelist [ ogg_curfile ] );
}
break;
}
}