mirror of
https://github.com/mudita/MuditaOS.git
synced 2026-01-17 02:19:06 -05:00
Verify stack usage of: - ServiceAudio - application music player - decoder worker Change if possible to a realistic value keeping a 30% margin. Stack usage reduction is limited due to an extensive stack usage of every call to the filesystem, which uses around 2 kB of stack. Signed-off-by: Marcin Smoczyński <smoczynski.marcin@gmail.com>
120 lines
3.5 KiB
C++
120 lines
3.5 KiB
C++
// Copyright (c) 2017-2020, Mudita Sp. z.o.o. All rights reserved.
|
|
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
|
|
|
|
#include "DecoderWorker.hpp"
|
|
#include "Audio/decoder/Decoder.hpp"
|
|
|
|
audio::DecoderWorker::DecoderWorker(Stream *audioStreamOut, Decoder *decoder, EndOfFileCallback endOfFileCallback)
|
|
: sys::Worker(DecoderWorker::workerName, DecoderWorker::workerPriority, stackDepth), audioStreamOut(audioStreamOut),
|
|
decoder(decoder), endOfFileCallback(endOfFileCallback),
|
|
bufferSize(audioStreamOut->getBlockSize() / sizeof(BufferInternalType))
|
|
{}
|
|
|
|
audio::DecoderWorker::~DecoderWorker()
|
|
{
|
|
audioStreamOut->unregisterListeners(queueListener.get());
|
|
}
|
|
|
|
auto audio::DecoderWorker::init(std::list<sys::WorkerQueueInfo> queues) -> bool
|
|
{
|
|
std::list<sys::WorkerQueueInfo> list{
|
|
{listenerQueueName, StreamQueuedEventsListener::listenerElementSize, listenerQueueCapacity}};
|
|
|
|
auto isSuccessful = Worker::init(list);
|
|
|
|
queueListener = std::make_unique<StreamQueuedEventsListener>(getQueueByName(listenerQueueName));
|
|
if (!queueListener) {
|
|
return false;
|
|
}
|
|
|
|
audioStreamOut->registerListener(queueListener.get());
|
|
|
|
decoderBuffer = std::make_unique<BufferInternalType[]>(bufferSize);
|
|
if (!decoderBuffer) {
|
|
return false;
|
|
}
|
|
|
|
return isSuccessful;
|
|
}
|
|
|
|
bool audio::DecoderWorker::handleMessage(uint32_t queueID)
|
|
{
|
|
auto queue = queues[queueID];
|
|
auto &queueName = queue->GetQueueName();
|
|
if (queueName == listenerQueueName && queueListener) {
|
|
auto event = queueListener->getEvent();
|
|
|
|
switch (event.second) {
|
|
case Stream::Event::StreamOverflow:
|
|
break;
|
|
case Stream::Event::StreamUnderFlow:
|
|
break;
|
|
case Stream::Event::NoEvent:
|
|
break;
|
|
case Stream::Event::StreamFull:
|
|
break;
|
|
case Stream::Event::StreamHalfUsed:
|
|
[[fallthrough]];
|
|
case Stream::Event::StreamEmpty:
|
|
pushAudioData();
|
|
}
|
|
}
|
|
else if (queueName == SERVICE_QUEUE_NAME) {
|
|
auto &serviceQueue = getServiceQueue();
|
|
sys::WorkerCommand cmd;
|
|
|
|
if (serviceQueue.Dequeue(&cmd)) {
|
|
switch (static_cast<Command>(cmd.command)) {
|
|
case Command::EnablePlayback: {
|
|
playbackEnabled = true;
|
|
stateSemaphore.Give();
|
|
pushAudioData();
|
|
break;
|
|
}
|
|
case Command::DisablePlayback: {
|
|
playbackEnabled = false;
|
|
stateSemaphore.Give();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void audio::DecoderWorker::pushAudioData()
|
|
{
|
|
auto samplesRead = 0;
|
|
|
|
while (!audioStreamOut->isFull() && playbackEnabled) {
|
|
samplesRead = decoder->decode(bufferSize, decoderBuffer.get());
|
|
|
|
if (samplesRead == 0) {
|
|
endOfFileCallback();
|
|
break;
|
|
}
|
|
|
|
if (!audioStreamOut->push(decoderBuffer.get(), samplesRead * sizeof(BufferInternalType))) {
|
|
LOG_FATAL("Decoder failed to push to stream.");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool audio::DecoderWorker::enablePlayback()
|
|
{
|
|
return sendCommand({.command = static_cast<uint32_t>(Command::EnablePlayback), .data = nullptr}) &&
|
|
stateChangeWait();
|
|
}
|
|
|
|
bool audio::DecoderWorker::disablePlayback()
|
|
{
|
|
return sendCommand({.command = static_cast<uint32_t>(Command::DisablePlayback), .data = nullptr}) &&
|
|
stateChangeWait();
|
|
}
|
|
|
|
bool audio::DecoderWorker::stateChangeWait()
|
|
{
|
|
return stateSemaphore.Take();
|
|
}
|