2012-01-21 21:57:41 +01:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright (C) 2010
|
|
|
|
* by Dimok
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* for WiiXplorer 2010
|
|
|
|
***************************************************************************/
|
|
|
|
#include <unistd.h>
|
|
|
|
#include "OggDecoder.hpp"
|
|
|
|
|
|
|
|
extern "C" int ogg_read(void * punt, int bytes, int blocks, int *f)
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
return ((CFile *) f)->read((u8 *) punt, bytes*blocks);
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int ogg_seek(int *f, ogg_int64_t offset, int mode)
|
|
|
|
{
|
|
|
|
return ((CFile *) f)->seek((u64) offset, mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int ogg_close(int *f)
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
((CFile*)f)->close();
|
2012-01-21 21:57:41 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" long ogg_tell(int *f)
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
return (long)((CFile *)f)->tell();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static ov_callbacks callbacks = {
|
2012-07-26 00:12:17 +02:00
|
|
|
(size_t (*)(void *, size_t, size_t, void *)) ogg_read,
|
|
|
|
(int (*)(void *, ogg_int64_t, int)) ogg_seek,
|
|
|
|
(int (*)(void *)) ogg_close,
|
|
|
|
(long (*)(void *)) ogg_tell
|
2012-01-21 21:57:41 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
OggDecoder::OggDecoder(const char * filepath)
|
2012-07-26 00:12:17 +02:00
|
|
|
: SoundDecoder(filepath)
|
2012-01-21 21:57:41 +01:00
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
SoundType = SOUND_OGG;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
if(!file_fd)
|
|
|
|
return;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
OpenFile();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
OggDecoder::OggDecoder(const u8 * snd, int len)
|
2012-07-26 00:12:17 +02:00
|
|
|
: SoundDecoder(snd, len)
|
2012-01-21 21:57:41 +01:00
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
SoundType = SOUND_OGG;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
if(!file_fd)
|
|
|
|
return;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
OpenFile();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
OggDecoder::~OggDecoder()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
ExitRequested = true;
|
|
|
|
while(Decoding)
|
|
|
|
usleep(100);
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
if(file_fd)
|
|
|
|
ov_clear(&ogg_file);
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void OggDecoder::OpenFile()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
if(ov_open_callbacks(file_fd, &ogg_file, NULL, 0, callbacks) < 0)
|
|
|
|
{
|
|
|
|
delete file_fd;
|
|
|
|
file_fd = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ogg_info = ov_info(&ogg_file, -1);
|
2020-05-26 00:17:50 +02:00
|
|
|
if(!ogg_info)
|
|
|
|
{
|
|
|
|
ov_clear(&ogg_file);
|
|
|
|
delete file_fd;
|
|
|
|
file_fd = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Format = ((ogg_info->channels == 2) ? VOICE_STEREO_16BIT : VOICE_MONO_16BIT);
|
|
|
|
SampleRate = ogg_info->rate;
|
2012-07-26 00:12:17 +02:00
|
|
|
Decode();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int OggDecoder::Rewind()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
if(!file_fd)
|
|
|
|
return -1;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
int ret = ov_time_seek(&ogg_file, 0);
|
|
|
|
CurPos = 0;
|
|
|
|
EndOfFile = false;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
return ret;
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
2012-09-15 23:20:22 +02:00
|
|
|
int OggDecoder::Tell()
|
|
|
|
{
|
|
|
|
return ov_raw_tell(&ogg_file);
|
|
|
|
}
|
|
|
|
|
|
|
|
int OggDecoder::Seek(int pos)
|
|
|
|
{
|
|
|
|
return ov_raw_seek(&ogg_file, pos);
|
|
|
|
}
|
|
|
|
|
2017-09-22 00:03:41 +02:00
|
|
|
int OggDecoder::Read(u8 * buffer, int buffer_size)
|
2012-01-21 21:57:41 +01:00
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
if(!file_fd)
|
|
|
|
return -1;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
int bitstream = 0;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
int read = ov_read(&ogg_file, (char *) buffer, buffer_size, &bitstream);
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
if(read > 0)
|
|
|
|
CurPos += read;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
return read;
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|