Config mode (#1433)

* Allow code to run without a token
* Reverted submodule change by mistake
* WIP force tcxo
* Added check to hide if not an R9
* Fixed comments
* Updated name
* Fixed crashing on TCXO switching
* Fixed the reboot issue
* Updated boot order
* Cleaned up comments
* added new recovery mode
* added IO
* implemented cpld mode change
* whitespace change
* renamed config mode
* inverted logic
* removed r9 dependency
* added disable external tcxo option to config mode
* fixed CLKIN detection for r9
* integrated tcxo setting into clock selection
Co-authored-by @jLynx
This commit is contained in:
Bernd Herzog 2023-09-04 17:46:07 +02:00 committed by GitHub
parent 4819a2f4e2
commit 62310ad9a9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 236 additions and 8 deletions

View File

@ -316,6 +316,7 @@ set(CPPSRC
# ui_replay_view.cpp # ui_replay_view.cpp
# ui_script.cpp # ui_script.cpp
ui_sd_card_debug.cpp ui_sd_card_debug.cpp
config_mode.cpp
${CPLD_20150901_DATA_CPP} ${CPLD_20150901_DATA_CPP}
${CPLD_20170522_DATA_CPP} ${CPLD_20170522_DATA_CPP}
${HACKRF_CPLD_DATA_CPP} ${HACKRF_CPLD_DATA_CPP}

View File

@ -169,6 +169,8 @@ SetRadioView::SetRadioView(
&button_save, &button_save,
&button_cancel}); &button_cancel});
add_children({&disable_external_tcxo});
SetFrequencyCorrectionModel model{ SetFrequencyCorrectionModel model{
static_cast<int8_t>(pmem::correction_ppb() / 1000), 0}; static_cast<int8_t>(pmem::correction_ppb() / 1000), 0};
@ -221,10 +223,13 @@ SetRadioView::SetRadioView(
send_system_refresh(); send_system_refresh();
}; };
disable_external_tcxo.set_value(pmem::config_disable_external_tcxo());
button_save.on_select = [this, &nav](Button&) { button_save.on_select = [this, &nav](Button&) {
const auto model = this->form_collect(); const auto model = this->form_collect();
pmem::set_correction_ppb(model.ppm * 1000); pmem::set_correction_ppb(model.ppm * 1000);
pmem::set_clkout_freq(model.freq); pmem::set_clkout_freq(model.freq);
pmem::set_config_disable_external_tcxo(disable_external_tcxo.value());
clock_manager.enable_clock_output(pmem::clkout_enabled()); clock_manager.enable_clock_output(pmem::clkout_enabled());
nav.pop(); nav.pop();
}; };

View File

@ -185,6 +185,11 @@ class SetRadioView : public View {
5, 5,
"Turn on bias voltage"}; "Turn on bias voltage"};
Checkbox disable_external_tcxo{
{18, 14 * 16},
5,
"Disable external TCXO"};
Button button_save{ Button button_save{
{2 * 8, 16 * 16, 12 * 8, 32}, {2 * 8, 16 * 16, 12 * 8, 32},
"Save"}; "Save"};

View File

@ -325,12 +325,18 @@ uint32_t ClockManager::measure_gp_clkin_frequency() {
} }
bool ClockManager::loss_of_signal() { bool ClockManager::loss_of_signal() {
return hackrf_r9 if (hackrf_r9) {
? clock_generator.plla_loss_of_signal() const auto frequency = measure_gp_clkin_frequency();
: clock_generator.clkin_loss_of_signal(); return (frequency < 9850000) || (frequency > 10150000);
} else {
return clock_generator.clkin_loss_of_signal();
}
} }
ClockManager::ReferenceSource ClockManager::detect_reference_source() { ClockManager::ReferenceSource ClockManager::detect_reference_source() {
if (portapack::persistent_memory::config_disable_external_tcxo())
return ReferenceSource::Xtal;
if (loss_of_signal()) { if (loss_of_signal()) {
// No external reference. Turn on PortaPack reference (if present). // No external reference. Turn on PortaPack reference (if present).
portapack_tcxo_enable(); portapack_tcxo_enable();

View File

@ -0,0 +1,127 @@
/*
* Copyright (C) 2023 Bernd Herzog
*
* 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 "config_mode.hpp"
#include "core_control.hpp"
#include "hackrf_gpio.hpp"
#include "portapack_hal.hpp"
void config_mode_blink_until_dfu();
void config_mode_set() {
portapack::persistent_memory::set_config_mode_storage(CONFIG_MODE_GUARD_VALUE);
}
bool config_mode_should_enter() {
return portapack::persistent_memory::config_mode_storage() == CONFIG_MODE_GUARD_VALUE;
}
void config_mode_clear() {
portapack::persistent_memory::set_config_mode_storage(CONFIG_MODE_NORMAL_VALUE);
}
uint32_t blink_patterns[] = {
0x00000000, // 0 Off
0xFFFFFFFF, // 1 On
0xF0F0F0F0, // 2 blink fast
0x00FF00FF, // 3 blink slow
0xFFF3FFF3 // 4 inverse blink slow
};
void config_mode_run() {
configure_pins_portapack();
portapack::gpio_dfu.input();
config_mode_blink_until_dfu();
auto last_dfu_btn = portapack::gpio_dfu.read();
portapack::persistent_memory::cache::init();
int32_t counter = 0;
int8_t blink_pattern_value = portapack::persistent_memory::config_cpld() +
(portapack::persistent_memory::config_disable_external_tcxo() ? 5 : 0);
while (true) {
auto dfu_btn = portapack::gpio_dfu.read();
auto dfu_clicked = last_dfu_btn == true && dfu_btn == false;
last_dfu_btn = dfu_btn;
if (dfu_clicked) {
int8_t value = portapack::persistent_memory::config_cpld() +
(portapack::persistent_memory::config_disable_external_tcxo() ? 5 : 0);
blink_pattern_value = value = (value + 1) % 10;
portapack::persistent_memory::set_config_cpld(value % 5);
portapack::persistent_memory::set_config_disable_external_tcxo((value / 5) == 1);
portapack::persistent_memory::cache::persist();
}
auto tx_blink_pattern = blink_patterns[blink_pattern_value % 5];
auto rx_blink_pattern = blink_patterns[blink_pattern_value / 5];
auto tx_value = ((tx_blink_pattern >> ((counter >> 0) & 31)) & 0x1) == 0x1;
if (tx_value) {
hackrf::one::led_tx.on();
} else {
hackrf::one::led_tx.off();
}
auto rx_value = ((rx_blink_pattern >> ((counter >> 0) & 31)) & 0x1) == 0x1;
if (rx_value) {
hackrf::one::led_rx.on();
} else {
hackrf::one::led_rx.off();
}
chThdSleepMilliseconds(100);
counter++;
}
}
void config_mode_blink_until_dfu() {
while (true) {
hackrf::one::led_tx.on();
hackrf::one::led_rx.on();
hackrf::one::led_usb.on();
chThdSleepMilliseconds(10);
hackrf::one::led_tx.off();
hackrf::one::led_rx.off();
hackrf::one::led_usb.off();
chThdSleepMilliseconds(115);
auto dfu_btn = portapack::gpio_dfu.read();
if (dfu_btn)
break;
}
while (true) {
chThdSleepMilliseconds(10);
auto dfu_btn = portapack::gpio_dfu.read();
if (!dfu_btn)
break;
}
chThdSleepMilliseconds(10);
}

View File

@ -0,0 +1,37 @@
/*
* Copyright (C) 2023 Bernd Herzog
*
* 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.
*/
#ifndef __CONFIG_MODE_H__
#define __CONFIG_MODE_H__
#include "ch.h"
#include "portapack.hpp"
#include "portapack_shared_memory.hpp"
#include "portapack_persistent_memory.hpp"
void config_mode_set();
bool config_mode_should_enter();
void config_mode_clear();
void config_mode_run();
#endif /* __CONFIG_MODE_H__ */

View File

@ -115,6 +115,7 @@ Continuous (Fox-oring)
#include "portapack.hpp" #include "portapack.hpp"
#include "portapack_shared_memory.hpp" #include "portapack_shared_memory.hpp"
#include "config_mode.hpp"
#include "message_queue.hpp" #include "message_queue.hpp"
@ -161,9 +162,17 @@ static void event_loop() {
} }
int main(void) { int main(void) {
if (config_mode_should_enter()) {
config_mode_clear();
config_mode_run();
}
config_mode_set();
first_if.init(); /* To avoid initial short Ant_DC_Bias pulse ,we need quick set up GP01_RFF507X =1 */ first_if.init(); /* To avoid initial short Ant_DC_Bias pulse ,we need quick set up GP01_RFF507X =1 */
if (portapack::init()) { if (portapack::init()) {
portapack::display.init(); portapack::display.init();
config_mode_clear();
// sdcStart(&SDCD1, nullptr); // Commented out as now happens in portapack.cpp // sdcStart(&SDCD1, nullptr); // Commented out as now happens in portapack.cpp

View File

@ -408,6 +408,10 @@ bool init() {
portapack::io.init(); portapack::io.init();
/* Cache some configuration data from persistent memory. */
persistent_memory::cache::init();
chThdSleepMilliseconds(10);
clock_manager.init_clock_generator(); clock_manager.init_clock_generator();
i2c0.stop(); i2c0.stop();
@ -467,9 +471,6 @@ bool init() {
controls_init(); controls_init();
chThdSleepMilliseconds(10); chThdSleepMilliseconds(10);
/* Cache some configuration data from persistent memory. */
persistent_memory::cache::init();
clock_manager.set_reference_ppb(persistent_memory::correction_ppb()); clock_manager.set_reference_ppb(persistent_memory::correction_ppb());
clock_manager.enable_if_clocks(); clock_manager.enable_if_clocks();
clock_manager.enable_codec_clocks(); clock_manager.enable_codec_clocks();

View File

@ -142,7 +142,7 @@ static_assert(sizeof(ui_config2_t) == sizeof(uint32_t));
struct misc_config_t { struct misc_config_t {
bool mute_audio : 1; bool mute_audio : 1;
bool disable_speaker : 1; bool disable_speaker : 1;
bool UNUSED_2 : 1; bool config_disable_external_tcxo : 1;
bool UNUSED_3 : 1; bool UNUSED_3 : 1;
bool UNUSED_4 : 1; bool UNUSED_4 : 1;
bool UNUSED_5 : 1; bool UNUSED_5 : 1;
@ -226,6 +226,9 @@ struct data_t {
// Additional UI settings. // Additional UI settings.
ui_config2_t ui_config2; ui_config2_t ui_config2;
// recovery mode magic value storage
uint32_t config_mode_storage;
constexpr data_t() constexpr data_t()
: structure_version(data_structure_version_enum::VERSION_CURRENT), : structure_version(data_structure_version_enum::VERSION_CURRENT),
target_frequency(target_frequency_reset_value), target_frequency(target_frequency_reset_value),
@ -273,7 +276,8 @@ struct data_t {
UNUSED_8(0), UNUSED_8(0),
headphone_volume_cb(-600), headphone_volume_cb(-600),
misc_config(), misc_config(),
ui_config2() { ui_config2(),
config_mode_storage(CONFIG_MODE_NORMAL_VALUE) {
} }
}; };
@ -363,6 +367,7 @@ void defaults() {
set_config_backlight_timer(backlight_config_t{}); set_config_backlight_timer(backlight_config_t{});
set_config_splash(true); set_config_splash(true);
set_config_disable_external_tcxo(false);
set_encoder_dial_sensitivity(DIAL_SENSITIVITY_NORMAL); set_encoder_dial_sensitivity(DIAL_SENSITIVITY_NORMAL);
set_config_speaker_disable(true); // Disable AK4951 speaker by default (in case of OpenSourceSDRLab H2) set_config_speaker_disable(true); // Disable AK4951 speaker by default (in case of OpenSourceSDRLab H2)
@ -380,6 +385,10 @@ void defaults() {
void init() { void init() {
const auto switches_state = get_switches_state(); const auto switches_state = get_switches_state();
// ignore for valid check
auto config_mode_backup = config_mode_storage();
set_config_mode_storage(CONFIG_MODE_NORMAL_VALUE);
if (!(switches_state[(size_t)ui::KeyEvent::Left] && switches_state[(size_t)ui::KeyEvent::Right]) && backup_ram->is_valid()) { if (!(switches_state[(size_t)ui::KeyEvent::Left] && switches_state[(size_t)ui::KeyEvent::Right]) && backup_ram->is_valid()) {
// Copy valid persistent data into cache. // Copy valid persistent data into cache.
cached_backup_ram = *backup_ram; cached_backup_ram = *backup_ram;
@ -395,6 +404,7 @@ void init() {
// Copy defaults into cache. // Copy defaults into cache.
defaults(); defaults();
} }
set_config_mode_storage(config_mode_backup);
} }
void persist() { void persist() {
@ -550,6 +560,10 @@ bool config_speaker_disable() {
return data->misc_config.disable_speaker; return data->misc_config.disable_speaker;
} }
bool config_disable_external_tcxo() {
return data->misc_config.config_disable_external_tcxo;
}
bool stealth_mode() { bool stealth_mode() {
return data->ui_config.stealth_mode; return data->ui_config.stealth_mode;
} }
@ -611,6 +625,10 @@ void set_config_speaker_disable(bool v) {
data->misc_config.disable_speaker = v; data->misc_config.disable_speaker = v;
} }
void set_config_disable_external_tcxo(bool v) {
data->misc_config.config_disable_external_tcxo = v;
}
void set_stealth_mode(bool v) { void set_stealth_mode(bool v) {
data->ui_config.stealth_mode = v; data->ui_config.stealth_mode = v;
} }
@ -840,6 +858,16 @@ void set_encoder_dial_sensitivity(uint8_t v) {
data->encoder_dial_sensitivity = v; data->encoder_dial_sensitivity = v;
} }
// Recovery mode magic value storage
static data_t* data_direct_access = reinterpret_cast<data_t*>(memory::map::backup_ram.base());
uint32_t config_mode_storage() {
return data_direct_access->config_mode_storage;
}
void set_config_mode_storage(uint32_t v) {
data_direct_access->config_mode_storage = v;
}
// PMem to sdcard settings // PMem to sdcard settings
bool should_use_sdcard_for_pmem() { bool should_use_sdcard_for_pmem() {
@ -965,6 +993,8 @@ bool debug_dump() {
// misc_config bits // misc_config bits
pmem_dump_file.write_line("misc_config config_audio_mute: " + to_string_dec_int(config_audio_mute())); pmem_dump_file.write_line("misc_config config_audio_mute: " + to_string_dec_int(config_audio_mute()));
pmem_dump_file.write_line("misc_config config_speaker_disable: " + to_string_dec_int(config_speaker_disable())); pmem_dump_file.write_line("misc_config config_speaker_disable: " + to_string_dec_int(config_speaker_disable()));
pmem_dump_file.write_line("ui_config config_disable_external_tcxo: " + to_string_dec_uint(config_disable_external_tcxo()));
// receiver_model // receiver_model
pmem_dump_file.write_line("\n[Receiver Model]"); pmem_dump_file.write_line("\n[Receiver Model]");
pmem_dump_file.write_line("target_frequency: " + to_string_dec_uint(receiver_model.target_frequency())); pmem_dump_file.write_line("target_frequency: " + to_string_dec_uint(receiver_model.target_frequency()));

View File

@ -173,6 +173,7 @@ void set_stealth_mode(const bool v);
uint8_t config_cpld(); uint8_t config_cpld();
void set_config_cpld(uint8_t i); void set_config_cpld(uint8_t i);
bool config_disable_external_tcxo();
bool config_splash(); bool config_splash();
bool config_converter(); bool config_converter();
bool config_updown_converter(); bool config_updown_converter();
@ -193,6 +194,7 @@ void set_gui_return_icon(bool v);
void set_load_app_settings(bool v); void set_load_app_settings(bool v);
void set_save_app_settings(bool v); void set_save_app_settings(bool v);
void set_show_bigger_qr_code(bool v); void set_show_bigger_qr_code(bool v);
void set_config_disable_external_tcxo(bool v);
void set_config_splash(bool v); void set_config_splash(bool v);
bool config_converter(); bool config_converter();
bool config_updown_converter(); bool config_updown_converter();
@ -218,6 +220,11 @@ void set_disable_touchscreen(bool v);
uint8_t config_encoder_dial_sensitivity(); uint8_t config_encoder_dial_sensitivity();
void set_encoder_dial_sensitivity(uint8_t v); void set_encoder_dial_sensitivity(uint8_t v);
#define CONFIG_MODE_GUARD_VALUE 2001
#define CONFIG_MODE_NORMAL_VALUE 1999
uint32_t config_mode_storage();
void set_config_mode_storage(uint32_t v);
uint32_t pocsag_last_address(); uint32_t pocsag_last_address();
void set_pocsag_last_address(uint32_t address); void set_pocsag_last_address(uint32_t address);