openFPGALoader/src/main.cpp

700 lines
19 KiB
C++
Raw Normal View History

2021-06-26 15:24:07 +02:00
// SPDX-License-Identifier: Apache-2.0
2019-09-26 18:29:20 +02:00
/*
* Copyright (C) 2019 Gwenhael Goavec-Merou <gwenhael.goavec-merou@trabucayre.com>
*/
2020-07-23 18:32:18 +02:00
#include "cxxopts.hpp"
2019-09-26 18:29:20 +02:00
#include <fstream>
#include <iomanip>
2019-09-26 18:29:20 +02:00
#include <iostream>
#include <map>
#include <sstream>
#include <string.h>
#include <unistd.h>
#include <vector>
#include "altera.hpp"
#include "anlogic.hpp"
2019-09-26 18:29:20 +02:00
#include "board.hpp"
#include "cable.hpp"
#include "device.hpp"
#include "dfu.hpp"
#include "display.hpp"
#include "efinix.hpp"
2020-10-06 08:38:24 +02:00
#include "ftdispi.hpp"
2019-12-06 07:29:28 +01:00
#include "gowin.hpp"
#include "ice40.hpp"
2019-11-18 16:04:50 +01:00
#include "lattice.hpp"
#include "jtag.hpp"
#include "part.hpp"
2020-10-06 08:38:24 +02:00
#include "spiFlash.hpp"
2020-10-07 07:47:46 +02:00
#include "rawParser.hpp"
2019-09-26 18:29:20 +02:00
#include "xilinx.hpp"
#define DEFAULT_FREQ 6000000
2019-09-26 18:29:20 +02:00
using namespace std;
struct arguments {
int8_t verbose;
bool reset, detect, verify;
2019-09-26 18:29:20 +02:00
unsigned int offset;
string bit_file;
string device;
2019-09-26 18:29:20 +02:00
string cable;
2020-09-29 14:16:30 +02:00
string ftdi_serial;
int ftdi_channel;
2020-05-25 10:31:37 +02:00
uint32_t freq;
2019-09-26 18:29:20 +02:00
string board;
bool pin_config;
bool list_cables;
bool list_boards;
bool list_fpga;
2021-02-18 21:09:34 +01:00
Device::prog_type_t prg_type;
bool is_list_command;
2020-10-06 08:38:24 +02:00
bool spi;
bool dfu;
string file_type;
string fpga_part;
2021-05-13 16:07:40 +02:00
string probe_firmware;
int index_chain;
unsigned int file_size;
2019-09-26 18:29:20 +02:00
};
int parse_opt(int argc, char **argv, struct arguments *args, jtag_pins_conf_t *pins_config);
void displaySupported(const struct arguments &args);
2019-09-26 18:29:20 +02:00
int main(int argc, char **argv)
{
cable_t cable;
target_board_t *board = NULL;
jtag_pins_conf_t pins_config = {0, 0, 0, 0};
2019-09-26 18:29:20 +02:00
/* command line args. */
struct arguments args = {0, false, false, false, 0, "", "", "-", "", -1,
DEFAULT_FREQ, "-", false, false, false, false, Device::WR_SRAM, false,
false, false, "", "", "", -1, 0};
2019-09-26 18:29:20 +02:00
/* parse arguments */
2020-07-23 18:32:18 +02:00
try {
if (parse_opt(argc, argv, &args, &pins_config))
2020-07-23 18:32:18 +02:00
return EXIT_SUCCESS;
} catch (std::exception &e) {
printError("Error in parse arg step");
return EXIT_FAILURE;
}
2019-09-26 18:29:20 +02:00
if (args.is_list_command) {
displaySupported(args);
return EXIT_SUCCESS;
}
2021-02-18 21:09:34 +01:00
if (args.prg_type == Device::WR_SRAM)
cout << "write to ram" << endl;
if (args.prg_type == Device::WR_FLASH)
cout << "write to flash" << endl;
2019-09-26 18:29:20 +02:00
if (args.board[0] != '-' && board_list.find(args.board) != board_list.end()) {
board = &(board_list[args.board]);
}
uint32_t freq = args.freq;
/* if a board name is specified try to use this to determine cable */
if (board) {
/* set pins config (only when user has not already provided
* configuration
*/
if (!args.pin_config) {
pins_config.tdi_pin = board->jtag_pins_config.tdi_pin;
pins_config.tdo_pin = board->jtag_pins_config.tdo_pin;
pins_config.tms_pin = board->jtag_pins_config.tms_pin;
pins_config.tck_pin = board->jtag_pins_config.tck_pin;
}
/* search for cable */
auto t = cable_list.find(board->cable_name);
2019-09-26 18:29:20 +02:00
if (t == cable_list.end()) {
cout << "Board " << args.board << " has not default cable" << endl;
} else {
if (args.cable[0] == '-') { // no use selection
args.cable = (*t).first; // use board default cable
} else {
cout << "Board default cable overridden with " << args.cable << endl;
}
}
/* Xilinx only: to write flash exact fpga model must be provided */
if (!board->fpga_part.empty() && !args.fpga_part.empty())
printInfo("Board default fpga part overridden with " + args.fpga_part);
else if (!board->fpga_part.empty() && args.fpga_part.empty())
args.fpga_part = board->fpga_part;
/* Some boards can override the default clock speed */
if (board->default_freq != CABLE_DEFAULT && freq == DEFAULT_FREQ)
freq = board->default_freq;
}
if (args.cable[0] == '-') { /* if no board and no cable */
if (args.verbose > 0)
cout << "No cable or board specified: using direct ft2232 interface" << endl;
2019-09-26 18:29:20 +02:00
args.cable = "ft2232";
}
auto select_cable = cable_list.find(args.cable);
if (select_cable == cable_list.end()) {
2020-07-11 07:47:27 +02:00
printError("error : " + args.cable + " not found");
2019-09-26 18:29:20 +02:00
return EXIT_FAILURE;
}
cable = select_cable->second;
if (args.ftdi_channel != -1) {
if (cable.type != MODE_FTDI_SERIAL && cable.type != MODE_FTDI_BITBANG){
printError("Error: FTDI channel param is for FTDI cables.");
return EXIT_FAILURE;
}
int mapping[] = {INTERFACE_A, INTERFACE_B, INTERFACE_C, INTERFACE_D};
cable.config.interface = mapping[args.ftdi_channel];
}
2020-10-07 07:37:40 +02:00
if (!args.ftdi_serial.empty()) {
2020-09-29 14:16:30 +02:00
if (cable.type != MODE_FTDI_SERIAL && cable.type != MODE_FTDI_BITBANG){
printError("Error: FTDI serial param is for FTDI cables.");
return EXIT_FAILURE;
}
}
2020-10-07 07:47:46 +02:00
/* FLASH direct access */
if (args.spi || (board && board->mode == COMM_SPI)) {
2020-10-07 07:47:46 +02:00
FtdiSpi *spi = NULL;
spi_pins_conf_t pins_config = board->spi_pins_config;
2020-10-07 07:47:46 +02:00
try {
spi = new FtdiSpi(cable.config, pins_config, freq, args.verbose > 0);
2020-10-07 07:47:46 +02:00
} catch (std::exception &e) {
printError("Error: Failed to claim cable");
return EXIT_FAILURE;
}
if (board->manufacturer == "efinix") {
Efinix target(spi, args.bit_file, args.file_type,
board->reset_pin, board->done_pin, args.verify, args.verbose);
if (args.prg_type == Device::RD_FLASH) {
if (args.file_size == 0) {
printError("Error: 0 size for dump");
} else {
target.dumpFlash(args.bit_file, args.offset, args.file_size);
}
} else {
target.program(args.offset);
}
} else if (board->manufacturer == "lattice") {
Ice40 target(spi, args.bit_file, args.file_type,
board->reset_pin, board->done_pin, args.verify, args.verbose);
2021-06-26 08:43:02 +02:00
if (args.prg_type == Device::RD_FLASH) {
if (args.file_size == 0) {
printError("Error: 0 size for dump");
} else {
target.dumpFlash(args.bit_file, args.offset, args.file_size);
}
} else {
target.program(args.offset);
}
} else {
RawParser *bit = NULL;
if (board->reset_pin) {
spi->gpio_set_output(board->reset_pin, true);
spi->gpio_clear(board->reset_pin, true);
2020-10-07 07:47:46 +02:00
}
SPIFlash flash((SPIInterface *)spi, args.verbose);
flash.power_up();
flash.reset();
flash.read_id();
if (args.prg_type != Device::RD_FLASH &&
(!args.bit_file.empty() || !args.file_type.empty())) {
printInfo("Open file " + args.bit_file + " ", false);
try {
bit = new RawParser(args.bit_file, false);
printSuccess("DONE");
} catch (std::exception &e) {
printError("FAIL");
delete spi;
return EXIT_FAILURE;
}
printInfo("Parse file ", false);
if (bit->parse() == EXIT_FAILURE) {
printError("FAIL");
delete spi;
return EXIT_FAILURE;
} else {
printSuccess("DONE");
}
flash.erase_and_prog(args.offset, bit->getData(), bit->getLength()/8);
if (args.verify)
flash.verify(args.offset, bit->getData(), bit->getLength() / 8);
delete bit;
} else if (args.prg_type == Device::RD_FLASH) {
flash.dump(args.bit_file, args.offset, args.file_size);
2020-10-07 07:47:46 +02:00
}
if (board->reset_pin)
spi->gpio_set(board->reset_pin, true);
2020-10-07 07:47:46 +02:00
}
delete spi;
return EXIT_SUCCESS;
}
/* ------------------- */
/* DFU access */
/* ------------------- */
if (args.dfu || (board && board->mode == COMM_DFU)) {
/* try to init DFU probe */
DFU *dfu = NULL;
try {
dfu = new DFU(args.bit_file, args.verbose);
} catch (std::exception &e) {
printError("DFU init failed with: " + string(e.what()));
return EXIT_FAILURE;
}
/* if verbose or detect: display device */
if (args.verbose > 0 || args.detect)
dfu->displayDFU();
/* if detect: stop */
if (args.detect)
return EXIT_SUCCESS;
try {
dfu->download();
} catch (std::exception &e) {
printError("DFU download failed with: " + string(e.what()));
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
2019-09-26 18:29:20 +02:00
/* jtag base */
Jtag *jtag;
2020-07-11 07:23:24 +02:00
try {
2021-05-13 16:07:40 +02:00
jtag = new Jtag(cable, &pins_config, args.device, args.ftdi_serial,
freq, false, args.probe_firmware);
2020-07-11 07:23:24 +02:00
} catch (std::exception &e) {
printError("JTAG init failed with: " + string(e.what()));
2020-07-11 07:23:24 +02:00
return EXIT_FAILURE;
}
2019-09-26 18:29:20 +02:00
/* chain detection */
vector<int> listDev = jtag->get_devices_list();
int found = listDev.size();
int idcode = -1, index = 0;
if (args.verbose > 0)
cout << "found " << std::to_string(found) << " devices" << endl;
/* in verbose mode or when detect
* display full chain with details
*/
if (args.verbose > 0 || args.detect) {
for (int i = 0; i < found; i++) {
int t = listDev[i];
printf("index %d:\n", i);
if (fpga_list.find(t) != fpga_list.end()) {
printf("\tidcode 0x%x\n\tmanufacturer %s\n\tmodel %s\n\tfamily %s\n",
t,
fpga_list[t].manufacturer.c_str(),
fpga_list[t].model.c_str(),
fpga_list[t].family.c_str());
printf("\tirlength %d\n", fpga_list[t].irlength);
} else if (misc_dev_list.find(t) != misc_dev_list.end()) {
printf("\tidcode 0x%x\n\ttype %s\n\tirlength %d\n",
t,
misc_dev_list[t].name.c_str(),
misc_dev_list[t].irlength);
}
}
if (args.detect == true) {
delete jtag;
return EXIT_SUCCESS;
}
}
if (found != 0) {
if (args.index_chain == -1) {
for (int i = 0; i < found; i++) {
if (fpga_list.find(listDev[i]) != fpga_list.end()) {
index = i;
if (idcode != -1) {
printError("Error: more than one FPGA found");
printError("Use --index-chain to force selection");
for (int i = 0; i < found; i++)
printf("0x%08x\n", listDev[i]);
delete(jtag);
return EXIT_FAILURE;
} else {
idcode = listDev[i];
}
}
}
} else {
index = args.index_chain;
if (index > found || index < 0) {
printError("wrong index for device in JTAG chain");
delete(jtag);
return EXIT_FAILURE;
}
idcode = listDev[index];
}
} else {
2020-07-11 07:47:27 +02:00
printError("Error: no device found");
2020-08-19 15:15:37 +02:00
delete(jtag);
2019-09-26 18:29:20 +02:00
return EXIT_FAILURE;
}
jtag->device_select(index);
/* check if selected device is supported
* mainly used in conjunction with --index-chain
*/
2019-09-26 18:29:20 +02:00
if (fpga_list.find(idcode) == fpga_list.end()) {
2019-12-06 07:29:28 +01:00
cerr << "Error: device " << hex << idcode << " not supported" << endl;
delete(jtag);
2020-07-11 07:47:27 +02:00
return EXIT_FAILURE;
}
2019-12-06 19:59:35 +01:00
string fab = fpga_list[idcode].manufacturer;
2019-09-26 18:29:20 +02:00
Device *fpga;
try {
if (fab == "xilinx") {
fpga = new Xilinx(jtag, args.bit_file, args.file_type,
args.prg_type, args.fpga_part, args.verify, args.verbose);
} else if (fab == "altera") {
fpga = new Altera(jtag, args.bit_file, args.file_type,
args.prg_type, args.fpga_part, args.verify, args.verbose);
} else if (fab == "anlogic") {
fpga = new Anlogic(jtag, args.bit_file, args.file_type,
args.prg_type, args.verify, args.verbose);
} else if (fab == "Gowin") {
fpga = new Gowin(jtag, args.bit_file, args.file_type,
args.prg_type, args.verify, args.verbose);
} else if (fab == "lattice") {
fpga = new Lattice(jtag, args.bit_file, args.file_type,
args.prg_type, args.verify, args.verbose);
} else {
printError("Error: manufacturer " + fab + " not supported");
delete(jtag);
return EXIT_FAILURE;
}
} catch (std::exception &e) {
printError("Error: Failed to claim FPGA device: " + string(e.what()));
delete(jtag);
return EXIT_FAILURE;
2019-09-26 18:29:20 +02:00
}
if ((!args.bit_file.empty() || !args.file_type.empty())
&& args.prg_type != Device::RD_FLASH) {
try {
fpga->program(args.offset);
} catch (std::exception &e) {
delete(fpga);
delete(jtag);
return EXIT_FAILURE;
}
}
if (args.prg_type == Device::RD_FLASH) {
if (args.file_size == 0) {
printError("Error: 0 size for dump");
} else {
fpga->dumpFlash(args.bit_file, args.offset, args.file_size);
}
}
if (args.reset)
fpga->reset();
delete(fpga);
2019-11-19 09:03:09 +01:00
delete(jtag);
2019-09-26 18:29:20 +02:00
}
// parse double from string in engineering notation
2020-05-25 10:31:37 +02:00
// can deal with postfixes k and m, add more when required
static int parse_eng(string arg, double *dst) {
2020-05-25 10:31:37 +02:00
try {
size_t end;
double base = stod(arg, &end);
if (end == arg.size()) {
*dst = base;
return 0;
} else if (end == (arg.size() - 1)) {
switch (arg.back()) {
case 'k': case 'K':
*dst = (uint32_t)(1e3 * base);
return 0;
case 'm': case 'M':
*dst = (uint32_t)(1e6 * base);
return 0;
default:
return EINVAL;
}
} else {
return EINVAL;
}
} catch (...) {
cerr << "error : speed: invalid format" << endl;
2020-05-25 10:31:37 +02:00
return EINVAL;
}
}
2019-09-26 18:29:20 +02:00
/* arguments parser */
int parse_opt(int argc, char **argv, struct arguments *args, jtag_pins_conf_t *pins_config)
2019-09-26 18:29:20 +02:00
{
2020-07-23 18:32:18 +02:00
string freqo;
vector<string> pins;
bool verbose, quiet;
2020-07-23 18:32:18 +02:00
try {
cxxopts::Options options(argv[0], "openFPGALoader -- a program to flash FPGA",
"<gwenhael.goavec-merou@trabucayre.com>");
options
.positional_help("BIT_FILE")
.show_positional_help();
options
.add_options()
("bitstream", "bitstream",
cxxopts::value<std::string>(args->bit_file))
("b,board", "board name, may be used instead of cable",
cxxopts::value<string>(args->board))
("c,cable", "jtag interface", cxxopts::value<string>(args->cable))
2020-09-29 14:16:30 +02:00
("ftdi-serial", "FTDI chip serial number", cxxopts::value<string>(args->ftdi_serial))
("ftdi-channel", "FTDI chip channel number (channels 0-3 map to A-D)", cxxopts::value<int>(args->ftdi_channel))
#ifdef USE_UDEV
2020-07-23 18:32:18 +02:00
("d,device", "device to use (/dev/ttyUSBx)",
cxxopts::value<string>(args->device))
#endif
2020-07-23 18:32:18 +02:00
("detect", "detect FPGA",
cxxopts::value<bool>(args->detect))
("dfu", "DFU mode", cxxopts::value<bool>(args->dfu))
("dump-flash", "Dump flash mode")
("file-size", "provides size in Byte to dump, must be used with dump-flash",
cxxopts::value<unsigned int>(args->file_size))
("file-type", "provides file type instead of let's deduced by using extension",
cxxopts::value<string>(args->file_type))
("fpga-part", "fpga model flavor + package", cxxopts::value<string>(args->fpga_part))
2020-07-23 18:32:18 +02:00
("freq", "jtag frequency (Hz)", cxxopts::value<string>(freqo))
("f,write-flash",
"write bitstream in flash (default: false, only for Gowin and ECP5 devices)")
("index-chain", "device index in JTAG-chain",
cxxopts::value<int>(args->index_chain))
2020-07-23 18:32:18 +02:00
("list-boards", "list all supported boards",
cxxopts::value<bool>(args->list_boards))
("list-cables", "list all supported cables",
cxxopts::value<bool>(args->list_cables))
("list-fpga", "list all supported FPGA",
cxxopts::value<bool>(args->list_fpga))
("m,write-sram",
"write bitstream in SRAM (default: true, only for Gowin and ECP5 devices)")
("o,offset", "start offset in EEPROM",
cxxopts::value<unsigned int>(args->offset))
("pins", "pin config (only for ft232R) TDI:TDO:TCK:TMS",
cxxopts::value<vector<string>>(pins))
2021-05-13 16:07:40 +02:00
("probe-firmware", "firmware for JTAG probe (usbBlasterII)",
cxxopts::value<string>(args->probe_firmware))
("quiet", "Produce quiet output (no progress bar)",
cxxopts::value<bool>(quiet))
2020-07-23 18:32:18 +02:00
("r,reset", "reset FPGA after operations",
cxxopts::value<bool>(args->reset))
2020-10-06 08:38:24 +02:00
("spi", "SPI mode (only for FTDI in serial mode)",
cxxopts::value<bool>(args->spi))
("v,verbose", "Produce verbose output", cxxopts::value<bool>(verbose))
2020-07-23 18:32:18 +02:00
("h,help", "Give this help list")
("verify", "Verify write operation (SPI Flash only)",
cxxopts::value<bool>(args->verify))
2020-07-23 18:32:18 +02:00
("V,Version", "Print program version");
options.parse_positional({"bitstream"});
auto result = options.parse(argc, argv);
if (result.count("help")) {
cout << options.help() << endl;
return 1;
2020-05-25 10:31:37 +02:00
}
2020-07-23 18:32:18 +02:00
if (verbose && quiet) {
printError("Error: can't select quiet and verbose mode in same time");
throw std::exception();
}
if (verbose)
args->verbose = 1;
if (quiet)
args->verbose = -1;
2020-07-23 18:32:18 +02:00
if (result.count("Version")) {
2020-12-17 13:58:30 +01:00
cout << "openFPGALoader " << VERSION << endl;
2020-07-23 18:32:18 +02:00
return 1;
2020-05-25 10:31:37 +02:00
}
2020-07-23 18:32:18 +02:00
if (result.count("write-flash") && result.count("write-sram") &&
result.count("dump-flash")) {
2020-07-23 18:32:18 +02:00
printError("Error: both write to flash and write to ram enabled");
throw std::exception();
}
2020-07-23 18:32:18 +02:00
2021-02-18 21:09:34 +01:00
if (result.count("write-flash"))
args->prg_type = Device::WR_FLASH;
else if (result.count("write-sram"))
args->prg_type = Device::WR_SRAM;
else if (result.count("dump-flash"))
args->prg_type = Device::RD_FLASH;
2020-07-23 18:32:18 +02:00
if (result.count("freq")) {
double freq;
if (parse_eng(freqo, &freq)) {
printError("Error: invalid format for --freq");
throw std::exception();
}
if (freq < 1) {
printError("Error: --freq must be positive");
throw std::exception();
}
args->freq = static_cast<uint32_t>(freq);
}
if (result.count("ftdi-channel")) {
if (args->ftdi_channel < 0 || args->ftdi_channel > 3) {
printError("Error: valid FTDI channels are 0-3.");
throw std::exception();
}
}
if (result.count("pins")) {
if (pins.size() != 4) {
printError("Error: pin_config need 4 pins");
throw std::exception();
}
static std::map <std::string, int> pins_list = {
{"TXD", FT232RL_TXD},
{"RXD", FT232RL_RXD},
{"RTS", FT232RL_RTS},
{"CTS", FT232RL_CTS},
{"DTR", FT232RL_DTR},
{"DSR", FT232RL_DSR},
{"DCD", FT232RL_DCD},
{"RI" , FT232RL_RI }};
for (int i = 0; i < 4; i++) {
int pin_num;
try {
pin_num = 1 << std::stoi(pins[i], nullptr, 0);
} catch (std::exception &e) {
if (pins_list.find(pins[i]) == pins_list.end()) {
printError("Invalid pin name");
throw std::exception();
}
pin_num = pins_list[pins[i]];
}
if (pin_num > FT232RL_RI || pin_num < FT232RL_TXD) {
printError("Invalid pin ID");
throw std::exception();
}
switch (i) {
case 0:
pins_config->tdi_pin = pin_num;
break;
case 1:
pins_config->tdo_pin = pin_num;
break;
case 2:
pins_config->tck_pin = pin_num;
break;
case 3:
pins_config->tms_pin = pin_num;
break;
}
}
args->pin_config = true;
}
2020-07-23 18:32:18 +02:00
if (args->list_cables || args->list_boards || args->list_fpga)
args->is_list_command = true;
if (args->bit_file.empty() &&
args->file_type.empty() &&
2020-07-23 18:32:18 +02:00
!args->is_list_command &&
!args->detect &&
!args->reset) {
printError("Error: bitfile not specified");
cout << options.help() << endl;
throw std::exception();
}
} catch (const cxxopts::OptionException& e) {
cerr << "Error parsing options: " << e.what() << endl;
throw std::exception();
2019-09-26 18:29:20 +02:00
}
2020-07-23 18:32:18 +02:00
2019-09-26 18:29:20 +02:00
return 0;
}
/* display list of cables, boards and devices supported */
void displaySupported(const struct arguments &args)
{
if (args.list_cables == true) {
stringstream t;
t << setw(15) << left << "cable name:" << "vid:pid";
printSuccess(t.str());
for (auto b = cable_list.begin(); b != cable_list.end(); b++) {
FTDIpp_MPSSE::mpsse_bit_config c = (*b).second.config;
stringstream ss;
ss << setw(15) << left << (*b).first;
ss << "0x" << hex << c.vid << ":" << c.pid;
printInfo(ss.str());
}
cout << endl;
}
if (args.list_boards) {
stringstream t;
t << setw(15) << left << "board name:" << "cable_name";
printSuccess(t.str());
for (auto b = board_list.begin(); b != board_list.end(); b++) {
stringstream ss;
target_board_t c = (*b).second;
ss << setw(15) << left << (*b).first << " " << c.cable_name;
printInfo(ss.str());
}
cout << endl;
}
if (args.list_fpga) {
stringstream t;
t << setw(12) << left << "IDCode" << setw(14) << "manufacturer";
t << setw(15) << "family" << setw(20) << "model";
printSuccess(t.str());
for (auto b = fpga_list.begin(); b != fpga_list.end(); b++) {
fpga_model fpga = (*b).second;
stringstream ss, idCode;
idCode << "0x" << hex << (*b).first;
ss << setw(12) << left << idCode.str();
ss << setw(14) << fpga.manufacturer << setw(15) << fpga.family;
ss << setw(20) << fpga.model;
printInfo(ss.str());
}
cout << endl;
}
}