libgui/source/sounds/SoundHandler.cpp

310 lines
9.8 KiB
C++
Raw Normal View History

2017-10-29 10:28:14 +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 <malloc.h>
2018-06-21 20:44:58 +02:00
#include <fs/CFile.hpp>
#include <sounds/SoundHandler.hpp>
#include <sounds/WavDecoder.hpp>
#include <sounds/Mp3Decoder.hpp>
#include <sounds/OggDecoder.hpp>
#include <sndcore2/core.h>
2017-10-29 10:28:14 +01:00
SoundHandler * SoundHandler::handlerInstance = NULL;
SoundHandler::SoundHandler()
2018-06-21 20:44:58 +02:00
: CThread(CThread::eAttributeAffCore1 | CThread::eAttributePinnedAff, 0, 0x8000) {
Decoding = false;
ExitRequested = false;
for(uint32_t i = 0; i < MAX_DECODERS; ++i) {
DecoderList[i] = NULL;
2017-10-29 10:28:14 +01:00
voiceList[i] = NULL;
}
resumeThread();
//! wait for initialization
while(!isThreadSuspended())
2018-06-21 20:44:58 +02:00
OSSleepTicks(OSMicrosecondsToTicks(1000));
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
SoundHandler::~SoundHandler() {
ExitRequested = true;
ThreadSignal();
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
ClearDecoderList();
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
void SoundHandler::AddDecoder(int32_t voice, const char * filepath) {
if(voice < 0 || voice >= MAX_DECODERS)
return;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
if(DecoderList[voice] != NULL)
RemoveDecoder(voice);
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
DecoderList[voice] = GetSoundDecoder(filepath);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
void SoundHandler::AddDecoder(int32_t voice, const uint8_t * snd, int32_t len) {
if(voice < 0 || voice >= MAX_DECODERS)
return;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
if(DecoderList[voice] != NULL)
RemoveDecoder(voice);
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
DecoderList[voice] = GetSoundDecoder(snd, len);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
void SoundHandler::RemoveDecoder(int32_t voice) {
if(voice < 0 || voice >= MAX_DECODERS)
return;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
if(DecoderList[voice] != NULL) {
if(voiceList[voice] && voiceList[voice]->getState() != Voice::STATE_STOPPED) {
2017-10-29 10:28:14 +01:00
if(voiceList[voice]->getState() != Voice::STATE_STOP)
voiceList[voice]->setState(Voice::STATE_STOP);
while(voiceList[voice]->getState() != Voice::STATE_STOPPED)
2018-06-21 20:44:58 +02:00
OSSleepTicks(OSMicrosecondsToTicks(1000));
2017-10-29 10:28:14 +01:00
}
SoundDecoder *decoder = DecoderList[voice];
decoder->Lock();
DecoderList[voice] = NULL;
decoder->Unlock();
2018-06-21 20:44:58 +02:00
delete decoder;
2017-10-29 10:28:14 +01:00
}
}
2018-06-21 20:44:58 +02:00
void SoundHandler::ClearDecoderList() {
for(uint32_t i = 0; i < MAX_DECODERS; ++i)
RemoveDecoder(i);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
static inline bool CheckMP3Signature(const uint8_t * buffer) {
const char MP3_Magic[][3] = {
{'I', 'D', '3'}, //'ID3'
{0xff, 0xfe}, //'MPEG ADTS, layer III, v1.0 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xff}, //'MPEG ADTS, layer III, v1.0', 'mp3', 'audio/mpeg'),
{0xff, 0xfa}, //'MPEG ADTS, layer III, v1.0 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xfb}, //'MPEG ADTS, layer III, v1.0', 'mp3', 'audio/mpeg'),
{0xff, 0xf2}, //'MPEG ADTS, layer III, v2.0 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xf3}, //'MPEG ADTS, layer III, v2.0', 'mp3', 'audio/mpeg'),
{0xff, 0xf4}, //'MPEG ADTS, layer III, v2.0 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xf5}, //'MPEG ADTS, layer III, v2.0', 'mp3', 'audio/mpeg'),
{0xff, 0xf6}, //'MPEG ADTS, layer III, v2.0 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xf7}, //'MPEG ADTS, layer III, v2.0', 'mp3', 'audio/mpeg'),
{0xff, 0xe2}, //'MPEG ADTS, layer III, v2.5 [protected]', 'mp3', 'audio/mpeg'),
{0xff, 0xe3}, //'MPEG ADTS, layer III, v2.5', 'mp3', 'audio/mpeg'),
};
if(buffer[0] == MP3_Magic[0][0] && buffer[1] == MP3_Magic[0][1] &&
buffer[2] == MP3_Magic[0][2]) {
return true;
}
for(int32_t i = 1; i < 13; i++) {
if(buffer[0] == MP3_Magic[i][0] && buffer[1] == MP3_Magic[i][1])
return true;
}
return false;
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
SoundDecoder * SoundHandler::GetSoundDecoder(const char * filepath) {
uint32_t magic;
CFile f(filepath, CFile::ReadOnly);
if(f.size() == 0)
return NULL;
do {
f.read((uint8_t *) &magic, 1);
} while(((uint8_t *) &magic)[0] == 0 && f.tell() < f.size());
if(f.tell() == f.size())
return NULL;
f.seek(f.tell()-1, SEEK_SET);
f.read((uint8_t *) &magic, 4);
f.close();
if(magic == 0x4f676753) { // 'OggS'
return new OggDecoder(filepath);
} else if(magic == 0x52494646) { // 'RIFF'
return new WavDecoder(filepath);
} else if(CheckMP3Signature((uint8_t *) &magic) == true) {
return new Mp3Decoder(filepath);
}
return new SoundDecoder(filepath);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
SoundDecoder * SoundHandler::GetSoundDecoder(const uint8_t * sound, int32_t length) {
const uint8_t * check = sound;
int32_t counter = 0;
while(check[0] == 0 && counter < length) {
check++;
counter++;
}
if(counter >= length)
return NULL;
uint32_t * magic = (uint32_t *) check;
if(magic[0] == 0x4f676753) { // 'OggS'
return new OggDecoder(sound, length);
} else if(magic[0] == 0x52494646) { // 'RIFF'
return new WavDecoder(sound, length);
} else if(CheckMP3Signature(check) == true) {
return new Mp3Decoder(sound, length);
}
return new SoundDecoder(sound, length);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
void SoundHandler::executeThread() {
/*// v2 sound lib can not properly end transition audio on old firmwares
2017-10-29 10:28:14 +01:00
if (OS_FIRMWARE >= 400 && OS_FIRMWARE <= 410)
{
ProperlyEndTransitionAudio();
2018-06-21 20:44:58 +02:00
}*/
2017-10-29 10:28:14 +01:00
//! initialize 48 kHz renderer
2018-06-21 20:44:58 +02:00
AXInitParams params;
memset(&params, 0, sizeof(params));
params.renderer = AX_INIT_RENDERER_48KHZ;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
// TODO: handle support for 3.1.0 with dynamic libs instead of static linking it
//if(AXInitWithParams != 0)
AXInitWithParams(&params);
//else
// AXInit();
2017-10-29 10:28:14 +01:00
// The problem with last voice on 500 was caused by it having priority 0
// We would need to change this priority distribution if for some reason
// we would need MAX_DECODERS > Voice::PRIO_MAX
2018-06-21 20:44:58 +02:00
for(uint32_t i = 0; i < MAX_DECODERS; ++i) {
int32_t priority = (MAX_DECODERS - i) * Voice::PRIO_MAX / MAX_DECODERS;
2017-10-29 10:28:14 +01:00
voiceList[i] = new Voice(priority); // allocate voice 0 with highest priority
}
2018-06-21 20:44:58 +02:00
AXRegisterAppFrameCallback(SoundHandler::axFrameCallback);
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
uint16_t i = 0;
while (!ExitRequested) {
suspendThread();
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
for(i = 0; i < MAX_DECODERS; ++i) {
if(DecoderList[i] == NULL)
continue;
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
Decoding = true;
if(DecoderList[i])
2017-10-29 10:28:14 +01:00
DecoderList[i]->Lock();
2018-06-21 20:44:58 +02:00
if(DecoderList[i])
2017-10-29 10:28:14 +01:00
DecoderList[i]->Decode();
2018-06-21 20:44:58 +02:00
if(DecoderList[i])
2017-10-29 10:28:14 +01:00
DecoderList[i]->Unlock();
2018-06-21 20:44:58 +02:00
}
Decoding = false;
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
for(uint32_t i = 0; i < MAX_DECODERS; ++i)
2017-10-29 10:28:14 +01:00
voiceList[i]->stop();
2018-06-21 20:44:58 +02:00
AXRegisterAppFrameCallback(NULL);
2017-10-29 10:28:14 +01:00
AXQuit();
2018-06-21 20:44:58 +02:00
for(uint32_t i = 0; i < MAX_DECODERS; ++i) {
2017-10-29 10:28:14 +01:00
delete voiceList[i];
voiceList[i] = NULL;
}
}
2018-06-21 20:44:58 +02:00
void SoundHandler::axFrameCallback(void) {
for (uint32_t i = 0; i < MAX_DECODERS; i++) {
2017-10-29 10:28:14 +01:00
Voice *voice = handlerInstance->getVoice(i);
2018-06-21 20:44:58 +02:00
switch (voice->getState()) {
default:
case Voice::STATE_STOPPED:
break;
case Voice::STATE_START: {
SoundDecoder * decoder = handlerInstance->getDecoder(i);
decoder->Lock();
if(decoder->IsBufferReady()) {
const uint8_t *buffer = decoder->GetBuffer();
const uint32_t bufferSize = decoder->GetBufferSize();
decoder->LoadNext();
const uint8_t *nextBuffer = NULL;
uint32_t nextBufferSize = 0;
if(decoder->IsBufferReady()) {
nextBuffer = decoder->GetBuffer();
nextBufferSize = decoder->GetBufferSize();
2017-10-29 10:28:14 +01:00
decoder->LoadNext();
2018-06-21 20:44:58 +02:00
}
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
voice->play(buffer, bufferSize, nextBuffer, nextBufferSize, decoder->GetFormat() & 0xff, decoder->GetSampleRate());
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
handlerInstance->ThreadSignal();
2017-10-29 10:28:14 +01:00
2018-06-21 20:44:58 +02:00
voice->setState(Voice::STATE_PLAYING);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
decoder->Unlock();
break;
}
case Voice::STATE_PLAYING:
if(voice->getInternState() == 1) {
if(voice->isBufferSwitched()) {
SoundDecoder * decoder = handlerInstance->getDecoder(i);
decoder->Lock();
if(decoder->IsBufferReady()) {
voice->setNextBuffer(decoder->GetBuffer(), decoder->GetBufferSize());
decoder->LoadNext();
handlerInstance->ThreadSignal();
} else if(decoder->IsEOF()) {
voice->setState(Voice::STATE_STOP);
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
decoder->Unlock();
2017-10-29 10:28:14 +01:00
}
2018-06-21 20:44:58 +02:00
} else {
2017-10-29 10:28:14 +01:00
voice->setState(Voice::STATE_STOPPED);
2018-06-21 20:44:58 +02:00
}
break;
case Voice::STATE_STOP:
if(voice->getInternState() != 0)
voice->stop();
voice->setState(Voice::STATE_STOPPED);
break;
2017-10-29 10:28:14 +01:00
}
}
}