/*
Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012
Her Majesty the Queen in Right of Canada (Communications Research
Center Canada)
Copyright (C) 2023
Matthias P. Braendli, matthias.braendli@mpb.li
http://opendigitalradio.org
*/
/*
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 .
*/
#include
#include
#include
#include "Utils.h"
#if defined(HAVE_PRCTL)
# include
#endif
#include
static void printHeader()
{
std::cerr << "ODR-DabMod version " <<
#if defined(GITVERSION)
GITVERSION
#else
VERSION
#endif
", compiled at " << __DATE__ << ", " << __TIME__ << std::endl;
std::cerr << "Compiled with features: " <<
#if defined(HAVE_ZEROMQ)
"zeromq " <<
#endif
#if defined(HAVE_OUTPUT_UHD)
"output_uhd " <<
#endif
#if defined(HAVE_SOAPYSDR)
"output_soapysdr " <<
#endif
#if defined(HAVE_LIMESDR)
"output_limesdr " <<
#endif
#if defined(__FAST_MATH__)
"fast-math " <<
#endif
#if defined(__SSE__)
"SSE " <<
#endif
#if defined(__ARM_NEON)
"NEON " <<
#endif
"\n";
}
void printUsage(const char* progName)
{
FILE* out = stderr;
fprintf(out, "Usage with configuration file:\n");
fprintf(out, "\t%s config_file.ini\n\n", progName);
fprintf(out, "Usage with command line options:\n");
fprintf(out, "\t%s"
" input"
" (-f filename -F format | -u uhddevice -F frequency)"
" [-o offset]"
"\n\t"
" [-G txgain]"
" [-T filter_taps_file]"
" [-a gain]"
" [-c clockrate]"
"\n\t"
" [-g gainMode]"
" [-m dabMode]"
" [-r samplingRate]"
" [-l]"
" [-h]"
"\n", progName);
fprintf(out, "Where:\n");
fprintf(out, "input: ETI input filename (default: stdin), or\n");
fprintf(out, " tcp://source:port for ETI-over-TCP input, or\n");
fprintf(out, " udp://:port for EDI input.\n");
fprintf(out, "-f name: Use file output with given filename. (use /dev/stdout for standard output)\n");
fprintf(out, "-F format: Set the output format (see doc/example.ini for formats) for the file output.\n");
fprintf(out, "-o: Set the timestamp offset added to the timestamp in the ETI. The offset is a double.\n");
fprintf(out, " Specifying this option has two implications: It enables synchronous transmission,\n"
" requiring an external REFCLK and PPS signal and frames that do not contain a valid timestamp\n"
" get muted.\n\n");
fprintf(out, "-u device: Use UHD output with given device string. (use "" for default device)\n");
fprintf(out, "-F frequency: Set the transmit frequency when using UHD output. (mandatory option when using UHD)\n");
fprintf(out, "-G txgain: Set the transmit gain for the UHD driver (default: 0)\n");
fprintf(out, "-T taps_file: Enable filtering before the output, using the specified file containing the filter taps.\n");
fprintf(out, " Use 'default' as taps_file to use the internal taps.\n");
fprintf(out, "-a gain: Apply digital amplitude gain.\n");
fprintf(out, "-c rate: Set the DAC clock rate and enable Cic Equalisation.\n");
fprintf(out, "-g gainmode: Set computation gain mode: fix, max or var\n");
fprintf(out, "-m mode: Set DAB mode: (0: auto, 1-4: force).\n");
fprintf(out, "-r rate: Set output sampling rate (default: 2048000).\n\n");
fprintf(out, "-l: Loop file when reach end of file.\n");
fprintf(out, "-h: Print this help.\n");
}
void printVersion(void)
{
FILE *out = stderr;
fprintf(out,
" ODR-DabMod is copyright (C) Her Majesty the Queen in Right of Canada,\n"
" 2005 -- 2012 Communications Research Centre (CRC),\n"
" and\n"
" Copyright (C) 2023 Matthias P. Braendli, matthias.braendli@mpb.li\n"
"\n"
" http://opendigitalradio.org\n"
"\n"
" ODR-DabMod is free software: you can redistribute it and/or modify it\n"
" under the terms of the GNU General Public License as published by the\n"
" Free Software Foundation, either version 3 of the License, or (at your\n"
" option) any later version.\n"
"\n"
" ODR-DabMod is distributed in the hope that it will be useful, but\n"
" WITHOUT ANY WARRANTY; without even the implied warranty of\n"
" MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU\n"
" General Public License for more details.\n"
"\n"
" You should have received a copy of the GNU General Public License along\n"
" with ODR-DabMod. If not, see .\n"
"\n"
);
}
void printStartupInfo()
{
printHeader();
}
void printModSettings(const mod_settings_t& mod_settings)
{
std::stringstream ss;
// Print settings
ss << "Input\n";
ss << " Type: " << mod_settings.inputTransport << "\n";
ss << " Source: " << mod_settings.inputName << "\n";
ss << "Output\n";
if (mod_settings.useFileOutput) {
ss << " Name: " << mod_settings.outputName << "\n";
}
#if defined(HAVE_OUTPUT_UHD)
else if (mod_settings.useUHDOutput) {
ss << " UHD\n" <<
" Device: " << mod_settings.sdr_device_config.device << "\n" <<
" Subdevice: " <<
mod_settings.sdr_device_config.subDevice << "\n" <<
" master_clock_rate: " <<
mod_settings.sdr_device_config.masterClockRate << "\n" <<
" refclk: " <<
mod_settings.sdr_device_config.refclk_src << "\n" <<
" pps source: " <<
mod_settings.sdr_device_config.pps_src << "\n";
}
#endif
#if defined(HAVE_SOAPYSDR)
else if (mod_settings.useSoapyOutput) {
ss << " SoapySDR\n"
" Device: " << mod_settings.sdr_device_config.device << "\n" <<
" master_clock_rate: " <<
mod_settings.sdr_device_config.masterClockRate << "\n";
}
#endif
#if defined(HAVE_DEXTER)
else if (mod_settings.useDexterOutput) {
ss << " PrecisionWave DEXTER\n";
}
#endif
#if defined(HAVE_LIMESDR)
else if (mod_settings.useLimeOutput) {
ss << " LimeSDR\n"
" Device: " << mod_settings.sdr_device_config.device << "\n" <<
" master_clock_rate: " <<
mod_settings.sdr_device_config.masterClockRate << "\n";
}
#endif
#if defined(HAVE_BLADERF)
else if (mod_settings.useBladeRFOutput) {
ss << " BladeRF\n"
" Device: " << mod_settings.sdr_device_config.device << "\n" <<
" refclk: " << mod_settings.sdr_device_config.refclk_src << "\n";
}
#endif
else if (mod_settings.useZeroMQOutput) {
ss << " ZeroMQ\n" <<
" Listening on: " << mod_settings.outputName << "\n" <<
" Socket type : " << mod_settings.zmqOutputSocketType << "\n";
}
ss << " Sampling rate: ";
if (mod_settings.outputRate > 1000) {
if (mod_settings.outputRate > 1000000) {
ss << std::fixed << std::setprecision(4) <<
mod_settings.outputRate / 1000000.0 <<
" MHz\n";
}
else {
ss << std::fixed << std::setprecision(4) <<
mod_settings.outputRate / 1000.0 <<
" kHz\n";
}
}
else {
ss << std::fixed << std::setprecision(4) <<
mod_settings.outputRate << " Hz\n";
}
fprintf(stderr, "%s", ss.str().c_str());
}
int set_realtime_prio(int prio)
{
// Set thread priority to realtime
const int policy = SCHED_RR;
sched_param sp;
sp.sched_priority = sched_get_priority_min(policy) + prio;
int ret = pthread_setschedparam(pthread_self(), policy, &sp);
return ret;
}
void set_thread_name(const char *name)
{
#if defined(HAVE_PRCTL)
prctl(PR_SET_NAME,name,0,0,0);
#endif
}
double parse_channel(const std::string& chan)
{
double freq;
if (chan == "5A") freq = 174928000;
else if (chan == "5B") freq = 176640000;
else if (chan == "5C") freq = 178352000;
else if (chan == "5D") freq = 180064000;
else if (chan == "6A") freq = 181936000;
else if (chan == "6B") freq = 183648000;
else if (chan == "6C") freq = 185360000;
else if (chan == "6D") freq = 187072000;
else if (chan == "7A") freq = 188928000;
else if (chan == "7B") freq = 190640000;
else if (chan == "7C") freq = 192352000;
else if (chan == "7D") freq = 194064000;
else if (chan == "8A") freq = 195936000;
else if (chan == "8B") freq = 197648000;
else if (chan == "8C") freq = 199360000;
else if (chan == "8D") freq = 201072000;
else if (chan == "9A") freq = 202928000;
else if (chan == "9B") freq = 204640000;
else if (chan == "9C") freq = 206352000;
else if (chan == "9D") freq = 208064000;
else if (chan == "10A") freq = 209936000;
else if (chan == "10B") freq = 211648000;
else if (chan == "10C") freq = 213360000;
else if (chan == "10D") freq = 215072000;
else if (chan == "11A") freq = 216928000;
else if (chan == "11B") freq = 218640000;
else if (chan == "11C") freq = 220352000;
else if (chan == "11D") freq = 222064000;
else if (chan == "12A") freq = 223936000;
else if (chan == "12B") freq = 225648000;
else if (chan == "12C") freq = 227360000;
else if (chan == "12D") freq = 229072000;
else if (chan == "13A") freq = 230784000;
else if (chan == "13B") freq = 232496000;
else if (chan == "13C") freq = 234208000;
else if (chan == "13D") freq = 235776000;
else if (chan == "13E") freq = 237488000;
else if (chan == "13F") freq = 239200000;
else {
std::cerr << "Channel " << chan << " does not exist in table\n";
throw std::out_of_range("channel out of range");
}
return freq;
}
std::optional convert_frequency_to_channel(double frequency)
{
const int freq = round(frequency);
std::string chan;
if (freq == 174928000) chan = "5A";
else if (freq == 176640000) chan = "5B";
else if (freq == 178352000) chan = "5C";
else if (freq == 180064000) chan = "5D";
else if (freq == 181936000) chan = "6A";
else if (freq == 183648000) chan = "6B";
else if (freq == 185360000) chan = "6C";
else if (freq == 187072000) chan = "6D";
else if (freq == 188928000) chan = "7A";
else if (freq == 190640000) chan = "7B";
else if (freq == 192352000) chan = "7C";
else if (freq == 194064000) chan = "7D";
else if (freq == 195936000) chan = "8A";
else if (freq == 197648000) chan = "8B";
else if (freq == 199360000) chan = "8C";
else if (freq == 201072000) chan = "8D";
else if (freq == 202928000) chan = "9A";
else if (freq == 204640000) chan = "9B";
else if (freq == 206352000) chan = "9C";
else if (freq == 208064000) chan = "9D";
else if (freq == 209936000) chan = "10A";
else if (freq == 211648000) chan = "10B";
else if (freq == 213360000) chan = "10C";
else if (freq == 215072000) chan = "10D";
else if (freq == 216928000) chan = "11A";
else if (freq == 218640000) chan = "11B";
else if (freq == 220352000) chan = "11C";
else if (freq == 222064000) chan = "11D";
else if (freq == 223936000) chan = "12A";
else if (freq == 225648000) chan = "12B";
else if (freq == 227360000) chan = "12C";
else if (freq == 229072000) chan = "12D";
else if (freq == 230784000) chan = "13A";
else if (freq == 232496000) chan = "13B";
else if (freq == 234208000) chan = "13C";
else if (freq == 235776000) chan = "13D";
else if (freq == 237488000) chan = "13E";
else if (freq == 239200000) chan = "13F";
else { return std::nullopt; }
return chan;
}
std::chrono::milliseconds transmission_frame_duration(unsigned int dabmode)
{
using namespace std::chrono;
switch (dabmode) {
case 1: return milliseconds(96);
case 2: return milliseconds(24);
case 3: return milliseconds(24);
case 4: return milliseconds(48);
default:
throw std::runtime_error("invalid DAB mode");
}
}
time_t get_clock_realtime_seconds()
{
struct timespec t;
if (clock_gettime(CLOCK_REALTIME, &t) != 0) {
throw std::runtime_error(std::string("Failed to retrieve CLOCK_REALTIME") + strerror(errno));
}
return t.tv_sec;
}