/*
Copyright (C) 2005, 2006, 2007, 2008, 2009, 2010 Her Majesty the
Queen in Right of Canada (Communications Research Center Canada)
Copyright (C) 2019
Matthias P. Braendli, matthias.braendli@mpb.li
Copyright (C) 2021
Steven Rossel, steven.rossel@bluewin.ch
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 "BladeRF.h"
#ifdef HAVE_BLADERF
#include
#include
#include
#include
#include
#include
#include "Log.h"
#include "Utils.h"
using namespace std;
namespace Output
{
BladeRF::BladeRF(SDRDeviceConfig &config) : SDRDevice(), m_conf(config)
{
etiLog.level(info) << "BladeRF:Creating the device with: " << m_conf.device;
struct bladerf_devinfo devinfo;
// init device infos
bladerf_init_devinfo(&devinfo); // this function does not return a status
int status = bladerf_open_with_devinfo(&m_device, &devinfo); // open device with info
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot open BladeRF output device");
}
if (m_conf.refclk_src == "pps")
{
status = bladerf_set_vctcxo_tamer_mode(m_device, BLADERF_VCTCXO_TAMER_1_PPS); // 1 PPS tames the clock
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot set BladeRF refclk to pps");
}
}
else if (m_conf.refclk_src == "10mhz")
{
status = bladerf_set_vctcxo_tamer_mode(m_device, BLADERF_VCTCXO_TAMER_10_MHZ); // 10 MHz tames the clock
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot set BladeRF refclk to 10 MHz");
}
}
status = bladerf_set_sample_rate(m_device, m_channel, (bladerf_sample_rate)m_conf.sampleRate, NULL);
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot set BladeRF sample rate");
}
bladerf_sample_rate host_sample_rate = 0;
status = bladerf_get_sample_rate(m_device, m_channel, &host_sample_rate);
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot get BladeRF sample rate");
}
etiLog.level(info) << "BladeRF sample rate set to " << std::to_string(host_sample_rate / 1000.0) << " kHz";
tune(m_conf.lo_offset, m_conf.frequency);
bladerf_frequency cur_frequency = 0;
status = bladerf_get_frequency(m_device, m_channel, &cur_frequency);
if(status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot get BladeRF frequency");
}
etiLog.level(info) << "BladeRF:Actual frequency: " << fixed << setprecision(3) << cur_frequency / 1000.0 << " kHz.";
status = bladerf_set_gain(m_device, m_channel, (bladerf_gain)m_conf.txgain); // gain in [dB]
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot set BladeRF gain");
}
bladerf_bandwidth cur_bandwidth = 0;
status = bladerf_set_bandwidth(m_device, m_channel, (bladerf_bandwidth)m_conf.bandwidth, &cur_bandwidth);
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot set BladeRF bandwidth");
}
/* ---------------------------- Streaming Config ---------------------------- */
const unsigned int num_buffers = 16; // Number of buffers to use in the underlying data stream
const unsigned int buffer_size = 8192; // "to hold 2048 samples for one channel, a buffer must be at least 8192 bytes large"
const unsigned int num_transfers = 8; // active USB transfers
const unsigned int timeout_ms = 3500;
/* Configure the device's x1 TX (SISO) channel for use with the
* synchronous interface. SC16 Q11 samples *without* metadata are used. */
status = bladerf_sync_config(m_device, BLADERF_TX_X1, BLADERF_FORMAT_SC16_Q11, num_buffers,
buffer_size, num_transfers, timeout_ms);
if (status != 0) {
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot setup BladeRF stream");
}
status = bladerf_enable_module(m_device, m_channel, true);
if(status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
throw runtime_error("Cannot enable BladeRF channel");
}
}
BladeRF::~BladeRF()
{
if (m_device != nullptr)
{
//bladerf_deinit_stream(m_stream); /* Asynchronous API function*/
bladerf_enable_module(m_device, m_channel, false);
bladerf_close(m_device);
}
}
void BladeRF::tune(double lo_offset, double frequency)
{
int status;
if (not m_device)
throw runtime_error("BladeRF device not set up");
if (lo_offset != 0)
{
etiLog.level(info) << "lo_offset cannot be set at "<< std::to_string(lo_offset) << " with BladeRF output, has to be 0"
<< "\nlo_offset is now set to 0";
m_conf.lo_offset = 0;
}
status = bladerf_set_frequency(m_device, m_channel, (bladerf_frequency)m_conf.frequency);
if (status < 0)
{
etiLog.level(error) << "Error setting BladeRF TX frequency: %s " << bladerf_strerror(status);
}
}
double BladeRF::get_tx_freq(void) const
{
if (not m_device)
throw runtime_error("Lime device not set up");
int status;
bladerf_frequency cur_frequency = 0;
status = bladerf_get_frequency(m_device, m_channel, &cur_frequency);
if (status < 0)
{
etiLog.level(error) << "Error getting BladeRF TX frequency: %s " << bladerf_strerror(status);
}
return (double)cur_frequency;
}
void BladeRF::set_txgain(double txgain)
{
m_conf.txgain = txgain;
if (not m_device)
throw runtime_error("Lime device not set up");
int status;
status = bladerf_set_gain(m_device, m_channel, (bladerf_gain)m_conf.txgain); // gain in [dB]
if (status < 0)
{
etiLog.level(error) << "Error making BladeRF device: %s " << bladerf_strerror(status);
}
}
double BladeRF::get_txgain(void) const
{
if (not m_device)
throw runtime_error("BladeRF device not set up");
bladerf_gain txgain = 0;
int status;
status = bladerf_get_gain(m_device, m_channel, &txgain);
if (status < 0)
{
etiLog.level(error) << "Error getting BladeRF TX gain: %s " << bladerf_strerror(status);
}
return (double)txgain;
}
void BladeRF::set_bandwidth(double bandwidth)
{
bladerf_set_bandwidth(m_device, m_channel, (bladerf_bandwidth)m_conf.bandwidth, NULL);
}
double BladeRF::get_bandwidth(void) const
{
bladerf_bandwidth bw;
bladerf_get_bandwidth(m_device, m_channel, &bw);
return (double)bw;
}
SDRDevice::run_statistics_t BladeRF::get_run_statistics(void) const
{
run_statistics_t rs;
rs["frames"] = num_frames_modulated;
return rs;
}
double BladeRF::get_real_secs(void) const
{
// TODO
return 0.0;
}
void BladeRF::set_rxgain(double rxgain)
{
// TODO
}
double BladeRF::get_rxgain(void) const
{
// TODO
return 0.0;
}
size_t BladeRF::receive_frame(
complexf *buf,
size_t num_samples,
frame_timestamp &ts,
double timeout_secs)
{
// TODO
return 0;
}
bool BladeRF::is_clk_source_ok()
{
// TODO
return true;
}
const char *BladeRF::device_name(void) const
{
return "BladeRF";
}
std::optional BladeRF::get_temperature(void) const
{
if (not m_device)
throw runtime_error("BladeRF device not set up");
float temp = 0.0;
int status = bladerf_get_rfic_temperature(m_device, &temp);
if (status >= 0) {
return (double)temp;
}
else {
etiLog.level(error) << "Error getting BladeRF temperature: %s " << bladerf_strerror(status);
return std::nullopt;
}
}
void BladeRF::transmit_frame(struct FrameData&& frame) // SC16 frames
{
const size_t num_samples = frame.buf.size() / (2*sizeof(int16_t));
const int status = bladerf_sync_tx(m_device, frame.buf.data(), num_samples, NULL, 0);
if (status < 0) {
etiLog.level(error) << "Error transmitting samples with BladeRF: %s " << bladerf_strerror(status);
throw runtime_error("Cannot transmit TX samples");
}
num_frames_modulated++;
}
} // namespace Output
#endif // HAVE_BLADERF