OpenFPGA/libopenfpga/libarchopenfpga/src/read_xml_circuit_library.cpp

733 lines
35 KiB
C++

/********************************************************************
* This file includes the top-level function of this library
* which reads an XML modeling OpenFPGA architecture to the associated
* data structures
*******************************************************************/
#include <string>
/* Headers from pugi XML library */
#include "pugixml.hpp"
#include "pugixml_util.hpp"
/* Headers from vtr util library */
#include "vtr_assert.h"
/* Headers from libarchfpga */
#include "arch_error.h"
#include "read_xml_util.h"
/* Headers from openfpgautil library */
#include "openfpga_tokenizer.h"
#include "read_xml_circuit_library.h"
/********************************************************************
* Convert string to the enumerate of model type
*******************************************************************/
static
e_circuit_model_type string_to_circuit_model_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_type>(itype);
}
}
/* Reach here, we have an invalid value, error out */
return NUM_CIRCUIT_MODEL_TYPES;
}
/********************************************************************
* Convert string to the enumerate of model type
*******************************************************************/
static
e_circuit_model_design_tech string_to_design_tech_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_DESIGN_TECH_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_DESIGN_TECH_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_design_tech>(itype);
}
}
return NUM_CIRCUIT_MODEL_DESIGN_TECH_TYPES;
}
/********************************************************************
* Convert string to the enumerate of buffer type
*******************************************************************/
static
e_circuit_model_buffer_type string_to_buffer_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_BUF_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_BUFFER_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_buffer_type>(itype);
}
}
return NUM_CIRCUIT_MODEL_BUF_TYPES;
}
/********************************************************************
* Convert string to the enumerate of pass-gate-logic type
*******************************************************************/
static
e_circuit_model_pass_gate_logic_type string_to_passgate_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_PASS_GATE_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_PASSGATE_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_pass_gate_logic_type>(itype);
}
}
return NUM_CIRCUIT_MODEL_PASS_GATE_TYPES;
}
/********************************************************************
* Convert string to the enumerate of multiplexer structure
*******************************************************************/
static
e_circuit_model_structure string_to_mux_structure_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_STRUCTURE_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_STRUCTURE_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_structure>(itype);
}
}
return NUM_CIRCUIT_MODEL_STRUCTURE_TYPES;
}
/********************************************************************
* Convert string to the enumerate of logic gate type
*******************************************************************/
static
e_circuit_model_gate_type string_to_gate_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_GATE_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_GATE_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_gate_type>(itype);
}
}
return NUM_CIRCUIT_MODEL_GATE_TYPES;
}
/********************************************************************
* Convert string to the enumerate of circuit model port type
*******************************************************************/
static
e_circuit_model_port_type string_to_circuit_model_port_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_PORT_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_PORT_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_port_type>(itype);
}
}
return NUM_CIRCUIT_MODEL_PORT_TYPES;
}
/********************************************************************
* Convert string to the enumerate of wire model type
*******************************************************************/
static
e_wire_model_type string_to_wire_model_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_WIRE_MODEL_TYPES; ++itype) {
if (std::string(WIRE_MODEL_TYPE_STRING[itype]) == type_string) {
return static_cast<e_wire_model_type>(itype);
}
}
return NUM_WIRE_MODEL_TYPES;
}
/********************************************************************
* Convert string to the enumerate of delay model type
*******************************************************************/
static
e_circuit_model_delay_type string_to_circuit_model_delay_type(const std::string& type_string) {
for (size_t itype = 0; itype < NUM_CIRCUIT_MODEL_DELAY_TYPES; ++itype) {
if (std::string(CIRCUIT_MODEL_DELAY_TYPE_STRING[itype]) == type_string) {
return static_cast<e_circuit_model_delay_type>(itype);
}
}
return NUM_CIRCUIT_MODEL_DELAY_TYPES;
}
/********************************************************************
* Parse XML codes of design technology of a circuit model to circuit library
*******************************************************************/
static
void read_xml_model_design_technology(pugi::xml_node& xml_model,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitModelId& model) {
auto xml_design_tech = get_single_child(xml_model, "design_technology", loc_data);
/* Identify if the circuit model power-gated */
circuit_lib.set_model_is_power_gated(model, get_attribute(xml_design_tech, "power_gated", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify the type of design technology */
const char* type_attr = get_attribute(xml_design_tech, "type", loc_data).value();
/* Translate the type of design technology to enumerate */
e_circuit_model_design_tech design_tech_type = string_to_design_tech_type(std::string(type_attr));
if (NUM_CIRCUIT_MODEL_DESIGN_TECH_TYPES == design_tech_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_design_tech),
"Invalid 'type' attribute '%s'\n",
type_attr);
}
circuit_lib.set_model_design_tech_type(model, design_tech_type);
/* Parse exclusive attributes for inverters and buffers */
if (CIRCUIT_MODEL_INVBUF == circuit_lib.model_type(model)) {
/* Identify the topology of the buffer */
const char* topology_attr = get_attribute(xml_design_tech, "topology", loc_data).value();
/* Translate the type of buffer to enumerate */
e_circuit_model_buffer_type buf_type = string_to_buffer_type(std::string(topology_attr));
if (NUM_CIRCUIT_MODEL_BUF_TYPES == buf_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_design_tech),
"Invalid 'topology' attribute '%s'\n",
topology_attr);
}
circuit_lib.set_buffer_type(model, buf_type);
/* Parse the others options:
* 1. size of buffer in the first stage
* 2. number of levels
* 3. driving strength per stage
*/
circuit_lib.set_buffer_size(model, get_attribute(xml_design_tech, "size", loc_data).as_float(0.));
circuit_lib.set_buffer_num_levels(model, get_attribute(xml_design_tech, "num_level", loc_data, pugiutil::ReqOpt::OPTIONAL).as_int(1));
circuit_lib.set_buffer_f_per_stage(model, get_attribute(xml_design_tech, "f_per_stage", loc_data, pugiutil::ReqOpt::OPTIONAL).as_int(4));
}
/* Parse exclusive attributes for pass-gate logics */
if (CIRCUIT_MODEL_PASSGATE == circuit_lib.model_type(model)) {
/* Identify the topology of the pass-gate logic */
const char* topology_attr = get_attribute(xml_design_tech, "topology", loc_data).value();
/* Translate the type of pass-gate logic to enumerate */
e_circuit_model_pass_gate_logic_type passgate_type = string_to_passgate_type(std::string(topology_attr));
if (NUM_CIRCUIT_MODEL_PASS_GATE_TYPES == passgate_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_design_tech),
"Invalid 'topology' attribute '%s'\n",
topology_attr);
}
circuit_lib.set_pass_gate_logic_type(model, passgate_type);
/* Parse the others options:
* 1. pmos size to be used in the pass gate logic
* 2. nmos size to be used in the pass gate logic
*/
circuit_lib.set_pass_gate_logic_pmos_size(model, get_attribute(xml_design_tech, "pmos_size", loc_data).as_float(0.));
circuit_lib.set_pass_gate_logic_nmos_size(model, get_attribute(xml_design_tech, "nmos_size", loc_data).as_float(0.));
}
/* Parse exclusive attributes for Look-Up Tables (LUTs) */
if (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model)) {
/* Identify if this is a fracturable LUT */
circuit_lib.set_lut_is_fracturable(model, get_attribute(xml_design_tech, "fracturable_lut", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Set default MUX-relate attributes as LUT contains a tree-like MUX */
circuit_lib.set_mux_structure(model, CIRCUIT_MODEL_STRUCTURE_TREE);
circuit_lib.set_mux_use_local_encoder(model, false);
circuit_lib.set_mux_use_advanced_rram_design(model, false);
}
/* Parse exclusive attributes for multiplexers */
if (CIRCUIT_MODEL_MUX == circuit_lib.model_type(model)) {
/* Set default values for multiplexer structure */
if (CIRCUIT_MODEL_DESIGN_CMOS == circuit_lib.design_tech_type(model)) {
circuit_lib.set_mux_structure(model, CIRCUIT_MODEL_STRUCTURE_TREE);
} else {
VTR_ASSERT_SAFE(CIRCUIT_MODEL_DESIGN_RRAM == circuit_lib.design_tech_type(model));
circuit_lib.set_mux_structure(model, CIRCUIT_MODEL_STRUCTURE_ONELEVEL);
}
/* Identify the topology of the multiplexer structure */
const char* structure_attr = get_attribute(xml_design_tech, "structure", loc_data).value();
/* Translate the type of multiplexer structure to enumerate */
e_circuit_model_structure mux_structure = string_to_mux_structure_type(std::string(structure_attr));
if (NUM_CIRCUIT_MODEL_STRUCTURE_TYPES == mux_structure) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_design_tech),
"Invalid 'structure' attribute '%s'\n",
structure_attr);
}
circuit_lib.set_mux_structure(model, mux_structure);
/* Parse the others options:
* 1. constant input values
* 2. number of levels if multi-level multiplexer structure is selected
* 3. if advanced ReRAM design is used
* 4. if local encoder is to be used
*/
if (true == get_attribute(xml_design_tech, "add_const_input", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false)) {
circuit_lib.set_mux_const_input_value(model, get_attribute(xml_design_tech, "const_input_val", loc_data).as_int(0));
}
if (CIRCUIT_MODEL_STRUCTURE_MULTILEVEL == circuit_lib.mux_structure(model)) {
circuit_lib.set_mux_num_levels(model, get_attribute(xml_design_tech, "num_level", loc_data).as_int(1));
/* Correction on the mux structure:
* if the number of level is set to 1 in a multi-level multiplexer,
* we change the mux structure to one-level
*/
if (1 == circuit_lib.mux_num_levels(model)) {
circuit_lib.set_mux_structure(model, CIRCUIT_MODEL_STRUCTURE_ONELEVEL);
}
}
circuit_lib.set_mux_use_advanced_rram_design(model, get_attribute(xml_design_tech, "advanced_rram_design", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
circuit_lib.set_mux_use_local_encoder(model, get_attribute(xml_design_tech, "local_encoder", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
}
/* Parse exclusive attributes for logic gates */
if (CIRCUIT_MODEL_GATE == circuit_lib.model_type(model)) {
/* Identify the topology of the logic gate */
const char* topology_attr = get_attribute(xml_design_tech, "topology", loc_data).value();
/* Translate the type of logic gate to enumerate */
e_circuit_model_gate_type gate_type = string_to_gate_type(std::string(topology_attr));
if (NUM_CIRCUIT_MODEL_GATE_TYPES == gate_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_design_tech),
"Invalid 'topology' attribute '%s'\n",
topology_attr);
}
circuit_lib.set_gate_type(model, gate_type);
}
/* Parse exclusive attributes for RRAM */
if (CIRCUIT_MODEL_DESIGN_RRAM == circuit_lib.design_tech_type(model)) {
circuit_lib.set_rram_rlrs(model, get_attribute(xml_design_tech, "ron", loc_data).as_float(0.));
circuit_lib.set_rram_rhrs(model, get_attribute(xml_design_tech, "roff", loc_data).as_float(0.));
circuit_lib.set_rram_wprog_set_pmos(model, get_attribute(xml_design_tech, "wprog_set_pmos", loc_data).as_float(0.));
circuit_lib.set_rram_wprog_set_nmos(model, get_attribute(xml_design_tech, "wprog_set_nmos", loc_data).as_float(0.));
circuit_lib.set_rram_wprog_reset_pmos(model, get_attribute(xml_design_tech, "wprog_reset_pmos", loc_data).as_float(0.));
circuit_lib.set_rram_wprog_reset_nmos(model, get_attribute(xml_design_tech, "wprog_reset_nmos", loc_data).as_float(0.));
}
}
/********************************************************************
* Parse XML codes of device technology of a circuit model to circuit library
*******************************************************************/
static
void read_xml_model_device_technology(pugi::xml_node& xml_model,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitModelId& model) {
auto xml_device_tech = get_single_child(xml_model, "device_technology", loc_data, pugiutil::ReqOpt::OPTIONAL);
if (!xml_device_tech) {
return;
}
/* Parse device model name */
const char* device_model_name_attr = get_attribute(xml_device_tech, "device_model_name", loc_data, pugiutil::ReqOpt::OPTIONAL).value();
if (nullptr != device_model_name_attr) {
circuit_lib.set_device_model_name(model, std::string(device_model_name_attr));
}
}
/********************************************************************
* This is a generic function to parse XML codes that describe
* a buffer of a circuit model to circuit library
* This function will return a string with the circuit model name
* linked to the buffer
* If the return is empty, it means that buffer does NOT exist
*******************************************************************/
static
std::string read_xml_buffer(pugi::xml_node& xml_buffer,
const pugiutil::loc_data& loc_data) {
bool buffer_existence = get_attribute(xml_buffer, "exist", loc_data).as_bool(false);
std::string buffer_circuit_model_name("");
if (true == buffer_existence) {
buffer_circuit_model_name = get_attribute(xml_buffer, "circuit_model_name", loc_data).as_string();
}
return buffer_circuit_model_name;
}
/********************************************************************
* Identify the output mask of the port in LUTs,
* by default it will be applied to each pin of this port
* This is only applicable to output ports of a LUT
*******************************************************************/
static
void read_xml_output_mask(pugi::xml_node& xml_port,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitPortId& port) {
std::string output_mask_attr = get_attribute(xml_port, "lut_output_mask", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string();
std::vector<size_t> mask_vector;
if (!output_mask_attr.empty()) {
/* Split the string with token ',' */
openfpga::StringToken string_tokenizer(get_attribute(xml_port, "lut_output_mask", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string(nullptr));
for (const std::string& mask_token : string_tokenizer.split(',')) {
mask_vector.push_back(std::atoi(mask_token.c_str()));
}
/* Make sure that the size of mask fits the port size */
if (circuit_lib.port_size(port) != mask_vector.size()) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_port),
"Invalid lut_output_mask attribute '%s'! It must match the port size (=%lu)\n",
output_mask_attr.c_str(), circuit_lib.port_size(port));
}
} else {
/* By default, we give a mask vector covering each pin of the port */
mask_vector.resize(circuit_lib.port_size(port));
std::iota(mask_vector.begin(), mask_vector.end(), 0);
}
circuit_lib.set_port_lut_output_mask(port, mask_vector);
}
/********************************************************************
* This is a generic function to parse XML codes that describe
* a port of a circuit model to circuit library
*******************************************************************/
static
void read_xml_circuit_port(pugi::xml_node& xml_port,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitModelId& model) {
/* Find the type of the circuit port
* so that we can add a new circuit port to circuit library
*/
const char* type_attr = get_attribute(xml_port, "type", loc_data).value();
/* Translate the type of circuit model to enumerate */
e_circuit_model_port_type port_type = string_to_circuit_model_port_type(std::string(type_attr));
if (NUM_CIRCUIT_MODEL_PORT_TYPES == port_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_port),
"Invalid 'type' attribute '%s'\n",
type_attr);
}
CircuitPortId port = circuit_lib.add_model_port(model, port_type);
/* Parse the name of the port */
circuit_lib.set_port_prefix(port, get_attribute(xml_port, "prefix", loc_data).as_string());
/* Parse the name of the port in cell library. By default, the lib_name is the same as port name */
const char* lib_name_attr = get_attribute(xml_port, "lib_name", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string(nullptr);
if (nullptr != lib_name_attr) {
circuit_lib.set_port_lib_name(port, get_attribute(xml_port, "lib_name", loc_data).as_string());
} else {
circuit_lib.set_port_lib_name(port, circuit_lib.port_prefix(port));
}
/* Parse the port size, by default it will be 1 */
circuit_lib.set_port_size(port, get_attribute(xml_port, "size", loc_data).as_int(1));
/* Identify if the port is for io, this is only applicable to INPUT ports.
* By default, it will NOT be a mode selection port
*/
circuit_lib.set_port_is_io(port, get_attribute(xml_port, "is_io", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify if the port is a data io, ONLY applicable to I/O port
* By default, it will NOT be a data io port
*/
if (true == circuit_lib.port_is_io(port)) {
circuit_lib.set_port_is_data_io(port, get_attribute(xml_port, "is_data_io", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
}
/* Identify if the port is for mode selection, this is only applicable to SRAM ports.
* By default, it will NOT be a mode selection port
*/
if (CIRCUIT_MODEL_PORT_SRAM == circuit_lib.port_type(port)) {
circuit_lib.set_port_is_mode_select(port, get_attribute(xml_port, "mode_select", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
}
/* Identify the default value of the port, by default it will be 0 */
circuit_lib.set_port_default_value(port, get_attribute(xml_port, "default_val", loc_data, pugiutil::ReqOpt::OPTIONAL).as_int(0));
/* Identify the tri-state map of the port, by default it will be nullptr
* This is only applicable to input ports of a LUT
*/
if ( (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model))
&& (CIRCUIT_MODEL_PORT_INPUT == circuit_lib.port_type(port)) ) {
const char* tri_state_map_attr = get_attribute(xml_port, "tri_state_map", loc_data, pugiutil::ReqOpt::OPTIONAL).value();
if (nullptr != tri_state_map_attr) {
circuit_lib.set_port_tri_state_map(port, get_attribute(xml_port, "tri_state_map", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string());
}
}
/* Identify the fracturable-level of the port in LUTs, by default it will be -1
* This is only applicable to output ports of a LUT
*/
if ( (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model))
&& (CIRCUIT_MODEL_PORT_OUTPUT == circuit_lib.port_type(port)) ) {
circuit_lib.set_port_lut_frac_level(port, get_attribute(xml_port, "lut_frac_level", loc_data, pugiutil::ReqOpt::OPTIONAL).as_int(-1));
}
/* Identify if the port carries a harden functionality rather than a reconfigurable port
* This is only applicable to LUT circuit models.
* The super LUT circuit model (whose netlists are supposed to be provided by users) contains
* some hard logic inside, e.g., a carry logic.
* By default, a port does NOT carry a hard functionality
*/
if (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model)) {
circuit_lib.set_port_is_harden_lut_port(port, get_attribute(xml_port, "is_harden_lut_port", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
}
/* Identify the output mask of the port in LUTs, by default it will be applied to each pin of this port
* This is only applicable to output ports of a LUT
*/
if ( (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model))
&& (CIRCUIT_MODEL_PORT_OUTPUT == circuit_lib.port_type(port)) ) {
read_xml_output_mask(xml_port, loc_data, circuit_lib, port);
}
/* Identify if the port is a global port, by default it is NOT */
circuit_lib.set_port_is_global(port, get_attribute(xml_port, "is_global", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify if the port is a reset port, by default it is NOT */
circuit_lib.set_port_is_reset(port, get_attribute(xml_port, "is_reset", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify if the port is a set port, by default it is NOT */
circuit_lib.set_port_is_set(port, get_attribute(xml_port, "is_set", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify if the port is in programming purpose, by default it is NOT */
circuit_lib.set_port_is_prog(port, get_attribute(xml_port, "is_prog", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Identify if the port is to enable programming for FPGAs, by default it is NOT */
circuit_lib.set_port_is_config_enable(port, get_attribute(xml_port, "is_config_enable", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Find the name of circuit model that this port is linked to */
circuit_lib.set_port_tri_state_model_name(port, get_attribute(xml_port, "circuit_model_name", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string());
/* Find the name of circuit model that port is used for inversion of signals,
* This is only applicable to BL/WL/BLB/WLB ports
*/
if ( (CIRCUIT_MODEL_PORT_BL == circuit_lib.port_type(port))
|| (CIRCUIT_MODEL_PORT_WL == circuit_lib.port_type(port))
|| (CIRCUIT_MODEL_PORT_BLB == circuit_lib.port_type(port))
|| (CIRCUIT_MODEL_PORT_WLB == circuit_lib.port_type(port)) ) {
circuit_lib.set_port_inv_model_name(port, get_attribute(xml_port, "inv_circuit_model_name", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string());
}
}
/********************************************************************
* This is a generic function to parse XML codes that describe
* RC parasitics for wire circuit model to circuit library
*******************************************************************/
static
void read_xml_wire_param(pugi::xml_node& xml_wire_param,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitModelId& model) {
/* Find the type of the wire model */
const char* type_attr = get_attribute(xml_wire_param, "model_type", loc_data).value();
/* Translate the type of circuit model to enumerate */
e_wire_model_type wire_model_type = string_to_wire_model_type(std::string(type_attr));
if (NUM_WIRE_MODEL_TYPES == wire_model_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_wire_param),
"Invalid 'type' attribute '%s'\n",
type_attr);
}
circuit_lib.set_wire_type(model, wire_model_type);
/* Parse the R and C values */
circuit_lib.set_wire_r(model, get_attribute(xml_wire_param, "R", loc_data).as_float(0.));
circuit_lib.set_wire_c(model, get_attribute(xml_wire_param, "C", loc_data).as_float(0.));
/* Parse the number of levels for the wire model */
circuit_lib.set_wire_num_level(model, get_attribute(xml_wire_param, "num_level", loc_data).as_int(0));
}
/********************************************************************
* This is a generic function to parse XML codes that describe
* a delay matrix for a circuit model to circuit library
*******************************************************************/
static
void read_xml_delay_matrix(pugi::xml_node& xml_delay_matrix,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib, const CircuitModelId& model) {
/* Find the type of the delay model, so that we can add to circuit library */
const char* type_attr = get_attribute(xml_delay_matrix, "type", loc_data).value();
/* Translate the type of delay matrix for a circuit model to enumerate */
e_circuit_model_delay_type delay_type = string_to_circuit_model_delay_type(std::string(type_attr));
if (NUM_CIRCUIT_MODEL_DELAY_TYPES == delay_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_delay_matrix),
"Invalid 'type' attribute '%s'\n",
type_attr);
}
circuit_lib.add_delay_info(model, delay_type);
/* Parse the input ports */
circuit_lib.set_delay_in_port_names(model, delay_type, get_attribute(xml_delay_matrix, "in_port", loc_data).as_string());
/* Parse the output ports */
circuit_lib.set_delay_out_port_names(model, delay_type, get_attribute(xml_delay_matrix, "out_port", loc_data).as_string());
/* Parse the delay values */
circuit_lib.set_delay_values(model, delay_type, std::string(xml_delay_matrix.child_value()));
}
/********************************************************************
* Parse XML codes of a circuit model to circuit library
*******************************************************************/
static
void read_xml_circuit_model(pugi::xml_node& xml_model,
const pugiutil::loc_data& loc_data,
CircuitLibrary& circuit_lib) {
/* Find the type of the circuit model
* so that we can add a new circuit model to circuit library
*/
const char* type_attr = get_attribute(xml_model, "type", loc_data).value();
/* Translate the type of circuit model to enumerate */
e_circuit_model_type model_type = string_to_circuit_model_type(std::string(type_attr));
if (NUM_CIRCUIT_MODEL_TYPES == model_type) {
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_model),
"Invalid 'type' attribute '%s'\n",
type_attr);
}
CircuitModelId model = circuit_lib.add_model(model_type);
/* Find the name of the circuit model */
const char* name_attr = get_attribute(xml_model, "name", loc_data).value();
circuit_lib.set_model_name(model, std::string(name_attr));
/* TODO: This attribute is going to be DEPRECATED
* Find the prefix of the circuit model
*/
const char* prefix_attr = get_attribute(xml_model, "prefix", loc_data).value();
circuit_lib.set_model_prefix(model, std::string(prefix_attr));
/* Find a SPICE netlist which is an optional attribute*/
circuit_lib.set_model_spice_netlist(model, get_attribute(xml_model, "spice_netlist", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string(""));
/* Find a Verilog netlist which is an optional attribute*/
circuit_lib.set_model_verilog_netlist(model, get_attribute(xml_model, "verilog_netlist", loc_data, pugiutil::ReqOpt::OPTIONAL).as_string(""));
/* Find if the circuit model is default in its type */
circuit_lib.set_model_is_default(model, get_attribute(xml_model, "is_default", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Find if the circuit model is should be dumped in structural verilog */
circuit_lib.set_model_dump_structural_verilog(model, get_attribute(xml_model, "dump_structural_verilog", loc_data, pugiutil::ReqOpt::OPTIONAL).as_bool(false));
/* Parse attributes under the <circuit_model> */
/* Design technology -related attributes */
read_xml_model_design_technology(xml_model, loc_data, circuit_lib, model);
/* Parse special buffer attributes required by LUTs only */
if (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model)) {
/* Input buffer of LUTs */
auto xml_input_buffer = get_single_child(xml_model, "lut_input_buffer", loc_data);
std::string input_buffer_circuit_model_name = read_xml_buffer(xml_input_buffer, loc_data);
circuit_lib.set_model_lut_input_buffer(model,
true != input_buffer_circuit_model_name.empty(),
input_buffer_circuit_model_name);
/* Input inverter of LUTs */
auto xml_input_inverter = get_single_child(xml_model, "lut_input_inverter", loc_data);
std::string input_inverter_circuit_model_name = read_xml_buffer(xml_input_inverter, loc_data);
circuit_lib.set_model_lut_input_inverter(model,
true != input_inverter_circuit_model_name.empty(),
input_inverter_circuit_model_name);
/* Intermediate buffer of LUTs */
auto xml_intermediate_buffer = get_single_child(xml_model, "lut_intermediate_buffer", loc_data, pugiutil::ReqOpt::OPTIONAL);
if (xml_intermediate_buffer) {
std::string intermediate_buffer_circuit_model_name = read_xml_buffer(xml_intermediate_buffer, loc_data);
circuit_lib.set_model_lut_intermediate_buffer(model,
true != intermediate_buffer_circuit_model_name.empty(),
intermediate_buffer_circuit_model_name);
/* If intermediate buffer is defined, try to find the location map */
if (true != intermediate_buffer_circuit_model_name.empty()) {
circuit_lib.set_model_lut_intermediate_buffer_location_map(model, get_attribute(xml_intermediate_buffer, "location_map", loc_data).as_string());
}
}
}
/* Parse device technology attributes
* This is applicable to only atom circuit models:
* - inverter/buffer
* - pass gate
* - logic gates
*/
if ((CIRCUIT_MODEL_INVBUF == circuit_lib.model_type(model))
|| (CIRCUIT_MODEL_PASSGATE == circuit_lib.model_type(model))
|| (CIRCUIT_MODEL_GATE == circuit_lib.model_type(model))) {
read_xml_model_device_technology(xml_model, loc_data, circuit_lib, model);
}
/* Input buffer attributes, NOT required for circuit models which are inverters or buffers */
if (CIRCUIT_MODEL_INVBUF != circuit_lib.model_type(model)) {
auto xml_input_buffer = get_single_child(xml_model, "input_buffer", loc_data);
std::string input_buffer_circuit_model_name = read_xml_buffer(xml_input_buffer, loc_data);
circuit_lib.set_model_input_buffer(model,
true != input_buffer_circuit_model_name.empty(),
input_buffer_circuit_model_name);
}
/* Output buffer attributes, NOT required for circuit models which are inverters or buffers */
if (CIRCUIT_MODEL_INVBUF != circuit_lib.model_type(model)) {
auto xml_output_buffer = get_single_child(xml_model, "output_buffer", loc_data);
std::string output_buffer_circuit_model_name = read_xml_buffer(xml_output_buffer, loc_data);
circuit_lib.set_model_output_buffer(model,
true != output_buffer_circuit_model_name.empty(),
output_buffer_circuit_model_name);
}
/* Pass-gate-logic attributes, required by LUT and MUX */
if ( (CIRCUIT_MODEL_LUT == circuit_lib.model_type(model))
|| (CIRCUIT_MODEL_MUX == circuit_lib.model_type(model)) ) {
auto xml_pass_gate_logic = get_single_child(xml_model, "pass_gate_logic", loc_data);
circuit_lib.set_model_pass_gate_logic(model, get_attribute(xml_pass_gate_logic, "circuit_model_name", loc_data).as_string());
}
/* Parse all the ports belonging to this circuit model
* We count the number of ports in total and then add one by one
*/
size_t num_ports = count_children(xml_model, "port", loc_data, pugiutil::ReqOpt::OPTIONAL);
if (0 < num_ports) {
pugi::xml_node xml_port = get_first_child(xml_model, "port", loc_data);
while (xml_port) {
read_xml_circuit_port(xml_port, loc_data, circuit_lib, model);
xml_port = xml_port.next_sibling(xml_port.name());
}
}
/* Parse the parasitics of wires */
if ( (CIRCUIT_MODEL_WIRE == circuit_lib.model_type(model))
|| (CIRCUIT_MODEL_CHAN_WIRE == circuit_lib.model_type(model)) ) {
auto xml_wire_param = get_single_child(xml_model, "wire_param", loc_data);
read_xml_wire_param(xml_wire_param, loc_data, circuit_lib, model);
}
/* Parse all the delay matrix if defined */
size_t num_delay_matrix = count_children(xml_model, "delay_matrix", loc_data, pugiutil::ReqOpt::OPTIONAL);
if (0 < num_delay_matrix) {
pugi::xml_node xml_delay_matrix = get_first_child(xml_model, "delay_matrix", loc_data);
while (xml_delay_matrix) {
read_xml_delay_matrix(xml_delay_matrix, loc_data, circuit_lib, model);
xml_delay_matrix = xml_delay_matrix.next_sibling(xml_delay_matrix.name());
}
}
}
/********************************************************************
* Parse XML codes about circuit models to circuit library
*******************************************************************/
CircuitLibrary read_xml_circuit_library(pugi::xml_node& Node,
const pugiutil::loc_data& loc_data) {
CircuitLibrary circuit_lib;
/* Iterate over the children under this node,
* each child should be named after circuit_model
*/
for (pugi::xml_node xml_model : Node.children()) {
/* Error out if the XML child has an invalid name! */
if (xml_model.name() != std::string("circuit_model")) {
bad_tag(xml_model, loc_data, Node, {"circuit_model"});
}
read_xml_circuit_model(xml_model, loc_data, circuit_lib);
}
return circuit_lib;
}