/*
 * Copyright (C) 2016 Jared Boone, ShareBrained Technology, Inc.
 *
 * 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 "capture_thread.hpp"

#include "baseband_api.hpp"
#include "buffer_exchange.hpp"

struct BasebandCapture {
	BasebandCapture(CaptureConfig* const config) {
		baseband::capture_start(config);
	}

	~BasebandCapture() {
		baseband::capture_stop();
	}
};

// CaptureThread //////////////////////////////////////////////////////////

CaptureThread::CaptureThread(
	std::unique_ptr<stream::Writer> writer,
	size_t write_size,
	size_t buffer_count,
	std::function<void()> success_callback,
	std::function<void(File::Error)> error_callback
) : config { write_size, buffer_count },
	writer { std::move(writer) },
	success_callback { std::move(success_callback) },
	error_callback { std::move(error_callback) }
{
	// Need significant stack for FATFS
	thread = chThdCreateFromHeap(NULL, 1024, NORMALPRIO + 10, CaptureThread::static_fn, this);
}

CaptureThread::~CaptureThread() {
	if( thread ) {
		chThdTerminate(thread);
		chThdWait(thread);
		thread = nullptr;
	}
}

msg_t CaptureThread::static_fn(void* arg) {
	auto obj = static_cast<CaptureThread*>(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<File::Error> CaptureThread::run() {
	BasebandCapture capture { &config };
	BufferExchange buffers { &config };

	while( !chThdShouldTerminate() ) {
		auto buffer = buffers.get();
		auto write_result = writer->write(buffer->data(), buffer->size());
		if( write_result.is_error() ) {
			return write_result.error();
		}
		buffer->empty();
		buffers.put(buffer);
	}

	return { };
}