#include "receiver.h"
#include "vfo.h"
#include "button_text.h"
+#include "ext.h"
static GtkWidget *parent_window=NULL;
if(gtk_toggle_button_get_active(widget)) {
active_receiver->agc=GPOINTER_TO_INT(data);
set_agc(active_receiver, active_receiver->agc);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
}
}
}
if (!found) {
fprintf(stderr,"MIDI device %s NOT FOUND!\n", myname);
+ return;
}
// Found our MIDI input device. Spawn off a thread reading data
ret = pthread_create(&midi_thread_id, NULL, midi_thread, NULL);
snd_pcm_sframes_t delay;
long rc;
long trim;
- int mode=modeUSB;
+ int txmode=get_tx_mode();
float *float_buffer;
gint32 *long_buffer;
gint16 *short_buffer;
- if(can_transmit) {
- mode=transmitter->mode;
- }
-
//
// We have to stop the stream here if a CW side tone may occur.
// This might cause underflows, but we cannot use audio_write
// to listen to this rx while transmitting.
//
- if (rx == active_receiver && isTransmitting() && (mode==modeCWU || mode==modeCWL)) {
+ if (rx == active_receiver && isTransmitting() && (txmode==modeCWU || txmode==modeCWL)) {
return 0;
}
static void cw_keyer_sidetone_frequency_value_changed_cb(GtkWidget *widget, gpointer data) {
cw_keyer_sidetone_frequency=gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(widget));
/*
- if(transmitter->mode==modeCWL || transmitter->mode==modeCWU) {
+ int txmode=get_tx_mode();
+ if(txmode==modeCWL || txmode==modeCWU) {
BANDSTACK_ENTRY *entry=bandstack_entry_get_current();
FILTER* band_filters=filters[entry->mode];
FILTER* band_filter=&band_filters[entry->filter];
return 0;
}
+//
+// ALL calls to vfo_update should go through g_idle_add(ext_vfo_update)
+// such that they can be filtered out if they come at high rate
+//
int ext_vfo_update(void *data) {
if (vfo_timeout ==0) {
vfo_timeout=g_timeout_add(100, vfo_timeout_cb, NULL);
i++;
if(steps[i]!=0) {
step=steps[i];
- vfo_update();
}
} else {
i--;
if(i>=0) {
step=steps[i];
- vfo_update();
}
}
}
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
}
SetRXAANRRun(active_receiver->id, active_receiver->nr);
SetRXAEMNRRun(active_receiver->id, active_receiver->nr2);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
}
SetEXTANBRun(active_receiver->id, active_receiver->nb);
SetEXTNOBRun(active_receiver->id, active_receiver->nb2);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
mode_settings[vfo[active_receiver->id].mode].snb=0;
}
SetRXASNBARun(active_receiver->id, active_receiver->snb);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
int ext_lock_update(void *data) {
locked=locked==1?0:1;
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
int ext_rit_update(void *data) {
vfo[active_receiver->id].rit_enabled=vfo[active_receiver->id].rit_enabled==1?0:1;
receiver_frequency_changed(active_receiver);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
int ext_rit_clear(void *data) {
vfo[active_receiver->id].rit=0;
receiver_frequency_changed(active_receiver);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
schedule_high_priority();
}
}
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
int ext_xit_clear(void *data) {
if(can_transmit) {
transmitter->xit=0;
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
}
return 0;
}
}
vfo[id].ctun_frequency=vfo[id].frequency;
set_offset(active_receiver,vfo[id].offset);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
active_receiver->agc=0;
}
set_agc(active_receiver, active_receiver->agc);
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
-int ext_split_update(void *data) {
+int ext_split_toggle(void *data) {
if(can_transmit) {
split=split==1?0:1;
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
- vfo_update();
+ tx_set_mode(transmitter,get_tx_mode());
+ g_idle_add(ext_vfo_update, NULL);
}
return 0;
}
schedule_high_priority();
schedule_receive_specific();
}
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
}
return 0;
}
sat_mode=SAT_NONE;
break;
}
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
function=0;
}
update_toolbar_labels();
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
return 0;
}
extern int ext_a_swap_b(void *data);
extern int ext_ctun_update(void *data);
extern int ext_agc_update(void *data);
-extern int ext_split_update(void *data);
+extern int ext_split_toggle(void *data);
extern int ext_cw_setup();
g_idle_add(ext_agc_update,NULL);
break;
case SPLIT:
- if(can_transmit) g_idle_add(ext_split_update,NULL);
+ if(can_transmit) g_idle_add(ext_split_toggle,NULL);
break;
case DIVERSITY:
g_idle_add(ext_diversity_update,GINT_TO_POINTER(0));
g_idle_add(ext_agc_update,NULL);
break;
case SPLIT:
- if(can_transmit) g_idle_add(ext_split_update,NULL);
+ if(can_transmit) g_idle_add(ext_split_toggle,NULL);
break;
case DIVERSITY:
g_idle_add(ext_diversity_update,GINT_TO_POINTER(0));
#include "iambic.h"
#include "transmitter.h"
#include "ext.h"
+#include "mode.h"
+#include "vfo.h"
static void* keyer_thread(void *arg);
static pthread_t keyer_thread_id;
int i;
int kdelay;
int old_volume;
+ int txmode;
#ifdef __APPLE__
struct timespec now;
#endif
// swallow any cw_events posted during the last "cw hang" time.
if (!kcwl && !kcwr) continue;
- if (!mox && cw_breakin) {
+ // check mode: to not induce RX/TX transition if not in CW mode
+ txmode=get_tx_mode();
+ if (!mox && cw_breakin && (txmode == modeCWU || txmode == modeCWL)) {
g_idle_add(ext_mox_update, (gpointer)(long) 1);
// Wait for mox, that is, wait for WDSP shutting down the RX and
// firing up the TX. This induces a small delay when hitting the key for
case MIDI_SPLIT: // only key supported
// toggle split mode
if (type == MIDI_KEY) {
- if(!split) {
- split=1;
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- split=0;
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
- g_idle_add(ext_vfo_update, NULL);
+ g_idle_add(ext_split_toggle, NULL);
}
break;
/////////////////////////////////////////////////////////// "SWAPRX"
static void new_protocol_general() {
BAND *band;
int rc;
+ int txvfo=get_tx_vfo();
pthread_mutex_lock(&general_mutex);
- if(split) {
- band=band_get_band(vfo[VFO_B].band);
- } else {
- band=band_get_band(vfo[VFO_A].band);
- }
+ band=band_get_band(vfo[txvfo].band);
memset(general_buffer, 0, sizeof(general_buffer));
general_buffer[0]=general_sequence>>24;
long long rxFrequency;
long long txFrequency;
long phase;
- int txmode;
int ddc;
- int txvfo;
+ int txvfo=get_tx_vfo();
+ int txmode=get_tx_mode();
if(data_socket==-1) {
return;
high_priority_buffer_to_radio[2]=high_priority_sequence>>8;
high_priority_buffer_to_radio[3]=high_priority_sequence;
- //
- // Determine VFO controlling the TX frequency
- // and the associated mode
- //
- if(active_receiver->id==VFO_A) {
- if(split) {
- txvfo=VFO_B;
- } else {
- txvfo=VFO_A;
- }
- } else {
- if(split) {
- txvfo=VFO_A;
- } else {
- txvfo=VFO_B;
- }
- }
- txmode=vfo[txvfo].mode;
-
high_priority_buffer_to_radio[4]=running;
//
// We need not set PTT of doing internal CW with break-in
high_priority_buffer_to_radio[345]=power&0xFF;
if(isTransmitting()) {
-
- if(split) {
- band=band_get_band(vfo[VFO_B].band);
- } else {
- band=band_get_band(vfo[VFO_A].band);
- }
+ band=band_get_band(vfo[txvfo].band);
high_priority_buffer_to_radio[1401]=band->OCtx<<1;
if(tune) {
if(OCmemory_tune_time!=0) {
static unsigned char last_50=0;
static void new_protocol_transmit_specific() {
- int mode;
+ int txmode=get_tx_mode();
int rc;
pthread_mutex_lock(&tx_spec_mutex);
transmit_specific_buffer[2]=tx_specific_sequence>>8;
transmit_specific_buffer[3]=tx_specific_sequence;
- if(split) {
- mode=vfo[1].mode;
- } else {
- mode=vfo[0].mode;
- }
transmit_specific_buffer[4]=1; // 1 DAC
transmit_specific_buffer[5]=0; // default no CW
- if ((mode==modeCWU || mode==modeCWL) && cw_keyer_internal) {
+ if ((txmode==modeCWU || txmode==modeCWL) && cw_keyer_internal) {
//
// Set this byte only if in CW, and if using the "internal" keyer
//
#endif
}
- //int tx_vfo=split?VFO_B:VFO_A;
- //if(vfo[tx_vfo].mode==modeCWL || vfo[tx_vfo].mode==modeCWU) {
- // local_ptt=local_ptt|dot|dash;
- //}
if(previous_ptt!=local_ptt) {
g_idle_add(ext_mox_update,(gpointer)(long)(local_ptt));
}
void new_protocol_cw_audio_samples(short left_audio_sample,short right_audio_sample) {
int rc;
- int mode=transmitter->mode;
+ int txmode=get_tx_mode();
//
// Only process samples if transmitting in CW
- if (isTransmitting() && (mode==modeCWU || mode==modeCWL)) {
+ if (isTransmitting() && (txmode==modeCWU || txmode==modeCWL)) {
// insert the samples
audiobuffer[audioindex++]=left_audio_sample>>8;
void new_protocol_audio_samples(RECEIVER *rx,short left_audio_sample,short right_audio_sample) {
int rc;
- int mode=transmitter->mode;
+ int txmode=get_tx_mode();
//
// Only process samples if NOT transmitting in CW
- if (isTransmitting() && (mode==modeCWU || mode==modeCWL)) return;
+ if (isTransmitting() && (txmode==modeCWU || txmode==modeCWL)) return;
// insert the samples
audiobuffer[audioindex++]=left_audio_sample>>8;
// indicates that we should use the TX frequency.
// We have to adjust by the offset for CTUN mode
//
- if(active_receiver->id==VFO_A) {
- vfonum = split ? VFO_B : VFO_A;
- } else {
- vfonum = split ? VFO_A : VFO_B;
- }
+ vfonum=get_tx_vfo();
freq=vfo[vfonum].frequency-vfo[vfonum].lo;
if (vfo[vfonum].ctun) freq += vfo[vfonum].offset;
if(transmitter->xit_enabled) {
int id=active_receiver->id;
- int tx_vfo=split?VFO_B:VFO_A;
-
int num_hpsdr_receivers=how_many_receivers();
int rxfdbk = rx_feedback_channel();
int txfdbk = tx_feedback_channel();
void ozy_send_buffer() {
- int mode;
+ int txmode=get_tx_mode();
+ int txvfo=get_tx_vfo();
int i;
BAND *band;
int num_hpsdr_receivers=how_many_receivers();
}
band=band_get_band(vfo[VFO_A].band);
if(isTransmitting()) {
- if(split) {
- band=band_get_band(vfo[VFO_B].band);
- }
+ band=band_get_band(vfo[txvfo].band);
output_buffer[C2]|=band->OCtx<<1;
if(tune) {
if(OCmemory_tune_time!=0) {
// Therefore, when in CW mode, send the TX drive level also when receiving.
// (it would be sufficient to do so only with internal CW).
//
- if(split) {
- mode=vfo[1].mode;
- } else {
- mode=vfo[0].mode;
- }
- if(isTransmitting() || (mode == modeCWU) || (mode == modeCWL)) {
+ if(isTransmitting() || (txmode == modeCWU) || (txmode == modeCWL)) {
if(tune && !transmitter->tune_use_drive) {
double fac=sqrt((double)transmitter->tune_percent * 0.01);
power=(int)((double)transmitter->drive_level*fac);
break;
case 7:
output_buffer[C0]=0x1E;
- if(split) {
- mode=vfo[1].mode;
- } else {
- mode=vfo[0].mode;
- }
output_buffer[C1]=0x00;
- if((mode==modeCWU || mode==modeCWL) && !tune && cw_keyer_internal && !transmitter->twotone) {
+ if((txmode==modeCWU || txmode==modeCWL) && !tune && cw_keyer_internal && !transmitter->twotone) {
output_buffer[C1]|=0x01;
}
output_buffer[C2]=cw_keyer_sidetone_volume;
}
// set mox
- if(split) {
- mode=vfo[1].mode;
- } else {
- mode=vfo[0].mode;
- }
if (isTransmitting()) {
- if(mode==modeCWU || mode==modeCWL) {
+ if(txmode==modeCWU || txmode==modeCWL) {
//
// For "internal" CW, we should not set
// the MOX bit, everything is done in the FPGA.
static void pa_value_changed_cb(GtkWidget *widget, gpointer data) {
BAND *band=(BAND *)data;
band->pa_calibration=gtk_spin_button_get_value(GTK_SPIN_BUTTON(widget));
- int v=VFO_A;
- if(split) v=VFO_B;
- int b=vfo[v].band;
+ int txvfo=get_tx_vfo();
+ int b=vfo[txvfo].band;
BAND *current=band_get_band(b);
if(band==current) {
calcDriveLevel();
}
}
- int mode=vfo[VFO_A].mode;
- if(split) {
- mode=vfo[VFO_B].mode;
- }
- pre_tune_mode=mode;
+ int txmode=get_tx_mode();
+ pre_tune_mode=txmode;
pre_tune_cw_internal=cw_keyer_internal;
//
// in LSB/DIGL, tune 1000 Hz below carrier
// all other (CW, AM, FM): tune on carrier freq.
//
- switch(mode) {
+ switch(txmode) {
case modeLSB:
case modeDIGL:
SetTXAPostGenToneFreq(transmitter->id,-(double)1000.0);
SetTXAPostGenMode(transmitter->id,0);
SetTXAPostGenRun(transmitter->id,1);
- switch(mode) {
+ switch(txmode) {
case modeCWL:
cw_keyer_internal=0;
tx_set_mode(transmitter,modeLSB);
static int calcLevel(double d) {
int level=0;
- int v=VFO_A;
- if(split) v=VFO_B;
+ int v=get_tx_vfo();
BAND *band=band_get_band(vfo[v].band);
double target_dbm = 10.0 * log10(d * 1000.0);
#endif
#include "gpio.h"
#include "vfo.h"
+#include "ext.h"
static GtkWidget *parent_window=NULL;
static GtkWidget *menu_b=NULL;
}
static void split_cb(GtkWidget *widget, gpointer data) {
- split=gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
- vfo_update();
+ int new=gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+ if (new != split) g_idle_add(ext_split_toggle, NULL);
}
//
-// call-able from outside, e.g. toolbar or MIDI
+// call-able from outside, e.g. toolbar or MIDI, through g_idle_add
//
void setDuplex() {
if(duplex) {
gtk_widget_destroy(transmitter->dialog);
reconfigure_transmitter(transmitter,display_width,rx_height*receivers);
}
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
}
static void duplex_cb(GtkWidget *widget, gpointer data) {
static void sat_cb(GtkWidget *widget, gpointer data) {
sat_mode=gtk_combo_box_get_active(GTK_COMBO_BOX(widget));
- vfo_update();
+ g_idle_add(ext_vfo_update, NULL);
}
static void load_filters(void) {
int rc;
double f;
- v=active_receiver->id;
- if(split) {
- v=active_receiver->id==0?1:0;
- }
+ v=get_tx_vfo();
if(soapy_device!=NULL) {
//f=(double)(vfo[v].frequency+vfo[v].ctun_frequency-vfo[v].lo_tx);
if(vfo[v].ctun) {
}
static void split_cb (GtkWidget *widget, gpointer data) {
- if(can_transmit) {
- split=split==1?0:1;
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
- g_idle_add(ext_vfo_update,NULL);
- }
+ g_idle_add(ext_split_toggle, NULL);
}
static void duplex_cb (GtkWidget *widget, gpointer data) {
void transmitter_set_out_of_band(TRANSMITTER *tx) {
tx->out_of_band=1;
+ g_idle_add(ext_vfo_update,NULL);
tx->out_of_band_timer_id=gdk_threads_add_timeout_full(G_PRIORITY_HIGH_IDLE,1000,update_out_of_band, tx, NULL);
}
TXASetMP(tx->id, tx->low_latency);
- int mode=vfo[VFO_A].mode;
- if(split) {
- mode=vfo[VFO_B].mode;
- }
-
SetTXABandpassWindow(tx->id, 1);
SetTXABandpassRun(tx->id, 1);
SetTXACompressorGain(tx->id, tx->compressor_level);
SetTXACompressorRun(tx->id, tx->compressor);
- tx_set_mode(tx,mode);
+ tx_set_mode(tx,get_tx_mode());
XCreateAnalyzer(tx->id, &rc, 262144, 1, 1, "");
if (rc != 0) {
}
void tx_set_filter(TRANSMITTER *tx,int low,int high) {
- int mode;
- if(split) {
- mode=vfo[1].mode;
- } else {
- mode=vfo[0].mode;
- }
+ int txmode=get_tx_mode();
- switch(mode) {
+ switch(txmode) {
case modeLSB:
case modeCWL:
case modeDIGL:
int display_width=gtk_widget_get_allocated_width (tx->panadapter);
int display_height=gtk_widget_get_allocated_height (tx->panadapter);
- // id = VFO which contains the TX frequency
- int id = active_receiver->id;
- if (split) {
- id = 1-id;
- }
+ int txvfo = get_tx_vfo();
+ int txmode = get_tx_mode();
+
samples=tx->pixel_samples;
hz_per_pixel=(double)tx->iq_output_rate/(double)tx->pixels;
// filter
- if (vfo[id].mode != modeCWU && vfo[id].mode != modeCWL) {
+ if (txmode != modeCWU && txmode != modeCWL) {
cairo_set_source_rgb (cr, 0.25, 0.25, 0.25);
filter_left=(double)display_width/2.0+((double)tx->filter_low/hz_per_pixel);
filter_right=(double)display_width/2.0+((double)tx->filter_high/hz_per_pixel);
//long long half=24000LL; //(long long)(tx->output_rate/2);
long long half=6000LL; //(long long)(tx->output_rate/2);
long long frequency;
- if(vfo[id].ctun) {
- frequency=vfo[id].ctun_frequency-vfo[id].lo_tx;
+ if(vfo[txvfo].ctun) {
+ frequency=vfo[txvfo].ctun_frequency-vfo[txvfo].lo_tx;
} else {
- frequency=vfo[id].frequency-vfo[id].lo_tx;
+ frequency=vfo[txvfo].frequency-vfo[txvfo].lo_tx;
}
double vfofreq=(double)display_width * 0.5;
if (!cw_is_on_vfo_freq) {
- if(vfo[id].mode==modeCWU) {
+ if(txmode==modeCWU) {
frequency+=(long long)cw_keyer_sidetone_frequency;
vfofreq -= (double) cw_keyer_sidetone_frequency/hz_per_pixel;
- } else if(vfo[id].mode==modeCWL) {
+ } else if(txmode==modeCWL) {
frequency-=(long long)cw_keyer_sidetone_frequency;
vfofreq += (double) cw_keyer_sidetone_frequency/hz_per_pixel;
}
// band edges
long long min_display=frequency-half;
long long max_display=frequency+half;
- int b=vfo[id].band;
+ int b=vfo[txvfo].band;
BAND *band=band_get_band(b);
if(band->frequencyMin!=0LL) {
cairo_set_source_rgb (cr, 1.0, 0.0, 0.0);
}
if(can_transmit) {
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
//
// If the band has changed, it is necessary to re-calculate
// the drive level. Furthermore, possibly the "PA disable"
}
if(can_transmit) {
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
}
//
// I do not think the band can change within this function.
break;
}
if(can_transmit) {
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
}
//
// changing modes may change BFO frequency
receiver_vfo_changed(receiver[1]);
}
if(can_transmit) {
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
}
g_idle_add(ext_vfo_update,NULL);
}
vfo[VFO_A].rit=vfo[VFO_B].rit;
receiver_vfo_changed(receiver[0]);
if(can_transmit) {
- if(!split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
}
g_idle_add(ext_vfo_update,NULL);
}
receiver_vfo_changed(receiver[1]);
}
if(can_transmit) {
- if(split) {
- tx_set_mode(transmitter,vfo[VFO_B].mode);
- } else {
- tx_set_mode(transmitter,vfo[VFO_A].mode);
- }
+ tx_set_mode(transmitter,get_tx_mode());
}
g_idle_add(ext_vfo_update,NULL);
}
void vfo_update() {
int id=active_receiver->id;
+ int txvfo=get_tx_vfo();
+
FILTER* band_filters=filters[vfo[id].mode];
FILTER* band_filter=&band_filters[vfo[id].filter];
if(vfo_surface) {
// If it is out-of-band, we display "Out of band" in red.
// Frequencies we are not transmitting on are displayed in green
// (dimmed if the freq. does not belong to the active receiver).
- // Depending on which receiver is the active one, and if we use split,
- // the following frequencies are used for transmitting (see old_protocol.c):
- // id == 0, split == 0 : TX freq = VFO_A
- // id == 0, split == 1 : TX freq = VFO_B
- // id == 1, split == 0 : TX freq = VFO_B
- // id == 1, split == 1 : TX freq = VFO_A
-
-
- long long af;
- if(isTransmitting() && !split) {
- if(vfo[0].ctun) {
- af=(double)(vfo[0].ctun_frequency-vfo[0].lo_tx);
- } else {
- af=(double)(vfo[0].frequency-vfo[0].lo_tx);
- }
- } else {
- if(vfo[0].ctun) {
- af=(double)(vfo[0].ctun_frequency);
- } else {
- af=(double)(vfo[0].frequency);
- }
- }
+
+ // Frequencies of VFO A and B
+
+ long long af = vfo[0].ctun ? vfo[0].ctun_frequency : vfo[0].frequency;
+ long long bf = vfo[1].ctun ? vfo[1].ctun_frequency : vfo[1].frequency;
+
+ int oob=0;
+ if (can_transmit) oob=transmitter->out_of_band;
sprintf(temp_text,"VFO A: %0lld.%06lld",af/(long long)1000000,af%(long long)1000000);
- if(isTransmitting() && ((id == 0 && !split) || (id == 1 && split))) {
- if (transmitter->out_of_band) sprintf(temp_text,"VFO A: Out of band");
+
+ if(txvfo == 0 && (isTransmitting() || oob)) {
+ if (oob) sprintf(temp_text,"VFO A: Out of band");
cairo_set_source_rgb(cr, 1.0, 0.0, 0.0);
} else {
if(id==0) {
cairo_set_font_size(cr, 22);
cairo_show_text(cr, temp_text);
-
- long long bf;
- if(isTransmitting() && split) {
- if(vfo[1].ctun) {
- bf=(double)(vfo[1].ctun_frequency-vfo[1].lo_tx);
- } else {
- bf=(double)(vfo[1].frequency-vfo[1].lo_tx);
- }
- } else {
- if(vfo[1].ctun) {
- bf=(double)(vfo[1].ctun_frequency);
- } else {
- bf=(double)(vfo[1].frequency);
- }
- }
sprintf(temp_text,"VFO B: %0lld.%06lld",bf/(long long)1000000,bf%(long long)1000000);
- if(isTransmitting() && ((id == 0 && split) || (id == 1 && !split))) {
- if (transmitter->out_of_band) sprintf(temp_text,"VFO B: Out of band");
+ if(txvfo == 1 && (isTransmitting() || oob)) {
+ if (oob) sprintf(temp_text,"VFO B: Out of band");
cairo_set_source_rgb(cr, 1.0, 0.0, 0.0);
} else {
if(id==1) {
cairo_set_source_rgb(cr, 1.0, 1.0, 0.0);
cairo_show_text(cr, temp_text);
- //
- // af: Frequency of VFO_A, bf: Frequency of VFO_B, without xit/rit etc.
- //
+
long long txfreq;
int txlow, txhigh;
- if ((id == 0 && ! split) || (id == 1 && split)) {
- // txfreq derived from VFO A
- txfreq=af;
- } else {
- txfreq=bf;
- }
- if (transmitter->xit_enabled) txfreq += transmitter->xit;
- //
- // In CW modes, the signal is generated explicitly (without WDSP),
- // so we can assume a zero-width filter. Note that the CW signal
- // can be on the VFO frequency or offset by the side tone freq.
- //
- switch (transmitter->mode) {
- case modeCWU:
- txlow = txhigh = cw_is_on_vfo_freq ? 0 : cw_keyer_sidetone_frequency;
- break;
- case modeCWL:
- txlow = txhigh = cw_is_on_vfo_freq ? 0 : -cw_keyer_sidetone_frequency;
- break;
- default:
- txlow =transmitter->filter_low;
- txhigh=transmitter->filter_high;
+ if (can_transmit) {
+ txfreq = (txvfo == 0) ? af : bf;
+ if (transmitter->xit_enabled) txfreq += transmitter->xit;
+ //
+ // In CW modes, the signal is generated explicitly (without WDSP),
+ // so we can assume a zero-width filter. Note that the CW signal
+ // can be on the VFO frequency or offset by the side tone freq.
+ //
+ switch (transmitter->mode) {
+ case modeCWU:
+ txlow = txhigh = cw_is_on_vfo_freq ? 0 : cw_keyer_sidetone_frequency;
+ break;
+ case modeCWL:
+ txlow = txhigh = cw_is_on_vfo_freq ? 0 : -cw_keyer_sidetone_frequency;
+ break;
+ default:
+ txlow =transmitter->filter_low;
+ txhigh=transmitter->filter_high;
break;
+ }
+ } else {
+ //
+ // If there is no transmitter, use VFO frequency of active receiver
+ //
+ txfreq = (id == 0) ? af : bf;
+ txlow = 0;
+ txhigh =0;
}
getFrequencyInfo(txfreq, txlow, txhigh);
/*
return vfo_panel;
}
+
+//
+// Some utility functions to get characteristics of the current
+// transmitter. These functions can be used even if there is no
+// transmitter (transmitter->mode may segfault).
+//
+
+int get_tx_vfo() {
+ int txvfo=active_receiver->id;
+ if (split) txvfo = 1 - txvfo;
+ return txvfo;
+}
+
+int get_tx_mode() {
+ int txvfo=active_receiver->id;
+ if (split) txvfo = 1 - txvfo;
+ if (can_transmit) {
+ return vfo[txvfo].mode;
+ } else {
+ return modeUSB;
+ }
+}
+
+long long get_tx_freq() {
+ int txvfo=active_receiver->id;
+ if (split) txvfo = 1 - txvfo;
+ if (vfo[txvfo].ctun) {
+ return vfo[txvfo].ctun_frequency;
+ } else {
+ return vfo[txvfo].frequency;
+ }
+}
extern void vfo_b_to_a();
extern void vfo_a_swap_b();
+extern int get_tx_vfo();
+extern int get_tx_mode();
+extern long long get_tx_freq();
+
extern void vfo_xvtr_changed();
#endif