mirror of
https://github.com/saymrwulf/uhd.git
synced 2026-05-16 21:10:10 +00:00
Note: Replacing everything with a lambda would be even better, but that can't be easily scripted so we'll do this as a first step to reduce the Boost footprint. This also removes occurences of #include <boost/bind.hpp>, and makes sure all usages of std::bind have an #include <functional>. clang-format wasn't always applied to minimize the changeset in this commit, however, it was applied to the blocks of #includes. Due to conflicts with other Boost libraries, the placeholders _1, _2, etc. could not be directly used, but had to be explicitly called out (as std::placeholders::_1, etc.). This makes the use of std::bind even uglier, which serves as another reminder that using std::bind (and even more so, boost::bind) should be avoided. nirio/rpc/rpc_client.cpp still contains a reference to boost::bind. It was not possible to remove it by simply doing a search and replace, so it will be removed in a separate commit.
145 lines
5.4 KiB
C++
145 lines
5.4 KiB
C++
//
|
|
// Copyright 2014 Ettus Research LLC
|
|
// Copyright 2018 Ettus Research, a National Instruments Company
|
|
//
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
//
|
|
|
|
#include <uhd/device.hpp>
|
|
#include <uhd/exception.hpp>
|
|
#include <uhd/types/time_spec.hpp>
|
|
#include <uhd/usrp/multi_usrp.hpp>
|
|
#include <uhd/usrp_clock/multi_usrp_clock.hpp>
|
|
#include <uhd/utils/safe_main.hpp>
|
|
#include <uhd/utils/thread.hpp>
|
|
#include <boost/format.hpp>
|
|
#include <boost/program_options.hpp>
|
|
#include <chrono>
|
|
#include <functional>
|
|
#include <iostream>
|
|
#include <thread>
|
|
|
|
namespace po = boost::program_options;
|
|
|
|
using namespace uhd::usrp_clock;
|
|
using namespace uhd::usrp;
|
|
|
|
void get_usrp_time(multi_usrp::sptr usrp, size_t mboard, std::vector<int64_t>* times)
|
|
{
|
|
(*times)[mboard] = usrp->get_time_now(mboard).get_full_secs();
|
|
}
|
|
|
|
int UHD_SAFE_MAIN(int argc, char* argv[])
|
|
{
|
|
// Variables to be set by command line options
|
|
std::string clock_args, usrp_args;
|
|
uint32_t max_interval, num_tests;
|
|
|
|
// Set up program options
|
|
po::options_description desc("Allowed options");
|
|
// clang-format off
|
|
desc.add_options()
|
|
("help", "Display this help message")
|
|
("clock-args", po::value<std::string>(&clock_args), "Clock device arguments")
|
|
("usrp-args", po::value<std::string>(&usrp_args), "USRP device arguments")
|
|
("max-interval", po::value<uint32_t>(&max_interval)->default_value(10000), "Maximum interval between comparisons (in ms)")
|
|
("num-tests", po::value<uint32_t>(&num_tests)->default_value(10), "Number of times to compare device times")
|
|
;
|
|
// clang-format on
|
|
po::variables_map vm;
|
|
po::store(po::parse_command_line(argc, argv, desc), vm);
|
|
po::notify(vm);
|
|
|
|
// Print the help message
|
|
if (vm.count("help")) {
|
|
std::cout << std::endl << "Test Clock Synchronization" << std::endl << std::endl;
|
|
|
|
std::cout << "This example shows how to use a clock device to" << std::endl
|
|
<< "synchronize the time on multiple USRP devices." << std::endl
|
|
<< std::endl;
|
|
|
|
std::cout << desc << std::endl;
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
// Create a Multi-USRP-Clock device (currently OctoClock only)
|
|
std::cout << boost::format("\nCreating the Clock device with: %s") % clock_args
|
|
<< std::endl;
|
|
multi_usrp_clock::sptr clock = multi_usrp_clock::make(clock_args);
|
|
|
|
// Make sure Clock configuration is correct
|
|
if (clock->get_sensor("gps_detected").value == "false") {
|
|
throw uhd::runtime_error("No GPSDO detected on Clock.");
|
|
}
|
|
if (clock->get_sensor("using_ref").value != "internal") {
|
|
throw uhd::runtime_error("Clock must be using an internal reference.");
|
|
}
|
|
|
|
// Create a Multi-USRP device
|
|
std::cout << boost::format("\nCreating the USRP device with: %s") % usrp_args
|
|
<< std::endl;
|
|
multi_usrp::sptr usrp = multi_usrp::make(usrp_args);
|
|
|
|
// Store USRP device serials for useful output
|
|
std::vector<std::string> serials;
|
|
for (size_t ch = 0; ch < usrp->get_num_mboards(); ch++) {
|
|
serials.push_back(usrp->get_usrp_tx_info(ch)["mboard_serial"]);
|
|
}
|
|
|
|
std::cout << std::endl << "Checking USRP devices for lock." << std::endl;
|
|
bool all_locked = true;
|
|
for (size_t ch = 0; ch < usrp->get_num_mboards(); ch++) {
|
|
std::string ref_locked = usrp->get_mboard_sensor("ref_locked", ch).value;
|
|
std::cout << boost::format(" * %s: %s") % serials[ch] % ref_locked << std::endl;
|
|
|
|
if (ref_locked != "true")
|
|
all_locked = false;
|
|
}
|
|
if (not all_locked)
|
|
std::cout << std::endl << "WARNING: One or more devices not locked." << std::endl;
|
|
|
|
// Get GPS time to initially set USRP devices
|
|
std::cout << std::endl
|
|
<< "Querying Clock for time and setting USRP times..." << std::endl
|
|
<< std::endl;
|
|
int64_t clock_time = clock->get_time();
|
|
usrp->set_time_next_pps(uhd::time_spec_t(double(clock_time + 1)));
|
|
srand((unsigned int)time(NULL));
|
|
|
|
std::cout << boost::format("Running %d comparisons at random intervals.") % num_tests
|
|
<< std::endl;
|
|
uint32_t num_matches = 0;
|
|
for (size_t i = 0; i < num_tests; i++) {
|
|
// Wait random time before querying
|
|
uint16_t wait_time = rand() % max_interval;
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(wait_time));
|
|
|
|
// Get all times before output
|
|
std::vector<int64_t> usrp_times(usrp->get_num_mboards());
|
|
boost::thread_group thread_group;
|
|
clock_time = clock->get_time();
|
|
for (size_t j = 0; j < usrp->get_num_mboards(); j++) {
|
|
thread_group.create_thread(std::bind(&get_usrp_time, usrp, j, &usrp_times));
|
|
}
|
|
// Wait for threads to complete
|
|
thread_group.join_all();
|
|
|
|
std::cout << boost::format("Comparison #%d") % (i + 1) << std::endl;
|
|
bool all_match = true;
|
|
std::cout << boost::format(" * Clock time: %d") % clock_time << std::endl;
|
|
for (size_t j = 0; j < usrp->get_num_mboards(); j++) {
|
|
std::cout << boost::format(" * %s time: %d") % serials[j] % usrp_times[j]
|
|
<< std::endl;
|
|
if (usrp_times[j] != clock_time)
|
|
all_match = false;
|
|
}
|
|
if (all_match)
|
|
num_matches++;
|
|
}
|
|
|
|
std::cout << std::endl
|
|
<< boost::format("Number of matches: %d/%d") % num_matches % num_tests
|
|
<< std::endl;
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|