add streaming input support for FLAC

git-svn-id: https://svn.musicpd.org/mpd/trunk@2912 09075e82-0dd4-0310-85a5-a0d7c8717e4f
This commit is contained in:
Eric Wong 2005-01-29 23:46:06 +00:00
parent fc4dc6c229
commit 0cf6cc4ada

View File

@ -43,8 +43,7 @@ typedef struct {
FLAC__uint64 position; FLAC__uint64 position;
OutputBuffer * cb; OutputBuffer * cb;
DecoderControl * dc; DecoderControl * dc;
InputStream inStream; InputStream * inStream;
char * path;
ReplayGainInfo * replayGainInfo; ReplayGainInfo * replayGainInfo;
} FlacData; } FlacData;
@ -53,7 +52,7 @@ typedef struct {
int flacSendChunk(FlacData * data); int flacSendChunk(FlacData * data);
void flacError(const FLAC__SeekableStreamDecoder *, void flacError(const FLAC__SeekableStreamDecoder *,
FLAC__StreamDecoderErrorStatus, void *); FLAC__StreamDecoderErrorStatus, void *);
void flacPrintErroredState(FLAC__SeekableStreamDecoderState state, char * file); void flacPrintErroredState(FLAC__SeekableStreamDecoderState state);
void flacMetadata(const FLAC__SeekableStreamDecoder *, void flacMetadata(const FLAC__SeekableStreamDecoder *,
const FLAC__StreamMetadata *, void *); const FLAC__StreamMetadata *, void *);
FLAC__StreamDecoderWriteStatus flacWrite(const FLAC__SeekableStreamDecoder *, FLAC__StreamDecoderWriteStatus flacWrite(const FLAC__SeekableStreamDecoder *,
@ -69,12 +68,12 @@ FLAC__SeekableStreamDecoderLengthStatus flacLength(
const FLAC__SeekableStreamDecoder *, FLAC__uint64 *, void *); const FLAC__SeekableStreamDecoder *, FLAC__uint64 *, void *);
FLAC__bool flacEOF(const FLAC__SeekableStreamDecoder *, void *); FLAC__bool flacEOF(const FLAC__SeekableStreamDecoder *, void *);
int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) { int flac_decode(OutputBuffer * cb, DecoderControl * dc, InputStream * inStream)
{
FLAC__SeekableStreamDecoder * flacDec = NULL; FLAC__SeekableStreamDecoder * flacDec = NULL;
FlacData data; FlacData data;
int status = 1; int status = 1;
int ret =0; int ret =0;
int streamOpen = 0;
data.chunk_length = 0; data.chunk_length = 0;
data.time = 0; data.time = 0;
@ -82,16 +81,9 @@ int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) {
data.bitRate = 0; data.bitRate = 0;
data.cb = cb; data.cb = cb;
data.dc = dc; data.dc = dc;
data.inStream = inStream;
data.replayGainInfo = NULL; data.replayGainInfo = NULL;
data.path = path;
if(openInputStream(&(data.inStream), path)<0) {
ERROR("unable to open flac: %s\n", path);
ret = -1;
goto fail;
}
streamOpen = 1;
if(!(flacDec = FLAC__seekable_stream_decoder_new())) { if(!(flacDec = FLAC__seekable_stream_decoder_new())) {
ret = -1; ret = -1;
goto fail; goto fail;
@ -117,10 +109,9 @@ int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) {
status&=FLAC__seekable_stream_decoder_set_client_data(flacDec, status&=FLAC__seekable_stream_decoder_set_client_data(flacDec,
(void *)&data); (void *)&data);
if(!status) { if(!status) {
ERROR("flac problem before init(): %s\n", path); ERROR("flac problem before init()\n");
flacPrintErroredState( flacPrintErroredState(
FLAC__seekable_stream_decoder_get_state(flacDec), FLAC__seekable_stream_decoder_get_state(flacDec));
path);
ret = -1; ret = -1;
goto fail; goto fail;
} }
@ -128,19 +119,17 @@ int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) {
if(FLAC__seekable_stream_decoder_init(flacDec)!= if(FLAC__seekable_stream_decoder_init(flacDec)!=
FLAC__STREAM_DECODER_SEARCH_FOR_METADATA) FLAC__STREAM_DECODER_SEARCH_FOR_METADATA)
{ {
ERROR("flac problem doing init(): %s\n", path); ERROR("flac problem doing init()\n");
flacPrintErroredState( flacPrintErroredState(
FLAC__seekable_stream_decoder_get_state(flacDec), FLAC__seekable_stream_decoder_get_state(flacDec));
path);
ret = -1; ret = -1;
goto fail; goto fail;
} }
if(!FLAC__seekable_stream_decoder_process_until_end_of_metadata(flacDec)) { if(!FLAC__seekable_stream_decoder_process_until_end_of_metadata(flacDec)) {
ERROR("flac problem reading metadata: %s\n", path); ERROR("flac problem reading metadata\n");
flacPrintErroredState( flacPrintErroredState(
FLAC__seekable_stream_decoder_get_state(flacDec), FLAC__seekable_stream_decoder_get_state(flacDec));
path);
ret = -1; ret = -1;
goto fail; goto fail;
} }
@ -173,8 +162,7 @@ int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) {
/*FLAC__file_decoder_process_until_end_of_file(flacDec);*/ /*FLAC__file_decoder_process_until_end_of_file(flacDec);*/
if(!dc->stop) { if(!dc->stop) {
flacPrintErroredState( flacPrintErroredState(
FLAC__seekable_stream_decoder_get_state(flacDec), FLAC__seekable_stream_decoder_get_state(flacDec));
path);
FLAC__seekable_stream_decoder_finish(flacDec); FLAC__seekable_stream_decoder_finish(flacDec);
} }
/* send last little bit */ /* send last little bit */
@ -197,8 +185,8 @@ int flac_decode(OutputBuffer * cb, DecoderControl *dc, char * path) {
fail: fail:
if(data.replayGainInfo) freeReplayGainInfo(data.replayGainInfo); if(data.replayGainInfo) freeReplayGainInfo(data.replayGainInfo);
if(streamOpen) closeInputStream(&(data.inStream)); closeInputStream(inStream);
if(flacDec) FLAC__seekable_stream_decoder_delete(flacDec); if(flacDec) FLAC__seekable_stream_decoder_delete(flacDec);
return ret; return ret;
@ -209,7 +197,7 @@ FLAC__SeekableStreamDecoderReadStatus flacRead(
unsigned * bytes, void * fdata) { unsigned * bytes, void * fdata) {
FlacData * data = (FlacData *) fdata; FlacData * data = (FlacData *) fdata;
*bytes = readFromInputStream(&(data->inStream),(void *)buf,1,*bytes); *bytes = readFromInputStream(data->inStream,(void *)buf,1,*bytes);
if(*bytes==0) return FLAC__SEEKABLE_STREAM_DECODER_READ_STATUS_ERROR; if(*bytes==0) return FLAC__SEEKABLE_STREAM_DECODER_READ_STATUS_ERROR;
@ -222,7 +210,7 @@ FLAC__SeekableStreamDecoderSeekStatus flacSeek(
{ {
FlacData * data = (FlacData *) fdata; FlacData * data = (FlacData *) fdata;
if(seekInputStream(&(data->inStream),offset,SEEK_SET)<0) { if(seekInputStream(data->inStream,offset,SEEK_SET)<0) {
return FLAC__SEEKABLE_STREAM_DECODER_SEEK_STATUS_ERROR; return FLAC__SEEKABLE_STREAM_DECODER_SEEK_STATUS_ERROR;
} }
@ -235,7 +223,7 @@ FLAC__SeekableStreamDecoderTellStatus flacTell(
{ {
FlacData * data = (FlacData *) fdata; FlacData * data = (FlacData *) fdata;
*offset = data->inStream.offset; *offset = (long)(data->inStream->offset);
return FLAC__SEEKABLE_STREAM_DECODER_TELL_STATUS_OK; return FLAC__SEEKABLE_STREAM_DECODER_TELL_STATUS_OK;
} }
@ -246,7 +234,7 @@ FLAC__SeekableStreamDecoderLengthStatus flacLength(
{ {
FlacData * data = (FlacData *) fdata; FlacData * data = (FlacData *) fdata;
*length = data->inStream.size; *length = (size_t)(data->inStream->size);
return FLAC__SEEKABLE_STREAM_DECODER_LENGTH_STATUS_OK; return FLAC__SEEKABLE_STREAM_DECODER_LENGTH_STATUS_OK;
} }
@ -254,7 +242,7 @@ FLAC__SeekableStreamDecoderLengthStatus flacLength(
FLAC__bool flacEOF(const FLAC__SeekableStreamDecoder * flacDec, void * fdata) { FLAC__bool flacEOF(const FLAC__SeekableStreamDecoder * flacDec, void * fdata) {
FlacData * data = (FlacData *) fdata; FlacData * data = (FlacData *) fdata;
switch(inputStreamAtEOF(&(data->inStream))) { switch(inputStreamAtEOF(data->inStream)) {
case 1: case 1:
return true; return true;
default: default:
@ -270,43 +258,42 @@ void flacError(const FLAC__SeekableStreamDecoder *dec,
switch(status) { switch(status) {
case FLAC__STREAM_DECODER_ERROR_STATUS_LOST_SYNC: case FLAC__STREAM_DECODER_ERROR_STATUS_LOST_SYNC:
ERROR("flac lost sync: %s\n", data->path); ERROR("flac lost sync\n");
break; break;
case FLAC__STREAM_DECODER_ERROR_STATUS_BAD_HEADER: case FLAC__STREAM_DECODER_ERROR_STATUS_BAD_HEADER:
ERROR("bad header %s\n", data->path); ERROR("bad header\n");
break; break;
case FLAC__STREAM_DECODER_ERROR_STATUS_FRAME_CRC_MISMATCH: case FLAC__STREAM_DECODER_ERROR_STATUS_FRAME_CRC_MISMATCH:
ERROR("crc mismatch %s\n", data->path); ERROR("crc mismatch\n");
break; break;
default: default:
ERROR("unknown flac error %s\n", data->path); ERROR("unknown flac error\n");
} }
} }
void flacPrintErroredState(FLAC__SeekableStreamDecoderState state, void flacPrintErroredState(FLAC__SeekableStreamDecoderState state)
char * file)
{ {
switch(state) { switch(state) {
case FLAC__SEEKABLE_STREAM_DECODER_MEMORY_ALLOCATION_ERROR: case FLAC__SEEKABLE_STREAM_DECODER_MEMORY_ALLOCATION_ERROR:
ERROR("flac allocation error\n"); ERROR("flac allocation error\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_READ_ERROR: case FLAC__SEEKABLE_STREAM_DECODER_READ_ERROR:
ERROR("flac read error: %s\n",file); ERROR("flac read error\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_SEEK_ERROR: case FLAC__SEEKABLE_STREAM_DECODER_SEEK_ERROR:
ERROR("flac seek error: %s\n",file); ERROR("flac seek error\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_STREAM_DECODER_ERROR: case FLAC__SEEKABLE_STREAM_DECODER_STREAM_DECODER_ERROR:
ERROR("flac seekable stream error: %s\n",file); ERROR("flac seekable stream error\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_ALREADY_INITIALIZED: case FLAC__SEEKABLE_STREAM_DECODER_ALREADY_INITIALIZED:
ERROR("flac decoder already initialized: %s\n",file); ERROR("flac decoder already initialized\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_INVALID_CALLBACK: case FLAC__SEEKABLE_STREAM_DECODER_INVALID_CALLBACK:
ERROR("invalid flac callback\n"); ERROR("invalid flac callback\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_UNINITIALIZED: case FLAC__SEEKABLE_STREAM_DECODER_UNINITIALIZED:
ERROR("flac decoder uninitialized: %s\n",file); ERROR("flac decoder uninitialized\n");
break; break;
case FLAC__SEEKABLE_STREAM_DECODER_OK: case FLAC__SEEKABLE_STREAM_DECODER_OK:
case FLAC__SEEKABLE_STREAM_DECODER_SEEKING: case FLAC__SEEKABLE_STREAM_DECODER_SEEKING:
@ -567,18 +554,19 @@ MpdTag * flacTagDup(char * file) {
} }
char * flacSuffixes[] = {"flac", NULL}; char * flacSuffixes[] = {"flac", NULL};
char * flac_mime_types[] = {"application/x-flac", NULL};
InputPlugin flacPlugin = InputPlugin flacPlugin =
{ {
"flac", "flac",
NULL, NULL,
NULL, NULL,
NULL,
flac_decode, flac_decode,
NULL,
flacTagDup, flacTagDup,
INPUT_PLUGIN_STREAM_FILE, INPUT_PLUGIN_STREAM_URL | INPUT_PLUGIN_STREAM_FILE,
flacSuffixes, flacSuffixes,
NULL flac_mime_types
}; };
#else #else