blob: 786243644638a86e424921b203ef69a2d3cd6b48 [file] [log] [blame]
#include "ChipConfig.hpp"
#include "BitDatabase.hpp"
#include "Bitstream.hpp"
#include "Chip.hpp"
#include "Database.hpp"
#include "DatabasePath.hpp"
#include "Tile.hpp"
#include <iostream>
#include <boost/program_options.hpp>
#include <stdexcept>
#include <streambuf>
#include <fstream>
/* TODO:
* Support golden image properly
* Cleanup logging
* Verify inputs before starting, including that there's no overlap.
* Fill with 0xFF instead of 0x00
* Possibly intel hex file output instead?
* Error checking!
* Allow specifying inputs/addresses in any order and sort them correctly
* and inject golden image at the suitable position
*/
using namespace std;
boost::optional<uint32_t> convert_hexstring(std::string value_str)
{
boost::optional<uint32_t> ret;
uint32_t value = uint32_t(strtoul(value_str.c_str(), nullptr, 0));
if (value != 0)
ret = value;
return ret;
}
int main(int argc, char *argv[])
{
using namespace Trellis;
namespace po = boost::program_options;
std::string database_folder = get_database_path();
uint32_t flash_size_bytes = 0;
boost::optional<uint32_t> input_idcode;
boost::optional<uint32_t> output_idcode;
boost::optional<uint32_t> idcode;
uint32_t nextaddr = 0;
std::array<char, 4096> fillpattern;
std::fill(fillpattern.begin(), fillpattern.end(), 0xff);
po::options_description options("Allowed options");
options.add_options()("help,h", "show help");
options.add_options()("verbose,v", "verbose output");
options.add_options()("db", po::value<std::string>(), "Trellis database folder location");
options.add_options()("input", po::value<std::vector<std::string>>()->required(), "input bitstream file 0..N");
options.add_options()("address", po::value<std::vector<std::string>>()->required(), "address to place next bitstream at [1..N]");
options.add_options()("flashsize", po::value<std::uint32_t>()->required(), "Flash size in Mbits, e.g. 2, 4, 8, ..., 128");
options.add_options()("input-idcode", po::value<std::string>(), "IDCODE override for input file");
options.add_options()("output-idcode", po::value<std::string>(), "IDCODE override in output bitstreams");
po::positional_options_description pos;
options.add_options()("output", po::value<std::string>()->required(), "output bitstream file");
pos.add("output", 1);
po::variables_map vm;
try {
po::parsed_options parsed = po::command_line_parser(argc, argv).options(options).positional(pos).run();
po::store(parsed, vm);
po::notify(vm);
}
catch (std::exception &e) {
cerr << e.what() << endl << endl;
goto help;
}
if (vm.count("help")) {
help:
cerr << "Project Trellis - Open Source Tools for ECP5 FPGAs" << endl;
cerr << "ecpmulti: ECP5 multiboot bitstream assembler" << endl;
cerr << endl;
cerr << "Copyright (C) 2019 Jens Andersen <jens.andersen@gmail.com>" << endl;
cerr << endl;
cerr << options << endl;
return vm.count("help") ? 0 : 1;
}
if (vm.count("flashsize")) {
uint32_t flash_size_mbit = vm["flashsize"].as<uint32_t>();
flash_size_bytes = flash_size_mbit * 1024* 1024 / 8;
cout << "Using flashsize " << flash_size_mbit << "(" << flash_size_bytes << " bytes)" << endl;
}
auto inputs = vm.at("input").as<std::vector<std::string>>();
auto addresses = vm.at("address").as<std::vector<std::string>>();
if (inputs.size() != (addresses.size() + 1)) {
cerr << "Inputs " << inputs.size() << " offsets " << addresses.size() << endl;
cerr << "There must be one address specified per extra bitstream (>1)" << endl;
return 1;
}
if (vm.count("input-idcode")) {
input_idcode = convert_hexstring(vm.at("input-idcode").as<std::string>());
}
if (vm.count("output-idcode")) {
output_idcode = convert_hexstring(vm.at("output-idcode").as<std::string>());
}
if (vm.count("db")) {
database_folder = vm["db"].as<string>();
}
try {
load_database(database_folder);
} catch (runtime_error &e) {
cerr << "Failed to load Trellis database: " << e.what() << endl;
return 1;
}
ofstream out_file(vm["output"].as<string>(), ofstream::trunc);
if (!out_file) {
cerr << "Failed to open output file" << endl;
return 1;
}
const map<string, string> bs_options = {{"multiboot", "yes"} };
for(uint32_t i = 0; i < inputs.size(); i++) {
ifstream bitfile;
string filename = inputs.at(i);
uint32_t address = nextaddr << 16;
if (address > flash_size_bytes || address < out_file.tellp()) {
cerr << "Addresses must be ordered and smaller than flash size" << endl;
return 1;
}
cout << "Processing " << filename << " for address " << hex << address << endl;
bitfile.open(inputs[i], ios::binary);
Bitstream bs = Bitstream::read_bit(bitfile);
Chip c = bs.deserialise_chip(input_idcode);
if (!idcode.is_initialized())
idcode = c.info.idcode;
if (idcode != c.info.idcode) {
cerr << "All input files must be for the same model (" << *idcode << " != " << c.info.idcode << ")" << endl;
return 1;
}
if (i < addresses.size()) {
string offset_str = addresses.at(i);
boost::optional<uint32_t> next_addr_val = convert_hexstring(offset_str);
if (!next_addr_val.is_initialized()) {
cerr << "Invalid offset: " << offset_str << endl;
return 1;
}
nextaddr = ((*next_addr_val) & 0x00FF0000) >> 16;
} else
/* Point back to first bitstream */
nextaddr = 0;
auto tile_db = get_tile_bitdata(TileLocator{c.info.family, c.info.name, "EFB1_PICB1"});
WordSettingBits wsb = tile_db->get_data_for_setword("BOOTADDR");
auto tile = c.get_tiles_by_type("EFB1_PICB1");
if (tile.size() != 1) {
cerr << "EFB1_PICB1 Frame is wrong size. Can't proceed" << endl;
return 1;
}
for(uint32_t j=0; j < wsb.bits.size(); j++) {
auto bg = wsb.bits.at(j);
for (auto bit : bg.bits) {
bool value = (nextaddr & (1 << j)) > 0;
tile[0]->cram.set_bit(bit.frame, bit.bit, value);
}
}
uint32_t fill_size = address - out_file.tellp();
while(fill_size > 0) {
uint32_t batch_size = std::min(fill_size, (uint32_t)4096);
out_file.write(fillpattern.data(), batch_size);
fill_size -= batch_size;
}
/* Pad with 256 byte 0xFF to match Diamond tools.
* TN1216 indicates it has to do with garbage bytes coming out of SPI Flash sometimes */
out_file.write(fillpattern.data(), 256);
if (output_idcode.is_initialized()) {
c.info.idcode = *output_idcode;
}
bs = Bitstream::serialise_chip(c, bs_options);
bs.write_bit(out_file);
}
/* TODO: Support golden image
* 1) Inject golden image into right area of final bitfile
* 2) Use Bitstream::generate_jump(golden_addr) to generate a Bitstream
* 3) At end of file, (flash_size - 0xFF) write jump bitstream
*/
out_file.flush();
out_file.close();
return 0;
}