mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-17 22:50:51 +00:00
crystal clear playback on resampled streams :) (still borked when shared,
though :/)
This commit is contained in:
parent
64d5e4e053
commit
49571b257d
4 changed files with 110 additions and 48 deletions
|
@ -57,6 +57,7 @@ struct sfxbuffer_s {
|
||||||
unsigned int tail; // ring buffer tail position in sampels
|
unsigned int tail; // ring buffer tail position in sampels
|
||||||
unsigned int length; // length of buffer in samples
|
unsigned int length; // length of buffer in samples
|
||||||
unsigned int pos; // position of tail within full stream
|
unsigned int pos; // position of tail within full stream
|
||||||
|
unsigned int bps; // bytes per sample: 1 2 4 usually
|
||||||
void (*paint) (channel_t *ch, sfxbuffer_t *buffer, int count);
|
void (*paint) (channel_t *ch, sfxbuffer_t *buffer, int count);
|
||||||
void (*advance) (sfxbuffer_t *buffer, unsigned int count);
|
void (*advance) (sfxbuffer_t *buffer, unsigned int count);
|
||||||
sfx_t *sfx;
|
sfx_t *sfx;
|
||||||
|
@ -67,7 +68,7 @@ typedef struct sfxstream_s {
|
||||||
sfx_t *sfx;
|
sfx_t *sfx;
|
||||||
void *file;
|
void *file;
|
||||||
wavinfo_t wavinfo;
|
wavinfo_t wavinfo;
|
||||||
void (*resample)(sfxbuffer_t *, byte *, int);
|
void (*resample)(sfxbuffer_t *, byte *, int, void *);
|
||||||
int (*read)(void *file, byte *data, int bytes, wavinfo_t *info);
|
int (*read)(void *file, byte *data, int bytes, wavinfo_t *info);
|
||||||
int (*seek)(void *file, int pos, wavinfo_t *info);
|
int (*seek)(void *file, int pos, wavinfo_t *info);
|
||||||
sfxbuffer_t buffer;
|
sfxbuffer_t buffer;
|
||||||
|
@ -121,8 +122,8 @@ void SND_LocalSound (const char *s);
|
||||||
void SND_BlockSound (void);
|
void SND_BlockSound (void);
|
||||||
void SND_UnblockSound (void);
|
void SND_UnblockSound (void);
|
||||||
|
|
||||||
void SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length);
|
void SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length, void *prev);
|
||||||
void SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length);
|
void SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length, void *prev);
|
||||||
sfxbuffer_t *SND_GetCache (long samples, int rate, int inwidth, int channels,
|
sfxbuffer_t *SND_GetCache (long samples, int rate, int inwidth, int channels,
|
||||||
sfxblock_t *block, cache_allocator_t allocator);
|
sfxblock_t *block, cache_allocator_t allocator);
|
||||||
|
|
||||||
|
|
|
@ -48,6 +48,8 @@ static __attribute__ ((unused)) const char rcsid[] =
|
||||||
#include "compat.h"
|
#include "compat.h"
|
||||||
#include "snd_render.h"
|
#include "snd_render.h"
|
||||||
|
|
||||||
|
#define SAMPLE_GAP 4
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
byte left;
|
byte left;
|
||||||
byte right;
|
byte right;
|
||||||
|
@ -111,12 +113,16 @@ SND_StreamWavinfo (sfx_t *sfx)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
read_samples (sfxbuffer_t *buffer, int count)
|
read_samples (sfxbuffer_t *buffer, int count, void *prev)
|
||||||
{
|
{
|
||||||
|
|
||||||
if (buffer->head + count > buffer->length) {
|
if (buffer->head + count > buffer->length) {
|
||||||
|
int s = (buffer->length - 1);
|
||||||
|
|
||||||
count -= buffer->length - buffer->head;
|
count -= buffer->length - buffer->head;
|
||||||
read_samples (buffer, buffer->length - buffer->head);
|
read_samples (buffer, buffer->length - buffer->head, prev);
|
||||||
read_samples (buffer, count);
|
prev = buffer->data + s * buffer->bps;
|
||||||
|
read_samples (buffer, count, prev);
|
||||||
} else {
|
} else {
|
||||||
byte *data;
|
byte *data;
|
||||||
float stepscale;
|
float stepscale;
|
||||||
|
@ -133,7 +139,7 @@ read_samples (sfxbuffer_t *buffer, int count)
|
||||||
|
|
||||||
if (stream->read (stream->file, data, size, info) < size)
|
if (stream->read (stream->file, data, size, info) < size)
|
||||||
Sys_Printf ("%s\n", sfx->name);
|
Sys_Printf ("%s\n", sfx->name);
|
||||||
stream->resample (buffer, data, samples);
|
stream->resample (buffer, data, samples, prev);
|
||||||
buffer->head += count;
|
buffer->head += count;
|
||||||
if (buffer->head >= buffer->length)
|
if (buffer->head >= buffer->length)
|
||||||
buffer->head -= buffer->length;
|
buffer->head -= buffer->length;
|
||||||
|
@ -149,6 +155,7 @@ SND_StreamAdvance (sfxbuffer_t *buffer, unsigned int count)
|
||||||
sfx_t *sfx = buffer->sfx;
|
sfx_t *sfx = buffer->sfx;
|
||||||
sfxstream_t *stream = (sfxstream_t *) sfx->data;
|
sfxstream_t *stream = (sfxstream_t *) sfx->data;
|
||||||
wavinfo_t *info = &stream->wavinfo;
|
wavinfo_t *info = &stream->wavinfo;
|
||||||
|
void *prev;
|
||||||
|
|
||||||
stepscale = (float) info->rate / shm->speed; // usually 0.5, 1, or 2
|
stepscale = (float) info->rate / shm->speed; // usually 0.5, 1, or 2
|
||||||
|
|
||||||
|
@ -201,7 +208,7 @@ SND_StreamAdvance (sfxbuffer_t *buffer, unsigned int count)
|
||||||
}
|
}
|
||||||
|
|
||||||
// find out how many samples can be read into the buffer
|
// find out how many samples can be read into the buffer
|
||||||
samples = buffer->tail - buffer->head - 1;
|
samples = buffer->tail - buffer->head - SAMPLE_GAP;
|
||||||
if (buffer->tail <= buffer->head)
|
if (buffer->tail <= buffer->head)
|
||||||
samples += buffer->length;
|
samples += buffer->length;
|
||||||
|
|
||||||
|
@ -214,11 +221,27 @@ SND_StreamAdvance (sfxbuffer_t *buffer, unsigned int count)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (samples) {
|
if (samples) {
|
||||||
read_samples (buffer, samples);
|
if (buffer->head != buffer->tail) {
|
||||||
|
int s = buffer->head - 1;
|
||||||
|
if (!buffer->head)
|
||||||
|
s += buffer->length;
|
||||||
|
prev = buffer->data + s * buffer->bps;
|
||||||
|
} else {
|
||||||
|
prev = 0;
|
||||||
|
}
|
||||||
|
read_samples (buffer, samples, prev);
|
||||||
}
|
}
|
||||||
if (loop_samples) {
|
if (loop_samples) {
|
||||||
|
if (buffer->head != buffer->tail) {
|
||||||
|
int s = buffer->head - 1;
|
||||||
|
if (!buffer->head)
|
||||||
|
s += buffer->length;
|
||||||
|
prev = buffer->data + s * buffer->bps;
|
||||||
|
} else {
|
||||||
|
prev = 0;
|
||||||
|
}
|
||||||
stream->seek (stream->file, info->loopstart, info);
|
stream->seek (stream->file, info->loopstart, info);
|
||||||
read_samples (buffer, loop_samples);
|
read_samples (buffer, loop_samples, prev);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -294,22 +317,32 @@ SND_GetCache (long samples, int rate, int inwidth, int channels,
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length)
|
SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length, void *prev)
|
||||||
{
|
{
|
||||||
unsigned char *ib, *ob;
|
byte *ib, *ob, *pb;
|
||||||
int fracstep, outcount, sample, samplefrac, srcsample, i;
|
int fracstep, outcount, sample, samplefrac, srcsample, i;
|
||||||
float stepscale;
|
float stepscale;
|
||||||
short *is, *os;
|
short *is, *os, *ps;
|
||||||
wavinfo_t *info = sc->sfx->wavinfo (sc->sfx);
|
wavinfo_t *info = sc->sfx->wavinfo (sc->sfx);
|
||||||
int inwidth = info->width;
|
int inwidth = info->width;
|
||||||
int inrate = info->rate;
|
int inrate = info->rate;
|
||||||
int outwidth;
|
int outwidth;
|
||||||
|
short zero_s[1];
|
||||||
|
byte zero_b[1];
|
||||||
|
|
||||||
is = (short *) data;
|
is = (short *) data;
|
||||||
os = (short *) sc->data;
|
os = (short *) sc->data;
|
||||||
ib = data;
|
ib = data;
|
||||||
ob = sc->data;
|
ob = sc->data;
|
||||||
|
|
||||||
|
ps = zero_s;
|
||||||
|
pb = zero_b;
|
||||||
|
|
||||||
|
if (!prev) {
|
||||||
|
zero_s[0] = 0;
|
||||||
|
zero_b[0] = 128;
|
||||||
|
}
|
||||||
|
|
||||||
os += sc->head;
|
os += sc->head;
|
||||||
ob += sc->head;
|
ob += sc->head;
|
||||||
|
|
||||||
|
@ -325,11 +358,19 @@ SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length)
|
||||||
sc->sfx->loopstart = (unsigned int)-1;
|
sc->sfx->loopstart = (unsigned int)-1;
|
||||||
|
|
||||||
if (snd_loadas8bit->int_val) {
|
if (snd_loadas8bit->int_val) {
|
||||||
outwidth = 1;
|
sc->bps = outwidth = 1;
|
||||||
sc->paint = SND_PaintChannelFrom8;
|
sc->paint = SND_PaintChannelFrom8;
|
||||||
|
if (prev) {
|
||||||
|
zero_s[0] = ((char *)prev)[0];
|
||||||
|
zero_b[0] = ((char *)prev)[0] + 128;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
outwidth = 2;
|
sc->bps = outwidth = 2;
|
||||||
sc->paint = SND_PaintChannelFrom16;
|
sc->paint = SND_PaintChannelFrom16;
|
||||||
|
if (prev) {
|
||||||
|
zero_s[0] = ((short *)prev)[0];
|
||||||
|
zero_b[0] = (((short *)prev)[0] >> 8) + 128;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!length)
|
if (!length)
|
||||||
|
@ -364,15 +405,13 @@ SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length)
|
||||||
int s1, s2;
|
int s1, s2;
|
||||||
|
|
||||||
if (inwidth == 2) {
|
if (inwidth == 2) {
|
||||||
s2 = s1 = LittleShort (is[0]);
|
s1 = LittleShort (*ps);
|
||||||
if (i < length - 1)
|
s2 = LittleShort (*is);
|
||||||
s2 = LittleShort (is[1]);
|
ps = is++;
|
||||||
is++;
|
|
||||||
} else {
|
} else {
|
||||||
s2 = s1 = (ib[0] - 128) << 8;
|
s1 = (*pb - 128) << 8;
|
||||||
if (i < length - 1)
|
s2 = (*ib - 128) << 8;
|
||||||
s2 = (ib[1] - 128) << 8;
|
pb = ib++;
|
||||||
ib++;
|
|
||||||
}
|
}
|
||||||
for (j = 0; j < points; j++) {
|
for (j = 0; j < points; j++) {
|
||||||
sample = s1 + (s2 - s1) * ((float) j) / points;
|
sample = s1 + (s2 - s1) * ((float) j) / points;
|
||||||
|
@ -414,22 +453,32 @@ SND_ResampleMono (sfxbuffer_t *sc, byte *data, int length)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length)
|
SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length, void *prev)
|
||||||
{
|
{
|
||||||
int fracstep, outcount, sl, sr, samplefrac, srcsample, i;
|
int fracstep, outcount, sl, sr, samplefrac, srcsample, i;
|
||||||
float stepscale;
|
float stepscale;
|
||||||
stereo8_t *ib, *ob;
|
stereo8_t *ib, *ob, *pb;
|
||||||
stereo16_t *is, *os;
|
stereo16_t *is, *os, *ps;
|
||||||
wavinfo_t *info = sc->sfx->wavinfo (sc->sfx);
|
wavinfo_t *info = sc->sfx->wavinfo (sc->sfx);
|
||||||
int inwidth = info->width;
|
int inwidth = info->width;
|
||||||
int inrate = info->rate;
|
int inrate = info->rate;
|
||||||
int outwidth;
|
int outwidth;
|
||||||
|
stereo16_t zero_s;
|
||||||
|
stereo8_t zero_b;
|
||||||
|
|
||||||
is = (stereo16_t *) data;
|
is = (stereo16_t *) data;
|
||||||
os = (stereo16_t *) sc->data;
|
os = (stereo16_t *) sc->data;
|
||||||
ib = (stereo8_t *) data;
|
ib = (stereo8_t *) data;
|
||||||
ob = (stereo8_t *) sc->data;
|
ob = (stereo8_t *) sc->data;
|
||||||
|
|
||||||
|
ps = &zero_s;
|
||||||
|
pb = &zero_b;
|
||||||
|
|
||||||
|
if (!prev) {
|
||||||
|
zero_s.left = zero_s.right = 0;
|
||||||
|
zero_b.left = zero_b.right = 128;
|
||||||
|
}
|
||||||
|
|
||||||
os += sc->head;
|
os += sc->head;
|
||||||
ob += sc->head;
|
ob += sc->head;
|
||||||
|
|
||||||
|
@ -446,9 +495,23 @@ SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length)
|
||||||
if (snd_loadas8bit->int_val) {
|
if (snd_loadas8bit->int_val) {
|
||||||
outwidth = 1;
|
outwidth = 1;
|
||||||
sc->paint = SND_PaintChannelStereo8;
|
sc->paint = SND_PaintChannelStereo8;
|
||||||
|
sc->bps = 2;
|
||||||
|
if (prev) {
|
||||||
|
zero_s.left = ((char *)prev)[0];
|
||||||
|
zero_s.right = ((char *)prev)[1];
|
||||||
|
zero_b.left = ((char *)prev)[0] + 128;
|
||||||
|
zero_b.right = ((char *)prev)[1] + 128;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
outwidth = 2;
|
outwidth = 2;
|
||||||
sc->paint = SND_PaintChannelStereo16;
|
sc->paint = SND_PaintChannelStereo16;
|
||||||
|
sc->bps = 4;
|
||||||
|
if (prev) {
|
||||||
|
zero_s.left = ((short *)prev)[0];
|
||||||
|
zero_s.right = ((short *)prev)[1];
|
||||||
|
zero_b.left = (((short *)prev)[0] >> 8) + 128;
|
||||||
|
zero_b.right = (((short *)prev)[1] >> 8) + 128;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!length)
|
if (!length)
|
||||||
|
@ -488,21 +551,17 @@ SND_ResampleStereo (sfxbuffer_t *sc, byte *data, int length)
|
||||||
int sr1, sr2;
|
int sr1, sr2;
|
||||||
|
|
||||||
if (inwidth == 2) {
|
if (inwidth == 2) {
|
||||||
sl2 = sl1 = LittleShort (is[0].left);
|
sl1 = LittleShort (ps->left);
|
||||||
sr2 = sr1 = LittleShort (is[0].right);
|
sr1 = LittleShort (ps->right);
|
||||||
if (i < length - 1) {
|
sl2 = LittleShort (is->left);
|
||||||
sl2 = LittleShort (is[1].left);
|
sr2 = LittleShort (is->right);
|
||||||
sr2 = LittleShort (is[1].right);
|
ps = is++;
|
||||||
}
|
|
||||||
is++;
|
|
||||||
} else {
|
} else {
|
||||||
sl2 = sl1 = (ib[0].left - 128) << 8;
|
sl1 = (pb->left - 128) << 8;
|
||||||
sr2 = sr1 = (ib[0].right - 128) << 8;
|
sr1 = (pb->right - 128) << 8;
|
||||||
if (i < length - 1) {
|
sl2 = (ib->left - 128) << 8;
|
||||||
sl2 = (ib[1].left - 128) << 8;
|
sr2 = (ib->right - 128) << 8;
|
||||||
sr2 = (ib[1].right - 128) << 8;
|
pb = ib++;
|
||||||
}
|
|
||||||
ib++;
|
|
||||||
}
|
}
|
||||||
for (j = 0; j < points; j++) {
|
for (j = 0; j < points; j++) {
|
||||||
sl = sl1 + (sl2 - sl1) * ((float) j) / points;
|
sl = sl1 + (sl2 - sl1) * ((float) j) / points;
|
||||||
|
|
|
@ -155,7 +155,7 @@ vorbis_load (OggVorbis_File *vf, sfxblock_t *block, cache_allocator_t allocator)
|
||||||
byte *data;
|
byte *data;
|
||||||
sfxbuffer_t *sc = 0;
|
sfxbuffer_t *sc = 0;
|
||||||
sfx_t *sfx = block->sfx;
|
sfx_t *sfx = block->sfx;
|
||||||
void (*resample)(sfxbuffer_t *, byte *, int);
|
void (*resample)(sfxbuffer_t *, byte *, int, void *);
|
||||||
wavinfo_t *info = &block->wavinfo;
|
wavinfo_t *info = &block->wavinfo;
|
||||||
|
|
||||||
switch (info->channels) {
|
switch (info->channels) {
|
||||||
|
@ -181,7 +181,7 @@ vorbis_load (OggVorbis_File *vf, sfxblock_t *block, cache_allocator_t allocator)
|
||||||
sc->sfx = sfx;
|
sc->sfx = sfx;
|
||||||
if (vorbis_read (vf, data, info->datalen) < 0)
|
if (vorbis_read (vf, data, info->datalen) < 0)
|
||||||
goto bail;
|
goto bail;
|
||||||
resample (sc, data, info->samples);
|
resample (sc, data, info->samples, 0);
|
||||||
sc->head = sc->length;
|
sc->head = sc->length;
|
||||||
bail:
|
bail:
|
||||||
if (data)
|
if (data)
|
||||||
|
@ -247,7 +247,8 @@ vorbis_stream (sfx_t *sfx, char *realname, OggVorbis_File *vf, wavinfo_t info)
|
||||||
int samples;
|
int samples;
|
||||||
int size;
|
int size;
|
||||||
|
|
||||||
samples = size = shm->speed * 0.3;
|
samples = shm->speed * 0.3;
|
||||||
|
size = samples = (samples + 3) & ~3;
|
||||||
if (!snd_loadas8bit->int_val)
|
if (!snd_loadas8bit->int_val)
|
||||||
size *= 2;
|
size *= 2;
|
||||||
if (info.channels == 2)
|
if (info.channels == 2)
|
||||||
|
@ -275,7 +276,7 @@ vorbis_stream (sfx_t *sfx, char *realname, OggVorbis_File *vf, wavinfo_t info)
|
||||||
stream->buffer.advance = SND_StreamAdvance;
|
stream->buffer.advance = SND_StreamAdvance;
|
||||||
stream->buffer.sfx = sfx;
|
stream->buffer.sfx = sfx;
|
||||||
|
|
||||||
stream->resample (&stream->buffer, 0, 0); // get sfx setup properly
|
stream->resample (&stream->buffer, 0, 0, 0); // get sfx setup properly
|
||||||
stream->seek (stream->file, 0, &stream->wavinfo);
|
stream->seek (stream->file, 0, &stream->wavinfo);
|
||||||
|
|
||||||
stream->buffer.advance (&stream->buffer, 0);
|
stream->buffer.advance (&stream->buffer, 0);
|
||||||
|
|
|
@ -72,9 +72,9 @@ wav_callback_load (void *object, cache_allocator_t allocator)
|
||||||
info->channels, block, allocator);
|
info->channels, block, allocator);
|
||||||
buffer->sfx = sfx;
|
buffer->sfx = sfx;
|
||||||
if (info->channels == 2)
|
if (info->channels == 2)
|
||||||
SND_ResampleStereo (buffer, data, info->samples);
|
SND_ResampleStereo (buffer, data, info->samples, 0);
|
||||||
else
|
else
|
||||||
SND_ResampleMono (buffer, data, info->samples);
|
SND_ResampleMono (buffer, data, info->samples, 0);
|
||||||
buffer->head = buffer->length;
|
buffer->head = buffer->length;
|
||||||
free (data);
|
free (data);
|
||||||
}
|
}
|
||||||
|
@ -118,7 +118,8 @@ wav_stream (sfx_t *sfx, char *realname, void *file, wavinfo_t info)
|
||||||
int samples;
|
int samples;
|
||||||
int size;
|
int size;
|
||||||
|
|
||||||
samples = size = shm->speed * 0.3;
|
samples = shm->speed * 0.3;
|
||||||
|
size = samples = (samples + 3) & ~3;
|
||||||
if (!snd_loadas8bit->int_val)
|
if (!snd_loadas8bit->int_val)
|
||||||
size *= 2;
|
size *= 2;
|
||||||
if (info.channels == 2)
|
if (info.channels == 2)
|
||||||
|
@ -145,7 +146,7 @@ wav_stream (sfx_t *sfx, char *realname, void *file, wavinfo_t info)
|
||||||
stream->buffer.advance = SND_StreamAdvance;
|
stream->buffer.advance = SND_StreamAdvance;
|
||||||
stream->buffer.sfx = sfx;
|
stream->buffer.sfx = sfx;
|
||||||
|
|
||||||
stream->resample (&stream->buffer, 0, 0); // get sfx setup properly
|
stream->resample (&stream->buffer, 0, 0, 0); // get sfx setup properly
|
||||||
stream->seek (stream->file, 0, &stream->wavinfo);
|
stream->seek (stream->file, 0, &stream->wavinfo);
|
||||||
|
|
||||||
stream->buffer.advance (&stream->buffer, 0);
|
stream->buffer.advance (&stream->buffer, 0);
|
||||||
|
|
Loading…
Reference in a new issue