aboutsummaryrefslogtreecommitdiffstats
path: root/host/utils/usrp_x3xx_fpga_burner.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'host/utils/usrp_x3xx_fpga_burner.cpp')
-rw-r--r--host/utils/usrp_x3xx_fpga_burner.cpp47
1 files changed, 31 insertions, 16 deletions
diff --git a/host/utils/usrp_x3xx_fpga_burner.cpp b/host/utils/usrp_x3xx_fpga_burner.cpp
index 07bc63559..2f0202e84 100644
--- a/host/utils/usrp_x3xx_fpga_burner.cpp
+++ b/host/utils/usrp_x3xx_fpga_burner.cpp
@@ -15,6 +15,7 @@
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
+#include <csignal>
#include <iostream>
#include <map>
#include <fstream>
@@ -81,6 +82,22 @@ namespace po = boost::program_options;
using namespace uhd;
using namespace uhd::transport;
+static int num_ctrl_c = 0;
+void sig_int_handler(int){
+ num_ctrl_c++;
+ if(num_ctrl_c == 1){
+ std::cout << std::endl << "Are you sure you want to abort the image burning? If you do, your "
+ "USRP-X series device will be bricked!" << std::endl
+ << "Press Ctrl+C again to abort the image burning procedure." << std::endl << std::endl;
+ }
+ else{
+ std::cout << std::endl << "Aborting. Your USRP X-Series device will be bricked." << std::endl
+ << "Refer to http://files.ettus.com/uhd_docs/manual/html/usrp_x3x0.html#use-jtag-to-load-fpga-images" << std::endl
+ << "for details on restoring your device." << std::endl;
+ exit(EXIT_FAILURE);
+ }
+}
+
typedef struct {
boost::uint32_t flags;
boost::uint32_t sector;
@@ -232,21 +249,16 @@ void ethernet_burn(udp_simple::sptr udp_transport, std::string fpga_path, bool v
throw std::runtime_error("Failed to start image burning! Did you specify the correct IP address? If so, power-cycle the device and try again.");
}
- std::cout << "Progress: " << std::flush;
-
- int percentage = -1;
- int last_percentage = -1;
size_t current_pos = 0;
+ size_t sectors = fpga_image_size / X300_FLASH_SECTOR_SIZE;
//Each sector
for(size_t i = 0; i < fpga_image_size; i += X300_FLASH_SECTOR_SIZE){
- //Print percentage at beginning of first sector after each 10%
- percentage = int(double(i)/double(fpga_image_size)*100);
- if((percentage != last_percentage) and (percentage % 10 == 0)){ //Don't print same percentage twice
- std::cout << percentage << "%..." << std::flush;
- }
- last_percentage = percentage;
+ //Print progress percentage at beginning of each sector
+ std::cout << "\rProgress: " << int(double(i)/double(fpga_image_size)*100)
+ << "% (" << (i / X300_FLASH_SECTOR_SIZE) << "/"
+ << sectors << " sectors)" << std::flush;
//Each packet
for(size_t j = i; (j < fpga_image_size and j < (i+X300_FLASH_SECTOR_SIZE)); j += X300_PACKET_SIZE_BYTES){
@@ -256,7 +268,7 @@ void ethernet_burn(udp_simple::sptr udp_transport, std::string fpga_path, bool v
if(verify) send_packet.flags |= X300_FPGA_PROG_FLAGS_VERIFY;
if(j == i) send_packet.flags |= X300_FPGA_PROG_FLAGS_ERASE; //Erase the sector before writing
send_packet.flags = htonx<boost::uint32_t>(send_packet.flags);
-
+
send_packet.sector = htonx<boost::uint32_t>(X300_FPGA_SECTOR_START + (i/X300_FLASH_SECTOR_SIZE));
send_packet.index = htonx<boost::uint32_t>((j % X300_FLASH_SECTOR_SIZE) / 2);
send_packet.size = htonx<boost::uint32_t>(X300_PACKET_SIZE_BYTES / 2);
@@ -297,10 +309,12 @@ void ethernet_burn(udp_simple::sptr udp_transport, std::string fpga_path, bool v
for(size_t k = 0; k < (X300_PACKET_SIZE_BYTES/2); k++){
send_packet.data[k] = htonx<boost::uint16_t>(send_packet.data[k]);
}
-
+
udp_transport->send(boost::asio::buffer(&send_packet, sizeof(send_packet)));
- udp_transport->recv(boost::asio::buffer(x300_data_in_mem), UDP_TIMEOUT);
+ if (udp_transport->recv(boost::asio::buffer(x300_data_in_mem), UDP_TIMEOUT) == 0)
+ throw std::runtime_error("Timed out waiting for ACK!");
+
const x300_fpga_update_data_t *update_data_in = reinterpret_cast<const x300_fpga_update_data_t *>(x300_data_in_mem);
if((ntohl(update_data_in->flags) & X300_FPGA_PROG_FLAGS_ERROR) == X300_FPGA_PROG_FLAGS_ERROR){
@@ -319,14 +333,15 @@ void ethernet_burn(udp_simple::sptr udp_transport, std::string fpga_path, bool v
memset(cleanup_packet.data, 0, sizeof(cleanup_packet.data));
udp_transport->send(boost::asio::buffer(&cleanup_packet, sizeof(cleanup_packet)));
- udp_transport->recv(boost::asio::buffer(x300_data_in_mem), UDP_TIMEOUT);
+ if (udp_transport->recv(boost::asio::buffer(x300_data_in_mem), UDP_TIMEOUT) == 0)
+ throw std::runtime_error("Timed out waiting for ACK!");
const x300_fpga_update_data_t *cleanup_data_in = reinterpret_cast<const x300_fpga_update_data_t *>(x300_data_in_mem);
if((ntohl(cleanup_data_in->flags) & X300_FPGA_PROG_FLAGS_ERROR) == X300_FPGA_PROG_FLAGS_ERROR){
throw std::runtime_error("Transfer or data verification failed!");
}
- std::cout << "100%" << std::endl;
+ std::cout << "\rProgress: " << "100% (" << sectors << "/" << sectors << " sectors)" << std::endl;
}
void pcie_burn(std::string resource, std::string rpc_port, std::string fpga_path)
@@ -460,7 +475,6 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){
}
}
-
/*
* Check validity of image through extension
*/
@@ -469,6 +483,7 @@ int UHD_SAFE_MAIN(int argc, char *argv[]){
throw std::runtime_error("The image filename must end in .bin, .bit, or .lvbitx.");
}
+ std::signal(SIGINT, &sig_int_handler);
if(vm.count("addr")){
udp_simple::sptr udp_transport = udp_simple::make_connected(ip_addr, BOOST_STRINGIZE(X300_FPGA_PROG_UDP_PORT));