/* * Copyright (C) 2016 Jared Boone, ShareBrained Technology, Inc. * Copyright (C) 2016 Furrtek * * This file is part of PortaPack. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, * Boston, MA 02110-1301, USA. */ #include "replay_thread.hpp" #include "baseband_api.hpp" #include "buffer_exchange.hpp" struct BasebandReplay { BasebandReplay(ReplayConfig* const config) { baseband::replay_start(config); } ~BasebandReplay() { baseband::replay_stop(); } }; // ReplayThread /////////////////////////////////////////////////////////// ReplayThread::ReplayThread( std::unique_ptr<stream::Reader> reader, size_t read_size, size_t buffer_count, bool* ready_signal, std::function<void(uint32_t return_code)> terminate_callback ) : config { read_size, buffer_count }, reader { std::move(reader) }, ready_sig { ready_signal }, terminate_callback { std::move(terminate_callback) } { // Need significant stack for FATFS thread = chThdCreateFromHeap(NULL, 1024, NORMALPRIO + 10, ReplayThread::static_fn, this); } ReplayThread::~ReplayThread() { if( thread ) { chThdTerminate(thread); chThdWait(thread); thread = nullptr; } } msg_t ReplayThread::static_fn(void* arg) { auto obj = static_cast<ReplayThread*>(arg); const auto return_code = obj->run(); if( obj->terminate_callback ) { obj->terminate_callback(return_code); } return 0; } uint32_t ReplayThread::run() { BasebandReplay replay { &config }; BufferExchange buffers { &config }; StreamBuffer* prefill_buffer { nullptr }; // Wait for FIFOs to be allocated in baseband // Wait for ui_replay_view to tell us that the buffers are ready (awful :( ) while (!(*ready_sig)) { chThdSleep(100); }; // While empty buffers fifo is not empty... while (!buffers.empty()) { prefill_buffer = buffers.get_prefill(); if (prefill_buffer == nullptr) { buffers.put_app(prefill_buffer); } else { size_t blocks = config.read_size / 512; for (size_t c = 0; c < blocks; c++) { auto read_result = reader->read(&((uint8_t*)prefill_buffer->data())[c * 512], 512); if( read_result.is_error() ) { return READ_ERROR; } } prefill_buffer->set_size(config.read_size); buffers.put(prefill_buffer); } }; baseband::set_fifo_data(nullptr); while( !chThdShouldTerminate() ) { auto buffer = buffers.get(); auto read_result = reader->read(buffer->data(), buffer->capacity()); if( read_result.is_error() ) { return READ_ERROR; } else { if (read_result.value() == 0) { return END_OF_FILE; } } buffer->set_size(buffer->capacity()); buffers.put(buffer); } return TERMINATED; }