mirror of
https://github.com/TerryCavanagh/VVVVVV.git
synced 2025-01-05 16:39:44 +01:00
Add support for sound OGG files
This adds support for OGG files as sound effects (via renaming them to the wrong .wav file extension), because in previous versions of the game, SDL_mixer didn't care what the file extension was, and so some people relied on this, as described in #900. This is accomplished by copy-pasting the OGG loading code for music tracks. For a bit of cleanliness, I put the WAV and OGG loading code in separate functions. This is mostly the same code, except that because sound effects don't loop and can't be paused or resumed, there's no reserve buffer, and there's no data for loop points. Also, for some reason, the music loading code divided by 20 in the `size` calculation. I found that this prematurely cut off sound effects, and that it made more sense to just not do the division. I don't know why it was there, but removing it works. Also also, some OGG files don't work with this. Namely, ones produced by FFmpeg. To test this, I just extracted 0levelcomplete.ogg from vvvvvvmusic.vvv and replaced terminal.wav with it. And it works, so hopefully I won't have to touch audio code again, although I might if someone complains about this. But either way, I'm committing this because it's better than it was before. Fixes #900.
This commit is contained in:
parent
05ed7e041c
commit
ba7519106f
1 changed files with 72 additions and 7 deletions
|
@ -95,18 +95,32 @@ class SoundTrack
|
||||||
public:
|
public:
|
||||||
SoundTrack(const char* fileName)
|
SoundTrack(const char* fileName)
|
||||||
{
|
{
|
||||||
unsigned char *mem;
|
unsigned char* mem;
|
||||||
size_t length;
|
size_t length;
|
||||||
SDL_AudioSpec spec;
|
|
||||||
SDL_RWops *fileIn;
|
|
||||||
SDL_zerop(this);
|
|
||||||
FILESYSTEM_loadAssetToMemory(fileName, &mem, &length);
|
FILESYSTEM_loadAssetToMemory(fileName, &mem, &length);
|
||||||
if (mem == NULL)
|
if (mem == NULL)
|
||||||
{
|
{
|
||||||
vlog_error("Unable to load WAV file %s", fileName);
|
vlog_error("Unable to load sound file %s", fileName);
|
||||||
SDL_assert(0 && "WAV file missing!");
|
SDL_assert(0 && "Sound file missing!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SDL_zerop(this);
|
||||||
|
if (SDL_memcmp(mem, "OggS", 4) == 0)
|
||||||
|
{
|
||||||
|
LoadOGG(fileName, mem, length);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
LoadWAV(fileName, mem, length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadWAV(const char* fileName, unsigned char* mem, const size_t length)
|
||||||
|
{
|
||||||
|
SDL_AudioSpec spec;
|
||||||
|
SDL_RWops *fileIn;
|
||||||
fileIn = SDL_RWFromConstMem(mem, length);
|
fileIn = SDL_RWFromConstMem(mem, length);
|
||||||
if (SDL_LoadWAV_RW(fileIn, 1, &spec, &wav_buffer, &wav_length) == NULL)
|
if (SDL_LoadWAV_RW(fileIn, 1, &spec, &wav_buffer, &wav_length) == NULL)
|
||||||
{
|
{
|
||||||
|
@ -125,9 +139,41 @@ end:
|
||||||
VVV_free(mem);
|
VVV_free(mem);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void LoadOGG(const char* fileName, unsigned char* mem, const size_t length)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
stb_vorbis_info vorbis_info;
|
||||||
|
vorbis = stb_vorbis_open_memory(mem, length, &err, NULL);
|
||||||
|
if (vorbis == NULL)
|
||||||
|
{
|
||||||
|
vlog_error("Unable to create Vorbis handle for %s, error %d", fileName, err);
|
||||||
|
VVV_free(mem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
vorbis_info = stb_vorbis_get_info(vorbis);
|
||||||
|
format.wFormatTag = FAUDIO_FORMAT_IEEE_FLOAT;
|
||||||
|
format.wBitsPerSample = sizeof(float) * 8;
|
||||||
|
format.nChannels = vorbis_info.channels;
|
||||||
|
format.nSamplesPerSec = vorbis_info.sample_rate;
|
||||||
|
format.nBlockAlign = format.nChannels * format.wBitsPerSample;
|
||||||
|
format.nAvgBytesPerSec = format.nSamplesPerSec * format.nBlockAlign;
|
||||||
|
format.cbSize = 0;
|
||||||
|
|
||||||
|
channels = format.nChannels;
|
||||||
|
size = format.nAvgBytesPerSec;
|
||||||
|
decoded_buf_playing = (Uint8*) SDL_malloc(size);
|
||||||
|
|
||||||
|
ogg_file = mem;
|
||||||
|
valid = true;
|
||||||
|
}
|
||||||
|
|
||||||
void Dispose(void)
|
void Dispose(void)
|
||||||
{
|
{
|
||||||
VVV_free(wav_buffer);
|
VVV_free(wav_buffer);
|
||||||
|
|
||||||
|
VVV_free(decoded_buf_playing);
|
||||||
|
VVV_freefunc(stb_vorbis_close, vorbis);
|
||||||
|
VVV_free(ogg_file);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Play(void)
|
void Play(void)
|
||||||
|
@ -151,7 +197,7 @@ end:
|
||||||
VVV_freefunc(FAudioVoice_DestroyVoice, voices[i]);
|
VVV_freefunc(FAudioVoice_DestroyVoice, voices[i]);
|
||||||
FAudio_CreateSourceVoice(faudioctx, &voices[i], &format, 0, 2.0f, NULL, NULL, NULL);
|
FAudio_CreateSourceVoice(faudioctx, &voices[i], &format, 0, 2.0f, NULL, NULL, NULL);
|
||||||
}
|
}
|
||||||
const FAudioBuffer faudio_buffer = {
|
FAudioBuffer faudio_buffer = {
|
||||||
FAUDIO_END_OF_STREAM, /* Flags */
|
FAUDIO_END_OF_STREAM, /* Flags */
|
||||||
wav_length * 8, /* AudioBytes */
|
wav_length * 8, /* AudioBytes */
|
||||||
wav_buffer, /* AudioData */
|
wav_buffer, /* AudioData */
|
||||||
|
@ -162,6 +208,18 @@ end:
|
||||||
0, /* LoopCount */
|
0, /* LoopCount */
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
if (vorbis != NULL)
|
||||||
|
{
|
||||||
|
stb_vorbis_seek_start(vorbis);
|
||||||
|
faudio_buffer.PlayLength = stb_vorbis_get_samples_float_interleaved(
|
||||||
|
vorbis,
|
||||||
|
channels,
|
||||||
|
(float*) decoded_buf_playing,
|
||||||
|
size / sizeof(float)
|
||||||
|
);
|
||||||
|
faudio_buffer.AudioBytes = size;
|
||||||
|
faudio_buffer.pAudioData = decoded_buf_playing;
|
||||||
|
}
|
||||||
if (FAudioSourceVoice_SubmitSourceBuffer(voices[i], &faudio_buffer, NULL))
|
if (FAudioSourceVoice_SubmitSourceBuffer(voices[i], &faudio_buffer, NULL))
|
||||||
{
|
{
|
||||||
vlog_error("Unable to queue sound buffer");
|
vlog_error("Unable to queue sound buffer");
|
||||||
|
@ -241,6 +299,13 @@ end:
|
||||||
Uint8 *wav_buffer;
|
Uint8 *wav_buffer;
|
||||||
Uint32 wav_length;
|
Uint32 wav_length;
|
||||||
FAudioWaveFormatEx format;
|
FAudioWaveFormatEx format;
|
||||||
|
|
||||||
|
unsigned char* ogg_file;
|
||||||
|
stb_vorbis* vorbis;
|
||||||
|
int channels;
|
||||||
|
Uint32 size;
|
||||||
|
Uint8* decoded_buf_playing;
|
||||||
|
|
||||||
bool valid;
|
bool valid;
|
||||||
|
|
||||||
static FAudioSourceVoice** voices;
|
static FAudioSourceVoice** voices;
|
||||||
|
|
Loading…
Reference in a new issue