quakeforge/libs/audio/renderer/wav.c
Bill Currie 48f704c84e [sound] Use SYS_snd for all masked prints
SYS_dev is a holdover from when we had only the one flag and is not
meant to be used for tests (I seem to remember mentioning an audit was
necessary, but obviously forgotten). One step at a time, I guess :)
2022-06-06 12:39:54 +09:00

299 lines
6.6 KiB
C

/*
wav.c
wav file loading
Copyright (C) 1996-1997 Id Software, Inc.
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:
Free Software Foundation, Inc.
59 Temple Place - Suite 330
Boston, MA 02111-1307, USA
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_STRING_H
# include <string.h>
#endif
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
#include <stdlib.h>
#include "qfalloca.h"
#include "QF/cvar.h"
#include "QF/sound.h"
#include "QF/sys.h"
#include "QF/qendian.h"
#include "QF/quakefs.h"
#include "QF/riff.h"
#include "snd_internal.h"
#define FRAMES 1024
typedef struct {
float *data;
QFile *file;
} wav_file_t;
static sfxbuffer_t *
wav_callback_load (sfxblock_t *block)
{
const sfx_t *sfx = block->sfx;
const char *name = (const char *) block->file;
QFile *file;
int len, fdata_ofs;
byte *data;
float *fdata;
sfxbuffer_t *buffer = 0;
wavinfo_t *info = &block->wavinfo;
file = QFS_FOpenFile (name);
if (!file)
return 0;
Qseek (file, info->dataofs, SEEK_SET);
fdata_ofs = (info->datalen + sizeof (float) - 1) & ~(sizeof (float) - 1);
len = fdata_ofs + info->frames * info->channels * sizeof (float);
data = malloc (len);
if (!data)
goto bail;
fdata = (float *) (data + fdata_ofs);
Qread (file, data, info->datalen);
SND_Convert (data, fdata, info->frames, info->channels, info->width);
unsigned buffer_frames = SND_ResamplerFrames (sfx, info->frames);
buffer = SND_Memory_AllocBuffer (buffer_frames * info->channels);
if (!buffer)
goto bail;
buffer->size = buffer_frames * info->channels;
buffer->channels = info->channels;
buffer->sfx_length = info->frames;
buffer->block = block;
SND_SetPaint (buffer);
SND_SetupResampler (buffer, 0);
SND_Resample (buffer, fdata, info->frames);
buffer->head = buffer->size;
bail:
free (data);
Qclose (file);
return buffer;
}
static void
wav_cache (sfx_t *sfx, char *realname, void *file, wavinfo_t info)
{
Qclose (file);
SND_SFX_Block (sfx, realname, info, wav_callback_load);
}
static long
wav_stream_read (void *file, float **buf)
{
sfxstream_t *stream = (sfxstream_t *) file;
wavinfo_t *info = &stream->wavinfo;
wav_file_t *wf = (wav_file_t *) stream->file;
int res;
int len = FRAMES * info->channels * info->width;
byte *data = alloca (len);
if (!wf->data)
wf->data = malloc (FRAMES * info->channels * sizeof (float));
res = Qread (wf->file, data, len);
if (res <= 0) {
stream->error = 1;
return 0;
}
res /= (info->channels * info->width);
SND_Convert (data, wf->data, res, info->channels, info->width);
*buf = wf->data;
return res;
}
static int
wav_stream_seek (sfxstream_t *stream, int pos)
{
wavinfo_t *info = &stream->wavinfo;
wav_file_t *wf = (wav_file_t *) stream->file;
pos *= info->width * info->channels;
pos += info->dataofs;
return Qseek (wf->file, pos, SEEK_SET);
}
static void
wav_stream_close (sfxbuffer_t *buffer)
{
sfxstream_t *stream = buffer->stream;
wav_file_t *wf = (wav_file_t *) stream->file;
Qclose (wf->file);
if (wf->data)
free (wf->data);
free (wf);
SND_SFX_StreamClose (stream);
}
static sfxbuffer_t *
wav_stream_open (sfx_t *sfx)
{
sfxstream_t *stream = sfx->stream;
QFile *file;
wav_file_t *wf;
file = QFS_FOpenFile (stream->file);
if (!file)
return 0;
wf = calloc (sizeof (wav_file_t), 1);
wf->file = file;
return SND_SFX_StreamOpen (sfx, wf, wav_stream_read, wav_stream_seek,
wav_stream_close);
}
static void
wav_stream (sfx_t *sfx, char *realname, void *file, wavinfo_t info)
{
Qclose (file);
SND_SFX_Stream (sfx, realname, info, wav_stream_open);
}
static wavinfo_t
wav_get_info (QFile *file)
{
riff_t *riff;
riff_d_chunk_t **ck;
riff_format_t *fmt;
riff_d_format_t *dfmt = 0;
riff_data_t *data = 0;
riff_cue_t *cue;
riff_d_cue_t *dcue = 0;
riff_list_t *list;
riff_d_chunk_t **lck;
//riff_ltxt_t *ltxt;
//riff_d_ltxt_t *dltxt = 0;
wavinfo_t info;
memset (&info, 0, sizeof (info));
if (!(riff = riff_read (file))) {
Sys_Printf ("bad riff file\n");
return info;
}
for (ck = riff->chunks; *ck; ck++) {
RIFF_SWITCH ((*ck)->name) {
case RIFF_CASE ('f','m','t',' '):
fmt = (riff_format_t *) *ck;
dfmt = (riff_d_format_t *) fmt->fdata;
break;
case RIFF_CASE ('d','a','t','a'):
data = (riff_data_t *) *ck;
break;
case RIFF_CASE ('c','u','e',' '):
cue = (riff_cue_t *) *ck;
dcue = cue->cue;
break;
case RIFF_CASE ('L','I','S','T'):
list = (riff_list_t *) *ck;
RIFF_SWITCH (list->name) {
case RIFF_CASE ('a','d','t','l'):
for (lck = list->chunks; *lck; lck++) {
RIFF_SWITCH ((*lck)->name) {
case RIFF_CASE ('l','t','x','t'):
//ltxt = (riff_ltxt_t *) *lck;
//dltxt = &ltxt->ltxt;
break;
}
}
break;
}
break;
}
}
if (!dfmt) {
Sys_Printf ("missing format chunk\n");
goto bail;
}
if (dfmt->format_tag != 1) {
Sys_Printf ("not Microsoft PCM\n");
goto bail;
}
if (dfmt->channels < 1 || dfmt->channels > 8) {
Sys_Printf ("unsupported channel count\n");
goto bail;
}
if (!data) {
Sys_Printf ("missing data chunk\n");
goto bail;
}
info.rate = dfmt->samples_per_sec;
info.width = dfmt->bits_per_sample / 8;
info.channels = dfmt->channels;
info.frames = 0;
info.frames = data->ck.len / (info.width * info.channels);
if (dcue && dcue->count) {
if (dcue->cue_points[0].sample_offset < info.frames) {
info.loopstart = dcue->cue_points[0].sample_offset;
}
if (dcue->count > 1) {
info.frames = min (info.frames, dcue->cue_points[1].sample_offset);
}
//if (dltxt)
// info.frames = info.loopstart + dltxt->len;
} else {
info.loopstart = -1;
}
info.dataofs = *(int *)data->data;
info.datalen = data->ck.len;
bail:
riff_free (riff);
return info;
}
int
SND_LoadWav (QFile *file, sfx_t *sfx, char *realname)
{
wavinfo_t info;
info = wav_get_info (file);
if (!info.rate) {
return -1;
}
if (info.frames / info.rate < 3) {
Sys_MaskPrintf (SYS_snd, "cache %s\n", realname);
wav_cache (sfx, realname, file, info);
} else {
Sys_MaskPrintf (SYS_snd, "stream %s\n", realname);
wav_stream (sfx, realname, file, info);
}
return 0;
}