// // Copyright 2016 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 #include "n230_fw_host_iface.h" #include "n230_impl.hpp" using namespace uhd; using namespace uhd::usrp; using namespace uhd::transport; struct xil_bitfile_hdr_t { xil_bitfile_hdr_t(): valid(false), userid(0), product(""), fpga(""), timestamp(""), filesize(0) {} bool valid; uint32_t userid; std::string product; std::string fpga; std::string timestamp; uint32_t filesize; }; static inline uint16_t _to_uint16(uint8_t* buf) { return (static_cast(buf[0]) << 8) | (static_cast(buf[1]) << 0); } static inline uint32_t _to_uint32(uint8_t* buf) { return (static_cast(buf[0]) << 24) | (static_cast(buf[1]) << 16) | (static_cast(buf[2]) << 8) | (static_cast(buf[3]) << 0); } static void _parse_bitfile_header(const std::string& filepath, xil_bitfile_hdr_t& hdr) { // Read header into memory std::ifstream img_file(filepath.c_str(), std::ios::binary); static const size_t MAX_HDR_SIZE = 1024; boost::scoped_array hdr_buf(new char[MAX_HDR_SIZE]); img_file.seekg(0, std::ios::beg); img_file.read(hdr_buf.get(), MAX_HDR_SIZE); img_file.close(); //Parse header size_t ptr = 0; uint8_t* buf = reinterpret_cast(hdr_buf.get()); //Shortcut uint8_t signature[10] = {0x00, 0x09, 0x0f, 0xf0, 0x0f, 0xf0, 0x0f, 0xf0, 0x0f, 0xf0}; if (memcmp(buf, signature, 10) == 0) { //Validate signature ptr += _to_uint16(buf + ptr) + 2; ptr += _to_uint16(buf + ptr) + 1; std::string fields[4]; for (size_t i = 0; i < 4; i++) { size_t key = buf[ptr++] - 'a'; uint16_t len = _to_uint16(buf + ptr); ptr += 2; fields[key] = std::string(reinterpret_cast(buf + ptr), size_t(len)); ptr += len; } hdr.filesize = _to_uint32(buf + ++ptr); hdr.fpga = fields[1]; hdr.timestamp = fields[2] + std::string(" ") + fields[3]; std::vector tokens; boost::split(tokens, fields[0], boost::is_any_of(";")); if (tokens.size() == 3) { hdr.product = tokens[0]; std::vector uidtokens; boost::split(uidtokens, tokens[1], boost::is_any_of("=")); if (uidtokens.size() == 2 and uidtokens[0] == "UserID") { std::stringstream stream; stream << uidtokens[1]; stream >> std::hex >> hdr.userid; hdr.valid = true; } } } } static size_t _send_and_recv( udp_simple::sptr xport, n230_flash_prog_t& out, n230_flash_prog_t& in) { static uint32_t seqno = 0; out.seq = htonx(++seqno); xport->send(boost::asio::buffer(&out, sizeof(n230_flash_prog_t))); size_t len = xport->recv(boost::asio::buffer(&in, udp_simple::mtu), 0.5); if (len != sizeof(n230_flash_prog_t) or ntohx(in.seq) != seqno) { throw uhd::io_error("Error communicating with the device."); } return len; } static bool n230_image_loader(const image_loader::image_loader_args_t &loader_args){ // Run discovery routine and ensure that exactly one N230 is specified device_addrs_t devs = usrp::n230::n230_impl::n230_find(loader_args.args); if (devs.size() == 0 or !loader_args.load_fpga) return false; if (devs.size() > 1) { throw uhd::runtime_error("Multiple devices match the specified args. To avoid accidentally updating the " "wrong device, please narrow the search by specifying a unique \"addr\" argument."); } device_addr_t dev = devs[0]; // Sanity check the specified bitfile std::string fpga_img_path = loader_args.fpga_path; bool fpga_path_specified = !loader_args.fpga_path.empty(); if (not fpga_path_specified) { fpga_img_path = ( fs::path(uhd::get_pkg_path()) / "share" / "uhd" / "images" / "usrp_n230_fpga.bit" ).string(); } if (not boost::filesystem::exists(fpga_img_path)) { if (fpga_path_specified) { throw uhd::runtime_error(str(boost::format("The file \"%s\" does not exist.") % fpga_img_path)); } else { throw uhd::runtime_error(str(boost::format( "Could not find the default FPGA image: %s.\n" "Either specify the --fpga-path argument or download the latest prebuilt images:\n" "%s\n") % fpga_img_path % print_utility_error("uhd_images_downloader.py"))); } } xil_bitfile_hdr_t hdr; _parse_bitfile_header(fpga_img_path, hdr); // Create a UDP communication link udp_simple::sptr udp_xport = udp_simple::make_connected(dev["addr"],BOOST_STRINGIZE(N230_FW_COMMS_FLASH_PROG_PORT)); if (hdr.valid and hdr.product == "n230") { if (hdr.userid != 0x5AFE0000) { std::cout << boost::format("Unit: USRP N230 (%s, %s)\n-- FPGA Image: %s\n") % dev["addr"] % dev["serial"] % fpga_img_path; // Write image std::ifstream image(fpga_img_path.c_str(), std::ios::binary); size_t image_size = boost::filesystem::file_size(fpga_img_path); static const size_t SECTOR_SIZE = 65536; static const size_t IMAGE_BASE = 0x400000; n230_flash_prog_t out, in; size_t bytes_written = 0; while (bytes_written < image_size) { size_t payload_size = std::min(image_size - bytes_written, N230_FLASH_COMM_MAX_PAYLOAD_SIZE); if (bytes_written % SECTOR_SIZE == 0) { out.flags = htonx(N230_FLASH_COMM_FLAGS_ACK|N230_FLASH_COMM_CMD_ERASE_FPGA); out.offset = htonx(bytes_written + IMAGE_BASE); out.size = htonx(payload_size); _send_and_recv(udp_xport, out, in); } out.flags = htonx(N230_FLASH_COMM_FLAGS_ACK|N230_FLASH_COMM_CMD_WRITE_FPGA); out.offset = htonx(bytes_written + IMAGE_BASE); out.size = htonx(payload_size); image.read((char*)out.data, payload_size); _send_and_recv(udp_xport, out, in); bytes_written += ntohx(in.size); std::cout << boost::format("\r-- Loading FPGA image: %d%%") % (int(double(bytes_written) / double(image_size) * 100.0)) << std::flush; } std::cout << std::endl << "FPGA image loaded successfully." << std::endl; std::cout << std::endl << "Power-cycle the device to run the image." << std::endl; return true; } else { throw uhd::runtime_error("This utility cannot burn a failsafe image!"); } } else { throw uhd::runtime_error(str(boost::format("The file at path \"%s\" is not a valid USRP N230 FPGA image.") % fpga_img_path)); } } UHD_STATIC_BLOCK(register_n230_image_loader){ std::string recovery_instructions = "Aborting. Your USRP N230 device will likely boot in safe mode.\n" "Please re-run this command with the additional \"safe_mode\" device argument\n" "to recover your device."; image_loader::register_image_loader("n230", n230_image_loader, recovery_instructions); }