yquake2remaster/src/client/sound/ogg.c

866 lines
16 KiB
C
Raw Normal View History

2009-10-03 16:08:34 +00:00
/*
* 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 the backends as
* if they were normal "raw" samples. At this moment only background
* music playback and in theory .cin movie file playback is supported.
*
* =======================================================================
*/
2009-10-03 16:08:34 +00:00
2012-04-25 08:24:38 +00:00
#ifdef OGG
#define OV_EXCLUDE_STATIC_CALLBACKS
#ifndef _WIN32
2009-10-03 16:08:34 +00:00
#include <sys/time.h>
#endif
2009-10-03 16:08:34 +00:00
#include <errno.h>
#include <vorbis/vorbisfile.h>
#include "../header/client.h"
2010-06-19 19:10:31 +00:00
#include "header/local.h"
#include "header/vorbis.h"
2009-10-03 16:08:34 +00:00
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. */
cvar_t *ogg_ignoretrack0; /* Toggle track 0 playing */
OggVorbis_File ovFile; /* Ogg Vorbis file. */
vorbis_info *ogg_info; /* Ogg Vorbis file information */
int ogg_numbufs; /* Number of buffers for OpenAL */
2009-10-03 16:08:34 +00:00
/*
* Initialize the Ogg Vorbis subsystem.
*/
void
OGG_Init(void)
{
cvar_t *cv; /* Cvar pointer. */
2009-10-03 16:08:34 +00:00
if (ogg_started)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
Com_Printf("Starting Ogg Vorbis.\n");
2009-10-03 16:08:34 +00:00
/* Skip initialization if disabled. */
cv = Cvar_Get("ogg_enable", "1", CVAR_ARCHIVE);
if (cv->value != 1)
{
Com_Printf("Ogg Vorbis not initializing.\n");
2009-10-03 16:08:34 +00:00
return;
}
if (bigendien == true)
{
ogg_bigendian = 1;
}
2009-10-03 16:08:34 +00:00
/* 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);
ogg_ignoretrack0 = Cvar_Get("ogg_ignoretrack0", "0", CVAR_ARCHIVE);
2009-10-03 16:08:34 +00:00
/* 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);
2009-10-03 16:08:34 +00:00
/* Build list of files. */
ogg_numfiles = 0;
if (ogg_playlist->string[0] != '\0')
{
OGG_LoadPlaylist(ogg_playlist->string);
}
if (ogg_numfiles == 0)
{
2009-10-03 16:08:34 +00:00
OGG_LoadFileList();
}
2009-10-03 16:08:34 +00:00
/* 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(). */
2009-10-03 16:08:34 +00:00
OGG_Shutdown();
return;
}
/* Initialize variables. */
if (ogg_first_init)
{
2009-10-03 16:08:34 +00:00
ogg_buffer = NULL;
ogg_curfile = -1;
ogg_info = NULL;
2009-10-03 16:08:34 +00:00
ogg_status = STOP;
ogg_first_init = false;
}
ogg_started = true;
Com_Printf("%d Ogg Vorbis files found.\n", ogg_numfiles);
2009-10-03 16:08:34 +00:00
/* Autoplay support. */
if (ogg_autoplay->string[0] != '\0')
{
OGG_ParseCmd(ogg_autoplay->string);
}
2009-10-03 16:08:34 +00:00
}
/*
* Shutdown the Ogg Vorbis subsystem.
*/
void
OGG_Shutdown(void)
{
if (!ogg_started)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
Com_Printf("Shutting down Ogg Vorbis.\n");
2009-10-03 16:08:34 +00:00
OGG_Stop();
/* Free the list of files. */
FS_FreeList(ogg_filelist, ogg_numfiles + 1);
2009-10-03 16:08:34 +00:00
/* 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");
2009-10-03 16:08:34 +00:00
ogg_started = false;
}
/*
* Reinitialize the Ogg Vorbis subsystem.
*/
void
OGG_Reinit(void)
{
2009-10-03 16:08:34 +00:00
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;
}
2009-10-03 16:08:34 +00:00
res = false;
if ((size = FS_LoadFile(name, (void **)&buffer)) > 0)
{
if (ov_test(NULL, &ovf, (char *)buffer, size) == 0)
{
2009-10-03 16:08:34 +00:00
res = true;
ov_clear(&ovf);
2009-10-03 16:08:34 +00:00
}
FS_FreeFile(buffer);
2009-10-03 16:08:34 +00:00
}
return res;
2009-10-03 16:08:34 +00:00
}
/*
* 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). */
2009-10-03 16:08:34 +00:00
/* Check if the file is seekable. */
if (ov_seekable(&ovFile) == 0)
{
Com_Printf("OGG_Seek: file is not seekable.\n");
2009-10-03 16:08:34 +00:00
return;
}
/* Get file information. */
pos = ov_time_tell(&ovFile);
total = ov_time_total(&ovFile, -1);
2009-10-03 16:08:34 +00:00
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;
2009-10-03 16:08:34 +00:00
}
}
/*
* 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. */
2009-10-03 16:08:34 +00:00
/* Get file list. */
list = FS_ListFiles2(va("%s/*.ogg", OGG_DIR),
&ogg_numfiles, 0, SFF_SUBDIR | SFF_HIDDEN |
SFF_SYSTEM);
2009-10-03 16:08:34 +00:00
ogg_numfiles--;
/* Check if there are posible Ogg files. */
if (list == NULL)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
/* Allocate list of files. */
ogg_filelist = malloc(sizeof(char *) * ogg_numfiles);
2009-10-03 16:08:34 +00:00
/* 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]);
2009-10-03 16:08:34 +00:00
continue;
}
ogg_filelist[j++] = list[i];
2009-10-03 16:08:34 +00:00
}
/* Free the file list. */
free(list);
2009-10-03 16:08:34 +00:00
/* Adjust the list size (remove
space for invalid music files). */
2009-10-03 16:08:34 +00:00
ogg_numfiles = j;
ogg_filelist = realloc(ogg_filelist, sizeof(char *) * ogg_numfiles);
2009-10-03 16:08:34 +00:00
}
/*
* 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. */
2009-10-03 16:08:34 +00:00
/* 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));
2009-10-03 16:08:34 +00:00
return;
}
/* Count the files in playlist. */
for (ptr = strtok((char *)buffer, "\n");
ptr != NULL;
ptr = strtok(NULL, "\n"))
{
if ((byte *)ptr != buffer)
{
ptr[-1] = '\n';
}
if (OGG_Check(va("%s/%s", OGG_DIR, ptr)))
{
2009-10-03 16:08:34 +00:00
ogg_numfiles++;
}
2009-10-03 16:08:34 +00:00
}
/* Allocate file list. */
ogg_filelist = malloc(sizeof(char *) * ogg_numfiles);
2009-10-03 16:08:34 +00:00
i = 0;
for (ptr = strtok((char *)buffer, "\n");
ptr != NULL;
ptr = strtok(NULL, "\n"))
{
if (OGG_Check(va("%s/%s", OGG_DIR, ptr)))
{
ogg_filelist[i++] = strdup(va("%s/%s", OGG_DIR, ptr));
}
}
2009-10-03 16:08:34 +00:00
/* Free file buffer. */
FS_FreeFile(buffer);
2009-10-03 16:08:34 +00:00
}
/*
* Play Ogg Vorbis file (with absolute or relative index).
*/
qboolean
OGG_Open(ogg_seek_t type, int offset)
{
int size; /* File size. */
int pos = -1; /* Absolute position. */
int res; /* Error indicator. */
2009-10-03 16:08:34 +00:00
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;
2009-10-03 16:08:34 +00:00
}
/* Check running music. */
if (ogg_status == PLAY)
{
if (ogg_curfile == pos)
{
return true;
}
2009-10-03 16:08:34 +00:00
else
{
2009-10-03 16:08:34 +00:00
OGG_Stop();
}
2009-10-03 16:08:34 +00:00
}
/* 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;
2009-10-03 16:08:34 +00:00
}
/* 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;
2009-10-03 16:08:34 +00:00
}
/* Play file. */
ovSection = 0;
ogg_curfile = pos;
ogg_status = PLAY;
return true;
2009-10-03 16:08:34 +00:00
}
/*
* 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', and ogg_ignoretrack0 is set to 0, stop playback */
if ((!strncmp(filename, "00", sizeof(char) * 3)) && ogg_ignoretrack0->value == 0)
{
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)
{
2009-10-03 16:08:34 +00:00
break;
}
}
2009-10-03 16:08:34 +00:00
if (i < ogg_numfiles)
{
return OGG_Open(ABS, i);
}
else
{
Com_Printf("OGG_OpenName: '%s' not in the list.\n", filename);
return false;
2009-10-03 16:08:34 +00:00
}
}
/*
* Play a portion of the currently opened file.
*/
int
OGG_Read(void)
{
int res; /* Number of bytes read. */
2009-10-03 16:08:34 +00:00
/* 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);
2009-10-03 16:08:34 +00:00
/* Check for end of file. */
if (res == 0)
{
2009-10-03 16:08:34 +00:00
OGG_Stop();
OGG_Sequence();
}
return res;
2009-10-03 16:08:34 +00:00
}
/*
* 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, randk() % 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");
2009-10-03 16:08:34 +00:00
}
}
/*
* Stop playing the current file.
*/
void
OGG_Stop(void)
{
if (ogg_status == STOP)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
#ifdef USE_OPENAL
if (sound_started == SS_OAL)
{
AL_UnqueueRawSamples();
}
#endif
ov_clear(&ovFile);
2009-10-03 16:08:34 +00:00
ogg_status = STOP;
ogg_info = NULL;
ogg_numbufs = 0;
2009-10-03 16:08:34 +00:00
if (ogg_buffer != NULL)
{
FS_FreeFile(ogg_buffer);
2009-10-03 16:08:34 +00:00
ogg_buffer = NULL;
}
}
/*
* Stream music.
*/
void
OGG_Stream(void)
{
if (!ogg_started)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
if (ogg_status == PLAY)
{
#ifdef USE_OPENAL
if (sound_started == SS_OAL)
{
/* Calculate the number of buffers used
for storing decoded OGG/Vorbis data.
We take the number of active buffers
at startup (at this point most of the
samples should be precached and loaded
into buffers) and add 64. Empircal
testing showed, that at most times
at least 52 buffers remain available
for OGG/Vorbis, enough for about 3
2012-04-29 13:57:33 +00:00
seconds playback. The music won't
stutter as long as the framerate
stayes over 1 FPS. */
if (ogg_numbufs == 0)
{
ogg_numbufs = active_buffers + 64;
}
/* active_buffers are all active OpenAL buffers,
buffering normal sfx _and_ ogg/vorbis samples. */
while (active_buffers <= ogg_numbufs)
{
OGG_Read();
}
}
else /* using SDL */
#endif
{
if (sound_started == SS_SDL)
{
/* Read that number samples into the buffer, that
were played since the last call to this function.
This keeps the buffer at all times at an "optimal"
fill level. */
while (paintedtime + MAX_RAW_SAMPLES - 2048 > s_rawend)
{
OGG_Read();
}
}
} /* using SDL */
} /* ogg_status == PLAY */
2009-10-03 16:08:34 +00:00
}
/*
* List Ogg Vorbis files.
*/
void
OGG_ListCmd(void)
{
2009-10-03 16:08:34 +00:00
int i;
for (i = 0; i < ogg_numfiles; i++)
{
Com_Printf("%d %s\n", i + 1, ogg_filelist[i]);
}
2009-10-03 16:08:34 +00:00
Com_Printf("%d Ogg Vorbis files.\n", ogg_numfiles);
2009-10-03 16:08:34 +00:00
}
/*
* Parse play controls.
*/
void
OGG_ParseCmd(char *arg)
{
int n;
cvar_t *ogg_enable;
2009-10-03 16:08:34 +00:00
ogg_enable = Cvar_Get("ogg_enable", "1", CVAR_ARCHIVE);
switch (arg[0])
{
case '#':
n = (int)strtol(arg + 1, (char **)NULL, 10) - 1;
OGG_Open(ABS, n);
break;
case '?':
OGG_Open(ABS, randk() % ogg_numfiles);
break;
case '>':
if (strlen(arg) > 1)
{
OGG_Open(REL, (int)strtol(arg + 1, (char **)NULL, 10));
}
else
{
OGG_Open(REL, 1);
}
break;
case '<':
if (strlen(arg) > 1)
{
OGG_Open(REL, -(int)strtol(arg + 1, (char **)NULL, 10));
}
else
{
OGG_Open(REL, -1);
}
break;
default:
if (ogg_enable->value != 0)
{
OGG_OpenName(arg);
}
break;
2009-10-03 16:08:34 +00:00
}
}
/*
* Pause current song.
*/
void
OGG_PauseCmd(void)
{
if (ogg_status == PLAY)
{
2009-10-03 16:08:34 +00:00
ogg_status = PAUSE;
ogg_numbufs = 0;
}
#ifdef USE_OPENAL
if (sound_started == SS_OAL)
{
AL_UnqueueRawSamples();
}
#endif
2009-10-03 16:08:34 +00:00
}
/*
* Play control.
*/
void
OGG_PlayCmd(void)
{
if (Cmd_Argc() < 2)
{
Com_Printf("Usage: ogg_play {filename | #n | ? | >n | <n}\n");
2009-10-03 16:08:34 +00:00
return;
}
OGG_ParseCmd(Cmd_Argv(1));
2009-10-03 16:08:34 +00:00
}
/*
* Resume current song.
*/
void
OGG_ResumeCmd(void)
{
if (ogg_status == PAUSE)
{
2009-10-03 16:08:34 +00:00
ogg_status = PLAY;
}
2009-10-03 16:08:34 +00:00
}
/*
* Change position in the file being played.
*/
void
OGG_SeekCmd(void)
{
if (ogg_status != STOP)
{
2009-10-03 16:08:34 +00:00
return;
}
2009-10-03 16:08:34 +00:00
if (Cmd_Argc() < 2)
{
Com_Printf("Usage: ogg_seek {n | <n | >n}\n");
2009-10-03 16:08:34 +00:00
return;
}
switch (Cmd_Argv(1)[0])
{
2009-10-03 16:08:34 +00:00
case '>':
OGG_Seek(REL, strtod(Cmd_Argv(1) + 1, (char **)NULL));
2009-10-03 16:08:34 +00:00
break;
case '<':
OGG_Seek(REL, -strtod(Cmd_Argv(1) + 1, (char **)NULL));
2009-10-03 16:08:34 +00:00
break;
default:
OGG_Seek(ABS, strtod(Cmd_Argv(1), (char **)NULL));
2009-10-03 16:08:34 +00:00
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;
2009-10-03 16:08:34 +00:00
}
}
#endif /* OGG */