mirror of
https://github.com/Shpoike/Quakespasm.git
synced 2024-11-14 00:50:38 +00:00
187 lines
5 KiB
C
187 lines
5 KiB
C
/* tracker music (module file) decoding support using libxmp >= v4.2.0
|
|
* https://sourceforge.net/projects/xmp/
|
|
* https://github.com/libxmp/libxmp.git
|
|
*
|
|
* Copyright (C) 2016 O.Sezer <sezero@users.sourceforge.net>
|
|
*
|
|
* 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_XMP)
|
|
#include "snd_codec.h"
|
|
#include "snd_codeci.h"
|
|
#include "snd_xmp.h"
|
|
#if defined(_WIN32) && defined(LIBXMP_STATIC)
|
|
#define BUILDING_STATIC
|
|
#endif
|
|
#include <xmp.h>
|
|
#if ((XMP_VERCODE+0) < 0x040200)
|
|
#error libxmp version 4.2 or newer is required
|
|
#endif
|
|
|
|
static qboolean S_XMP_CodecInitialize (void)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
static void S_XMP_CodecShutdown (void)
|
|
{
|
|
}
|
|
|
|
#if (XMP_VERCODE >= 0x040500)
|
|
static unsigned long xmp_fread(void *dest, unsigned long len, unsigned long nmemb, void *f)
|
|
{
|
|
return FS_fread(dest, len, nmemb, (fshandle_t *)f);
|
|
}
|
|
static int xmp_fseek(void *f, long offset, int whence)
|
|
{
|
|
return FS_fseek((fshandle_t *)f, offset, whence);
|
|
}
|
|
static long xmp_ftell(void *f)
|
|
{
|
|
return FS_ftell((fshandle_t *)f);
|
|
}
|
|
#endif
|
|
|
|
static qboolean S_XMP_CodecOpenStream (snd_stream_t *stream)
|
|
{
|
|
/* need to load the whole file into memory and pass it to libxmp
|
|
* using xmp_load_module_from_memory() which requires libxmp >= 4.2.
|
|
* libxmp-4.0/4.1 only have xmp_load_module() which accepts a file
|
|
* name which isn't good with files in containers like paks, etc.
|
|
* On the other hand, libxmp >= 4.5 introduces file callbacks: use
|
|
* if available. */
|
|
xmp_context c;
|
|
#if (XMP_VERCODE >= 0x040500)
|
|
struct xmp_callbacks file_callbacks = {
|
|
xmp_fread, xmp_fseek, xmp_ftell, NULL
|
|
};
|
|
#else
|
|
byte *moddata;
|
|
long len;
|
|
int mark;
|
|
#endif
|
|
int fmt;
|
|
|
|
c = xmp_create_context();
|
|
if (c == NULL)
|
|
return false;
|
|
|
|
#if (XMP_VERCODE >= 0x040500)
|
|
if (xmp_load_module_from_callbacks(c, &stream->fh, file_callbacks) < 0) {
|
|
Con_DPrintf("Could not load module %s\n", stream->name);
|
|
goto err1;
|
|
}
|
|
#else
|
|
len = FS_filelength (&stream->fh);
|
|
mark = Hunk_LowMark();
|
|
moddata = (byte *) Hunk_Alloc(len);
|
|
FS_fread(moddata, 1, len, &stream->fh);
|
|
if (xmp_load_module_from_memory(c, moddata, len) < 0) {
|
|
Hunk_FreeToLowMark(mark);
|
|
Con_DPrintf("Could not load module %s\n", stream->name);
|
|
goto err1;
|
|
}
|
|
Hunk_FreeToLowMark(mark); /* free original file data */
|
|
#endif
|
|
|
|
stream->priv = c;
|
|
if (shm->speed > XMP_MAX_SRATE)
|
|
stream->info.rate = 44100;
|
|
else if (shm->speed < XMP_MIN_SRATE)
|
|
stream->info.rate = 11025;
|
|
else stream->info.rate = shm->speed;
|
|
stream->info.bits = shm->samplebits;
|
|
stream->info.width = stream->info.bits / 8;
|
|
stream->info.channels = shm->channels;
|
|
|
|
fmt = 0;
|
|
if (stream->info.channels == 1)
|
|
fmt |= XMP_FORMAT_MONO;
|
|
if (stream->info.width == 1)
|
|
fmt |= XMP_FORMAT_8BIT|XMP_FORMAT_UNSIGNED;
|
|
if (xmp_start_player(c, stream->info.rate, fmt) < 0)
|
|
goto err2;
|
|
|
|
/* interpolation type, default is XMP_INTERP_LINEAR */
|
|
xmp_set_player(c, XMP_PLAYER_INTERP, XMP_INTERP_SPLINE);
|
|
|
|
return true;
|
|
|
|
err2: xmp_release_module(c);
|
|
err1: xmp_free_context(c);
|
|
return false;
|
|
}
|
|
|
|
static int S_XMP_CodecReadStream (snd_stream_t *stream, int bytes, void *buffer)
|
|
{
|
|
int r;
|
|
/* xmp_play_buffer() requires libxmp >= 4.1. it will write
|
|
* native-endian pcm data to the buffer. if the data write
|
|
* is partial, the rest of the buffer will be zero-filled.
|
|
* the last param is the max number that the current sequence
|
|
* of song will be looped, or 0 to disable loop checking. */
|
|
r = xmp_play_buffer((xmp_context)stream->priv, buffer, bytes, !stream->loop);
|
|
if (r == 0) {
|
|
return bytes;
|
|
}
|
|
if (r == -XMP_END) {
|
|
Con_DPrintf("XMP EOF\n");
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
static void S_XMP_CodecCloseStream (snd_stream_t *stream)
|
|
{
|
|
xmp_context c = (xmp_context)stream->priv;
|
|
xmp_end_player(c);
|
|
xmp_release_module(c);
|
|
xmp_free_context(c);
|
|
S_CodecUtilClose(&stream);
|
|
}
|
|
|
|
static int S_XMP_CodecJumpToOrder (snd_stream_t *stream, int to)
|
|
{
|
|
return xmp_set_position((xmp_context)stream->priv, to);
|
|
}
|
|
|
|
static int S_XMP_CodecRewindStream (snd_stream_t *stream)
|
|
{
|
|
int ret = xmp_seek_time((xmp_context)stream->priv, 0);
|
|
if (ret < 0) return ret;
|
|
xmp_play_buffer((xmp_context)stream->priv, NULL, 0, 0); /* reset internal state */
|
|
return 0;
|
|
}
|
|
|
|
snd_codec_t xmp_codec =
|
|
{
|
|
CODECTYPE_MOD,
|
|
true, /* always available. */
|
|
"s3m",
|
|
S_XMP_CodecInitialize,
|
|
S_XMP_CodecShutdown,
|
|
S_XMP_CodecOpenStream,
|
|
S_XMP_CodecReadStream,
|
|
S_XMP_CodecRewindStream,
|
|
S_XMP_CodecJumpToOrder,
|
|
S_XMP_CodecCloseStream,
|
|
NULL
|
|
};
|
|
|
|
#endif /* USE_CODEC_XMP */
|