2012-01-21 21:57:41 +01:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright (C) 2010
|
|
|
|
* by Dimok
|
|
|
|
*
|
|
|
|
* 3Band resampling thanks to libmad
|
|
|
|
*
|
|
|
|
* 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 <gccore.h>
|
2020-05-26 00:17:50 +02:00
|
|
|
#include <malloc.h>
|
2012-01-21 21:57:41 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include "SoundDecoder.hpp"
|
2020-05-26 00:17:50 +02:00
|
|
|
#include "MusicPlayer.hpp"
|
|
|
|
|
|
|
|
static const u32 FixedPointShift = 15;
|
|
|
|
static const u32 FixedPointScale = 1 << FixedPointShift;
|
2012-01-21 21:57:41 +01:00
|
|
|
|
|
|
|
SoundDecoder::SoundDecoder()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
file_fd = NULL;
|
|
|
|
Init();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SoundDecoder::SoundDecoder(const char * filepath)
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
file_fd = new CFile(filepath, "rb");
|
|
|
|
Init();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SoundDecoder::SoundDecoder(const u8 * buffer, int size)
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
file_fd = new CFile(buffer, size);
|
|
|
|
Init();
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
SoundDecoder::~SoundDecoder()
|
|
|
|
{
|
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)
|
|
|
|
delete file_fd;
|
|
|
|
file_fd = NULL;
|
2020-05-26 00:17:50 +02:00
|
|
|
|
|
|
|
if(ResampleBuffer)
|
|
|
|
free(ResampleBuffer);
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundDecoder::Init()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
SoundType = SOUND_RAW;
|
2012-08-16 19:18:15 +02:00
|
|
|
SoundBlocks = 8;
|
|
|
|
SoundBlockSize = 8192;
|
2020-05-26 00:17:50 +02:00
|
|
|
ResampleTo48kHz = MusicPlayer.ResampleSetting;
|
2012-07-26 00:12:17 +02:00
|
|
|
CurPos = 0;
|
2012-08-12 23:26:24 +02:00
|
|
|
LoopStart = 0;
|
2012-08-31 22:01:53 +02:00
|
|
|
LoopEnd = 0;
|
2012-07-26 00:12:17 +02:00
|
|
|
Loop = false;
|
|
|
|
EndOfFile = false;
|
|
|
|
Decoding = false;
|
|
|
|
ExitRequested = false;
|
|
|
|
SoundBuffer.SetBufferBlockSize(SoundBlockSize);
|
|
|
|
SoundBuffer.Resize(SoundBlocks);
|
2020-05-26 00:17:50 +02:00
|
|
|
ResampleBuffer = NULL;
|
|
|
|
ResampleRatio = 0;
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int SoundDecoder::Rewind()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
CurPos = 0;
|
|
|
|
EndOfFile = false;
|
|
|
|
file_fd->rewind();
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
return 0;
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|
2017-09-22 00:03:41 +02:00
|
|
|
int SoundDecoder::Read(u8 * buffer, int buffer_size)
|
2012-01-21 21:57:41 +01:00
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
int ret = file_fd->read(buffer, buffer_size);
|
|
|
|
CurPos += ret;
|
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
|
|
|
}
|
|
|
|
|
2020-05-26 00:17:50 +02:00
|
|
|
void SoundDecoder::EnableUpsample(void)
|
|
|
|
{
|
|
|
|
if( (ResampleBuffer == NULL)
|
|
|
|
&& IsStereo() && Is16Bit()
|
|
|
|
&& SampleRate != 32000
|
|
|
|
&& SampleRate != 48000)
|
|
|
|
{
|
|
|
|
ResampleBuffer = (u8*)memalign(32, SoundBlockSize);
|
|
|
|
ResampleRatio = ( FixedPointScale * SampleRate ) / 48000;
|
|
|
|
SoundBlockSize = ( SoundBlockSize * ResampleRatio ) / FixedPointScale;
|
|
|
|
SoundBlockSize &= ~0x03;
|
|
|
|
// set new sample rate
|
|
|
|
SampleRate = 48000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundDecoder::Upsample(s16 *src, s16 *dst, u32 nr_src_samples, u32 nr_dst_samples)
|
|
|
|
{
|
|
|
|
int timer = 0;
|
|
|
|
|
|
|
|
for(u32 i = 0, n = 0; i < nr_dst_samples; i += 2)
|
|
|
|
{
|
|
|
|
if((n+3) < nr_src_samples) {
|
|
|
|
// simple fixed point linear interpolation
|
|
|
|
dst[i] = src[n] + ( ((src[n+2] - src[n] ) * timer) >> FixedPointShift );
|
|
|
|
dst[i+1] = src[n+1] + ( ((src[n+3] - src[n+1]) * timer) >> FixedPointShift );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dst[i] = src[n];
|
|
|
|
dst[i+1] = src[n+1];
|
|
|
|
}
|
|
|
|
|
|
|
|
timer += ResampleRatio;
|
|
|
|
|
|
|
|
if(timer >= (int)FixedPointScale) {
|
|
|
|
n += 2;
|
|
|
|
timer -= FixedPointScale;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-21 21:57:41 +01:00
|
|
|
void SoundDecoder::Decode()
|
|
|
|
{
|
2012-07-26 00:12:17 +02:00
|
|
|
if(!file_fd || ExitRequested || EndOfFile)
|
|
|
|
return;
|
|
|
|
|
|
|
|
u16 newWhich = SoundBuffer.Which();
|
|
|
|
u16 i = 0;
|
|
|
|
for(i = 0; i < SoundBuffer.Size()-2; i++)
|
|
|
|
{
|
|
|
|
if(!SoundBuffer.IsBufferReady(newWhich))
|
|
|
|
break;
|
|
|
|
|
|
|
|
newWhich = (newWhich+1) % SoundBuffer.Size();
|
|
|
|
}
|
|
|
|
|
|
|
|
if(i == SoundBuffer.Size()-2)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Decoding = true;
|
|
|
|
|
|
|
|
int done = 0;
|
|
|
|
u8 *write_buf = SoundBuffer.GetBuffer(newWhich);
|
|
|
|
if(!write_buf)
|
|
|
|
{
|
|
|
|
ExitRequested = true;
|
|
|
|
Decoding = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-26 00:17:50 +02:00
|
|
|
//*******************************************
|
|
|
|
if(ResampleTo48kHz && !ResampleBuffer)
|
|
|
|
EnableUpsample();
|
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
while(done < SoundBlockSize)
|
|
|
|
{
|
2017-09-22 00:03:41 +02:00
|
|
|
int ret = Read(&write_buf[done], SoundBlockSize-done);
|
2012-07-26 00:12:17 +02:00
|
|
|
|
|
|
|
if(ret <= 0)
|
|
|
|
{
|
2012-08-31 22:01:53 +02:00
|
|
|
if(Loop || LoopStart || LoopEnd)
|
2012-07-26 00:12:17 +02:00
|
|
|
{
|
|
|
|
Rewind();
|
2012-08-31 22:01:53 +02:00
|
|
|
if(LoopStart)
|
|
|
|
CurPos = LoopStart;
|
2012-08-11 21:47:13 +02:00
|
|
|
continue;
|
|
|
|
}
|
2012-07-26 00:12:17 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
EndOfFile = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
done += ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(done > 0)
|
|
|
|
{
|
2020-05-26 00:17:50 +02:00
|
|
|
// check if we need to resample
|
|
|
|
if(ResampleBuffer && ResampleRatio)
|
|
|
|
{
|
|
|
|
memcpy(ResampleBuffer, write_buf, done);
|
|
|
|
|
|
|
|
int src_samples = done >> 1;
|
|
|
|
int dest_samples = ( src_samples * FixedPointScale ) / ResampleRatio;
|
|
|
|
dest_samples &= ~0x01;
|
|
|
|
Upsample((s16*)ResampleBuffer, (s16*)write_buf, src_samples, dest_samples);
|
|
|
|
done = dest_samples << 1;
|
|
|
|
}
|
2012-07-26 00:12:17 +02:00
|
|
|
SoundBuffer.SetBufferSize(newWhich, done);
|
|
|
|
SoundBuffer.SetBufferReady(newWhich, true);
|
|
|
|
}
|
2012-01-21 21:57:41 +01:00
|
|
|
|
|
|
|
if(!SoundBuffer.IsBufferReady((newWhich+1) % SoundBuffer.Size()))
|
2012-07-26 00:12:17 +02:00
|
|
|
Decode();
|
2012-01-21 21:57:41 +01:00
|
|
|
|
2012-07-26 00:12:17 +02:00
|
|
|
Decoding = false;
|
2012-01-21 21:57:41 +01:00
|
|
|
}
|
|
|
|
|