/*
Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010 Her Majesty the
Queen in Right of Canada (Communications Research Center Canada)
Copyright (C) 2014, 2015
Matthias P. Braendli, matthias.braendli@mpb.li
http://opendigitalradio.org
DESCRIPTION:
It is an output driver for the USRP family of devices, and uses the UHD
library. This version is multi-threaded. A separate thread sends the data to
the device.
Data between the modulator and the UHD thread is exchanged by swapping
buffers at a synchronisation barrier.
*/
/*
This file is part of ODR-DabMod.
ODR-DabMod 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, either version 3 of the
License, or (at your option) any later version.
ODR-DabMod 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 for more details.
You should have received a copy of the GNU General Public License
along with ODR-DabMod. If not, see .
*/
#ifndef OUTPUT_UHD_H
#define OUTPUT_UHD_H
#define FAKE_UHD 0
#ifdef HAVE_CONFIG_H
# include
#endif
#ifdef HAVE_OUTPUT_UHD
#include
#include
#include
#include
#include
#include
#include
#include
#include "Log.h"
#include "ModOutput.h"
#include "EtiReader.h"
#include "TimestampDecoder.h"
#include "RemoteControl.h"
#include
#include
#define MDEBUG(fmt, args...) fprintf (LOG, fmt , ## args)
//#define MDEBUG(fmt, args...)
// If the timestamp is further in the future than
// 100 seconds, abort
#define TIMESTAMP_ABORT_FUTURE 100
// Add a delay to increase buffers when
// frames are too far in the future
#define TIMESTAMP_MARGIN_FUTURE 0.5
typedef std::complex complexf;
struct UHDWorkerFrameData {
// Buffer holding frame data
void* buf;
// Full timestamp
struct frame_timestamp ts;
};
struct fct_discontinuity_error : public std::exception
{
const char* what () const throw ()
{
return "FCT discontinuity detected";
}
};
enum refclk_lock_loss_behaviour_t { CRASH, IGNORE };
struct UHDWorkerData {
bool running;
bool failed_due_to_fct;
#if FAKE_UHD == 0
uhd::usrp::multi_usrp::sptr myUsrp;
#endif
unsigned sampleRate;
// Double buffering between the two threads
// Each buffer contains one OFDM frame, and it's
// associated timestamp
// A full timestamp contains a TIST according to standard
// and time information within MNSC with tx_second.
bool sourceContainsTimestamp;
// When working with timestamps, mute the frames that
// do not have a timestamp
bool muteNoTimestamps;
struct UHDWorkerFrameData frame0;
struct UHDWorkerFrameData frame1;
size_t bufsize; // in bytes
// If we want to verify loss of refclk
bool check_refclk_loss;
// If we want to check for the gps_timelock sensor
bool check_gpsfix;
bool gpsdo_is_ettus; // Set to false in case the ODR LEA-M8F board is used
// muting set by remote control
bool muting;
// A barrier to synchronise the two threads
std::shared_ptr sync_barrier;
// What to do when the reference clock PLL loses lock
refclk_lock_loss_behaviour_t refclk_lock_loss_behaviour;
// What transmission mode we're using defines by how
// much the FCT should increment for each
// transmission frame.
int fct_increment;
};
class UHDWorker {
public:
void start(struct UHDWorkerData *uhdworkerdata) {
uwd = uhdworkerdata;
uwd->running = true;
uwd->failed_due_to_fct = false;
uhd_thread = boost::thread(&UHDWorker::process_errhandler, this);
}
void stop() {
if (uwd) {
uwd->running = false;
}
uhd_thread.interrupt();
uhd_thread.join();
}
private:
// Asynchronous message statistics
int num_underflows;
int num_late_packets;
bool fct_discontinuity;
int expected_next_fct;
uhd::tx_metadata_t md;
time_t tx_second;
double pps_offset;
double last_pps;
void print_async_metadata(const struct UHDWorkerFrameData *frame);
void handle_frame(const struct UHDWorkerFrameData *frame);
void tx_frame(const struct UHDWorkerFrameData *frame);
struct UHDWorkerData *uwd;
boost::thread uhd_thread;
uhd::tx_streamer::sptr myTxStream;
void process();
void process_errhandler();
};
/* This structure is used as initial configuration for OutputUHD */
struct OutputUHDConfig {
std::string device;
std::string usrpType; // e.g. b100, b200, usrp2
// The USRP1 can accept two daughterboards
std::string subDevice; // e.g. A:0
long masterClockRate;
unsigned sampleRate;
double frequency;
double txgain;
bool enableSync;
bool muteNoTimestamps;
unsigned dabMode;
unsigned maxGPSHoldoverTime;
/* allowed values : auto, int, sma, mimo */
std::string refclk_src;
/* allowed values : int, sma, mimo */
std::string pps_src;
/* allowed values : pos, neg */
std::string pps_polarity;
/* What to do when the reference clock PLL loses lock */
refclk_lock_loss_behaviour_t refclk_lock_loss_behaviour;
};
class OutputUHD: public ModOutput, public RemoteControllable {
public:
OutputUHD(const OutputUHDConfig& config);
~OutputUHD();
int process(Buffer* dataIn, Buffer* dataOut);
const char* name() { return "OutputUHD"; }
void setETIReader(EtiReader *etiReader) {
myEtiReader = etiReader;
}
/*********** REMOTE CONTROL ***************/
/* virtual void enrol_at(BaseRemoteController& controller)
* is inherited
*/
/* Base function to set parameters. */
virtual void set_parameter(const std::string& parameter,
const std::string& value);
/* Getting a parameter always returns a string. */
virtual const std::string get_parameter(
const std::string& parameter) const;
protected:
OutputUHD(const OutputUHD& other);
OutputUHD& operator=(const OutputUHD& other);
EtiReader *myEtiReader;
OutputUHDConfig myConf;
uhd::usrp::multi_usrp::sptr myUsrp;
std::shared_ptr mySyncBarrier;
UHDWorker worker;
bool first_run;
bool gps_fix_verified;
struct UHDWorkerData uwd;
int activebuffer;
// muting can only be changed using the remote control
bool myMuting;
private:
// Resize the internal delay buffer according to the dabMode and
// the sample rate.
void SetDelayBuffer(unsigned int dabMode);
// data
int myStaticDelayUs; // static delay in microseconds
int myTFDurationMs; // TF duration in milliseconds
std::vector myDelayBuf;
size_t lastLen;
// GPS Fix check variables
int num_checks_without_gps_fix;
struct timespec first_gps_fix_check;
struct timespec last_gps_fix_check;
struct timespec time_last_frame;
boost::packaged_task gps_fix_pt;
boost::unique_future gps_fix_future;
boost::thread gps_fix_task;
// Wait time in seconds to get fix
static const int initial_gps_fix_wait = 180;
// Interval for checking the GPS at runtime
static constexpr double gps_fix_check_interval = 10.0; // seconds
void check_gps();
void set_usrp_time();
void initial_gps_check();
};
#endif // HAVE_OUTPUT_UHD
#endif // OUTPUT_UHD_H