mirror of
https://github.com/Shpoike/Quakespasm.git
synced 2024-11-10 23:41:46 +00:00
170 lines
3.9 KiB
C
170 lines
3.9 KiB
C
|
/*
|
||
|
* Ogg/Vorbis streaming music support, adapted from several open source
|
||
|
* Quake engine based projects with many modifications.
|
||
|
*
|
||
|
* 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.,
|
||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "quakedef.h"
|
||
|
|
||
|
#if defined(USE_CODEC_VORBIS)
|
||
|
#include "snd_codec.h"
|
||
|
#include "snd_codeci.h"
|
||
|
#include "snd_vorbis.h"
|
||
|
#define OV_EXCLUDE_STATIC_CALLBACKS
|
||
|
#include <vorbis/vorbisfile.h>
|
||
|
|
||
|
/* The OGG codec can return the samples in a number of different
|
||
|
* formats, we use the standard signed short format. */
|
||
|
#define OGG_SAMPLEWIDTH 2
|
||
|
#define OGG_SIGNED_DATA 1
|
||
|
|
||
|
static int ogg_bigendian = 0;
|
||
|
|
||
|
/* CALLBACK FUNCTIONS: */
|
||
|
|
||
|
static int ovc_fclose (void *f)
|
||
|
{
|
||
|
return 0; /* we fclose() elsewhere. */
|
||
|
}
|
||
|
|
||
|
static int ovc_fseek (void *f, ogg_int64_t off, int whence)
|
||
|
{
|
||
|
if (f == NULL) return (-1);
|
||
|
return FS_fseek((fshandle_t *)f, (long) off, whence);
|
||
|
}
|
||
|
|
||
|
static const ov_callbacks ovc_qfs =
|
||
|
{
|
||
|
(size_t (*)(void *, size_t, size_t, void *)) FS_fread,
|
||
|
(int (*)(void *, ogg_int64_t, int)) ovc_fseek,
|
||
|
(int (*)(void *)) ovc_fclose,
|
||
|
(long (*)(void *)) FS_ftell
|
||
|
};
|
||
|
|
||
|
static qboolean S_OGG_CodecInitialize (void)
|
||
|
{
|
||
|
ogg_bigendian = (bigendien == true) ? 1 : 0;
|
||
|
ogg_codec.initialized = true;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
static void S_OGG_CodecShutdown (void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static snd_stream_t *S_OGG_CodecOpenStream (const char *filename)
|
||
|
{
|
||
|
snd_stream_t *stream;
|
||
|
OggVorbis_File *ovFile;
|
||
|
vorbis_info *ogg_info;
|
||
|
int res;
|
||
|
|
||
|
stream = S_CodecUtilOpen(filename, &ogg_codec);
|
||
|
if (!stream)
|
||
|
return NULL;
|
||
|
|
||
|
ovFile = (OggVorbis_File *) Z_Malloc(sizeof(OggVorbis_File));
|
||
|
res = ov_open_callbacks(&stream->fh, ovFile, NULL, 0, ovc_qfs);
|
||
|
if (res < 0)
|
||
|
{
|
||
|
Con_Printf("%s is not a valid Ogg Vorbis file (error %i).\n", filename, res);
|
||
|
Z_Free(ovFile);
|
||
|
S_CodecUtilClose(&stream);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
stream->priv = ovFile;
|
||
|
|
||
|
if (!ov_seekable(ovFile))
|
||
|
{
|
||
|
Con_Printf("OGG_Open: stream %s not seekable.\n", filename);
|
||
|
ov_clear(ovFile);
|
||
|
Z_Free(ovFile);
|
||
|
S_CodecUtilClose(&stream);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
ogg_info = ov_info(ovFile, 0);
|
||
|
if (!ogg_info)
|
||
|
{
|
||
|
Con_Printf("Unable to get stream information for %s.\n", filename);
|
||
|
ov_clear(ovFile);
|
||
|
Z_Free(ovFile);
|
||
|
S_CodecUtilClose(&stream);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
stream->info.rate = ogg_info->rate;
|
||
|
stream->info.channels = ogg_info->channels;
|
||
|
stream->info.width = OGG_SAMPLEWIDTH;
|
||
|
|
||
|
return stream;
|
||
|
}
|
||
|
|
||
|
static int S_OGG_CodecReadStream (snd_stream_t *stream, int bytes, void *buffer)
|
||
|
{
|
||
|
int section; /* FIXME: handle section changes */
|
||
|
int cnt, res, rem;
|
||
|
char * ptr;
|
||
|
|
||
|
cnt = 0; rem = bytes;
|
||
|
ptr = (char *) buffer;
|
||
|
while (1)
|
||
|
{
|
||
|
res = ov_read((OggVorbis_File *) stream->priv, ptr, rem, ogg_bigendian,
|
||
|
OGG_SAMPLEWIDTH, OGG_SIGNED_DATA, §ion);
|
||
|
if (res <= 0)
|
||
|
break;
|
||
|
rem -= res;
|
||
|
cnt += res;
|
||
|
if (rem <= 0)
|
||
|
break;
|
||
|
ptr += res;
|
||
|
}
|
||
|
return (res >= 0) ? cnt : res;
|
||
|
}
|
||
|
|
||
|
static void S_OGG_CodecCloseStream (snd_stream_t *stream)
|
||
|
{
|
||
|
ov_clear((OggVorbis_File *)stream->priv);
|
||
|
Z_Free(stream->priv);
|
||
|
S_CodecUtilClose(&stream);
|
||
|
}
|
||
|
|
||
|
static int S_OGG_CodecRewindStream (snd_stream_t *stream)
|
||
|
{
|
||
|
return ov_time_seek ((OggVorbis_File *)stream->priv, 0.0);
|
||
|
}
|
||
|
|
||
|
snd_codec_t ogg_codec =
|
||
|
{
|
||
|
CODECTYPE_OGG,
|
||
|
false,
|
||
|
".ogg",
|
||
|
S_OGG_CodecInitialize,
|
||
|
S_OGG_CodecShutdown,
|
||
|
S_OGG_CodecOpenStream,
|
||
|
S_OGG_CodecReadStream,
|
||
|
S_OGG_CodecRewindStream,
|
||
|
S_OGG_CodecCloseStream,
|
||
|
NULL
|
||
|
};
|
||
|
|
||
|
#endif /* USE_CODEC_VORBIS */
|
||
|
|