# 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
int action;
GtkWidget *button;
gulong signal_id;
+ struct _choice *previous;
} CHOICE;
static GtkWidget *dialog;
int action_dialog(GtkWidget *parent,int filter,int currentAction) {
int i,j;
+ CHOICE *previous=NULL;
+ CHOICE *choice=NULL;
action=currentAction;
previous_button=NULL;
for(i=0;i<ACTIONS;i++) {
if((ActionTable[i].type&filter) || (ActionTable[i].type==TYPE_NONE)) {
GtkWidget *button=gtk_toggle_button_new_with_label(ActionTable[i].str);
- gtk_widget_set_name(button,"small_button");
+ gtk_widget_set_name(button,"small_toggle_button");
gtk_grid_attach(GTK_GRID(grid),button,j%GRID_WIDTH,j/GRID_WIDTH,1,1);
if(ActionTable[i].action==currentAction) {
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button),TRUE);
}
- CHOICE *choice=g_new0(CHOICE,1);
+ choice=g_new0(CHOICE,1);
choice->action=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;
if(result!=GTK_RESPONSE_ACCEPT) {
action=currentAction;
}
+ // free up choice structures
+ while(previous!=NULL) {
+ choice=previous;
+ previous=choice->previous;
+ g_free(choice);
+ }
return action;
}
{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;
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:
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:
value=10000.0;
}
transmitter->xit=(int)value;
+ transmitter->xit_enabled=(value!=0);
if(protocol==NEW_PROTOCOL) {
schedule_high_priority();
}
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);
}
}
value=10000.0;
}
transmitter->xit=(int)value;
+ transmitter->xit_enabled=(value!=0);
if(protocol==NEW_PROTOCOL) {
schedule_high_priority();
}
value=10000.0;
}
transmitter->xit=(int)value;
+ transmitter->xit_enabled=(value!=0);
if(protocol==NEW_PROTOCOL) {
schedule_high_priority();
}
}
break;
}
- g_free(data);
+ if(free_action) {
+ g_free(data);
+ }
return 0;
}
#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() {
#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;
static int init(void *data) {
char wisdom_directory[1024];
+ int counter;
g_print("%s\n",__FUNCTION__);
//
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
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.
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);
}
}
+ 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;
}
}
+ 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;
}
}
+ txFrequency+=calibration;
+
phase=(long)((4294967296.0*(double)txFrequency)/122880000.0);
if(isTransmitting() && transmitter->puresignal) {
}
}
}
+ freq+=calibration;
return freq;
}
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;
static GtkWidget *waterfall;
static GtkWidget *audio_waterfall;
+// RX and TX calibration
+long long calibration=0LL;
+
/*
#ifdef GPIO
static GtkWidget *encoders;
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);
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);
extern GtkWidget *fixed;
+extern long long calibration;
+
extern char property_path[];
#define NONE 0
*/
#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));
}
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;
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), "<b>Frequency\nCalibration(Hz):</b>");
+ 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), "<b>RX Gain Calibration:</b>");
gtk_grid_attach(GTK_GRID(grid),rx_gain_label,col,row,1,1);
g_signal_connect(PA_enable_b,"toggled",G_CALLBACK(PA_enable_cb),NULL);
}
- row++;
}
+ row++;
if(row>temp_row) temp_row=row;
#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;
#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;
} 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++;
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);
}
} 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);
}