Handle file sources with mpg123

This commit is contained in:
Chris Robinson 2014-06-19 20:14:16 -07:00
parent 1310a4a814
commit 364065f121
3 changed files with 157 additions and 6 deletions

View file

@ -549,6 +549,14 @@ SoundDecoder *SoundRenderer::CreateDecoder(const BYTE *sfxdata, int length)
SoundDecoder* SoundRenderer::CreateDecoder(const char *fname, int offset, int length) SoundDecoder* SoundRenderer::CreateDecoder(const char *fname, int offset, int length)
{ {
SoundDecoder *decoder = NULL; SoundDecoder *decoder = NULL;
#ifdef HAVE_MPG123
decoder = new MPG123Decoder;
if(!decoder->open(fname, offset, length))
{
delete decoder;
decoder = NULL;
}
#endif
#ifdef HAVE_SNDFILE #ifdef HAVE_SNDFILE
decoder = new SndFileDecoder; decoder = new SndFileDecoder;
if(!decoder->open(fname, offset, length)) if(!decoder->open(fname, offset, length))

View file

@ -149,7 +149,7 @@ struct MPG123Decoder : public SoundDecoder
virtual bool seek(size_t ms_offset); virtual bool seek(size_t ms_offset);
virtual size_t getSampleOffset(); virtual size_t getSampleOffset();
MPG123Decoder() : MPG123(0) { } MPG123Decoder() : MPG123(0), File(0) { }
virtual ~MPG123Decoder(); virtual ~MPG123Decoder();
protected: protected:
@ -160,6 +160,12 @@ private:
mpg123_handle *MPG123; mpg123_handle *MPG123;
bool Done; bool Done;
FILE *File;
size_t FileLength;
size_t FileOffset;
static off_t file_lseek(void *handle, off_t offset, int whence);
static ssize_t file_read(void *handle, void *buffer, size_t bytes);
const char *MemData; const char *MemData;
size_t MemLength; size_t MemLength;
size_t MemPos; size_t MemPos;

View file

@ -4,6 +4,59 @@
static bool inited = false; static bool inited = false;
off_t MPG123Decoder::file_lseek(void *handle, off_t offset, int whence)
{
MPG123Decoder *self = reinterpret_cast<MPG123Decoder*>(handle);
long cur = ftell(self->File);
if(cur < 0) return -1;
switch(whence)
{
case SEEK_SET:
if(offset < 0 || offset > (off_t)self->FileLength)
return -1;
cur = offset;
break;
case SEEK_CUR:
cur -= self->FileOffset;
if((offset > 0 && (off_t)(self->FileLength-cur) < offset) ||
(offset < 0 && (off_t)cur < -offset))
return -1;
cur += offset;
break;
case SEEK_END:
if(offset > 0 || -offset > (off_t)self->FileLength)
return -1;
cur = self->FileLength + offset;
break;
default:
return -1;
}
if(fseek(self->File, cur + self->FileOffset, SEEK_SET) != 0)
return -1;
return cur;
}
ssize_t MPG123Decoder::file_read(void *handle, void *buffer, size_t bytes)
{
MPG123Decoder *self = reinterpret_cast<MPG123Decoder*>(handle);
long cur = ftell(self->File);
if(cur < 0) return -1;
cur -= self->FileOffset;
if(bytes > (size_t)(self->FileLength-cur))
bytes = self->FileLength-cur;
return fread(buffer, 1, bytes, self->File);
}
off_t MPG123Decoder::mem_lseek(void *handle, off_t offset, int whence) off_t MPG123Decoder::mem_lseek(void *handle, off_t offset, int whence)
{ {
MPG123Decoder *self = reinterpret_cast<MPG123Decoder*>(handle); MPG123Decoder *self = reinterpret_cast<MPG123Decoder*>(handle);
@ -58,6 +111,9 @@ MPG123Decoder::~MPG123Decoder()
mpg123_delete(MPG123); mpg123_delete(MPG123);
MPG123 = 0; MPG123 = 0;
} }
if(File)
fclose(File);
File = 0;
} }
bool MPG123Decoder::open(const char *data, size_t length) bool MPG123Decoder::open(const char *data, size_t length)
@ -135,6 +191,92 @@ bool MPG123Decoder::open(const char *data, size_t length)
return false; return false;
} }
bool MPG123Decoder::open(const char *fname, size_t offset, size_t length)
{
if(!inited)
{
if(mpg123_init() != MPG123_OK)
return false;
inited = true;
}
FileLength = length;
FileOffset = offset;
File = fopen(fname, "rb");
if(!File || fseek(File, FileOffset, SEEK_SET) != 0)
return false;
char data[10];
if(file_read(this, data, 10) != 10)
return false;
// Check for ID3 tags and skip them
if(memcmp(data, "ID3", 3) == 0 &&
(BYTE)data[3] <= 4 && (BYTE)data[4] != 0xff &&
(data[5]&0x0f) == 0 && (data[6]&0x80) == 0 &&
(data[7]&0x80) == 0 && (data[8]&0x80) == 0 &&
(data[9]&0x80) == 0)
{
// ID3v2
int start_offset;
start_offset = (data[6]<<21) | (data[7]<<14) |
(data[8]<< 7) | (data[9] );
start_offset += ((data[5]&0x10) ? 20 : 10);
offset += start_offset;
length -= start_offset;
}
if(file_lseek(this, -128, SEEK_END) > 0 && memcmp(data, "TAG", 3) == 0) // ID3v1
length -= 128;
FileLength = length;
FileOffset = offset;
if(file_lseek(this, 0, SEEK_SET) != 0)
return false;
// Check for a frame header
bool frame_ok = false;
if(file_read(this, data, 3) != 3)
{
if((BYTE)data[0] == 0xff &&
((data[1]&0xfe) == 0xfa/*MPEG-1*/ || (data[1]&0xfe) == 0xf2/*MPEG-2*/))
{
int brate_idx = (data[2]>>4) & 0x0f;
int srate_idx = (data[2]>>2) & 0x03;
if(brate_idx != 0 && brate_idx != 15 && srate_idx != 3)
frame_ok = (file_lseek(this, 0, SEEK_SET) == 0);
}
}
if(frame_ok)
{
MPG123 = mpg123_new(NULL, NULL);
if(mpg123_replace_reader_handle(MPG123, file_read, file_lseek, NULL) == MPG123_OK &&
mpg123_open_handle(MPG123, this) == MPG123_OK)
{
int enc, channels;
long srate;
if(mpg123_getformat(MPG123, &srate, &channels, &enc) == MPG123_OK)
{
if((channels == 1 || channels == 2) && srate > 0 &&
mpg123_format_none(MPG123) == MPG123_OK &&
mpg123_format(MPG123, srate, channels, MPG123_ENC_SIGNED_16) == MPG123_OK)
{
// All OK
Done = false;
return true;
}
}
mpg123_close(MPG123);
}
mpg123_delete(MPG123);
MPG123 = 0;
}
return false;
}
void MPG123Decoder::getInfo(int *samplerate, ChannelConfig *chans, SampleType *type) void MPG123Decoder::getInfo(int *samplerate, ChannelConfig *chans, SampleType *type)
{ {
int enc = 0, channels = 0; int enc = 0, channels = 0;
@ -173,11 +315,6 @@ size_t MPG123Decoder::read(char *buffer, size_t bytes)
return amt; return amt;
} }
bool MPG123Decoder::open(const char *fname, size_t offset, size_t length)
{
return false;
}
bool MPG123Decoder::seek(size_t ms_offset) bool MPG123Decoder::seek(size_t ms_offset)
{ {
int enc, channels; int enc, channels;