// // Copyright 2010-2011 Ettus Research LLC // Copyright 2018 Ettus Research, a National Instruments Company // // SPDX-License-Identifier: GPL-3.0-or-later // #include #include #include #include #include #include #include #include #include #include #include using namespace uhd; using namespace uhd::usrp; namespace po = boost::program_options; uhd::fs_path get_radio_path(uhd::property_tree::sptr tree, std::string& slot) { // RFNoC devices: if (tree->exists("/blocks")) { if (slot.empty()) { slot = "0"; } const uhd::fs_path radio_path = uhd::fs_path("/blocks/0/Radio#" + slot) / "dboard"; if (!tree->exists(radio_path)) { throw uhd::assertion_error("Invalid slot index: " + slot); } return radio_path; } // Other devices: const uhd::fs_path db_root = "/mboards/0/dboards"; std::vector dboard_names = tree->list(db_root); if (dboard_names.size() == 1 and slot.empty()) { slot = dboard_names.front(); } uhd::assert_has(dboard_names, slot, "dboard slot name"); return db_root / slot; } int UHD_SAFE_MAIN(int argc, char* argv[]) { // command line variables std::string args, slot, unit; po::options_description desc("Allowed options"); // clang-format off desc.add_options() ("help", "help message") ("args", po::value(&args)->default_value(""), "device address args [default = \"\"]") ("slot", po::value(&slot)->default_value(""), "dboard slot name/index [default is blank for automatic]") ("unit", po::value(&unit)->default_value(""), "which unit [RX, TX, or GDB]") ("id", po::value(), "dboard id to burn, omit for readback") ("ser", po::value(), "serial to burn, omit for readback") ("rev", po::value(), "revision to burn, omit for readback") ; // 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 << boost::format("USRP Burn Daughterboard EEPROM %s") % desc << std::endl; std::cout << "Omit the ID argument to perform readback,\n" "Or specify a new ID to burn into the EEPROM.\n" << std::endl; return EXIT_SUCCESS; } if (unit.empty()) { std::cout << "Error: --unit must be specified!" << std::endl; return EXIT_FAILURE; } // make the device and extract the dboard w/ property auto dev = multi_usrp::make(args); auto tree = dev->get_tree(); const auto radio_path = get_radio_path(tree, slot); std::cout << "Reading " << unit << " EEPROM on dboard " << slot << "..." << std::endl; boost::to_lower(unit); const uhd::fs_path db_path = radio_path / (unit + "_eeprom"); dboard_eeprom_t db_eeprom = tree->access(db_path).get(); //------------- handle the dboard ID -----------------------------// if (vm.count("id")) { db_eeprom.id = dboard_id_t::from_string(vm["id"].as()); tree->access(db_path).set(db_eeprom); } std::cout << boost::format(" Current ID: %s") % db_eeprom.id.to_pp_string() << std::endl; //------------- handle the dboard serial--------------------------// if (vm.count("ser")) { db_eeprom.serial = vm["ser"].as(); tree->access(db_path).set(db_eeprom); } std::cout << boost::format(" Current serial: \"%s\"") % db_eeprom.serial << std::endl; //------------- handle the dboard revision------------------------// if (vm.count("rev")) { db_eeprom.revision = vm["rev"].as(); tree->access(db_path).set(db_eeprom); } std::cout << boost::format(" Current revision: \"%s\"") % db_eeprom.revision << std::endl; std::cout << " Done" << std::endl << std::endl; return EXIT_SUCCESS; }