2015-09-09 03:13:24 -04:00
///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2015 Edouard Griffiths, F4EXB //
// //
// This program is free software; you can redistribute it and/or modify //
// it under the terms of the GNU General Public License as published by //
// the Free Software Foundation as version 3 of the License, or //
// //
// This program is distributed in the hope that it will be useful, //
// but WITHOUT ANY WARRANTY; without even the implied warranty of //
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
// GNU General Public License V3 for more details. //
// //
// You should have received a copy of the GNU General Public License //
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
///////////////////////////////////////////////////////////////////////////////////
# include <string.h>
# include <errno.h>
# include <QDebug>
# include "airspygui.h"
# include "airspyinput.h"
# include "util/simpleserializer.h"
# include "dsp/dspcommands.h"
# include "airspyserializer.h"
# include "airspythread.h"
MESSAGE_CLASS_DEFINITION ( AirspyInput : : MsgConfigureAirspy , Message )
MESSAGE_CLASS_DEFINITION ( AirspyInput : : MsgReportAirspy , Message )
AirspyInput : : Settings : : Settings ( ) :
m_centerFrequency ( 435000 * 1000 ) ,
2015-09-09 23:53:37 -04:00
m_devSampleRateIndex ( 0 ) ,
2015-09-09 03:13:24 -04:00
m_LOppmTenths ( 0 ) ,
2015-09-12 03:57:10 -04:00
m_lnaGain ( 14 ) ,
m_mixerGain ( 15 ) ,
m_vgaGain ( 4 ) ,
2015-09-09 03:13:24 -04:00
m_log2Decim ( 0 ) ,
2015-09-12 03:57:10 -04:00
m_fcPos ( FC_POS_CENTER ) ,
2015-09-09 03:13:24 -04:00
m_biasT ( false )
{
}
void AirspyInput : : Settings : : resetToDefaults ( )
{
m_centerFrequency = 435000 * 1000 ;
2015-09-09 23:53:37 -04:00
m_devSampleRateIndex = 0 ;
2015-09-09 03:13:24 -04:00
m_LOppmTenths = 0 ;
2015-09-12 03:57:10 -04:00
m_lnaGain = 14 ;
m_mixerGain = 15 ;
m_vgaGain = 4 ;
2015-09-09 03:13:24 -04:00
m_log2Decim = 0 ;
2015-09-12 03:57:10 -04:00
m_fcPos = FC_POS_CENTER ;
2015-09-09 03:13:24 -04:00
m_biasT = false ;
}
QByteArray AirspyInput : : Settings : : serialize ( ) const
{
AirspySerializer : : AirspyData data ;
data . m_data . m_frequency = m_centerFrequency ;
data . m_LOppmTenths = m_LOppmTenths ;
2015-09-09 23:53:37 -04:00
data . m_sampleRateIndex = m_devSampleRateIndex ;
2015-09-09 03:13:24 -04:00
data . m_log2Decim = m_log2Decim ;
2015-09-09 23:53:37 -04:00
data . m_fcPos = ( qint32 ) m_fcPos ;
2015-09-09 03:13:24 -04:00
data . m_lnaGain = m_lnaGain ;
data . m_mixerGain = m_mixerGain ;
data . m_vgaGain = m_vgaGain ;
data . m_biasT = m_biasT ;
QByteArray byteArray ;
AirspySerializer : : writeSerializedData ( data , byteArray ) ;
return byteArray ;
}
bool AirspyInput : : Settings : : deserialize ( const QByteArray & serializedData )
{
AirspySerializer : : AirspyData data ;
bool valid = AirspySerializer : : readSerializedData ( serializedData , data ) ;
m_centerFrequency = data . m_data . m_frequency ;
m_LOppmTenths = data . m_LOppmTenths ;
2015-09-09 23:53:37 -04:00
m_devSampleRateIndex = data . m_sampleRateIndex ;
2015-09-09 03:13:24 -04:00
m_log2Decim = data . m_log2Decim ;
2015-09-09 23:53:37 -04:00
m_fcPos = ( fcPos_t ) data . m_fcPos ;
2015-09-09 03:13:24 -04:00
m_lnaGain = data . m_lnaGain ;
m_mixerGain = data . m_mixerGain ;
m_vgaGain = data . m_vgaGain ;
m_biasT = data . m_biasT ;
return valid ;
}
AirspyInput : : AirspyInput ( ) :
m_settings ( ) ,
m_dev ( 0 ) ,
m_airspyThread ( 0 ) ,
m_deviceDescription ( " Airspy " )
{
2015-09-10 21:02:02 -04:00
m_sampleRates . push_back ( 10000000 ) ;
m_sampleRates . push_back ( 2500000 ) ;
2015-09-09 03:13:24 -04:00
}
AirspyInput : : ~ AirspyInput ( )
{
stop ( ) ;
}
bool AirspyInput : : init ( const Message & cmd )
{
return false ;
}
bool AirspyInput : : start ( int device )
{
QMutexLocker mutexLocker ( & m_mutex ) ;
airspy_error rc ;
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_init ( ) ;
2015-09-09 03:13:24 -04:00
if ( rc ! = AIRSPY_SUCCESS )
{
qCritical ( " AirspyInput::start: failed to initiate Airspy library %s " , airspy_error_name ( rc ) ) ;
}
if ( m_dev ! = 0 )
{
stop ( ) ;
}
2015-09-13 05:56:14 -04:00
if ( ! m_sampleFifo . setSize ( 1 < < 19 ) )
2015-09-09 03:13:24 -04:00
{
qCritical ( " AirspyInput::start: could not allocate SampleFifo " ) ;
return false ;
}
2015-09-10 21:02:02 -04:00
if ( ( m_dev = open_airspy_from_sequence ( device ) ) = = 0 )
2015-09-09 03:13:24 -04:00
{
2015-09-10 21:02:02 -04:00
qCritical ( " AirspyInput::start: could not open Airspy #%d " , device ) ;
2015-09-09 03:13:24 -04:00
return false ;
}
2015-09-10 21:02:02 -04:00
# ifdef LIBAIRSPY_DYN_RATES
2015-09-09 03:13:24 -04:00
uint32_t nbSampleRates ;
2015-09-09 23:53:37 -04:00
uint32_t * sampleRates ;
2015-09-09 03:13:24 -04:00
airspy_get_samplerates ( m_dev , & nbSampleRates , 0 ) ;
sampleRates = new uint32_t [ nbSampleRates ] ;
airspy_get_samplerates ( m_dev , sampleRates , nbSampleRates ) ;
2015-09-09 23:53:37 -04:00
if ( nbSampleRates = = 0 )
{
qCritical ( " AirspyInput::start: could not obtain Airspy sample rates " ) ;
return false ;
}
2015-09-10 21:02:02 -04:00
m_sampleRates . clear ( ) ;
2015-09-09 03:13:24 -04:00
for ( int i = 0 ; i < nbSampleRates ; i + + )
{
m_sampleRates . push_back ( sampleRates [ i ] ) ;
2015-09-10 21:02:02 -04:00
qDebug ( " AirspyInput::start: sampleRates[%d] = %u Hz " , i , sampleRates [ i ] ) ;
2015-09-09 03:13:24 -04:00
}
delete [ ] sampleRates ;
2015-09-23 20:50:24 -04:00
# else
m_sampleRates . clear ( ) ;
m_sampleRates . push_back ( 10000000 ) ;
m_sampleRates . push_back ( 2500000 ) ;
2015-09-09 03:13:24 -04:00
# endif
2015-09-10 21:02:02 -04:00
MsgReportAirspy * message = MsgReportAirspy : : create ( m_sampleRates ) ;
getOutputMessageQueueToGUI ( ) - > push ( message ) ;
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_set_sample_type ( m_dev , AIRSPY_SAMPLE_INT16_IQ ) ;
if ( rc ! = AIRSPY_SUCCESS )
{
qCritical ( " AirspyInput::start: could not set sample type to INT16_IQ " ) ;
return false ;
}
2015-09-10 21:02:02 -04:00
if ( ( m_airspyThread = new AirspyThread ( m_dev , & m_sampleFifo ) ) = = 0 )
{
qFatal ( " AirspyInput::start: out of memory " ) ;
stop ( ) ;
return false ;
2015-09-09 03:13:24 -04:00
}
m_airspyThread - > startWork ( ) ;
mutexLocker . unlock ( ) ;
2015-09-10 21:02:02 -04:00
applySettings ( m_settings , true ) ;
2015-09-09 03:13:24 -04:00
qDebug ( " AirspyInput::startInput: started " ) ;
return true ;
}
void AirspyInput : : stop ( )
{
2015-09-10 21:02:02 -04:00
qDebug ( " AirspyInput::stop " ) ;
2015-09-09 03:13:24 -04:00
QMutexLocker mutexLocker ( & m_mutex ) ;
if ( m_airspyThread ! = 0 )
{
m_airspyThread - > stopWork ( ) ;
delete m_airspyThread ;
m_airspyThread = 0 ;
}
if ( m_dev ! = 0 )
{
airspy_stop_rx ( m_dev ) ;
airspy_close ( m_dev ) ;
m_dev = 0 ;
}
airspy_exit ( ) ;
}
const QString & AirspyInput : : getDeviceDescription ( ) const
{
return m_deviceDescription ;
}
int AirspyInput : : getSampleRate ( ) const
{
2015-09-09 23:53:37 -04:00
int rate = m_sampleRates [ m_settings . m_devSampleRateIndex ] ;
2015-09-09 03:13:24 -04:00
return ( rate / ( 1 < < m_settings . m_log2Decim ) ) ;
}
quint64 AirspyInput : : getCenterFrequency ( ) const
{
return m_settings . m_centerFrequency ;
}
bool AirspyInput : : handleMessage ( const Message & message )
{
if ( MsgConfigureAirspy : : match ( message ) )
{
MsgConfigureAirspy & conf = ( MsgConfigureAirspy & ) message ;
qDebug ( ) < < " AirspyInput::handleMessage: MsgConfigureAirspy " ;
2015-09-10 21:02:02 -04:00
bool success = applySettings ( conf . getSettings ( ) , false ) ;
if ( ! success )
2015-09-09 03:13:24 -04:00
{
qDebug ( " Airspy config error " ) ;
}
return true ;
}
else
{
return false ;
}
}
2015-09-09 23:53:37 -04:00
void AirspyInput : : setCenterFrequency ( quint64 freq_hz )
{
freq_hz + = ( freq_hz * m_settings . m_LOppmTenths ) / 10000000ULL ;
airspy_error rc = ( airspy_error ) airspy_set_freq ( m_dev , static_cast < uint32_t > ( freq_hz ) ) ;
if ( rc ! = AIRSPY_SUCCESS )
{
qWarning ( " AirspyInput::setCenterFrequency: could not frequency to %llu Hz " , freq_hz ) ;
}
else
{
qWarning ( " AirspyInput::setCenterFrequency: frequency set to %llu Hz " , freq_hz ) ;
}
}
2015-09-09 03:13:24 -04:00
bool AirspyInput : : applySettings ( const Settings & settings , bool force )
{
2015-09-10 21:02:02 -04:00
QMutexLocker mutexLocker ( & m_mutex ) ;
2015-09-09 03:13:24 -04:00
bool forwardChange = false ;
airspy_error rc ;
2015-09-10 21:02:02 -04:00
qDebug ( ) < < " AirspyInput::applySettings " ;
2015-09-09 03:13:24 -04:00
2015-09-09 23:53:37 -04:00
if ( ( m_settings . m_devSampleRateIndex ! = settings . m_devSampleRateIndex ) | | force )
{
2015-09-10 21:02:02 -04:00
forwardChange = true ;
2015-09-09 23:53:37 -04:00
if ( settings . m_devSampleRateIndex < m_sampleRates . size ( ) )
{
m_settings . m_devSampleRateIndex = settings . m_devSampleRateIndex ;
}
else
{
2015-09-10 21:02:02 -04:00
m_settings . m_devSampleRateIndex = m_sampleRates . size ( ) - 1 ;
2015-09-09 23:53:37 -04:00
}
if ( m_dev ! = 0 )
{
2015-09-10 21:02:02 -04:00
rc = ( airspy_error ) airspy_set_samplerate ( m_dev , static_cast < airspy_samplerate_t > ( m_settings . m_devSampleRateIndex ) ) ;
2015-09-09 23:53:37 -04:00
if ( rc ! = AIRSPY_SUCCESS )
{
qCritical ( " AirspyInput::applySettings: could not set sample rate index %u (%d S/s): %s " , m_settings . m_devSampleRateIndex , m_sampleRates [ m_settings . m_devSampleRateIndex ] , airspy_error_name ( rc ) ) ;
}
else
{
qDebug ( " AirspyInput::applySettings: sample rate set to index: %u (%d S/s) " , m_settings . m_devSampleRateIndex , m_sampleRates [ m_settings . m_devSampleRateIndex ] ) ;
m_airspyThread - > setSamplerate ( m_sampleRates [ m_settings . m_devSampleRateIndex ] ) ;
}
}
}
if ( ( m_settings . m_log2Decim ! = settings . m_log2Decim ) | | force )
{
m_settings . m_log2Decim = settings . m_log2Decim ;
forwardChange = true ;
if ( m_dev ! = 0 )
{
m_airspyThread - > setLog2Decimation ( m_settings . m_log2Decim ) ;
qDebug ( ) < < " AirspyInput: set decimation to " < < ( 1 < < m_settings . m_log2Decim ) ;
}
}
if ( ( m_settings . m_fcPos ! = settings . m_fcPos ) | | force )
{
m_settings . m_fcPos = settings . m_fcPos ;
if ( m_dev ! = 0 )
{
m_airspyThread - > setFcPos ( ( int ) m_settings . m_fcPos ) ;
qDebug ( ) < < " AirspyInput: set fc pos (enum) to " < < ( int ) m_settings . m_fcPos ;
}
}
2015-09-10 21:02:02 -04:00
qint64 deviceCenterFrequency = m_settings . m_centerFrequency ;
qint64 f_img = deviceCenterFrequency ;
quint32 devSampleRate = m_sampleRates [ m_settings . m_devSampleRateIndex ] ;
2015-09-11 02:23:04 -04:00
if ( force | | ( m_settings . m_centerFrequency ! = settings . m_centerFrequency ) | |
( m_settings . m_LOppmTenths ! = settings . m_LOppmTenths ) )
2015-09-09 23:53:37 -04:00
{
m_settings . m_centerFrequency = settings . m_centerFrequency ;
2015-09-11 02:23:04 -04:00
m_settings . m_LOppmTenths = settings . m_LOppmTenths ;
2015-09-09 23:53:37 -04:00
if ( ( m_settings . m_log2Decim = = 0 ) | | ( m_settings . m_fcPos = = FC_POS_CENTER ) )
{
deviceCenterFrequency = m_settings . m_centerFrequency ;
f_img = deviceCenterFrequency ;
}
else
{
if ( m_settings . m_fcPos = = FC_POS_INFRA )
{
deviceCenterFrequency = m_settings . m_centerFrequency + ( devSampleRate / 4 ) ;
f_img = deviceCenterFrequency + devSampleRate / 2 ;
}
else if ( m_settings . m_fcPos = = FC_POS_SUPRA )
{
deviceCenterFrequency = m_settings . m_centerFrequency - ( devSampleRate / 4 ) ;
f_img = deviceCenterFrequency - devSampleRate / 2 ;
}
}
if ( m_dev ! = 0 )
{
setCenterFrequency ( deviceCenterFrequency ) ;
qDebug ( ) < < " AirspyInput::applySettings: center freq: " < < m_settings . m_centerFrequency < < " Hz "
< < " device center freq: " < < deviceCenterFrequency < < " Hz "
< < " device sample rate: " < < devSampleRate < < " Hz "
< < " Actual sample rate: " < < devSampleRate / ( 1 < < m_settings . m_log2Decim ) < < " Hz "
< < " img: " < < f_img < < " Hz " ;
}
forwardChange = true ;
}
2015-09-09 03:13:24 -04:00
if ( ( m_settings . m_lnaGain ! = settings . m_lnaGain ) | | force )
{
m_settings . m_lnaGain = settings . m_lnaGain ;
if ( m_dev ! = 0 )
{
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_set_lna_gain ( m_dev , m_settings . m_lnaGain ) ;
2015-09-09 03:13:24 -04:00
if ( rc ! = AIRSPY_SUCCESS )
{
qDebug ( " AirspyInput::applySettings: airspy_set_lna_gain failed: %s " , airspy_error_name ( rc ) ) ;
}
else
{
qDebug ( ) < < " AirspyInput:applySettings: LNA gain set to " < < m_settings . m_lnaGain ;
}
}
}
if ( ( m_settings . m_mixerGain ! = settings . m_mixerGain ) | | force )
{
m_settings . m_mixerGain = settings . m_mixerGain ;
if ( m_dev ! = 0 )
{
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_set_mixer_gain ( m_dev , m_settings . m_mixerGain ) ;
2015-09-09 03:13:24 -04:00
if ( rc ! = AIRSPY_SUCCESS )
{
qDebug ( " AirspyInput::applySettings: airspy_set_mixer_gain failed: %s " , airspy_error_name ( rc ) ) ;
}
else
{
qDebug ( ) < < " AirspyInput:applySettings: mixer gain set to " < < m_settings . m_mixerGain ;
}
}
}
if ( ( m_settings . m_vgaGain ! = settings . m_vgaGain ) | | force )
{
m_settings . m_vgaGain = settings . m_vgaGain ;
if ( m_dev ! = 0 )
{
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_set_vga_gain ( m_dev , m_settings . m_vgaGain ) ;
2015-09-09 03:13:24 -04:00
if ( rc ! = AIRSPY_SUCCESS )
{
qDebug ( " AirspyInput::applySettings: airspy_set_vga_gain failed: %s " , airspy_error_name ( rc ) ) ;
}
else
{
qDebug ( ) < < " AirspyInput:applySettings: VGA gain set to " < < m_settings . m_vgaGain ;
}
}
}
2015-09-09 23:53:37 -04:00
if ( ( m_settings . m_biasT ! = settings . m_biasT ) | | force )
2015-09-09 03:13:24 -04:00
{
2015-09-09 23:53:37 -04:00
m_settings . m_biasT = settings . m_biasT ;
2015-09-09 03:13:24 -04:00
if ( m_dev ! = 0 )
{
2015-09-09 23:53:37 -04:00
rc = ( airspy_error ) airspy_set_rf_bias ( m_dev , ( m_settings . m_biasT ? 1 : 0 ) ) ;
2015-09-09 03:13:24 -04:00
2015-09-09 23:53:37 -04:00
if ( rc ! = AIRSPY_SUCCESS )
2015-09-09 03:13:24 -04:00
{
2015-09-09 23:53:37 -04:00
qDebug ( " AirspyInput::applySettings: airspy_set_rf_bias failed: %s " , airspy_error_name ( rc ) ) ;
2015-09-09 03:13:24 -04:00
}
else
{
2015-09-09 23:53:37 -04:00
qDebug ( ) < < " AirspyInput:applySettings: bias tee set to " < < m_settings . m_biasT ;
2015-09-09 03:13:24 -04:00
}
}
}
if ( forwardChange )
{
2015-09-09 23:53:37 -04:00
int sampleRate = devSampleRate / ( 1 < < m_settings . m_log2Decim ) ;
2015-09-09 03:13:24 -04:00
DSPSignalNotification * notif = new DSPSignalNotification ( sampleRate , m_settings . m_centerFrequency ) ;
getOutputMessageQueue ( ) - > push ( notif ) ;
}
return true ;
}
struct airspy_device * AirspyInput : : open_airspy_from_sequence ( int sequence )
{
2015-09-10 21:02:02 -04:00
airspy_read_partid_serialno_t read_partid_serialno ;
struct airspy_device * devinfo , * retdev = 0 ;
uint32_t serial_msb = 0 ;
uint32_t serial_lsb = 0 ;
airspy_error rc ;
int i ;
2015-09-09 03:13:24 -04:00
2015-09-10 21:02:02 -04:00
for ( int i = 0 ; i < AIRSPY_MAX_DEVICE ; i + + )
2015-09-09 03:13:24 -04:00
{
2015-09-10 21:02:02 -04:00
rc = ( airspy_error ) airspy_open ( & devinfo ) ;
2015-09-09 03:13:24 -04:00
2015-09-09 23:53:37 -04:00
if ( rc = = AIRSPY_SUCCESS )
2015-09-09 03:13:24 -04:00
{
2015-09-09 23:53:37 -04:00
if ( i = = sequence )
{
return devinfo ;
}
2015-09-09 03:13:24 -04:00
}
else
{
2015-09-10 21:02:02 -04:00
break ;
2015-09-09 03:13:24 -04:00
}
}
return 0 ;
}