//
// Copyright 2010,2014,2016 Ettus Research LLC
//
// 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, either version 3 of the License, or
// (at your option) any later version.
//
// 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 for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see .
//
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "usrp1_eeprom.h"
#include "b100_eeprom.h"
#ifdef UHD_PLATFORM_LINUX
#include // syscall constants
#include // O_NONBLOCK
#include
#include
#include // for std::strerror
#endif //UHD_PLATFORM_LINUX
const std::string FX2_VENDOR_ID("0x04b4");
const std::string FX2_PRODUCT_ID("0x8613");
namespace po = boost::program_options;
int UHD_SAFE_MAIN(int argc, char *argv[]){
std::string type;
std::string image;
po::options_description desc("Allowed options");
desc.add_options()
("help", "help message")
("image", po::value(), "BIN image file; if not specified, use built-in image")
("vid", po::value(), "VID of device to program")
("pid", po::value(), "PID of device to program")
("type", po::value(&type), "device type (usrp1 or b100, required if using built-in image)")
;
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 << boost::format("USRP EEPROM initialization %s") % desc << std::endl;
return EXIT_FAILURE;
}
#ifdef UHD_PLATFORM_LINUX
//can't find an uninitialized usrp with this mystery usbtest in the way...
std::string module("usbtest");
std::ifstream modules("/proc/modules");
bool module_found = false;
std::string module_line;
while(std::getline(modules, module_line) && (!module_found)) {
module_found = boost::starts_with(module_line, module);
}
if(module_found) {
std::cout << boost::format("Found the '%s' module. Unloading it.\n" ) % module;
int fail = syscall(__NR_delete_module, module.c_str(), O_NONBLOCK);
if(fail)
std::cerr << ( boost::format("Removing the '%s' module failed with error '%s'.\n") % module % std::strerror(errno) );
}
#endif //UHD_PLATFORM_LINUX
//load the options into the address
uhd::device_addr_t device_addr;
device_addr["type"] = type;
if(vm.count("vid") or vm.count("pid")) {
if(not (vm.count("vid") and vm.count("pid") and vm.count("type"))) {
std::cerr << "ERROR: Must specify vid, pid, and type if specifying any of the two former args" << std::endl;
} else {
device_addr["vid"] = vm["vid"].as();
device_addr["pid"] = vm["pid"].as();
device_addr["type"] = vm["type"].as();
}
} else {
device_addr["vid"] = FX2_VENDOR_ID;
device_addr["pid"] = FX2_PRODUCT_ID;
}
if(vm.count("image")) {
//if specified, use external image file
image = vm["image"].as();
} else {
//if not specified, use built-ins; requires user to define type
size_t image_len;
unsigned const char* image_data;
if(!vm.count("type")) {
std::cerr << boost::format("ERROR: Image file not specified and type of device not given. Cannot use built-in images.\n");
return EXIT_FAILURE;
}
std::cout << boost::format("Using built-in image for \"%s\".\n") % type;
if(vm["type"].as() == "usrp1") {
image_len = usrp1_eeprom_bin_len;
image_data = usrp1_eeprom_bin;
} else if(vm["type"].as() == "b100") {
image_len = b100_eeprom_bin_len;
image_data = b100_eeprom_bin;
} else {
std::cerr << boost::format("ERROR: Unsupported device type \"%s\" specified and no EEPROM image file given.\n") % type;
return EXIT_FAILURE;
}
//get temporary file name, and write image to that.
image = boost::filesystem::unique_path().string();
std::ofstream tmp_image(image, std::ofstream::binary);
tmp_image.write((const char*)image_data, image_len);
tmp_image.close();
}
//find and create a control transport to do the writing.
uhd::device_addrs_t found_addrs = uhd::device::find(device_addr, uhd::device::USRP);
if (found_addrs.size() == 0){
std::cerr << "No USRP devices found" << std::endl;
return EXIT_FAILURE;
}
for (size_t i = 0; i < found_addrs.size(); i++){
std::cout << "Writing EEPROM data..." << std::endl;
//uhd::device_addrs_t devs = uhd::device::find(found_addrs[i]);
uhd::device::sptr dev = uhd::device::make(found_addrs[i], uhd::device::USRP);
uhd::property_tree::sptr tree = dev->get_tree();
tree->access("/mboards/0/load_eeprom").set(image);
}
//delete temporary image file if we created one
if(!vm.count("image")) {
boost::filesystem::remove(image);
}
std::cout << "Power-cycle the usrp for the changes to take effect." << std::endl;
return EXIT_SUCCESS;
}