/*
 * Copyright (C) 2014 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 "receiver_model.hpp"

#include "portapack_shared_memory.hpp"
#include "radio.hpp"
#include "wm8731.hpp"

// TODO: Nasty. Put this in an #include somewhere, or a shared system state
// object?

extern wolfson::wm8731::WM8731 audio_codec;

rf::Frequency ReceiverModel::tuning_frequency() const {
	return tuning_frequency_;
}

void ReceiverModel::set_tuning_frequency(rf::Frequency f) {
	tuning_frequency_ = f;
	update_tuning_frequency();
}

rf::Frequency ReceiverModel::frequency_step() const {
	return frequency_step_;
}

void ReceiverModel::set_frequency_step(rf::Frequency f) {
	frequency_step_ = f;
}

int32_t ReceiverModel::reference_ppm_correction() const {
	return shared_memory.correction_ppm;
}

void ReceiverModel::set_reference_ppm_correction(int32_t v) {
	shared_memory.correction_ppm = v;
	update_tuning_frequency();
}

bool ReceiverModel::rf_amp() const {
	return rf_amp_;
}

void ReceiverModel::set_rf_amp(bool enabled) {
	rf_amp_ = enabled;
	update_rf_amp();
}

int32_t ReceiverModel::lna() const {
	return lna_gain_db_;
}

void ReceiverModel::set_lna(int32_t v_db) {
	lna_gain_db_ = v_db;
	update_lna();
}

uint32_t ReceiverModel::baseband_bandwidth() const {
	return baseband_bandwidth_;
}

void ReceiverModel::set_baseband_bandwidth(uint32_t v) {
	baseband_bandwidth_ = v;
	update_baseband_bandwidth();
}

int32_t ReceiverModel::vga() const {
	return vga_gain_db_;
}

void ReceiverModel::set_vga(int32_t v_db) {
	vga_gain_db_ = v_db;
	update_vga();
}

uint32_t ReceiverModel::sampling_rate() const {
	return baseband_configuration.sampling_rate;
}

void ReceiverModel::set_sampling_rate(uint32_t hz) {
	baseband_configuration.sampling_rate = hz;
	update_baseband_configuration();
}

uint32_t ReceiverModel::modulation() const {
	return baseband_configuration.mode;
}

void ReceiverModel::set_modulation(uint32_t v) {
	baseband_configuration.mode = v;
	update_modulation();
}

volume_t ReceiverModel::headphone_volume() const {
	return headphone_volume_;
}

void ReceiverModel::set_headphone_volume(volume_t v) {
	headphone_volume_ = v;
	update_headphone_volume();
}

uint32_t ReceiverModel::baseband_oversampling() const {
	// TODO: Rename decimation_factor.
	return baseband_configuration.decimation_factor;
}

void ReceiverModel::set_baseband_oversampling(uint32_t v) {
	baseband_configuration.decimation_factor = v;
	update_baseband_configuration();
}

void ReceiverModel::enable() {
	radio::set_direction(rf::Direction::Receive);
	update_tuning_frequency();
	update_rf_amp();
	update_lna();
	update_vga();
	update_baseband_bandwidth();
	update_baseband_configuration();
	radio::streaming_enable();

	update_headphone_volume();
}

void ReceiverModel::disable() {
	/* TODO: This is a dumb hack to stop baseband from working so hard. */
	BasebandConfigurationMessage message {
		.configuration = {
			.mode = -1,
			.sampling_rate = 0,
			.decimation_factor = 1,
		}
	};
	shared_memory.baseband_queue.push(&message);
	while( !message.is_free() );

	radio::disable();
}

int32_t ReceiverModel::tuning_offset() {
	return -(sampling_rate() / 4);
}

void ReceiverModel::update_tuning_frequency() {
	radio::set_tuning_frequency(tuning_frequency_ + tuning_offset());
}

void ReceiverModel::update_rf_amp() {
	radio::set_rf_amp(rf_amp_);
}

void ReceiverModel::update_lna() {
	radio::set_lna_gain(lna_gain_db_);
}

void ReceiverModel::update_baseband_bandwidth() {
	radio::set_baseband_filter_bandwidth(baseband_bandwidth_);
}

void ReceiverModel::update_vga() {
	radio::set_vga_gain(vga_gain_db_);
}

void ReceiverModel::update_modulation() {
	update_baseband_configuration();
}

void ReceiverModel::update_baseband_configuration() {
	clock_manager.set_sampling_frequency(sampling_rate() * baseband_oversampling());
	update_tuning_frequency();
	radio::set_baseband_decimation_by(baseband_oversampling());

	BasebandConfigurationMessage message { baseband_configuration };
	shared_memory.baseband_queue.push(&message);

	// Block until message is consumed, since we allocated it on the stack.
	while( !message.is_free() );

	if( baseband_configuration.mode == 3 ) {
		update_fsk_configuration();
	}
}

void ReceiverModel::update_headphone_volume() {
	// TODO: Manipulating audio codec here, and in ui_receiver.cpp. Good to do
	// both?

	audio_codec.set_headphone_volume(headphone_volume_);
}

static constexpr FSKConfiguration fsk_configuration_ais = {
	.symbol_rate = 9600,
	.access_code = 0b01010101010101010101111110,
	.access_code_length = 26,
	.access_code_tolerance = 1,
	.packet_length = 256,
};

static constexpr FSKConfiguration fsk_configuration_tpms_a = {
	.symbol_rate = 19200,
	.access_code = 0b0101010101010101010101010110,
	.access_code_length = 28,
	.access_code_tolerance = 1,
	.packet_length = 160,
};

void ReceiverModel::update_fsk_configuration() {
	FSKConfigurationMessage message { fsk_configuration_ais };
	shared_memory.baseband_queue.push(&message);

	// Block until message is consumed, since we allocated it on the stack.
	while( !message.is_free() );
}