mirror of
https://github.com/saitohirga/WSJT-X.git
synced 2024-11-26 22:28:41 -05:00
821 lines
24 KiB
C++
821 lines
24 KiB
C++
#include "OmniRigTransceiver.hpp"
|
|
|
|
#include <QDebug>
|
|
#include <objbase.h>
|
|
#include <QThread>
|
|
#include <QEventLoop>
|
|
|
|
#include "qt_helpers.hpp"
|
|
|
|
#include "moc_OmniRigTransceiver.cpp"
|
|
|
|
namespace
|
|
{
|
|
auto constexpr OmniRig_transceiver_one_name = "OmniRig Rig 1";
|
|
auto constexpr OmniRig_transceiver_two_name = "OmniRig Rig 2";
|
|
}
|
|
|
|
auto OmniRigTransceiver::map_mode (OmniRig::RigParamX param) -> MODE
|
|
{
|
|
if (param & OmniRig::PM_CW_U)
|
|
{
|
|
return CW_R;
|
|
}
|
|
else if (param & OmniRig::PM_CW_L)
|
|
{
|
|
return CW;
|
|
}
|
|
else if (param & OmniRig::PM_SSB_U)
|
|
{
|
|
return USB;
|
|
}
|
|
else if (param & OmniRig::PM_SSB_L)
|
|
{
|
|
return LSB;
|
|
}
|
|
else if (param & OmniRig::PM_DIG_U)
|
|
{
|
|
return DIG_U;
|
|
}
|
|
else if (param & OmniRig::PM_DIG_L)
|
|
{
|
|
return DIG_L;
|
|
}
|
|
else if (param & OmniRig::PM_AM)
|
|
{
|
|
return AM;
|
|
}
|
|
else if (param & OmniRig::PM_FM)
|
|
{
|
|
return FM;
|
|
}
|
|
CAT_ERROR ("unrecognized mode");
|
|
throw_qstring (tr ("OmniRig: unrecognized mode"));
|
|
return UNK;
|
|
}
|
|
|
|
OmniRig::RigParamX OmniRigTransceiver::map_mode (MODE mode)
|
|
{
|
|
switch (mode)
|
|
{
|
|
case AM: return OmniRig::PM_AM;
|
|
case CW: return OmniRig::PM_CW_L;
|
|
case CW_R: return OmniRig::PM_CW_U;
|
|
case USB: return OmniRig::PM_SSB_U;
|
|
case LSB: return OmniRig::PM_SSB_L;
|
|
case FSK: return OmniRig::PM_DIG_L;
|
|
case FSK_R: return OmniRig::PM_DIG_U;
|
|
case DIG_L: return OmniRig::PM_DIG_L;
|
|
case DIG_U: return OmniRig::PM_DIG_U;
|
|
case FM: return OmniRig::PM_FM;
|
|
case DIG_FM: return OmniRig::PM_FM;
|
|
default: break;
|
|
}
|
|
return OmniRig::PM_SSB_U; // quieten compiler grumble
|
|
}
|
|
|
|
void OmniRigTransceiver::register_transceivers (logger_type *,
|
|
TransceiverFactory::Transceivers * registry,
|
|
int id1, int id2)
|
|
{
|
|
(*registry)[OmniRig_transceiver_one_name] = TransceiverFactory::Capabilities {
|
|
id1
|
|
, TransceiverFactory::Capabilities::none // COM isn't serial or network
|
|
, true // does PTT
|
|
, false // doesn't select mic/data (use OmniRig config file)
|
|
, true // can remote control RTS nd DTR
|
|
, true // asynchronous interface
|
|
};
|
|
(*registry)[OmniRig_transceiver_two_name] = TransceiverFactory::Capabilities {
|
|
id2
|
|
, TransceiverFactory::Capabilities::none // COM isn't serial or network
|
|
, true // does PTT
|
|
, false // doesn't select mic/data (use OmniRig config file)
|
|
, true // can remote control RTS nd DTR
|
|
, true // asynchronous interface
|
|
};
|
|
}
|
|
|
|
OmniRigTransceiver::OmniRigTransceiver (logger_type * logger,
|
|
std::unique_ptr<TransceiverBase> wrapped,
|
|
RigNumber n, TransceiverFactory::PTTMethod ptt_type,
|
|
QString const& ptt_port, QObject * parent)
|
|
: TransceiverBase {logger, parent}
|
|
, wrapped_ {std::move (wrapped)}
|
|
, use_for_ptt_ {TransceiverFactory::PTT_method_CAT == ptt_type || ("CAT" == ptt_port && (TransceiverFactory::PTT_method_RTS == ptt_type || TransceiverFactory::PTT_method_DTR == ptt_type))}
|
|
, ptt_type_ {ptt_type}
|
|
, rig_number_ {n}
|
|
, readable_params_ {0}
|
|
, writable_params_ {0}
|
|
, send_update_signal_ {false}
|
|
, reversed_ {false}
|
|
{
|
|
CoInitializeEx (nullptr, 0 /*COINIT_APARTMENTTHREADED*/); // required because Qt only does this for GUI thread
|
|
}
|
|
|
|
OmniRigTransceiver::~OmniRigTransceiver ()
|
|
{
|
|
CoUninitialize ();
|
|
}
|
|
|
|
// returns false on time out
|
|
bool OmniRigTransceiver::await_notification_with_timeout (int timeout)
|
|
{
|
|
QEventLoop el;
|
|
connect (this, &OmniRigTransceiver::notified, &el, [&el] () {el.exit (1);});
|
|
QTimer::singleShot (timeout, Qt::CoarseTimer, &el, [&el] () {el.exit (0);});
|
|
return 1 == el.exec (); // wait for notify or timer
|
|
}
|
|
|
|
int OmniRigTransceiver::do_start ()
|
|
{
|
|
CAT_TRACE ("starting");
|
|
|
|
if (wrapped_) wrapped_->start (0);
|
|
|
|
omni_rig_.reset (new OmniRig::OmniRigX {this});
|
|
if (omni_rig_->isNull ())
|
|
{
|
|
CAT_ERROR ("failed to start COM server");
|
|
throw_qstring (tr ("Failed to start OmniRig COM server"));
|
|
}
|
|
|
|
// COM/OLE exceptions get signaled
|
|
connect (&*omni_rig_, SIGNAL (exception (int, QString, QString, QString)), this, SLOT (handle_COM_exception (int, QString, QString, QString)));
|
|
|
|
// IOmniRigXEvent interface signals
|
|
connect (&*omni_rig_, SIGNAL (VisibleChange ()), this, SLOT (handle_visible_change ()));
|
|
connect (&*omni_rig_, SIGNAL (RigTypeChange (int)), this, SLOT (handle_rig_type_change (int)));
|
|
connect (&*omni_rig_, SIGNAL (StatusChange (int)), this, SLOT (handle_status_change (int)));
|
|
connect (&*omni_rig_, SIGNAL (ParamsChange (int, int)), this, SLOT (handle_params_change (int, int)));
|
|
connect (&*omni_rig_
|
|
, SIGNAL (CustomReply (int, QVariant const&, QVariant const&))
|
|
, this, SLOT (handle_custom_reply (int, QVariant const&, QVariant const&)));
|
|
|
|
CAT_INFO ("OmniRig s/w version: " << omni_rig_->SoftwareVersion ()
|
|
<< "i/f version: " << omni_rig_->InterfaceVersion ());
|
|
|
|
// fetch the interface of the RigX CoClass and instantiate a proxy object
|
|
switch (rig_number_)
|
|
{
|
|
case One: rig_.reset (new OmniRig::RigX (omni_rig_->Rig1 ())); break;
|
|
case Two: rig_.reset (new OmniRig::RigX (omni_rig_->Rig2 ())); break;
|
|
}
|
|
|
|
Q_ASSERT (rig_);
|
|
Q_ASSERT (!rig_->isNull ());
|
|
|
|
// COM/OLE exceptions get signaled
|
|
connect (&*rig_, SIGNAL (exception (int, QString, QString, QString)), this, SLOT (handle_COM_exception (int, QString, QString, QString)));
|
|
|
|
offline_timer_.reset (new QTimer); // instantiate here as
|
|
// constructor runs in wrong
|
|
// thread
|
|
offline_timer_->setSingleShot (true);
|
|
connect (offline_timer_.data (), &QTimer::timeout, [this] () {offline ("Rig went offline");});
|
|
|
|
if (use_for_ptt_ && (TransceiverFactory::PTT_method_DTR == ptt_type_ || TransceiverFactory::PTT_method_RTS == ptt_type_))
|
|
{
|
|
// fetch the interface for the serial port if we need it for PTT
|
|
port_.reset (new OmniRig::PortBits (rig_->PortBits ()));
|
|
|
|
Q_ASSERT (port_);
|
|
Q_ASSERT (!port_->isNull ());
|
|
|
|
// COM/OLE exceptions get signaled
|
|
connect (&*port_, SIGNAL (exception (int, QString, QString, QString)), this, SLOT (handle_COM_exception (int, QString, QString, QString)));
|
|
|
|
CAT_TRACE ("OmniRig RTS state: " << port_->Rts ());
|
|
|
|
// remove locking because it doesn't seem to work properly
|
|
// if (!port_->Lock ()) // try to take exclusive use of the OmniRig serial port for PTT
|
|
// {
|
|
// CAT_WARNING ("Failed to get exclusive use of serial port for PTT from OmniRig");
|
|
// }
|
|
|
|
// start off so we don't accidentally key the radio
|
|
if (TransceiverFactory::PTT_method_DTR == ptt_type_)
|
|
{
|
|
port_->SetDtr (false);
|
|
}
|
|
else // RTS
|
|
{
|
|
port_->SetRts (false);
|
|
}
|
|
}
|
|
|
|
rig_type_ = rig_->RigType ();
|
|
readable_params_ = rig_->ReadableParams ();
|
|
writable_params_ = rig_->WriteableParams ();
|
|
|
|
CAT_INFO (QString {"OmniRig initial rig type: %1 readable params=0x%2 writable params=0x%3 for rig %4"}
|
|
.arg (rig_type_)
|
|
.arg (readable_params_, 8, 16, QChar ('0'))
|
|
.arg (writable_params_, 8, 16, QChar ('0'))
|
|
.arg (rig_number_));
|
|
for (int i = 0; i < 5; ++i)
|
|
{
|
|
if (OmniRig::ST_ONLINE == rig_->Status ())
|
|
{
|
|
break;
|
|
}
|
|
await_notification_with_timeout (1000);
|
|
}
|
|
if (OmniRig::ST_ONLINE != rig_->Status ())
|
|
{
|
|
throw_qstring ("OmniRig: " + rig_->StatusStr ());
|
|
}
|
|
QThread::msleep (500); // leave some time for Omni-Rig to get
|
|
// the rig status for the 1st. time
|
|
auto f = rig_->GetRxFrequency ();
|
|
for (int i = 0; (f == 0) && (i < 5); ++i)
|
|
{
|
|
await_notification_with_timeout (1000);
|
|
f = rig_->GetRxFrequency ();
|
|
}
|
|
update_rx_frequency (f);
|
|
int resolution {0};
|
|
if (OmniRig::PM_UNKNOWN == rig_->Vfo ()
|
|
&& (writable_params_ & (OmniRig::PM_VFOA | OmniRig::PM_VFOB))
|
|
== (OmniRig::PM_VFOA | OmniRig::PM_VFOB))
|
|
{
|
|
// start with VFO A (probably MAIN) on rigs that we
|
|
// can't query VFO but can set explicitly
|
|
rig_->SetVfo (OmniRig::PM_VFOA);
|
|
}
|
|
f = state ().frequency ();
|
|
if (f % 10) return resolution; // 1Hz resolution
|
|
auto test_frequency = f - f % 100 + 55;
|
|
if (OmniRig::PM_FREQ & writable_params_)
|
|
{
|
|
rig_->SetFreq (test_frequency);
|
|
}
|
|
else if (reversed_ && (OmniRig::PM_FREQB & writable_params_))
|
|
{
|
|
rig_->SetFreqB (test_frequency);
|
|
}
|
|
else if (!reversed_ && (OmniRig::PM_FREQA & writable_params_))
|
|
{
|
|
rig_->SetFreqA (test_frequency);
|
|
}
|
|
else
|
|
{
|
|
throw_qstring (tr ("OmniRig: don't know how to set rig frequency"));
|
|
}
|
|
if (!await_notification_with_timeout (1000))
|
|
{
|
|
CAT_ERROR ("do_start 1: wait timed out");
|
|
throw_qstring (tr ("OmniRig: timeout waiting for update from rig"));
|
|
}
|
|
switch (rig_->GetRxFrequency () - test_frequency)
|
|
{
|
|
case -5: resolution = -1; break; // 10Hz truncated
|
|
case 5: resolution = 1; break; // 10Hz rounded
|
|
case -15: resolution = -2; break; // 20Hz truncated
|
|
case -55: resolution = -2; break; // 100Hz truncated
|
|
case 45: resolution = 2; break; // 100Hz rounded
|
|
}
|
|
if (1 == resolution) // may be 20Hz rounded
|
|
{
|
|
test_frequency = f - f % 100 + 51;
|
|
if (OmniRig::PM_FREQ & writable_params_)
|
|
{
|
|
rig_->SetFreq (test_frequency);
|
|
}
|
|
else if (reversed_ && (OmniRig::PM_FREQB & writable_params_))
|
|
{
|
|
rig_->SetFreqB (test_frequency);
|
|
}
|
|
else if (!reversed_ && (OmniRig::PM_FREQA & writable_params_))
|
|
{
|
|
rig_->SetFreqA (test_frequency);
|
|
}
|
|
if (!await_notification_with_timeout (2000))
|
|
{
|
|
CAT_ERROR ("do_start 2: wait timed out");
|
|
throw_qstring (tr ("OmniRig: timeout waiting for update from rig"));
|
|
}
|
|
if (9 == rig_->GetRxFrequency () - test_frequency)
|
|
{
|
|
resolution = 2; // 20Hz rounded
|
|
}
|
|
}
|
|
if (OmniRig::PM_FREQ & writable_params_)
|
|
{
|
|
rig_->SetFreq (f);
|
|
}
|
|
else if (reversed_ && (OmniRig::PM_FREQB & writable_params_))
|
|
{
|
|
rig_->SetFreqB (f);
|
|
}
|
|
else if (!reversed_ && (OmniRig::PM_FREQA & writable_params_))
|
|
{
|
|
rig_->SetFreqA (f);
|
|
}
|
|
update_rx_frequency (f);
|
|
return resolution;
|
|
}
|
|
|
|
void OmniRigTransceiver::do_stop ()
|
|
{
|
|
QThread::msleep (200); // leave some time for pending
|
|
// commands at the server end
|
|
|
|
offline_timer_.reset (); // destroy here rather than in
|
|
// destructor as destructor runs in
|
|
// wrong thread
|
|
|
|
if (port_ && !port_->isNull ())
|
|
{
|
|
// port_->Unlock (); // release serial port
|
|
port_->clear ();
|
|
port_.reset ();
|
|
}
|
|
if (omni_rig_ && !omni_rig_->isNull ())
|
|
{
|
|
if (rig_ && !rig_->isNull ())
|
|
{
|
|
rig_->clear ();
|
|
rig_.reset ();
|
|
}
|
|
omni_rig_->clear ();
|
|
omni_rig_.reset ();
|
|
}
|
|
|
|
if (wrapped_) wrapped_->stop ();
|
|
|
|
CAT_TRACE ("stopped");
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_COM_exception (int code, QString source, QString desc, QString help)
|
|
{
|
|
CAT_ERROR ((QString::number (code) + " at " + source + ": " + desc + " (" + help + ')'));
|
|
throw_qstring (tr ("OmniRig COM/OLE error: %1 at %2: %3 (%4)").arg (QString::number (code)).arg (source). arg (desc). arg (help));
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_visible_change ()
|
|
{
|
|
if (!omni_rig_ || omni_rig_->isNull ()) return;
|
|
CAT_TRACE ("visibility change: visibility =" << omni_rig_->DialogVisible ());
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_rig_type_change (int rig_number)
|
|
{
|
|
CAT_TRACE ("rig type change: rig =" << rig_number);
|
|
if (rig_number_ == rig_number)
|
|
{
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
readable_params_ = rig_->ReadableParams ();
|
|
writable_params_ = rig_->WriteableParams ();
|
|
CAT_INFO (QString {"rig type change to: %1 readable params = 0x%2 writable params = 0x%3 for rig %4"}
|
|
.arg (rig_->RigType ())
|
|
.arg (readable_params_, 8, 16, QChar ('0'))
|
|
.arg (writable_params_, 8, 16, QChar ('0'))
|
|
.arg (rig_number));
|
|
}
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_status_change (int rig_number)
|
|
{
|
|
CAT_TRACE (QString {"status change for rig %1"}.arg (rig_number));
|
|
if (rig_number_ == rig_number)
|
|
{
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
auto const& status = rig_->StatusStr ();
|
|
CAT_TRACE ("OmniRig status change: new status = " << status);
|
|
if (OmniRig::ST_ONLINE != rig_->Status ())
|
|
{
|
|
if (!offline_timer_->isActive ())
|
|
{
|
|
// Omni-Rig is prone to reporting the rig offline and
|
|
// then recovering autonomously, so we will give it a
|
|
// few seconds to make its mind up
|
|
offline_timer_->start (10000);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
offline_timer_->stop (); // good to go again
|
|
Q_EMIT notified ();
|
|
}
|
|
// else
|
|
// {
|
|
// update_rx_frequency (rig_->GetRxFrequency ());
|
|
// update_complete ();
|
|
// CAT_TRACE ("frequency:" << state ().frequency ());
|
|
// }
|
|
}
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_params_change (int rig_number, int params)
|
|
{
|
|
CAT_TRACE (QString {"params change: params=0x%1 for rig %2"}
|
|
.arg (params, 8, 16, QChar ('0'))
|
|
.arg (rig_number)
|
|
<< "state before:" << state ());
|
|
if (rig_number_ == rig_number)
|
|
{
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
// starting_ = false;
|
|
TransceiverState old_state {state ()};
|
|
auto need_frequency = false;
|
|
|
|
if (params & OmniRig::PM_VFOAA)
|
|
{
|
|
CAT_TRACE ("VFOAA");
|
|
update_split (false);
|
|
reversed_ = false;
|
|
update_rx_frequency (rig_->FreqA ());
|
|
update_other_frequency (rig_->FreqB ());
|
|
}
|
|
if (params & OmniRig::PM_VFOAB)
|
|
{
|
|
CAT_TRACE ("VFOAB");
|
|
update_split (true);
|
|
reversed_ = false;
|
|
update_rx_frequency (rig_->FreqA ());
|
|
update_other_frequency (rig_->FreqB ());
|
|
}
|
|
if (params & OmniRig::PM_VFOBA)
|
|
{
|
|
CAT_TRACE ("VFOBA");
|
|
update_split (true);
|
|
reversed_ = true;
|
|
update_other_frequency (rig_->FreqA ());
|
|
update_rx_frequency (rig_->FreqB ());
|
|
}
|
|
if (params & OmniRig::PM_VFOBB)
|
|
{
|
|
CAT_TRACE ("VFOBB");
|
|
update_split (false);
|
|
reversed_ = true;
|
|
update_other_frequency (rig_->FreqA ());
|
|
update_rx_frequency (rig_->FreqB ());
|
|
}
|
|
if (params & OmniRig::PM_VFOA)
|
|
{
|
|
CAT_TRACE ("VFOA");
|
|
reversed_ = false;
|
|
need_frequency = true;
|
|
}
|
|
if (params & OmniRig::PM_VFOB)
|
|
{
|
|
CAT_TRACE ("VFOB");
|
|
reversed_ = true;
|
|
need_frequency = true;
|
|
}
|
|
|
|
if (params & OmniRig::PM_FREQ)
|
|
{
|
|
CAT_TRACE ("FREQ");
|
|
need_frequency = true;
|
|
}
|
|
if (params & OmniRig::PM_FREQA)
|
|
{
|
|
auto f = rig_->FreqA ();
|
|
CAT_TRACE ("FREQA = " << f);
|
|
if (reversed_)
|
|
{
|
|
update_other_frequency (f);
|
|
}
|
|
else
|
|
{
|
|
update_rx_frequency (f);
|
|
}
|
|
}
|
|
if (params & OmniRig::PM_FREQB)
|
|
{
|
|
auto f = rig_->FreqB ();
|
|
CAT_TRACE ("FREQB = " << f);
|
|
if (reversed_)
|
|
{
|
|
update_rx_frequency (f);
|
|
}
|
|
else
|
|
{
|
|
update_other_frequency (f);
|
|
}
|
|
}
|
|
if (need_frequency)
|
|
{
|
|
if (readable_params_ & OmniRig::PM_FREQA)
|
|
{
|
|
auto f = rig_->FreqA ();
|
|
if (f)
|
|
{
|
|
CAT_TRACE ("FREQA = " << f);
|
|
if (reversed_)
|
|
{
|
|
update_other_frequency (f);
|
|
}
|
|
else
|
|
{
|
|
update_rx_frequency (f);
|
|
}
|
|
}
|
|
}
|
|
if (readable_params_ & OmniRig::PM_FREQB)
|
|
{
|
|
auto f = rig_->FreqB ();
|
|
if (f)
|
|
{
|
|
CAT_TRACE ("FREQB = " << f);
|
|
if (reversed_)
|
|
{
|
|
update_rx_frequency (f);
|
|
}
|
|
else
|
|
{
|
|
update_other_frequency (f);
|
|
}
|
|
}
|
|
}
|
|
if (readable_params_ & OmniRig::PM_FREQ && !state ().ptt ())
|
|
{
|
|
auto f = rig_->Freq ();
|
|
if (f)
|
|
{
|
|
CAT_TRACE ("FREQ = " << f);
|
|
update_rx_frequency (f);
|
|
}
|
|
}
|
|
}
|
|
if (params & OmniRig::PM_PITCH)
|
|
{
|
|
CAT_TRACE ("PITCH");
|
|
}
|
|
if (params & OmniRig::PM_RITOFFSET)
|
|
{
|
|
CAT_TRACE ("RITOFFSET");
|
|
}
|
|
if (params & OmniRig::PM_RIT0)
|
|
{
|
|
CAT_TRACE ("RIT0");
|
|
}
|
|
if (params & OmniRig::PM_VFOEQUAL)
|
|
{
|
|
auto f = readable_params_ & OmniRig::PM_FREQA ? rig_->FreqA () : rig_->Freq ();
|
|
auto m = map_mode (rig_->Mode ());
|
|
CAT_TRACE (QString {"VFOEQUAL f=%1 m=%2"}.arg (f).arg (m));
|
|
update_rx_frequency (f);
|
|
update_other_frequency (f);
|
|
update_mode (m);
|
|
}
|
|
if (params & OmniRig::PM_VFOSWAP)
|
|
{
|
|
CAT_TRACE ("VFOSWAP");
|
|
auto f = state ().tx_frequency ();
|
|
update_other_frequency (state ().frequency ());
|
|
update_rx_frequency (f);
|
|
update_mode (map_mode (rig_->Mode ()));
|
|
}
|
|
if (params & OmniRig::PM_SPLITON)
|
|
{
|
|
CAT_TRACE ("SPLITON");
|
|
update_split (true);
|
|
}
|
|
if (params & OmniRig::PM_SPLITOFF)
|
|
{
|
|
CAT_TRACE ("SPLITOFF");
|
|
update_split (false);
|
|
}
|
|
if (params & OmniRig::PM_RITON)
|
|
{
|
|
CAT_TRACE ("RITON");
|
|
}
|
|
if (params & OmniRig::PM_RITOFF)
|
|
{
|
|
CAT_TRACE ("RITOFF");
|
|
}
|
|
if (params & OmniRig::PM_XITON)
|
|
{
|
|
CAT_TRACE ("XITON");
|
|
}
|
|
if (params & OmniRig::PM_XITOFF)
|
|
{
|
|
CAT_TRACE ("XITOFF");
|
|
}
|
|
if (params & OmniRig::PM_RX)
|
|
{
|
|
CAT_TRACE ("RX");
|
|
update_PTT (false);
|
|
}
|
|
if (params & OmniRig::PM_TX)
|
|
{
|
|
CAT_TRACE ("TX");
|
|
update_PTT ();
|
|
}
|
|
if (params & OmniRig::PM_CW_U)
|
|
{
|
|
CAT_TRACE ("CW-R");
|
|
update_mode (CW_R);
|
|
}
|
|
if (params & OmniRig::PM_CW_L)
|
|
{
|
|
CAT_TRACE ("CW");
|
|
update_mode (CW);
|
|
}
|
|
if (params & OmniRig::PM_SSB_U)
|
|
{
|
|
CAT_TRACE ("USB");
|
|
update_mode (USB);
|
|
}
|
|
if (params & OmniRig::PM_SSB_L)
|
|
{
|
|
CAT_TRACE ("LSB");
|
|
update_mode (LSB);
|
|
}
|
|
if (params & OmniRig::PM_DIG_U)
|
|
{
|
|
CAT_TRACE ("DATA-U");
|
|
update_mode (DIG_U);
|
|
}
|
|
if (params & OmniRig::PM_DIG_L)
|
|
{
|
|
CAT_TRACE ("DATA-L");
|
|
update_mode (DIG_L);
|
|
}
|
|
if (params & OmniRig::PM_AM)
|
|
{
|
|
CAT_TRACE ("AM");
|
|
update_mode (AM);
|
|
}
|
|
if (params & OmniRig::PM_FM)
|
|
{
|
|
CAT_TRACE ("FM");
|
|
update_mode (FM);
|
|
}
|
|
|
|
if (old_state != state () || send_update_signal_)
|
|
{
|
|
update_complete ();
|
|
send_update_signal_ = false;
|
|
}
|
|
CAT_TRACE ("OmniRig params change: state after:" << state ());
|
|
}
|
|
Q_EMIT notified ();
|
|
}
|
|
|
|
void OmniRigTransceiver::handle_custom_reply (int rig_number, QVariant const& command, QVariant const& reply)
|
|
{
|
|
(void)command;
|
|
(void)reply;
|
|
|
|
if (rig_number_ == rig_number)
|
|
{
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
CAT_TRACE ("custom command" << command.toString ()
|
|
<< "with reply" << reply.toString ()
|
|
<< QString ("for rig %1").arg (rig_number));
|
|
CAT_TRACE ("rig number:" << rig_number_ << ':' << state ());
|
|
}
|
|
}
|
|
|
|
void OmniRigTransceiver::do_ptt (bool on)
|
|
{
|
|
CAT_TRACE (on << state ());
|
|
if (use_for_ptt_ && TransceiverFactory::PTT_method_CAT == ptt_type_)
|
|
{
|
|
CAT_TRACE ("set PTT");
|
|
if (rig_ && !rig_->isNull ())
|
|
{
|
|
rig_->SetTx (on ? OmniRig::PM_TX : OmniRig::PM_RX);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (port_ && !port_->isNull ())
|
|
{
|
|
if (TransceiverFactory::PTT_method_RTS == ptt_type_)
|
|
{
|
|
CAT_TRACE ("set RTS");
|
|
port_->SetRts (on);
|
|
}
|
|
else // "DTR"
|
|
{
|
|
CAT_TRACE ("set DTR");
|
|
port_->SetDtr (on);
|
|
}
|
|
}
|
|
else if (wrapped_)
|
|
{
|
|
CAT_TRACE ("set PTT using basic transceiver");
|
|
TransceiverState new_state {wrapped_->state ()};
|
|
new_state.ptt (on);
|
|
wrapped_->set (new_state, 0);
|
|
}
|
|
}
|
|
update_PTT (on);
|
|
}
|
|
|
|
void OmniRigTransceiver::do_frequency (Frequency f, MODE m, bool /*no_ignore*/)
|
|
{
|
|
CAT_TRACE (f << state ());
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
if (UNK != m)
|
|
{
|
|
do_mode (m);
|
|
}
|
|
if (OmniRig::PM_FREQ & writable_params_)
|
|
{
|
|
rig_->SetFreq (f);
|
|
update_rx_frequency (f);
|
|
}
|
|
else if (reversed_ && (OmniRig::PM_FREQB & writable_params_))
|
|
{
|
|
rig_->SetFreqB (f);
|
|
update_rx_frequency (f);
|
|
}
|
|
else if (!reversed_ && (OmniRig::PM_FREQA & writable_params_))
|
|
{
|
|
rig_->SetFreqA (f);
|
|
update_rx_frequency (f);
|
|
}
|
|
else
|
|
{
|
|
throw_qstring (tr ("OmniRig: don't know how to set rig frequency"));
|
|
}
|
|
}
|
|
|
|
void OmniRigTransceiver::do_tx_frequency (Frequency tx, MODE m, bool /*no_ignore*/)
|
|
{
|
|
CAT_TRACE (tx << state ());
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
bool split {tx != 0};
|
|
if (split)
|
|
{
|
|
if (UNK != m)
|
|
{
|
|
do_mode (m);
|
|
if (OmniRig::PM_UNKNOWN == rig_->Vfo ())
|
|
{
|
|
if (writable_params_ & OmniRig::PM_VFOEQUAL)
|
|
{
|
|
// nothing to do here because OmniRig will use VFO
|
|
// equalize to set the mode of the Tx VFO for us
|
|
}
|
|
else if ((writable_params_ & (OmniRig::PM_VFOA | OmniRig::PM_VFOB))
|
|
== (OmniRig::PM_VFOA | OmniRig::PM_VFOB))
|
|
{
|
|
rig_->SetVfo (OmniRig::PM_VFOB);
|
|
do_mode (m);
|
|
rig_->SetVfo (OmniRig::PM_VFOA);
|
|
}
|
|
else if (writable_params_ & OmniRig::PM_VFOSWAP)
|
|
{
|
|
rig_->SetVfo (OmniRig::PM_VFOSWAP);
|
|
do_mode (m);
|
|
rig_->SetVfo (OmniRig::PM_VFOSWAP);
|
|
}
|
|
}
|
|
}
|
|
CAT_TRACE ("set SPLIT mode on");
|
|
rig_->SetSplitMode (state ().frequency (), tx);
|
|
update_other_frequency (tx);
|
|
update_split (true);
|
|
}
|
|
else
|
|
{
|
|
CAT_TRACE ("set SPLIT mode off");
|
|
rig_->SetSimplexMode (state ().frequency ());
|
|
update_split (false);
|
|
}
|
|
bool notify {false};
|
|
if (readable_params_ & OmniRig::PM_FREQ || !(readable_params_ & (OmniRig::PM_FREQA | OmniRig::PM_FREQB)))
|
|
{
|
|
update_other_frequency (tx); // async updates won't return this
|
|
// so just store it and hope
|
|
// operator doesn't change the
|
|
// "back" VFO on rig
|
|
notify = true;
|
|
}
|
|
if (!((OmniRig::PM_VFOAB | OmniRig::PM_VFOBA | OmniRig::PM_SPLITON) & readable_params_))
|
|
{
|
|
CAT_TRACE ("setting SPLIT manually");
|
|
update_split (split); // we can't read it so just set and
|
|
// hope op doesn't change it
|
|
notify = true;
|
|
}
|
|
if (notify)
|
|
{
|
|
update_complete ();
|
|
}
|
|
}
|
|
|
|
void OmniRigTransceiver::do_mode (MODE mode)
|
|
{
|
|
CAT_TRACE (mode << state ());
|
|
if (!rig_ || rig_->isNull ()) return;
|
|
// TODO: G4WJS OmniRig doesn't seem to have any capability of tracking/setting VFO B mode
|
|
auto mapped = map_mode (mode);
|
|
if (mapped & writable_params_)
|
|
{
|
|
rig_->SetMode (mapped);
|
|
update_mode (mode);
|
|
}
|
|
else
|
|
{
|
|
offline ("OmniRig invalid mode");
|
|
}
|
|
}
|