/* * 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 reader, size_t read_size, size_t buffer_count, bool* ready_signal, std::function success_callback, std::function error_callback ) : config { read_size, buffer_count }, reader { std::move(reader) }, ready_sig { ready_signal }, success_callback { std::move(success_callback) }, error_callback { std::move(error_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(arg); const auto error = obj->run(); if( error.is_valid() && obj->error_callback ) { obj->error_callback(error.value()); } else { if( obj->success_callback ) { obj->success_callback(); } } return 0; } Optional 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 = 16384 / 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_result.error(); } } prefill_buffer->set_size(16384); 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_result.error(); } else { if (read_result.value() == 0) { return { }; } } buffer->set_size(buffer->capacity()); buffers.put(buffer); } return { }; }