/*
 * 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;
}