Recon repeat delay (#1699)

- Recon repeat delay, range: from 0s to 254s, timer is played before each TX
- fixed missing '<' '>' around button names for DELETE and REMOVE name to indicate that they can also be used to skip in the list
- added missing pmem recon dump entries
This commit is contained in:
gullradriel 2023-12-30 22:32:41 +01:00 committed by GitHub
parent 94cdb16ca9
commit 2893c031ab
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 76 additions and 16 deletions

View File

@ -102,7 +102,7 @@ void ReconView::set_loop_config(bool v) {
persistent_memory::set_recon_continuous(continuous); persistent_memory::set_recon_continuous(continuous);
} }
void ReconView::recon_stop_recording() { void ReconView::recon_stop_recording(bool exiting) {
if (is_recording) { if (is_recording) {
if (field_mode.selected_index_value() == SPEC_MODULATION) if (field_mode.selected_index_value() == SPEC_MODULATION)
button_audio_app.set_text("RAW"); button_audio_app.set_text("RAW");
@ -113,14 +113,14 @@ void ReconView::recon_stop_recording() {
button_config.set_style(&Styles::white); button_config.set_style(&Styles::white);
is_recording = false; is_recording = false;
// repeater mode // repeater mode
if (persistent_memory::recon_repeat_recorded()) { if (!exiting && persistent_memory::recon_repeat_recorded()) {
start_repeat(); start_repeat();
} }
} }
} }
void ReconView::clear_freqlist_for_ui_action() { void ReconView::clear_freqlist_for_ui_action() {
recon_stop_recording(); recon_stop_recording(false);
if (field_mode.selected_index_value() != SPEC_MODULATION) if (field_mode.selected_index_value() != SPEC_MODULATION)
audio::output::stop(); audio::output::stop();
// flag to detect and reload frequency_list // flag to detect and reload frequency_list
@ -308,9 +308,16 @@ void ReconView::focus() {
} }
ReconView::~ReconView() { ReconView::~ReconView() {
recon_stop_recording(); if (recon_tx) {
replay_thread.reset();
}
recon_stop_recording(true);
if (field_mode.selected_index_value() != SPEC_MODULATION) if (field_mode.selected_index_value() != SPEC_MODULATION)
audio::output::stop(); audio::output::stop();
transmitter_model.disable();
receiver_model.disable(); receiver_model.disable();
baseband::shutdown(); baseband::shutdown();
} }
@ -542,7 +549,7 @@ ReconView::ReconView(NavigationView& nav)
}; };
button_manual_recon.on_select = [this](Button&) { button_manual_recon.on_select = [this](Button&) {
button_remove.set_text("DELETE"); button_remove.set_text("<DELETE>");
button_add.hidden(false); button_add.hidden(false);
scanner_mode = false; scanner_mode = false;
manual_mode = true; manual_mode = true;
@ -636,12 +643,12 @@ ReconView::ReconView(NavigationView& nav)
scanner_mode = false; scanner_mode = false;
button_scanner_mode.set_style(&Styles::blue); button_scanner_mode.set_style(&Styles::blue);
button_scanner_mode.set_text("RECON"); button_scanner_mode.set_text("RECON");
button_remove.set_text("REMOVE"); button_remove.set_text("<REMOVE>");
} else { } else {
scanner_mode = true; scanner_mode = true;
button_scanner_mode.set_style(&Styles::red); button_scanner_mode.set_style(&Styles::red);
button_scanner_mode.set_text("SCAN"); button_scanner_mode.set_text("SCAN");
button_remove.set_text("DELETE"); button_remove.set_text("<DELETE>");
} }
frequency_file_load(); frequency_file_load();
if (autostart) { if (autostart) {
@ -844,7 +851,7 @@ void ReconView::on_statistics_update(const ChannelStatistics& statistics) {
if (status != 1) { if (status != 1) {
status = 1; status = 1;
if (wait != 0) { if (wait != 0) {
recon_stop_recording(); recon_stop_recording(false);
if (field_mode.selected_index_value() != SPEC_MODULATION) if (field_mode.selected_index_value() != SPEC_MODULATION)
audio::output::stop(); audio::output::stop();
} }
@ -1145,7 +1152,7 @@ size_t ReconView::change_mode(freqman_index_t new_mod) {
return 0; return 0;
field_mode.on_change = [this](size_t, OptionsField::value_t) {}; field_mode.on_change = [this](size_t, OptionsField::value_t) {};
field_bw.on_change = [this](size_t, OptionsField::value_t) {}; field_bw.on_change = [this](size_t, OptionsField::value_t) {};
recon_stop_recording(); recon_stop_recording(false);
if (record_view != nullptr) { if (record_view != nullptr) {
remove_child(record_view.get()); remove_child(record_view.get());
record_view.reset(); record_view.reset();
@ -1377,10 +1384,31 @@ void ReconView::start_repeat() {
repeat_file_error(rawfile, "Can't open file to send to thread"); repeat_file_error(rawfile, "Can't open file to send to thread");
return; return;
} }
repeat_ready_signal = true; // wait for TX if needed (hackish, direct screen update since the UI will be blocked)
repeat_cur_rep++; if (persistent_memory::recon_repeat_delay() > 0) {
uint8_t delay = persistent_memory::recon_repeat_delay();
Painter p;
while (delay > 0) {
std::string delay_message = "TX DELAY: " + to_string_dec_uint(delay) + "s";
// update display information
p.fill_rectangle({0, (SCREEN_H / 2) - 16, SCREEN_W, 64}, Color::light_grey());
p.draw_string({(SCREEN_W / 2) - 7 * 8, SCREEN_H / 2}, Styles::red, delay_message);
// sleep 1 second
chThdSleepMilliseconds(1000);
// decre delay
if (delay > 0)
delay = delay - 1;
else
break;
}
}
// ReplayThread starts immediately on construction; must be set before creating. // ReplayThread starts immediately on construction; must be set before creating.
repeat_ready_signal = true;
repeat_cur_rep++;
replay_thread = std::make_unique<ReplayThread>( replay_thread = std::make_unique<ReplayThread>(
std::move(reader), std::move(reader),
/* read_size */ repeat_read_size, /* read_size */ repeat_read_size,

View File

@ -110,7 +110,7 @@ class ReconView : public View {
void load_persisted_settings(); void load_persisted_settings();
bool recon_save_freq(const std::filesystem::path& path, size_t index, bool warn_if_exists); bool recon_save_freq(const std::filesystem::path& path, size_t index, bool warn_if_exists);
// placeholder for possible void recon_start_recording(); // placeholder for possible void recon_start_recording();
void recon_stop_recording(); void recon_stop_recording(bool exiting);
// Returns true if 'current_index' is in bounds of frequency_list. // Returns true if 'current_index' is in bounds of frequency_list.
bool current_is_valid(); bool current_is_valid();

View File

@ -106,6 +106,7 @@ void ReconSetupViewMore::save() {
persistent_memory::set_recon_repeat_nb(field_repeat_nb.value()); persistent_memory::set_recon_repeat_nb(field_repeat_nb.value());
persistent_memory::set_recon_repeat_amp(checkbox_repeat_amp.value()); persistent_memory::set_recon_repeat_amp(checkbox_repeat_amp.value());
persistent_memory::set_recon_repeat_gain(field_repeat_gain.value()); persistent_memory::set_recon_repeat_gain(field_repeat_gain.value());
persistent_memory::set_recon_repeat_delay(field_repeat_delay.value());
}; };
void ReconSetupViewMain::focus() { void ReconSetupViewMain::focus() {
@ -128,7 +129,9 @@ ReconSetupViewMore::ReconSetupViewMore(NavigationView& nav, Rect parent_rect)
&field_repeat_nb, &field_repeat_nb,
&checkbox_repeat_amp, &checkbox_repeat_amp,
&text_repeat_gain, &text_repeat_gain,
&field_repeat_gain}); &field_repeat_gain,
&text_repeat_delay,
&field_repeat_delay});
// tx options have to be in yellow to inform the users that activating them will make the device transmit // tx options have to be in yellow to inform the users that activating them will make the device transmit
checkbox_repeat_recorded.set_style(&Styles::yellow); checkbox_repeat_recorded.set_style(&Styles::yellow);
@ -137,6 +140,8 @@ ReconSetupViewMore::ReconSetupViewMore(NavigationView& nav, Rect parent_rect)
checkbox_repeat_amp.set_style(&Styles::yellow); checkbox_repeat_amp.set_style(&Styles::yellow);
text_repeat_gain.set_style(&Styles::yellow); text_repeat_gain.set_style(&Styles::yellow);
field_repeat_gain.set_style(&Styles::yellow); field_repeat_gain.set_style(&Styles::yellow);
text_repeat_delay.set_style(&Styles::yellow);
field_repeat_delay.set_style(&Styles::yellow);
checkbox_load_freqs.set_value(persistent_memory::recon_load_freqs()); checkbox_load_freqs.set_value(persistent_memory::recon_load_freqs());
checkbox_load_repeaters.set_value(persistent_memory::recon_load_repeaters()); checkbox_load_repeaters.set_value(persistent_memory::recon_load_repeaters());
@ -148,6 +153,7 @@ ReconSetupViewMore::ReconSetupViewMore(NavigationView& nav, Rect parent_rect)
checkbox_repeat_amp.set_value(persistent_memory::recon_repeat_amp()); checkbox_repeat_amp.set_value(persistent_memory::recon_repeat_amp());
field_repeat_nb.set_value(persistent_memory::recon_repeat_nb()); field_repeat_nb.set_value(persistent_memory::recon_repeat_nb());
field_repeat_gain.set_value(persistent_memory::recon_repeat_gain()); field_repeat_gain.set_value(persistent_memory::recon_repeat_gain());
field_repeat_delay.set_value(persistent_memory::recon_repeat_delay());
// tx warning modal // tx warning modal
checkbox_repeat_recorded.on_select = [this, &nav](Checkbox&, bool v) { checkbox_repeat_recorded.on_select = [this, &nav](Checkbox&, bool v) {

View File

@ -158,7 +158,7 @@ class ReconSetupViewMore : public View {
"nb:"}; "nb:"};
NumberField field_repeat_nb{ NumberField field_repeat_nb{
{18 * 8, 165}, {17 * 8, 165},
2, 2,
{1, 99}, {1, 99},
1, 1,
@ -171,16 +171,28 @@ class ReconSetupViewMore : public View {
"AMP,"}; "AMP,"};
Text text_repeat_gain{ Text text_repeat_gain{
{10 * 8, 196, 5 * 8, 22}, {9 * 8, 196, 5 * 8, 22},
"GAIN:"}; "GAIN:"};
NumberField field_repeat_gain{ NumberField field_repeat_gain{
{16 * 8, 196}, {14 * 8, 196},
2, 2,
{0, 47}, {0, 47},
1, 1,
' ', ' ',
}; };
Text text_repeat_delay{
{16 * 8, 196, 8 * 8, 22},
", delay:"};
NumberField field_repeat_delay{
{24 * 8, 196},
3,
{0, 254},
1,
' ',
};
}; };
class ReconSetupView : public View { class ReconSetupView : public View {

View File

@ -197,6 +197,7 @@ struct data_t {
uint64_t recon_config; uint64_t recon_config;
int8_t recon_repeat_nb; int8_t recon_repeat_nb;
int8_t recon_repeat_gain; int8_t recon_repeat_gain;
uint8_t recon_repeat_delay;
// enable or disable converter // enable or disable converter
bool converter; bool converter;
@ -263,6 +264,7 @@ struct data_t {
recon_config(0), recon_config(0),
recon_repeat_nb(0), recon_repeat_nb(0),
recon_repeat_gain(0), recon_repeat_gain(0),
recon_repeat_delay(0),
converter(false), converter(false),
updown_converter(false), updown_converter(false),
@ -405,6 +407,7 @@ void defaults() {
set_recon_repeat_amp(false); set_recon_repeat_amp(false);
set_recon_repeat_gain(35); set_recon_repeat_gain(35);
set_recon_repeat_nb(3); set_recon_repeat_nb(3);
set_recon_repeat_delay(1);
set_config_sdcard_high_speed_io(false, true); set_config_sdcard_high_speed_io(false, true);
} }
@ -766,6 +769,9 @@ int8_t recon_repeat_nb() {
int8_t recon_repeat_gain() { int8_t recon_repeat_gain() {
return data->recon_repeat_gain; return data->recon_repeat_gain;
} }
uint8_t recon_repeat_delay() {
return data->recon_repeat_delay;
}
bool recon_repeat_amp() { bool recon_repeat_amp() {
return (data->recon_config & 0x00100000UL) ? true : false; return (data->recon_config & 0x00100000UL) ? true : false;
} }
@ -812,6 +818,9 @@ void set_recon_repeat_nb(const int8_t v) {
void set_recon_repeat_gain(const int8_t v) { void set_recon_repeat_gain(const int8_t v) {
data->recon_repeat_gain = v; data->recon_repeat_gain = v;
} }
void set_recon_repeat_delay(const uint8_t v) {
data->recon_repeat_delay = v;
}
void set_recon_repeat_amp(const bool v) { void set_recon_repeat_amp(const bool v) {
data->recon_config = (data->recon_config & ~0x00100000UL) | (v << 20); data->recon_config = (data->recon_config & ~0x00100000UL) | (v << 20);
} }
@ -1028,6 +1037,9 @@ bool debug_dump() {
pmem_dump_file.write_line("tone_mix: " + to_string_dec_uint(data->tone_mix)); pmem_dump_file.write_line("tone_mix: " + to_string_dec_uint(data->tone_mix));
pmem_dump_file.write_line("hardware_config: " + to_string_dec_uint(data->hardware_config)); pmem_dump_file.write_line("hardware_config: " + to_string_dec_uint(data->hardware_config));
pmem_dump_file.write_line("recon_config: 0x" + to_string_hex(data->recon_config, 16)); pmem_dump_file.write_line("recon_config: 0x" + to_string_hex(data->recon_config, 16));
pmem_dump_file.write_line("recon_repeat_nb: " + to_string_dec_int(data->recon_repeat_nb, 16));
pmem_dump_file.write_line("recon_repeat_gain:" + to_string_hex(data->recon_config, 16));
pmem_dump_file.write_line("recon_repeat_delay:" + to_string_hex(data->recon_config, 16));
pmem_dump_file.write_line("converter: " + to_string_dec_int(data->converter)); pmem_dump_file.write_line("converter: " + to_string_dec_int(data->converter));
pmem_dump_file.write_line("updown_converter: " + to_string_dec_int(data->updown_converter)); pmem_dump_file.write_line("updown_converter: " + to_string_dec_int(data->updown_converter));
pmem_dump_file.write_line("updown_frequency_rx_correction: " + to_string_dec_int(data->updown_frequency_rx_correction)); pmem_dump_file.write_line("updown_frequency_rx_correction: " + to_string_dec_int(data->updown_frequency_rx_correction));

View File

@ -255,6 +255,7 @@ int8_t recon_repeat_gain();
bool recon_repeat_amp(); bool recon_repeat_amp();
bool recon_load_hamradios(); bool recon_load_hamradios();
bool recon_match_mode(); bool recon_match_mode();
uint8_t recon_repeat_delay();
void set_recon_autosave_freqs(const bool v); void set_recon_autosave_freqs(const bool v);
void set_recon_autostart_recon(const bool v); void set_recon_autostart_recon(const bool v);
void set_recon_continuous(const bool v); void set_recon_continuous(const bool v);
@ -270,6 +271,7 @@ void set_recon_repeat_amp(const bool v);
void set_recon_load_hamradios(const bool v); void set_recon_load_hamradios(const bool v);
void set_recon_load_repeaters(const bool v); void set_recon_load_repeaters(const bool v);
void set_recon_match_mode(const bool v); void set_recon_match_mode(const bool v);
void set_recon_repeat_delay(const uint8_t v);
/* UI Config 2 */ /* UI Config 2 */
bool ui_hide_speaker(); bool ui_hide_speaker();