mirror of
https://github.com/Rockbox/rockbox.git
synced 2025-10-24 07:27:39 -04:00
There are some real bugs in here, but we're drowning in warnings. Change-Id: I7c2c0eafc8426327521bdd8a3ac2d3742ac16864
520 lines
13 KiB
C
520 lines
13 KiB
C
/*
|
|
SDL_mixer: An audio mixer library based on the SDL library
|
|
Copyright (C) 1997-2012 Sam Lantinga <slouken@libsdl.org>
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
warranty. In no event will the authors be held liable for any damages
|
|
arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
including commercial applications, and to alter it and redistribute it
|
|
freely, subject to the following restrictions:
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
claim that you wrote the original software. If you use this software
|
|
in a product, an acknowledgment in the product documentation would be
|
|
appreciated but is not required.
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
misrepresented as being the original software.
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
/* This file supports streaming WAV files, without volume adjustment */
|
|
|
|
#include "SDL_audio.h"
|
|
#include "SDL_mutex.h"
|
|
#include "SDL_rwops.h"
|
|
#include "SDL_endian.h"
|
|
|
|
#include "SDL_mixer.h"
|
|
#include "wavestream.h"
|
|
|
|
/*
|
|
Taken with permission from SDL_wave.h, part of the SDL library,
|
|
available at: http://www.libsdl.org/
|
|
and placed under the same license as this mixer library.
|
|
*/
|
|
|
|
/* WAVE files are little-endian */
|
|
|
|
/*******************************************/
|
|
/* Define values for Microsoft WAVE format */
|
|
/*******************************************/
|
|
#define RIFF 0x46464952 /* "RIFF" */
|
|
#define WAVE 0x45564157 /* "WAVE" */
|
|
#define FACT 0x74636166 /* "fact" */
|
|
#define LIST 0x5453494c /* "LIST" */
|
|
#define FMT 0x20746D66 /* "fmt " */
|
|
#define DATA 0x61746164 /* "data" */
|
|
#define PCM_CODE 1
|
|
#define ADPCM_CODE 2
|
|
#define WAVE_MONO 1
|
|
#define WAVE_STEREO 2
|
|
|
|
/* Normally, these three chunks come consecutively in a WAVE file */
|
|
typedef struct WaveFMT {
|
|
/* Not saved in the chunk we read:
|
|
Uint32 FMTchunk;
|
|
Uint32 fmtlen;
|
|
*/
|
|
Uint16 encoding;
|
|
Uint16 channels; /* 1 = mono, 2 = stereo */
|
|
Uint32 frequency; /* One of 11025, 22050, or 44100 Hz */
|
|
Uint32 byterate; /* Average bytes per second */
|
|
Uint16 blockalign; /* Bytes per sample block */
|
|
Uint16 bitspersample; /* One of 8, 12, 16, or 4 for ADPCM */
|
|
} WaveFMT;
|
|
|
|
/* The general chunk found in the WAVE file */
|
|
typedef struct Chunk {
|
|
Uint32 magic;
|
|
Uint32 length;
|
|
Uint8 *data; /* Data includes magic and length */
|
|
} Chunk;
|
|
|
|
/*********************************************/
|
|
/* Define values for AIFF (IFF audio) format */
|
|
/*********************************************/
|
|
#define FORM 0x4d524f46 /* "FORM" */
|
|
#define AIFF 0x46464941 /* "AIFF" */
|
|
#define SSND 0x444e5353 /* "SSND" */
|
|
#define COMM 0x4d4d4f43 /* "COMM" */
|
|
|
|
|
|
/* Currently we only support a single stream at a time */
|
|
static WAVStream *music = NULL;
|
|
|
|
/* This is the format of the audio mixer data */
|
|
static SDL_AudioSpec mixer;
|
|
static int wavestream_volume = MIX_MAX_VOLUME;
|
|
|
|
/* Function to load the WAV/AIFF stream */
|
|
static SDL_RWops *LoadWAVStream (SDL_RWops *rw, SDL_AudioSpec *spec,
|
|
long *start, long *stop);
|
|
static SDL_RWops *LoadAIFFStream (SDL_RWops *rw, SDL_AudioSpec *spec,
|
|
long *start, long *stop);
|
|
|
|
/* Initialize the WAVStream player, with the given mixer settings
|
|
This function returns 0, or -1 if there was an error.
|
|
*/
|
|
int WAVStream_Init(SDL_AudioSpec *mixerfmt)
|
|
{
|
|
mixer = *mixerfmt;
|
|
return(0);
|
|
}
|
|
|
|
void WAVStream_SetVolume(int volume)
|
|
{
|
|
wavestream_volume = volume;
|
|
}
|
|
|
|
/* Load a WAV stream from the given RWops object */
|
|
WAVStream *WAVStream_LoadSong_RW(SDL_RWops *rw, const char *magic, int freerw)
|
|
{
|
|
WAVStream *wave;
|
|
SDL_AudioSpec wavespec;
|
|
|
|
if ( ! mixer.format ) {
|
|
Mix_SetError("WAV music output not started");
|
|
if ( freerw ) {
|
|
SDL_RWclose(rw);
|
|
}
|
|
return(NULL);
|
|
}
|
|
wave = (WAVStream *)SDL_malloc(sizeof *wave);
|
|
if ( wave ) {
|
|
memset(wave, 0, (sizeof *wave));
|
|
wave->freerw = freerw;
|
|
if ( strcmp(magic, "RIFF") == 0 ) {
|
|
wave->rw = LoadWAVStream(rw, &wavespec,
|
|
&wave->start, &wave->stop);
|
|
} else
|
|
if ( strcmp(magic, "FORM") == 0 ) {
|
|
wave->rw = LoadAIFFStream(rw, &wavespec,
|
|
&wave->start, &wave->stop);
|
|
} else {
|
|
Mix_SetError("Unknown WAVE format");
|
|
}
|
|
if ( wave->rw == NULL ) {
|
|
SDL_free(wave);
|
|
if ( freerw ) {
|
|
SDL_RWclose(rw);
|
|
}
|
|
return(NULL);
|
|
}
|
|
SDL_BuildAudioCVT(&wave->cvt,
|
|
wavespec.format, wavespec.channels, wavespec.freq,
|
|
mixer.format, mixer.channels, mixer.freq);
|
|
} else {
|
|
SDL_OutOfMemory();
|
|
if ( freerw ) {
|
|
SDL_RWclose(rw);
|
|
}
|
|
return(NULL);
|
|
}
|
|
return(wave);
|
|
}
|
|
|
|
/* Start playback of a given WAV stream */
|
|
void WAVStream_Start(WAVStream *wave)
|
|
{
|
|
SDL_RWseek (wave->rw, wave->start, RW_SEEK_SET);
|
|
music = wave;
|
|
}
|
|
|
|
/* Play some of a stream previously started with WAVStream_Start() */
|
|
int WAVStream_PlaySome(Uint8 *stream, int len)
|
|
{
|
|
long pos;
|
|
int left = 0;
|
|
|
|
if ( music && ((pos=SDL_RWtell(music->rw)) < music->stop) ) {
|
|
if ( music->cvt.needed ) {
|
|
int original_len;
|
|
|
|
original_len=(int)((double)len/music->cvt.len_ratio);
|
|
if ( music->cvt.len != original_len ) {
|
|
int worksize;
|
|
if ( music->cvt.buf != NULL ) {
|
|
SDL_free(music->cvt.buf);
|
|
}
|
|
worksize = original_len*music->cvt.len_mult;
|
|
music->cvt.buf=(Uint8 *)SDL_malloc(worksize);
|
|
if ( music->cvt.buf == NULL ) {
|
|
return 0;
|
|
}
|
|
music->cvt.len = original_len;
|
|
}
|
|
if ( (music->stop - pos) < original_len ) {
|
|
left = (original_len - (music->stop - pos));
|
|
original_len -= left;
|
|
left = (int)((double)left*music->cvt.len_ratio);
|
|
}
|
|
original_len = SDL_RWread(music->rw, music->cvt.buf,1,original_len);
|
|
/* At least at the time of writing, SDL_ConvertAudio()
|
|
does byte-order swapping starting at the end of the
|
|
buffer. Thus, if we are reading 16-bit samples, we
|
|
had better make damn sure that we get an even
|
|
number of bytes, or we'll get garbage.
|
|
*/
|
|
if ( (music->cvt.src_format & 0x0010) && (original_len & 1) ) {
|
|
original_len--;
|
|
}
|
|
music->cvt.len = original_len;
|
|
SDL_ConvertAudio(&music->cvt);
|
|
SDL_MixAudio(stream, music->cvt.buf, music->cvt.len_cvt, wavestream_volume);
|
|
} else {
|
|
Uint8 *data;
|
|
if ( (music->stop - pos) < len ) {
|
|
left = (len - (music->stop - pos));
|
|
len -= left;
|
|
}
|
|
data = SDL_stack_alloc(Uint8, len);
|
|
if (data)
|
|
{
|
|
SDL_RWread(music->rw, data, len, 1);
|
|
SDL_MixAudio(stream, data, len, wavestream_volume);
|
|
SDL_stack_free(data);
|
|
}
|
|
}
|
|
}
|
|
return left;
|
|
}
|
|
|
|
/* Stop playback of a stream previously started with WAVStream_Start() */
|
|
void WAVStream_Stop(void)
|
|
{
|
|
music = NULL;
|
|
}
|
|
|
|
/* Close the given WAV stream */
|
|
void WAVStream_FreeSong(WAVStream *wave)
|
|
{
|
|
if ( wave ) {
|
|
/* Clean up associated data */
|
|
if ( wave->cvt.buf ) {
|
|
SDL_free(wave->cvt.buf);
|
|
}
|
|
if ( wave->freerw ) {
|
|
SDL_RWclose(wave->rw);
|
|
}
|
|
SDL_free(wave);
|
|
}
|
|
}
|
|
|
|
/* Return non-zero if a stream is currently playing */
|
|
int WAVStream_Active(void)
|
|
{
|
|
int active;
|
|
|
|
active = 0;
|
|
if ( music && (SDL_RWtell(music->rw) < music->stop) ) {
|
|
active = 1;
|
|
}
|
|
return(active);
|
|
}
|
|
|
|
static int ReadChunk(SDL_RWops *src, Chunk *chunk, int read_data)
|
|
{
|
|
chunk->magic = SDL_ReadLE32(src);
|
|
chunk->length = SDL_ReadLE32(src);
|
|
if ( read_data ) {
|
|
chunk->data = (Uint8 *)SDL_malloc(chunk->length);
|
|
if ( chunk->data == NULL ) {
|
|
Mix_SetError("Out of memory");
|
|
return(-1);
|
|
}
|
|
if ( SDL_RWread(src, chunk->data, chunk->length, 1) != 1 ) {
|
|
Mix_SetError("Couldn't read chunk");
|
|
SDL_free(chunk->data);
|
|
return(-1);
|
|
}
|
|
} else {
|
|
SDL_RWseek(src, chunk->length, RW_SEEK_CUR);
|
|
}
|
|
return(chunk->length);
|
|
}
|
|
|
|
static SDL_RWops *LoadWAVStream (SDL_RWops *src, SDL_AudioSpec *spec,
|
|
long *start, long *stop)
|
|
{
|
|
int was_error;
|
|
Chunk chunk;
|
|
int lenread;
|
|
|
|
/* WAV magic header */
|
|
Uint32 RIFFchunk;
|
|
// Uint32 wavelen;
|
|
Uint32 WAVEmagic;
|
|
|
|
/* FMT chunk */
|
|
WaveFMT *format = NULL;
|
|
|
|
was_error = 0;
|
|
|
|
/* Check the magic header */
|
|
RIFFchunk = SDL_ReadLE32(src);
|
|
// wavelen = SDL_ReadLE32(src);
|
|
WAVEmagic = SDL_ReadLE32(src);
|
|
if ( (RIFFchunk != RIFF) || (WAVEmagic != WAVE) ) {
|
|
Mix_SetError("Unrecognized file type (not WAVE)");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
|
|
/* Read the audio data format chunk */
|
|
chunk.data = NULL;
|
|
do {
|
|
/* FIXME! Add this logic to SDL_LoadWAV_RW() */
|
|
if ( chunk.data ) {
|
|
SDL_free(chunk.data);
|
|
}
|
|
lenread = ReadChunk(src, &chunk, 1);
|
|
if ( lenread < 0 ) {
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
} while ( (chunk.magic == FACT) || (chunk.magic == LIST) );
|
|
|
|
/* Decode the audio data format */
|
|
format = (WaveFMT *)chunk.data;
|
|
if ( chunk.magic != FMT ) {
|
|
SDL_free(chunk.data);
|
|
Mix_SetError("Complex WAVE files not supported");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
switch (SDL_SwapLE16(format->encoding)) {
|
|
case PCM_CODE:
|
|
/* We can understand this */
|
|
break;
|
|
default:
|
|
Mix_SetError("Unknown WAVE data format");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
memset(spec, 0, (sizeof *spec));
|
|
spec->freq = SDL_SwapLE32(format->frequency);
|
|
switch (SDL_SwapLE16(format->bitspersample)) {
|
|
case 8:
|
|
spec->format = AUDIO_U8;
|
|
break;
|
|
case 16:
|
|
spec->format = AUDIO_S16;
|
|
break;
|
|
default:
|
|
Mix_SetError("Unknown PCM data format");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
spec->channels = (Uint8) SDL_SwapLE16(format->channels);
|
|
spec->samples = 4096; /* Good default buffer size */
|
|
|
|
/* Set the file offset to the DATA chunk data */
|
|
chunk.data = NULL;
|
|
do {
|
|
*start = SDL_RWtell(src) + 2*sizeof(Uint32);
|
|
lenread = ReadChunk(src, &chunk, 0);
|
|
if ( lenread < 0 ) {
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
} while ( chunk.magic != DATA );
|
|
*stop = SDL_RWtell(src);
|
|
|
|
done:
|
|
if ( format != NULL ) {
|
|
SDL_free(format);
|
|
}
|
|
if ( was_error ) {
|
|
return NULL;
|
|
}
|
|
return(src);
|
|
}
|
|
|
|
/* I couldn't get SANE_to_double() to work, so I stole this from libsndfile.
|
|
* I don't pretend to fully understand it.
|
|
*/
|
|
|
|
static Uint32 SANE_to_Uint32 (Uint8 *sanebuf)
|
|
{
|
|
/* Negative number? */
|
|
if (sanebuf[0] & 0x80)
|
|
return 0;
|
|
|
|
/* Less than 1? */
|
|
if (sanebuf[0] <= 0x3F)
|
|
return 1;
|
|
|
|
/* Way too big? */
|
|
if (sanebuf[0] > 0x40)
|
|
return 0x4000000;
|
|
|
|
/* Still too big? */
|
|
if (sanebuf[0] == 0x40 && sanebuf[1] > 0x1C)
|
|
return 800000000;
|
|
|
|
return ((sanebuf[2] << 23) | (sanebuf[3] << 15) | (sanebuf[4] << 7)
|
|
| (sanebuf[5] >> 1)) >> (29 - sanebuf[1]);
|
|
}
|
|
|
|
static SDL_RWops *LoadAIFFStream (SDL_RWops *src, SDL_AudioSpec *spec,
|
|
long *start, long *stop)
|
|
{
|
|
int was_error;
|
|
int found_SSND;
|
|
int found_COMM;
|
|
|
|
Uint32 chunk_type;
|
|
Uint32 chunk_length;
|
|
long next_chunk;
|
|
|
|
/* AIFF magic header */
|
|
Uint32 FORMchunk;
|
|
Uint32 AIFFmagic;
|
|
/* SSND chunk */
|
|
Uint32 offset;
|
|
// Uint32 blocksize;
|
|
/* COMM format chunk */
|
|
Uint16 channels = 0;
|
|
Uint32 numsamples = 0;
|
|
Uint16 samplesize = 0;
|
|
Uint8 sane_freq[10];
|
|
Uint32 frequency = 0;
|
|
|
|
was_error = 0;
|
|
|
|
/* Check the magic header */
|
|
FORMchunk = SDL_ReadLE32(src);
|
|
chunk_length = SDL_ReadBE32(src);
|
|
AIFFmagic = SDL_ReadLE32(src);
|
|
if ( (FORMchunk != FORM) || (AIFFmagic != AIFF) ) {
|
|
Mix_SetError("Unrecognized file type (not AIFF)");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
|
|
/* From what I understand of the specification, chunks may appear in
|
|
* any order, and we should just ignore unknown ones.
|
|
*
|
|
* TODO: Better sanity-checking. E.g. what happens if the AIFF file
|
|
* contains compressed sound data?
|
|
*/
|
|
|
|
found_SSND = 0;
|
|
found_COMM = 0;
|
|
|
|
do {
|
|
chunk_type = SDL_ReadLE32(src);
|
|
chunk_length = SDL_ReadBE32(src);
|
|
next_chunk = SDL_RWtell(src) + chunk_length;
|
|
|
|
/* Paranoia to avoid infinite loops */
|
|
if (chunk_length == 0)
|
|
break;
|
|
|
|
switch (chunk_type) {
|
|
case SSND:
|
|
found_SSND = 1;
|
|
offset = SDL_ReadBE32(src);
|
|
// blocksize = SDL_ReadBE32(src);
|
|
*start = SDL_RWtell(src) + offset;
|
|
break;
|
|
|
|
case COMM:
|
|
found_COMM = 1;
|
|
|
|
/* Read the audio data format chunk */
|
|
channels = SDL_ReadBE16(src);
|
|
numsamples = SDL_ReadBE32(src);
|
|
samplesize = SDL_ReadBE16(src);
|
|
SDL_RWread(src, sane_freq, sizeof(sane_freq), 1);
|
|
frequency = SANE_to_Uint32(sane_freq);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
} while ((!found_SSND || !found_COMM)
|
|
&& SDL_RWseek(src, next_chunk, RW_SEEK_SET) != -1);
|
|
|
|
if (!found_SSND) {
|
|
Mix_SetError("Bad AIFF file (no SSND chunk)");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
|
|
if (!found_COMM) {
|
|
Mix_SetError("Bad AIFF file (no COMM chunk)");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
|
|
*stop = *start + channels * numsamples * (samplesize / 8);
|
|
|
|
/* Decode the audio data format */
|
|
memset(spec, 0, (sizeof *spec));
|
|
spec->freq = frequency;
|
|
switch (samplesize) {
|
|
case 8:
|
|
spec->format = AUDIO_S8;
|
|
break;
|
|
case 16:
|
|
spec->format = AUDIO_S16MSB;
|
|
break;
|
|
default:
|
|
Mix_SetError("Unknown samplesize in data format");
|
|
was_error = 1;
|
|
goto done;
|
|
}
|
|
spec->channels = (Uint8) channels;
|
|
spec->samples = 4096; /* Good default buffer size */
|
|
|
|
done:
|
|
if ( was_error ) {
|
|
return NULL;
|
|
}
|
|
return(src);
|
|
}
|