[FPGA-Verilog] Upgraded testbench generator to support memory bank using flatten BL/WLs
This commit is contained in:
parent
386812777c
commit
a56d1f4fdb
|
@ -229,6 +229,8 @@ void print_verilog_top_testbench_config_protocol_port(std::fstream& fp,
|
|||
print_verilog_top_testbench_config_chain_port(fp, module_manager, top_module);
|
||||
break;
|
||||
case CONFIG_MEM_QL_MEMORY_BANK:
|
||||
print_verilog_top_testbench_ql_memory_bank_port(fp, module_manager, top_module, config_protocol);
|
||||
break;
|
||||
case CONFIG_MEM_MEMORY_BANK:
|
||||
print_verilog_top_testbench_memory_bank_port(fp, module_manager, top_module);
|
||||
break;
|
||||
|
@ -1111,7 +1113,10 @@ void print_verilog_top_testbench_configuration_protocol_stimulus(std::fstream& f
|
|||
case CONFIG_MEM_FRAME_BASED: {
|
||||
ModulePortId en_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_ENABLE_PORT_NAME));
|
||||
BasicPort en_port = module_manager.module_port(top_module, en_port_id);
|
||||
BasicPort en_port(std::string(DECODER_ENABLE_PORT_NAME), 1);
|
||||
if (en_port_id) {
|
||||
en_port = module_manager.module_port(top_module, en_port_id);
|
||||
}
|
||||
BasicPort en_register_port(std::string(en_port.get_name() + std::string(TOP_TB_CLOCK_REG_POSTFIX)), 1);
|
||||
print_verilog_comment(fp, std::string("---- Generate enable signal waveform -----"));
|
||||
print_verilog_shifted_clock_stimuli(fp, en_register_port,
|
||||
|
@ -1684,7 +1689,7 @@ void print_verilog_full_testbench_frame_decoder_bitstream(std::fstream& fp,
|
|||
static
|
||||
void print_verilog_full_testbench_bitstream(std::fstream& fp,
|
||||
const std::string& bitstream_file,
|
||||
const e_config_protocol_type& config_protocol_type,
|
||||
const ConfigProtocol& config_protocol,
|
||||
const bool& fast_configuration,
|
||||
const bool& bit_value_to_skip,
|
||||
const ModuleManager& module_manager,
|
||||
|
@ -1693,7 +1698,7 @@ void print_verilog_full_testbench_bitstream(std::fstream& fp,
|
|||
const FabricBitstream& fabric_bitstream) {
|
||||
|
||||
/* Branch on the type of configuration protocol */
|
||||
switch (config_protocol_type) {
|
||||
switch (config_protocol.type()) {
|
||||
case CONFIG_MEM_STANDALONE:
|
||||
print_verilog_full_testbench_vanilla_bitstream(fp,
|
||||
bitstream_file,
|
||||
|
@ -1719,6 +1724,7 @@ void print_verilog_full_testbench_bitstream(std::fstream& fp,
|
|||
break;
|
||||
case CONFIG_MEM_QL_MEMORY_BANK:
|
||||
print_verilog_full_testbench_ql_memory_bank_bitstream(fp, bitstream_file,
|
||||
config_protocol,
|
||||
fast_configuration,
|
||||
bit_value_to_skip,
|
||||
module_manager, top_module,
|
||||
|
@ -1998,7 +2004,7 @@ int print_verilog_full_testbench(const ModuleManager& module_manager,
|
|||
/* load bitstream to FPGA fabric in a configuration phase */
|
||||
print_verilog_full_testbench_bitstream(fp,
|
||||
bitstream_file,
|
||||
config_protocol.type(),
|
||||
config_protocol,
|
||||
apply_fast_configuration,
|
||||
bit_value_to_skip,
|
||||
module_manager, top_module,
|
||||
|
|
|
@ -36,7 +36,258 @@
|
|||
/* begin namespace openfpga */
|
||||
namespace openfpga {
|
||||
|
||||
void print_verilog_full_testbench_ql_memory_bank_bitstream(std::fstream& fp,
|
||||
void print_verilog_top_testbench_ql_memory_bank_port(std::fstream& fp,
|
||||
const ModuleManager& module_manager,
|
||||
const ModuleId& top_module,
|
||||
const ConfigProtocol& config_protocol) {
|
||||
/* Validate the file stream */
|
||||
valid_file_stream(fp);
|
||||
|
||||
/* Print the address port for the Bit-Line decoder here */
|
||||
if (BLWL_PROTOCOL_DECODER == config_protocol.bl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Address port for Bit-Line decoder -----"));
|
||||
ModulePortId bl_addr_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_BL_ADDRESS_PORT_NAME));
|
||||
BasicPort bl_addr_port = module_manager.module_port(top_module, bl_addr_port_id);
|
||||
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, bl_addr_port) << ";" << std::endl;
|
||||
} else if (BLWL_PROTOCOL_FLATTEN == config_protocol.bl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Bit-Line ports -----"));
|
||||
for (const ConfigRegionId& region : module_manager.regions(top_module)) {
|
||||
ModulePortId bl_port_id = module_manager.find_module_port(top_module,
|
||||
generate_regional_blwl_port_name(std::string(MEMORY_BL_PORT_NAME), region));
|
||||
BasicPort bl_port = module_manager.module_port(top_module, bl_port_id);
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, bl_port) << ";" << std::endl;
|
||||
}
|
||||
} else {
|
||||
VTR_ASSERT(BLWL_PROTOCOL_SHIFT_REGISTER == config_protocol.bl_protocol_type());
|
||||
/* TODO */
|
||||
}
|
||||
|
||||
/* Print the address port for the Word-Line decoder here */
|
||||
if (BLWL_PROTOCOL_DECODER == config_protocol.wl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Address port for Word-Line decoder -----"));
|
||||
ModulePortId wl_addr_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_WL_ADDRESS_PORT_NAME));
|
||||
BasicPort wl_addr_port = module_manager.module_port(top_module, wl_addr_port_id);
|
||||
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, wl_addr_port) << ";" << std::endl;
|
||||
} else if (BLWL_PROTOCOL_FLATTEN == config_protocol.wl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Word-Line ports -----"));
|
||||
for (const ConfigRegionId& region : module_manager.regions(top_module)) {
|
||||
ModulePortId wl_port_id = module_manager.find_module_port(top_module,
|
||||
generate_regional_blwl_port_name(std::string(MEMORY_WL_PORT_NAME), region));
|
||||
BasicPort wl_port = module_manager.module_port(top_module, wl_port_id);
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, wl_port) << ";" << std::endl;
|
||||
}
|
||||
} else {
|
||||
VTR_ASSERT(BLWL_PROTOCOL_SHIFT_REGISTER == config_protocol.wl_protocol_type());
|
||||
/* TODO */
|
||||
}
|
||||
|
||||
/* Print the data-input port: only available when BL has a decoder */
|
||||
if (BLWL_PROTOCOL_DECODER == config_protocol.bl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Data input port for memory decoders -----"));
|
||||
ModulePortId din_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_DATA_IN_PORT_NAME));
|
||||
BasicPort din_port = module_manager.module_port(top_module, din_port_id);
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, din_port) << ";" << std::endl;
|
||||
}
|
||||
|
||||
/* Print the optional readback port for the decoder here */
|
||||
if (BLWL_PROTOCOL_DECODER == config_protocol.wl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Readback port for memory decoders -----"));
|
||||
ModulePortId readback_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_READBACK_PORT_NAME));
|
||||
if (readback_port_id) {
|
||||
BasicPort readback_port = module_manager.module_port(top_module, readback_port_id);
|
||||
fp << generate_verilog_port(VERILOG_PORT_WIRE, readback_port) << ";" << std::endl;
|
||||
/* Disable readback in full testbenches */
|
||||
print_verilog_wire_constant_values(fp, readback_port, std::vector<size_t>(readback_port.get_width(), 0));
|
||||
}
|
||||
} else if (BLWL_PROTOCOL_FLATTEN == config_protocol.wl_protocol_type()) {
|
||||
print_verilog_comment(fp, std::string("---- Word line read ports -----"));
|
||||
for (const ConfigRegionId& region : module_manager.regions(top_module)) {
|
||||
ModulePortId wlr_port_id = module_manager.find_module_port(top_module,
|
||||
generate_regional_blwl_port_name(std::string(MEMORY_WLR_PORT_NAME), region));
|
||||
if (wlr_port_id) {
|
||||
BasicPort wlr_port = module_manager.module_port(top_module, wlr_port_id);
|
||||
fp << generate_verilog_port(VERILOG_PORT_WIRE, wlr_port) << ";" << std::endl;
|
||||
/* Disable readback in full testbenches */
|
||||
print_verilog_wire_constant_values(fp, wlr_port, std::vector<size_t>(wlr_port.get_width(), 0));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* Generate enable signal waveform here:
|
||||
* which is a 90 degree phase shift than the programming clock
|
||||
*/
|
||||
print_verilog_comment(fp, std::string("---- Wire enable port of memory decoders -----"));
|
||||
ModulePortId en_port_id = module_manager.find_module_port(top_module,
|
||||
std::string(DECODER_ENABLE_PORT_NAME));
|
||||
BasicPort en_port(std::string(DECODER_ENABLE_PORT_NAME), 1);
|
||||
if (en_port_id) {
|
||||
en_port = module_manager.module_port(top_module, en_port_id);
|
||||
}
|
||||
BasicPort en_register_port(std::string(en_port.get_name() + std::string(TOP_TB_CLOCK_REG_POSTFIX)), 1);
|
||||
|
||||
BasicPort config_done_port(std::string(TOP_TB_CONFIG_DONE_PORT_NAME), 1);
|
||||
|
||||
fp << generate_verilog_port(VERILOG_PORT_WIRE, en_port) << ";" << std::endl;
|
||||
fp << generate_verilog_port(VERILOG_PORT_REG, en_register_port) << ";" << std::endl;
|
||||
|
||||
write_tab_to_file(fp, 1);
|
||||
fp << "assign ";
|
||||
fp << generate_verilog_port(VERILOG_PORT_CONKT, en_port);
|
||||
fp << "= ";
|
||||
fp << "~" << generate_verilog_port(VERILOG_PORT_CONKT, en_register_port);
|
||||
fp << " & ";
|
||||
fp << "~" << generate_verilog_port(VERILOG_PORT_CONKT, config_done_port);
|
||||
fp << ";" << std::endl;
|
||||
}
|
||||
|
||||
/* Verilog codes to load bitstream from a bit file for memory bank using flatten BL/WLs */
|
||||
static
|
||||
void print_verilog_full_testbench_ql_memory_bank_flatten_bitstream(std::fstream& fp,
|
||||
const std::string& bitstream_file,
|
||||
const bool& fast_configuration,
|
||||
const bool& bit_value_to_skip,
|
||||
const ModuleManager& module_manager,
|
||||
const ModuleId& top_module,
|
||||
const FabricBitstream& fabric_bitstream) {
|
||||
/* Validate the file stream */
|
||||
valid_file_stream(fp);
|
||||
|
||||
/* No fast configuration available in this configuration protocol. Give a warning */
|
||||
if (true == fast_configuration) {
|
||||
VTR_LOG_WARN("Fast configuration is not available for flatten BL protocol");
|
||||
}
|
||||
|
||||
/* Reorganize the fabric bitstream by the same address across regions */
|
||||
MemoryBankFlattenFabricBitstream fabric_bits_by_addr = build_memory_bank_flatten_fabric_bitstream(fabric_bitstream,
|
||||
bit_value_to_skip);
|
||||
|
||||
/* Feed address and data input pair one by one
|
||||
* Note: the first cycle is reserved for programming reset
|
||||
* We should give dummy values
|
||||
*/
|
||||
std::vector<BasicPort> bl_ports;
|
||||
for (const ConfigRegionId& region : module_manager.regions(top_module)) {
|
||||
ModulePortId cur_bl_port_id = module_manager.find_module_port(top_module,
|
||||
generate_regional_blwl_port_name(std::string(MEMORY_BL_PORT_NAME), region));
|
||||
bl_ports.push_back(module_manager.module_port(top_module, cur_bl_port_id));
|
||||
}
|
||||
|
||||
std::vector<BasicPort> wl_ports;
|
||||
for (const ConfigRegionId& region : module_manager.regions(top_module)) {
|
||||
ModulePortId cur_wl_port_id = module_manager.find_module_port(top_module,
|
||||
generate_regional_blwl_port_name(std::string(MEMORY_WL_PORT_NAME), region));
|
||||
wl_ports.push_back(module_manager.module_port(top_module, cur_wl_port_id));
|
||||
}
|
||||
|
||||
/* Calculate the total size of BL/WL ports */
|
||||
size_t bl_port_width = 0;
|
||||
for (const BasicPort& bl_port : bl_ports) {
|
||||
bl_port_width += bl_port.get_width();
|
||||
}
|
||||
size_t wl_port_width = 0;
|
||||
for (const BasicPort& wl_port : wl_ports) {
|
||||
wl_port_width += wl_port.get_width();
|
||||
}
|
||||
|
||||
std::vector<size_t> initial_bl_values(bl_port_width, 0);
|
||||
std::vector<size_t> initial_wl_values(wl_port_width, 0);
|
||||
|
||||
/* Define a constant for the bitstream length */
|
||||
print_verilog_define_flag(fp, std::string(TOP_TB_BITSTREAM_LENGTH_VARIABLE), fabric_bits_by_addr.size());
|
||||
print_verilog_define_flag(fp, std::string(TOP_TB_BITSTREAM_WIDTH_VARIABLE), bl_port_width + wl_port_width);
|
||||
|
||||
/* Declare local variables for bitstream loading in Verilog */
|
||||
print_verilog_comment(fp, "----- Virtual memory to store the bitstream from external file -----");
|
||||
fp << "reg [0:`" << TOP_TB_BITSTREAM_WIDTH_VARIABLE << " - 1] ";
|
||||
fp << TOP_TB_BITSTREAM_MEM_REG_NAME << "[0:`" << TOP_TB_BITSTREAM_LENGTH_VARIABLE << " - 1];";
|
||||
fp << std::endl;
|
||||
|
||||
fp << "reg [$clog2(`" << TOP_TB_BITSTREAM_LENGTH_VARIABLE << "):0] " << TOP_TB_BITSTREAM_INDEX_REG_NAME << ";" << std::endl;
|
||||
|
||||
print_verilog_comment(fp, "----- Preload bitstream file to a virtual memory -----");
|
||||
fp << "initial begin" << std::endl;
|
||||
fp << "\t";
|
||||
fp << "$readmemb(\"" << bitstream_file << "\", " << TOP_TB_BITSTREAM_MEM_REG_NAME << ");";
|
||||
fp << std::endl;
|
||||
|
||||
print_verilog_comment(fp, "----- Bit-Line Address port default input -----");
|
||||
fp << "\t";
|
||||
fp << generate_verilog_ports_constant_values(bl_ports, initial_bl_values);
|
||||
fp << ";";
|
||||
fp << std::endl;
|
||||
|
||||
print_verilog_comment(fp, "----- Word-Line Address port default input -----");
|
||||
fp << "\t";
|
||||
fp << generate_verilog_ports_constant_values(wl_ports, initial_wl_values);
|
||||
fp << ";";
|
||||
fp << std::endl;
|
||||
|
||||
fp << "\t";
|
||||
fp << TOP_TB_BITSTREAM_INDEX_REG_NAME << " <= 0";
|
||||
fp << ";";
|
||||
fp << std::endl;
|
||||
|
||||
fp << "end";
|
||||
fp << std::endl;
|
||||
|
||||
print_verilog_comment(fp, "----- Begin bitstream loading during configuration phase -----");
|
||||
BasicPort prog_clock_port(std::string(TOP_TB_PROG_CLOCK_PORT_NAME) + std::string(TOP_TB_CLOCK_REG_POSTFIX), 1);
|
||||
fp << "always";
|
||||
fp << " @(negedge " << generate_verilog_port(VERILOG_PORT_CONKT, prog_clock_port) << ")";
|
||||
fp << " begin";
|
||||
fp << std::endl;
|
||||
|
||||
fp << "\t";
|
||||
fp << "if (";
|
||||
fp << TOP_TB_BITSTREAM_INDEX_REG_NAME;
|
||||
fp << " >= ";
|
||||
fp << "`" << TOP_TB_BITSTREAM_LENGTH_VARIABLE;
|
||||
fp << ") begin";
|
||||
fp << std::endl;
|
||||
|
||||
BasicPort config_done_port(std::string(TOP_TB_CONFIG_DONE_PORT_NAME), 1);
|
||||
fp << "\t\t";
|
||||
std::vector<size_t> config_done_final_values(config_done_port.get_width(), 1);
|
||||
fp << generate_verilog_port_constant_values(config_done_port, config_done_final_values, true);
|
||||
fp << ";" << std::endl;
|
||||
|
||||
fp << "\t";
|
||||
fp << "end else begin";
|
||||
fp << std::endl;
|
||||
|
||||
std::vector<BasicPort> blwl_ports = bl_ports;
|
||||
blwl_ports.insert(blwl_ports.end(), wl_ports.begin(), wl_ports.end());
|
||||
fp << "\t\t";
|
||||
fp << generate_verilog_ports(blwl_ports);
|
||||
fp << " <= ";
|
||||
fp << TOP_TB_BITSTREAM_MEM_REG_NAME << "[" << TOP_TB_BITSTREAM_INDEX_REG_NAME << "]";
|
||||
fp << ";" << std::endl;
|
||||
|
||||
fp << "\t\t";
|
||||
fp << TOP_TB_BITSTREAM_INDEX_REG_NAME;
|
||||
fp << " <= ";
|
||||
fp << TOP_TB_BITSTREAM_INDEX_REG_NAME << " + 1";
|
||||
fp << ";" << std::endl;
|
||||
|
||||
fp << "\t";
|
||||
fp << "end";
|
||||
fp << std::endl;
|
||||
|
||||
fp << "end";
|
||||
fp << std::endl;
|
||||
|
||||
print_verilog_comment(fp, "----- End bitstream loading during configuration phase -----");
|
||||
}
|
||||
|
||||
/* Verilog codes to load bitstream from a bit file for memory bank using BL/WL decoders */
|
||||
static
|
||||
void print_verilog_full_testbench_ql_memory_bank_decoder_bitstream(std::fstream& fp,
|
||||
const std::string& bitstream_file,
|
||||
const bool& fast_configuration,
|
||||
const bool& bit_value_to_skip,
|
||||
|
@ -172,4 +423,29 @@ void print_verilog_full_testbench_ql_memory_bank_bitstream(std::fstream& fp,
|
|||
print_verilog_comment(fp, "----- End bitstream loading during configuration phase -----");
|
||||
}
|
||||
|
||||
void print_verilog_full_testbench_ql_memory_bank_bitstream(std::fstream& fp,
|
||||
const std::string& bitstream_file,
|
||||
const ConfigProtocol& config_protocol,
|
||||
const bool& fast_configuration,
|
||||
const bool& bit_value_to_skip,
|
||||
const ModuleManager& module_manager,
|
||||
const ModuleId& top_module,
|
||||
const FabricBitstream& fabric_bitstream) {
|
||||
if ( (BLWL_PROTOCOL_DECODER == config_protocol.bl_protocol_type())
|
||||
&& (BLWL_PROTOCOL_DECODER == config_protocol.wl_protocol_type()) ) {
|
||||
print_verilog_full_testbench_ql_memory_bank_decoder_bitstream(fp, bitstream_file,
|
||||
fast_configuration,
|
||||
bit_value_to_skip,
|
||||
module_manager, top_module,
|
||||
fabric_bitstream);
|
||||
} else if ( (BLWL_PROTOCOL_FLATTEN == config_protocol.bl_protocol_type())
|
||||
&& (BLWL_PROTOCOL_FLATTEN == config_protocol.wl_protocol_type()) ) {
|
||||
print_verilog_full_testbench_ql_memory_bank_flatten_bitstream(fp, bitstream_file,
|
||||
fast_configuration,
|
||||
bit_value_to_skip,
|
||||
module_manager, top_module,
|
||||
fabric_bitstream);
|
||||
}
|
||||
}
|
||||
|
||||
} /* end namespace openfpga */
|
||||
|
|
|
@ -26,12 +26,21 @@
|
|||
/* begin namespace openfpga */
|
||||
namespace openfpga {
|
||||
|
||||
/**
|
||||
* @brief Print local wires for memory bank configuration protocols
|
||||
*/
|
||||
void print_verilog_top_testbench_ql_memory_bank_port(std::fstream& fp,
|
||||
const ModuleManager& module_manager,
|
||||
const ModuleId& top_module,
|
||||
const ConfigProtocol& config_protocol);
|
||||
|
||||
/**
|
||||
* @brief Print stimulus for a FPGA fabric with a memory bank configuration protocol
|
||||
* where configuration bits are programming in serial (one by one)
|
||||
*/
|
||||
void print_verilog_full_testbench_ql_memory_bank_bitstream(std::fstream& fp,
|
||||
const std::string& bitstream_file,
|
||||
const ConfigProtocol& config_protocol,
|
||||
const bool& fast_configuration,
|
||||
const bool& bit_value_to_skip,
|
||||
const ModuleManager& module_manager,
|
||||
|
|
|
@ -721,7 +721,7 @@ std::string generate_verilog_constant_values(const std::vector<size_t>& const_va
|
|||
}
|
||||
|
||||
/********************************************************************
|
||||
* Generate a verilog port with a deposite of constant values
|
||||
* Generate a verilog port with a deposit of constant values
|
||||
********************************************************************/
|
||||
std::string generate_verilog_port_constant_values(const BasicPort& output_port,
|
||||
const std::vector<size_t>& const_values,
|
||||
|
@ -742,6 +742,32 @@ std::string generate_verilog_port_constant_values(const BasicPort& output_port,
|
|||
return port_str;
|
||||
}
|
||||
|
||||
/********************************************************************
|
||||
* Generate a list of verilog ports with a deposit of constant values
|
||||
********************************************************************/
|
||||
std::string generate_verilog_ports_constant_values(const std::vector<BasicPort>& output_ports,
|
||||
const std::vector<size_t>& const_values,
|
||||
const bool& is_register) {
|
||||
std::string port_str;
|
||||
|
||||
/* Must check: the port width matches */
|
||||
size_t total_width = 0;
|
||||
for (const BasicPort& port : output_ports) {
|
||||
total_width += port.get_width();
|
||||
}
|
||||
VTR_ASSERT( const_values.size() == total_width );
|
||||
|
||||
port_str = generate_verilog_ports(output_ports);
|
||||
if (is_register) {
|
||||
port_str += " <= ";
|
||||
} else {
|
||||
VTR_ASSERT_SAFE(!is_register);
|
||||
port_str += " = ";
|
||||
}
|
||||
port_str += generate_verilog_constant_values(const_values);
|
||||
return port_str;
|
||||
}
|
||||
|
||||
/********************************************************************
|
||||
* Generate a wire connection, that assigns constant values to a
|
||||
* Verilog port
|
||||
|
|
|
@ -111,6 +111,10 @@ std::string generate_verilog_port_constant_values(const BasicPort& output_port,
|
|||
const std::vector<size_t>& const_values,
|
||||
const bool& is_register = false);
|
||||
|
||||
std::string generate_verilog_ports_constant_values(const std::vector<BasicPort>& output_ports,
|
||||
const std::vector<size_t>& const_values,
|
||||
const bool& is_register = false);
|
||||
|
||||
void print_verilog_wire_constant_values(std::fstream& fp,
|
||||
const BasicPort& output_port,
|
||||
const std::vector<size_t>& const_values);
|
||||
|
|
Loading…
Reference in New Issue