/* * Copyright (C) 2015 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 "ui_fsk_rx.hpp" #include "audio.hpp" #include "baseband_api.hpp" #include "portapack_persistent_memory.hpp" #include "string_format.hpp" #include "utility.hpp" #include "ui_freqman.hpp" using namespace portapack; namespace pmem = portapack::persistent_memory; void FskRxLogger::log_raw_data(const std::string& data, const uint32_t frequency) { std::string entry = "Raw: F:" + to_string_dec_uint(frequency) + "Hz"; // // Raw hex dump of all the codewords // for (size_t c = 0; c < 16; c++) // entry += to_string_hex(packet[c], 8) + " "; log_file.write_entry(data + entry); } void FskRxLogger::log_decoded(Timestamp timestamp, const std::string& text) { log_file.write_entry(timestamp, text); } namespace ui { //--------------------------------------------------------------------------------------------------------------- // Console View //--------------------------------------------------------------------------------------------------------------- FskRxAppConsoleView::FskRxAppConsoleView(NavigationView& nav, Rect parent_rect) : View(parent_rect), nav_{nav} { add_child(&console); }; void FskRxAppConsoleView::on_packet(uint32_t value, bool is_data) { if (is_data) { console.write(to_string_dec_uint(value) + " "); } } void FskRxAppConsoleView::on_show() { hidden(false); } void FskRxAppConsoleView::on_hide() { hidden(true); } FskRxAppConsoleView::~FskRxAppConsoleView() { } //--------------------------------------------------------------------------------------------------------------- // Spectrum View //--------------------------------------------------------------------------------------------------------------- FskRxAppView::FskRxAppView(NavigationView& nav, Rect parent_rect) : View(parent_rect), nav_{nav} { add_child(&waterfall); hidden(true); } FskRxAppView::~FskRxAppView() { } void FskRxAppView::focus() { } void FskRxAppView::on_show() { hidden(false); waterfall.start(); } void FskRxAppView::on_hide() { hidden(true); waterfall.stop(); } //--------------------------------------------------------------------------------------------------------------- // Base View //--------------------------------------------------------------------------------------------------------------- FskxRxMainView::FskxRxMainView(NavigationView& nav) : nav_{nav} { add_children({&tab_view, &view_data, &view_stream, &labels, &rssi, &channel, &field_rf_amp, &field_lna, &field_vga, &field_frequency, &deviation_frequency, &record_view}); baseband::run_image(portapack::spi_flash::image_tag_fskrx); // DEBUG record_view.on_error = [&nav](std::string message) { nav.display_modal("Error", message); }; deviation_frequency.on_change = [this](rf::Frequency f) { refresh_ui(f); }; // Set initial sampling rate /* Bandwidth of 2FSK is 2 * Deviation */ record_view.set_sampling_rate(initial_deviation * 2); field_frequency.set_value(initial_target_frequency); deviation_frequency.set_value(initial_deviation); logger.append(LOG_ROOT_DIR "/FSKRX.TXT"); baseband::set_fsk(initial_deviation); audio::output::start(); receiver_model.enable(); } void FskxRxMainView::handle_decoded(Timestamp timestamp, const std::string& prefix) { if (logging()) { logger.log_decoded(timestamp, prefix); } } void FskxRxMainView::refresh_ui(rf::Frequency deviationHz) { /* Nyquist would imply a sample rate of 2x bandwidth, but because the ADC * provides 2 values (I,Q), the sample_rate is equal to bandwidth here. */ /* Bandwidth of 2FSK is 2 * Deviation */ auto sample_rate = deviationHz * 2; /* base_rate (bandwidth) is used for FFT calculation and display LCD, and also in recording writing SD Card rate. */ /* ex. sampling_rate values, 4Mhz, when recording 500 kHz (BW) and fs 8 Mhz, when selected 1 Mhz BW ... */ /* ex. recording 500kHz BW to .C16 file, base_rate clock 500kHz x2(I,Q) x 2 bytes (int signed) =2MB/sec rate SD Card. */ if (!view_stream.hidden()) { view_stream.waterfall.stop(); } // record_view determines the correct oversampling to apply and returns the actual sample rate. // NB: record_view is what actually updates proc_capture baseband settings. auto actual_sample_rate = record_view.set_sampling_rate(sample_rate); // Update the radio model with the actual sampling rate. receiver_model.set_sampling_rate(actual_sample_rate); // Get suitable anti-aliasing BPF bandwidth for MAX2837 given the actual sample rate. auto anti_alias_filter_bandwidth = filter_bandwidth_for_sampling_rate(actual_sample_rate); receiver_model.set_baseband_bandwidth(anti_alias_filter_bandwidth); if (!view_stream.hidden()) { view_stream.waterfall.start(); } } void FskxRxMainView::focus() { field_frequency.focus(); } void FskxRxMainView::set_parent_rect(const Rect new_parent_rect) { View::set_parent_rect(new_parent_rect); ui::Rect waterfall_rect{0, 0, new_parent_rect.width(), new_parent_rect.height() - header_height}; view_stream.waterfall.set_parent_rect(waterfall_rect); } FskxRxMainView::~FskxRxMainView() { audio::output::stop(); receiver_model.disable(); baseband::shutdown(); } } /* namespace ui */