From f4cc29e0e46948cab786e36f03fa89389d21d15e Mon Sep 17 00:00:00 2001 From: c vw Date: Mon, 9 Aug 2021 10:26:00 +0200 Subject: [PATCH] Manual merge with John's update --- Makefile | 2 +- action_dialog.c | 16 ++++++++++++++-- actions.c | 43 +++++++++++++++++++++++++++++++++++++++++-- css.c | 17 +++++++++++++++++ encoder_menu.c | 8 -------- main.c | 12 ++++++++++-- midi3.c | 45 +++++++++++++++++++++------------------------ new_protocol.c | 6 ++++++ old_protocol.c | 1 + pulseaudio.c | 1 + radio.c | 11 ++++++++++- radio.h | 2 ++ radio_menu.c | 23 ++++++++++++++++++++--- switch_menu.c | 8 -------- toolbar_menu.c | 9 --------- vfo.c | 4 +++- 16 files changed, 147 insertions(+), 61 deletions(-) diff --git a/Makefile b/Makefile index 651843a..f852ff2 100644 --- a/Makefile +++ b/Makefile @@ -26,7 +26,7 @@ PURESIGNAL_INCLUDE=PURESIGNAL # USBOZY_INCLUDE=USBOZY # uncomment the line to below include support local CW keyer -#LOCALCW_INCLUDE=LOCALCW +LOCALCW_INCLUDE=LOCALCW # uncomment the line below for SoapySDR #SOAPYSDR_INCLUDE=SOAPYSDR diff --git a/action_dialog.c b/action_dialog.c index a60fd71..f60063d 100644 --- a/action_dialog.c +++ b/action_dialog.c @@ -7,6 +7,7 @@ typedef struct _choice { int action; GtkWidget *button; gulong signal_id; + struct _choice *previous; } CHOICE; static GtkWidget *dialog; @@ -26,6 +27,8 @@ static void action_select_cb(GtkWidget *widget,gpointer data) { int action_dialog(GtkWidget *parent,int filter,int currentAction) { int i,j; + CHOICE *previous=NULL; + CHOICE *choice=NULL; action=currentAction; previous_button=NULL; @@ -44,15 +47,18 @@ int action_dialog(GtkWidget *parent,int filter,int currentAction) { for(i=0;iaction=i; choice->button=button; choice->signal_id=g_signal_connect(button,"toggled",G_CALLBACK(action_select_cb),choice); + choice->previous=previous; + previous=choice; + if(ActionTable[i].action==currentAction) { previous_button=button; previous_signal_id=choice->signal_id; @@ -68,6 +74,12 @@ int action_dialog(GtkWidget *parent,int filter,int currentAction) { if(result!=GTK_RESPONSE_ACCEPT) { action=currentAction; } + // free up choice structures + while(previous!=NULL) { + choice=previous; + previous=choice->previous; + g_free(choice); + } return action; } diff --git a/actions.c b/actions.c index 370c76e..dceff4f 100644 --- a/actions.c +++ b/actions.c @@ -183,6 +183,21 @@ ACTION_TABLE ActionTable[] = { {ACTIONS, "", NULL, TYPE_NONE} }; +static gint timer=0; +static gboolean timer_released; + +static int timeout_cb(gpointer data) { + if(timer_released) { + g_free(data); + timer=0; + return FALSE; + } + // process the action; + process_action(data); + return TRUE; +} + + int process_action(void *data) { PROCESS_ACTION *a=(PROCESS_ACTION *)data; @@ -194,8 +209,9 @@ int process_action(void *data) { FILTER *filter; int new_val; int i; + gboolean free_action=TRUE; - g_print("%s: action=%d mode=%d value=%d\n",__FUNCTION__,a->action,a->mode,a->val); + //g_print("%s: action=%d mode=%d value=%d\n",__FUNCTION__,a->action,a->mode,a->val); switch(a->action) { case A_SWAP_B: @@ -956,11 +972,25 @@ int process_action(void *data) { case RIT_MINUS: if(a->mode==PRESSED) { vfo_rit(active_receiver->id,-1); + if(timer==0) { + timer=g_timeout_add(250,timeout_cb,a); + timer_released=FALSE; + } + free_action=FALSE; + } else { + timer_released=TRUE; } break; case RIT_PLUS: if(a->mode==PRESSED) { vfo_rit(active_receiver->id,1); + if(timer==0) { + timer=g_timeout_add(250,timeout_cb,a); + timer_released=FALSE; + } + free_action=FALSE; + } else { + timer_released=TRUE; } break; case RIT_RX1: @@ -1193,6 +1223,7 @@ int process_action(void *data) { value=10000.0; } transmitter->xit=(int)value; + transmitter->xit_enabled=(value!=0); if(protocol==NEW_PROTOCOL) { schedule_high_priority(); } @@ -1202,6 +1233,10 @@ int process_action(void *data) { if(a->mode==PRESSED) { if(can_transmit) { transmitter->xit=0; + transmitter->xit_enabled=0; + if(protocol==NEW_PROTOCOL) { + schedule_high_priority(); + } g_idle_add(ext_vfo_update, NULL); } } @@ -1228,6 +1263,7 @@ int process_action(void *data) { value=10000.0; } transmitter->xit=(int)value; + transmitter->xit_enabled=(value!=0); if(protocol==NEW_PROTOCOL) { schedule_high_priority(); } @@ -1246,6 +1282,7 @@ int process_action(void *data) { value=10000.0; } transmitter->xit=(int)value; + transmitter->xit_enabled=(value!=0); if(protocol==NEW_PROTOCOL) { schedule_high_priority(); } @@ -1275,7 +1312,9 @@ int process_action(void *data) { } break; } - g_free(data); + if(free_action) { + g_free(data); + } return 0; } diff --git a/css.c b/css.c index 810492c..16bfd7b 100644 --- a/css.c +++ b/css.c @@ -2,11 +2,28 @@ #include "css.h" char *css= +" @define-color TOGGLE_ON rgb(100%,0%,0%);\n" +" @define-color TOGGLE_OFF rgb(70%,70%,70%);\n" " #small_button {\n" " padding: 0;\n" " font-family: Sans;\n" " font-size: 15px;\n" " }\n" +" #small_toggle_button {\n" +" padding: 0;\n" +" font-family: Sans;\n" +" font-size: 15px;\n" +" background-image: none;\n" +" background-color: @TOGGLE_OFF;\n" +" }\n" +" #small_toggle_button:checked {\n" +" padding: 0;\n" +" font-family: Sans;\n" +" font-size: 15px;\n" +" background-image: none;\n" +" background-color: @TOGGLE_ON;\n" +" }\n" + ; void load_css() { diff --git a/encoder_menu.c b/encoder_menu.c index 2c1220f..66b6b2f 100644 --- a/encoder_menu.c +++ b/encoder_menu.c @@ -37,14 +37,6 @@ #include "gpio.h" #include "i2c.h" -typedef struct _choice { - int id; - int action; - GtkWidget *initial_button; - GtkWidget *button; - gulong signal_id; -} CHOICE; - static GtkWidget *parent_window=NULL; static GtkWidget *dialog=NULL; diff --git a/main.c b/main.c index d964e7f..15b031b 100644 --- a/main.c +++ b/main.c @@ -176,6 +176,7 @@ gboolean main_delete (GtkWidget *widget) { static int init(void *data) { char wisdom_directory[1024]; + int counter; g_print("%s\n",__FUNCTION__); @@ -198,11 +199,18 @@ static int init(void *data) { // char *c=getcwd(wisdom_directory, sizeof(wisdom_directory)); strcpy(&wisdom_directory[strlen(wisdom_directory)],"/"); - fprintf(stderr,"Securing wisdom file in directory: %s\n", wisdom_directory); - status_text("Creating FFTW Wisdom file ..."); + g_print("Securing wisdom file in directory: %s\n", wisdom_directory); wisdom_running=1; + counter=0; pthread_create(&wisdom_thread_id, NULL, wisdom_thread, wisdom_directory); while (wisdom_running) { + if (counter++ < 10) { + status_text("Checking FFTW Wisdom file ..."); + } else { + // if it takes longer than 1 sec, assume that WDSP + // is (re-) creating the wisdom file + status_text("Creating FFTW Wisdom file ..."); + } // wait for the wisdom thread to complete, meanwhile // handling any GTK events. usleep(100000); // 100ms diff --git a/midi3.c b/midi3.c index 55b3c4d..ab72d75 100644 --- a/midi3.c +++ b/midi3.c @@ -35,13 +35,22 @@ void DoTheMidi(int action, enum ACTIONtype type, int val) { int *ip; PROCESS_ACTION *a; - g_print("%s: action=%d type=%d val=%d\n",__FUNCTION__,action,type,val); + //g_print("%s: action=%d type=%d val=%d\n",__FUNCTION__,action,type,val); - // - // CW actions are time-critical, so they are handled HERE - // instead of invoking the GTK idle queue - // - if (action == CW_KEYER) { + switch(type) { + case MIDI_KEY: + // + // CW_LEFT, CW_RIGHT, and CW_KEYER have to be handled with + // minimum latency, so these are not put to the GTK idle queue + // but rather handled immediately + // + if(action==CW_LEFT || action==CW_RIGHT) { +#ifdef LOCALCW + keyer_event(action==CW_LEFT,val); +#else + g_print("MIDI CW key but compiled without LOCALCW\n"); +#endif + } else if (action == CW_KEYER) { // // This is a CW key-up/down which uses functions from the keyer // that by-pass the interrupt-driven standard action. @@ -63,24 +72,12 @@ void DoTheMidi(int action, enum ACTIONtype type, int val) { cw_key_down=0; cw_key_up=0; } - return; - } -#ifdef LOCALCW - if (action == CW_LEFT) { - keyer_event(1, val); - return; - } - if (action == CW_RIGHT) { - keyer_event(0, val); - return; - } -#endif - switch(type) { - case MIDI_KEY: - a=g_new(PROCESS_ACTION,1); - a->action=action; - a->mode=val?PRESSED:RELEASED; - g_idle_add(process_action,a); + } else { + a=g_new(PROCESS_ACTION,1); + a->action=action; + a->mode=val?PRESSED:RELEASED; + g_idle_add(process_action,a); + } break; case MIDI_KNOB: a=g_new(PROCESS_ACTION,1); diff --git a/new_protocol.c b/new_protocol.c index b4f4e37..8b9d479 100644 --- a/new_protocol.c +++ b/new_protocol.c @@ -782,6 +782,8 @@ static void new_protocol_high_priority() { } } + rxFrequency+=calibration; + phase=(long)((4294967296.0*(double)rxFrequency)/122880000.0); high_priority_buffer_to_radio[ 9]=phase>>24; high_priority_buffer_to_radio[10]=phase>>16; @@ -813,6 +815,8 @@ static void new_protocol_high_priority() { } } + rxFrequency+=calibration; + phase=(long)((4294967296.0*(double)rxFrequency)/122880000.0); high_priority_buffer_to_radio[9+(ddc*4)]=phase>>24; high_priority_buffer_to_radio[10+(ddc*4)]=phase>>16; @@ -839,6 +843,8 @@ static void new_protocol_high_priority() { } } + txFrequency+=calibration; + phase=(long)((4294967296.0*(double)txFrequency)/122880000.0); if(isTransmitting() && transmitter->puresignal) { diff --git a/old_protocol.c b/old_protocol.c index 91c9332..c3f04b3 100644 --- a/old_protocol.c +++ b/old_protocol.c @@ -761,6 +761,7 @@ static long long channel_freq(int chan) { } } } + freq+=calibration; return freq; } diff --git a/pulseaudio.c b/pulseaudio.c index d62c2bd..7670ed9 100644 --- a/pulseaudio.c +++ b/pulseaudio.c @@ -333,6 +333,7 @@ float audio_get_next_mic_sample() { g_mutex_lock(&audio_mutex); if ((mic_ring_buffer == NULL) || (mic_ring_read_pt == mic_ring_write_pt)) { // no buffer, or nothing in buffer: insert silence + //g_print("%s: no samples\n",__FUNCTION__); sample=0.0; } else { newpt = mic_ring_read_pt+1; diff --git a/radio.c b/radio.c index 3be79cf..0a4b918 100644 --- a/radio.c +++ b/radio.c @@ -108,6 +108,9 @@ static GtkWidget *panadapter; static GtkWidget *waterfall; static GtkWidget *audio_waterfall; +// RX and TX calibration +long long calibration=0LL; + /* #ifdef GPIO static GtkWidget *encoders; @@ -2132,8 +2135,11 @@ g_print("radioRestoreState: %s\n",property_path); value=getProperty("binaural"); if(value) binaural=atoi(value); + value=getProperty("calibration"); + if(value) calibration=atoll(value); + value=getProperty("frequencyB"); - if(value) frequencyB=atol(value); + if(value) frequencyB=atoll(value); value=getProperty("modeB"); if(value) modeB=atoi(value); @@ -2483,6 +2489,9 @@ g_print("radioSaveState: %s\n",property_path); sprintf(value,"%d",binaural); setProperty("binaural",value); + sprintf(value,"%lld",calibration); + setProperty("calibration",value); + sprintf(value,"%lld",frequencyB); setProperty("frequencyB",value); sprintf(value,"%d",modeB); diff --git a/radio.h b/radio.h index 93236e3..997af97 100644 --- a/radio.h +++ b/radio.h @@ -74,6 +74,8 @@ extern gboolean radio_is_remote; extern GtkWidget *fixed; +extern long long calibration; + extern char property_path[]; #define NONE 0 diff --git a/radio_menu.c b/radio_menu.c index 752e5d2..b55ecf3 100644 --- a/radio_menu.c +++ b/radio_menu.c @@ -167,6 +167,10 @@ static void dac0_gain_value_changed_cb(GtkWidget *widget, gpointer data) { */ #endif +static void calibration_value_changed_cb(GtkWidget *widget, gpointer data) { + calibration=(long long)gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(widget)); +} + static void rx_gain_calibration_value_changed_cb(GtkWidget *widget, gpointer data) { rx_gain_calibration=gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(widget)); } @@ -429,8 +433,10 @@ void radio_menu(GtkWidget *parent) { GtkWidget *content=gtk_dialog_get_content_area(GTK_DIALOG(dialog)); GtkWidget *grid=gtk_grid_new(); - gtk_grid_set_column_spacing (GTK_GRID(grid),10); + gtk_grid_set_column_spacing (GTK_GRID(grid),5); gtk_grid_set_row_spacing (GTK_GRID(grid),5); + gtk_grid_set_column_homogeneous (GTK_GRID(grid), FALSE); + gtk_grid_set_row_homogeneous (GTK_GRID(grid), FALSE); int col=0; int row=0; @@ -822,9 +828,20 @@ void radio_menu(GtkWidget *parent) { g_signal_connect(mute_rx_b,"toggled",G_CALLBACK(mute_rx_cb),NULL); row++; + col=0; + + GtkWidget *calibration_label=gtk_label_new(NULL); + gtk_label_set_markup(GTK_LABEL(calibration_label), "Frequency\nCalibration(Hz):"); + gtk_grid_attach(GTK_GRID(grid),calibration_label,col,row,1,1); + col++; + + GtkWidget *calibration_b=gtk_spin_button_new_with_range(-10000.0,10000.0,1.0); + gtk_spin_button_set_value(GTK_SPIN_BUTTON(calibration_b),(double)calibration); + gtk_grid_attach(GTK_GRID(grid),calibration_b,col,row,1,1); + g_signal_connect(calibration_b,"value_changed",G_CALLBACK(calibration_value_changed_cb),NULL); if(have_rx_gain) { - col=0; + col++; GtkWidget *rx_gain_label=gtk_label_new(NULL); gtk_label_set_markup(GTK_LABEL(rx_gain_label), "RX Gain Calibration:"); gtk_grid_attach(GTK_GRID(grid),rx_gain_label,col,row,1,1); @@ -844,8 +861,8 @@ void radio_menu(GtkWidget *parent) { g_signal_connect(PA_enable_b,"toggled",G_CALLBACK(PA_enable_cb),NULL); } - row++; } + row++; if(row>temp_row) temp_row=row; diff --git a/switch_menu.c b/switch_menu.c index 16d479e..94bee0d 100644 --- a/switch_menu.c +++ b/switch_menu.c @@ -38,14 +38,6 @@ #include "gpio.h" #include "i2c.h" -typedef struct _choice { - int sw; - int action; - GtkWidget *initial_button; - GtkWidget *button; - gulong signal_id; -} CHOICE; - static GtkWidget *parent_window=NULL; static GtkWidget *dialog=NULL; diff --git a/toolbar_menu.c b/toolbar_menu.c index 0297e36..4dd2cc5 100644 --- a/toolbar_menu.c +++ b/toolbar_menu.c @@ -39,14 +39,6 @@ #include "gpio.h" #include "i2c.h" -typedef struct _choice { - int sw; - int action; - GtkWidget *initial_button; - GtkWidget *button; - gulong signal_id; -} CHOICE; - static GtkWidget *parent_window=NULL; static GtkWidget *dialog=NULL; @@ -135,7 +127,6 @@ next_function_set: } else { widget=gtk_button_new_with_label(ActionTable[temp_switches[i].switch_function].str); g_signal_connect(widget,"button-press-event",G_CALLBACK(switch_cb),GINT_TO_POINTER(i)); -g_print("%s: %d\n",__FUNCTION__,i); } gtk_grid_attach(GTK_GRID(grid),widget,col,row,1,1); col++; diff --git a/vfo.c b/vfo.c index 7935ff7..8a5e6b6 100644 --- a/vfo.c +++ b/vfo.c @@ -1457,6 +1457,7 @@ void vfo_rit_update(int rx) { void vfo_rit_clear(int rx) { vfo[receiver[rx]->id].rit=0; + vfo[receiver[rx]->id].rit_enabled=0; receiver_frequency_changed(receiver[rx]); g_idle_add(ext_vfo_update, NULL); } @@ -1469,7 +1470,8 @@ void vfo_rit(int rx,int i) { } else if(value>10000.0) { value=10000.0; } - vfo[receiver[rx]->id].rit=(int)value; + vfo[receiver[rx]->id].rit=value; + vfo[receiver[rx]->id].rit_enabled=(value!=0); receiver_frequency_changed(receiver[rx]); g_idle_add(ext_vfo_update,NULL); } -- 2.45.2