2020-01-18 22:19:20 -06:00
|
|
|
/********************************************************************
|
|
|
|
* 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"
|
2023-04-21 10:45:35 -05:00
|
|
|
#include "config_protocol_xml_constants.h"
|
|
|
|
#include "openfpga_port_parser.h"
|
2023-04-22 02:12:38 -05:00
|
|
|
#include "read_xml_config_protocol.h"
|
2022-10-06 19:08:50 -05:00
|
|
|
#include "read_xml_util.h"
|
2020-01-18 22:19:20 -06:00
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Convert string to the enumerate of configuration protocol type
|
|
|
|
*******************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
static e_config_protocol_type string_to_config_protocol_type(
|
|
|
|
const std::string& type_string) {
|
2020-09-27 15:33:14 -05:00
|
|
|
for (size_t itype = 0; itype < NUM_CONFIG_PROTOCOL_TYPES; ++itype) {
|
|
|
|
if (std::string(CONFIG_PROTOCOL_TYPE_STRING[itype]) == type_string) {
|
2022-10-06 19:08:50 -05:00
|
|
|
return static_cast<e_config_protocol_type>(itype);
|
2020-09-27 15:33:14 -05:00
|
|
|
}
|
2020-01-18 22:19:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return NUM_CONFIG_PROTOCOL_TYPES;
|
|
|
|
}
|
|
|
|
|
2021-09-23 16:25:25 -05:00
|
|
|
/********************************************************************
|
|
|
|
* Convert string to the enumerate of BL/WL protocol type
|
|
|
|
*******************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
static e_blwl_protocol_type string_to_blwl_protocol_type(
|
|
|
|
const std::string& type_string) {
|
2021-09-23 16:25:25 -05:00
|
|
|
for (size_t itype = 0; itype < NUM_BLWL_PROTOCOL_TYPES; ++itype) {
|
|
|
|
if (std::string(BLWL_PROTOCOL_TYPE_STRING[itype]) == type_string) {
|
2022-10-06 19:08:50 -05:00
|
|
|
return static_cast<e_blwl_protocol_type>(itype);
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NUM_BLWL_PROTOCOL_TYPES;
|
|
|
|
}
|
|
|
|
|
2023-04-21 04:01:51 -05:00
|
|
|
/********************************************************************
|
2023-04-22 02:12:38 -05:00
|
|
|
* Parse XML codes of a <programming_clock> to an object of configuration
|
|
|
|
*protocol
|
2023-04-21 04:01:51 -05:00
|
|
|
*******************************************************************/
|
|
|
|
static void read_xml_ccff_prog_clock(pugi::xml_node& xml_progclk,
|
|
|
|
const pugiutil::loc_data& loc_data,
|
|
|
|
ConfigProtocol& config_protocol) {
|
|
|
|
/* Find the type of configuration protocol */
|
2023-04-21 10:45:35 -05:00
|
|
|
std::string port_attr =
|
2023-04-22 02:12:38 -05:00
|
|
|
get_attribute(xml_progclk, XML_CONFIG_PROTOCOL_CCFF_PROG_CLOCK_PORT_ATTR,
|
|
|
|
loc_data)
|
|
|
|
.as_string();
|
2023-04-21 04:01:51 -05:00
|
|
|
|
2023-04-21 10:45:35 -05:00
|
|
|
std::string indices_attr =
|
2023-04-22 02:12:38 -05:00
|
|
|
get_attribute(xml_progclk, XML_CONFIG_PROTOCOL_CCFF_PROG_CLOCK_INDICES_ATTR,
|
|
|
|
loc_data)
|
|
|
|
.as_string();
|
2023-04-21 10:45:35 -05:00
|
|
|
|
2023-04-22 02:12:38 -05:00
|
|
|
openfpga::BasicPort port = openfpga::PortParser(port_attr).port();
|
2023-04-21 10:45:35 -05:00
|
|
|
|
2023-04-22 02:12:38 -05:00
|
|
|
config_protocol.set_prog_clock_port_ccff_head_indices_pair(port,
|
|
|
|
indices_attr);
|
2023-04-21 04:01:51 -05:00
|
|
|
}
|
|
|
|
|
2021-09-23 16:25:25 -05:00
|
|
|
/********************************************************************
|
|
|
|
* Parse XML codes of a <bl> to an object of configuration protocol
|
|
|
|
*******************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
static void read_xml_bl_protocol(pugi::xml_node& xml_bl_protocol,
|
|
|
|
const pugiutil::loc_data& loc_data,
|
|
|
|
ConfigProtocol& config_protocol) {
|
2021-09-23 16:25:25 -05:00
|
|
|
/* Find the type of configuration protocol */
|
2022-10-06 19:08:50 -05:00
|
|
|
const char* type_attr =
|
|
|
|
get_attribute(xml_bl_protocol, "protocol", loc_data).value();
|
2021-09-23 16:25:25 -05:00
|
|
|
/* Translate the type of design technology to enumerate */
|
2022-10-06 19:08:50 -05:00
|
|
|
e_blwl_protocol_type blwl_protocol_type =
|
|
|
|
string_to_blwl_protocol_type(std::string(type_attr));
|
2021-09-23 16:25:25 -05:00
|
|
|
|
|
|
|
if (NUM_BLWL_PROTOCOL_TYPES == blwl_protocol_type) {
|
|
|
|
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_bl_protocol),
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid 'protocol' attribute '%s'\n", type_attr);
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
config_protocol.set_bl_protocol_type(blwl_protocol_type);
|
|
|
|
|
2021-09-28 16:20:35 -05:00
|
|
|
/* only applicable to shift-registor protocol
|
|
|
|
* - Find the memory model to build shift register chains
|
|
|
|
* - Find the number of shift register chains for each protocol
|
|
|
|
*/
|
2021-09-23 16:25:25 -05:00
|
|
|
if (BLWL_PROTOCOL_SHIFT_REGISTER == blwl_protocol_type) {
|
2022-10-06 19:08:50 -05:00
|
|
|
config_protocol.set_bl_memory_model_name(
|
|
|
|
get_attribute(xml_bl_protocol, "circuit_model_name", loc_data)
|
|
|
|
.as_string());
|
|
|
|
config_protocol.set_bl_num_banks(get_attribute(xml_bl_protocol, "num_banks",
|
|
|
|
loc_data,
|
|
|
|
pugiutil::ReqOpt::OPTIONAL)
|
|
|
|
.as_int(1));
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Parse XML codes of a <wl> to an object of configuration protocol
|
|
|
|
*******************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
static void read_xml_wl_protocol(pugi::xml_node& xml_wl_protocol,
|
|
|
|
const pugiutil::loc_data& loc_data,
|
|
|
|
ConfigProtocol& config_protocol) {
|
2021-09-23 16:25:25 -05:00
|
|
|
/* Find the type of configuration protocol */
|
2022-10-06 19:08:50 -05:00
|
|
|
const char* type_attr =
|
|
|
|
get_attribute(xml_wl_protocol, "protocol", loc_data).value();
|
2021-09-23 16:25:25 -05:00
|
|
|
/* Translate the type of design technology to enumerate */
|
2022-10-06 19:08:50 -05:00
|
|
|
e_blwl_protocol_type blwl_protocol_type =
|
|
|
|
string_to_blwl_protocol_type(std::string(type_attr));
|
2021-09-23 16:25:25 -05:00
|
|
|
|
|
|
|
if (NUM_BLWL_PROTOCOL_TYPES == blwl_protocol_type) {
|
|
|
|
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_wl_protocol),
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid 'protocol' attribute '%s'\n", type_attr);
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
config_protocol.set_wl_protocol_type(blwl_protocol_type);
|
|
|
|
|
2021-09-28 16:20:35 -05:00
|
|
|
/* only applicable to shift-registor protocol
|
|
|
|
* - Find the memory model to build shift register chains
|
|
|
|
* - Find the number of shift register chains for each protocol
|
|
|
|
*/
|
2021-09-23 16:25:25 -05:00
|
|
|
if (BLWL_PROTOCOL_SHIFT_REGISTER == blwl_protocol_type) {
|
2022-10-06 19:08:50 -05:00
|
|
|
config_protocol.set_wl_memory_model_name(
|
|
|
|
get_attribute(xml_wl_protocol, "circuit_model_name", loc_data)
|
|
|
|
.as_string());
|
|
|
|
config_protocol.set_wl_num_banks(get_attribute(xml_wl_protocol, "num_banks",
|
|
|
|
loc_data,
|
|
|
|
pugiutil::ReqOpt::OPTIONAL)
|
|
|
|
.as_int(1));
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-18 22:19:20 -06:00
|
|
|
/********************************************************************
|
|
|
|
* Parse XML codes of a <organization> to an object of configuration protocol
|
|
|
|
*******************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
static void read_xml_config_organization(pugi::xml_node& xml_config_orgz,
|
|
|
|
const pugiutil::loc_data& loc_data,
|
|
|
|
ConfigProtocol& config_protocol) {
|
2020-01-18 22:19:20 -06:00
|
|
|
/* Find the type of configuration protocol */
|
2022-10-06 19:08:50 -05:00
|
|
|
const char* type_attr =
|
|
|
|
get_attribute(xml_config_orgz, "type", loc_data).value();
|
2020-01-18 22:19:20 -06:00
|
|
|
/* Translate the type of design technology to enumerate */
|
2022-10-06 19:08:50 -05:00
|
|
|
e_config_protocol_type config_orgz_type =
|
|
|
|
string_to_config_protocol_type(std::string(type_attr));
|
2020-01-18 22:19:20 -06:00
|
|
|
|
|
|
|
if (NUM_CONFIG_PROTOCOL_TYPES == config_orgz_type) {
|
|
|
|
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_config_orgz),
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid 'type' attribute '%s'\n", type_attr);
|
2020-01-18 22:19:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
config_protocol.set_type(config_orgz_type);
|
|
|
|
|
2020-09-28 14:51:43 -05:00
|
|
|
/* Find the circuit model used by the configuration protocol */
|
2022-10-06 19:08:50 -05:00
|
|
|
config_protocol.set_memory_model_name(
|
|
|
|
get_attribute(xml_config_orgz, "circuit_model_name", loc_data).as_string());
|
2020-01-18 22:19:20 -06:00
|
|
|
|
2020-09-28 14:51:43 -05:00
|
|
|
/* Parse the number of configurable regions
|
2022-10-06 19:08:50 -05:00
|
|
|
* At least 1 region should be defined, otherwise error out
|
2020-09-28 14:51:43 -05:00
|
|
|
*/
|
2023-04-22 02:12:38 -05:00
|
|
|
config_protocol.set_num_regions(
|
|
|
|
get_attribute(xml_config_orgz, XML_CONFIG_PROTOCOL_NUM_REGIONS_ATTR,
|
|
|
|
loc_data, pugiutil::ReqOpt::OPTIONAL)
|
|
|
|
.as_int(1));
|
2020-09-28 14:51:43 -05:00
|
|
|
if (1 > config_protocol.num_regions()) {
|
|
|
|
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_config_orgz),
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid 'num_region=%d' definition. At least 1 region "
|
|
|
|
"should be defined!\n",
|
2020-09-28 14:51:43 -05:00
|
|
|
config_protocol.num_regions());
|
|
|
|
}
|
2021-09-23 16:25:25 -05:00
|
|
|
|
2023-04-21 04:01:51 -05:00
|
|
|
/* Parse Configuration chain protocols */
|
|
|
|
if (config_protocol.type() == CONFIG_MEM_SCAN_CHAIN) {
|
|
|
|
for (pugi::xml_node xml_progclk : xml_config_orgz.children()) {
|
|
|
|
/* Error out if the XML child has an invalid name! */
|
2023-04-22 02:12:38 -05:00
|
|
|
if (xml_progclk.name() !=
|
|
|
|
std::string(XML_CONFIG_PROTOCOL_CCFF_PROG_CLOCK_NODE_NAME)) {
|
2023-04-21 04:01:51 -05:00
|
|
|
bad_tag(xml_progclk, loc_data, xml_config_orgz, {"programming_clock"});
|
|
|
|
}
|
|
|
|
read_xml_ccff_prog_clock(xml_progclk, loc_data, config_protocol);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-23 16:25:25 -05:00
|
|
|
/* Parse BL & WL protocols */
|
|
|
|
if (config_protocol.type() == CONFIG_MEM_QL_MEMORY_BANK) {
|
2022-10-06 19:08:50 -05:00
|
|
|
pugi::xml_node xml_bl_protocol = get_single_child(
|
|
|
|
xml_config_orgz, "bl", loc_data, pugiutil::ReqOpt::OPTIONAL);
|
2021-09-24 14:03:35 -05:00
|
|
|
if (xml_bl_protocol) {
|
|
|
|
read_xml_bl_protocol(xml_bl_protocol, loc_data, config_protocol);
|
|
|
|
}
|
2021-09-23 16:25:25 -05:00
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
pugi::xml_node xml_wl_protocol = get_single_child(
|
|
|
|
xml_config_orgz, "wl", loc_data, pugiutil::ReqOpt::OPTIONAL);
|
2021-09-24 14:03:35 -05:00
|
|
|
if (xml_wl_protocol) {
|
|
|
|
read_xml_wl_protocol(xml_wl_protocol, loc_data, config_protocol);
|
|
|
|
}
|
2022-10-06 19:08:50 -05:00
|
|
|
|
|
|
|
/* Throw an execption if the BL/WL protocols are different. We currently do
|
|
|
|
* not support it! */
|
|
|
|
if (config_protocol.bl_protocol_type() !=
|
|
|
|
config_protocol.wl_protocol_type()) {
|
2021-10-05 16:08:01 -05:00
|
|
|
archfpga_throw(loc_data.filename_c_str(), loc_data.line(xml_config_orgz),
|
2022-10-06 19:08:50 -05:00
|
|
|
"Expect same type of protocol for both BL and WL! Other "
|
|
|
|
"combinations are not supported yet\n");
|
2021-10-05 16:08:01 -05:00
|
|
|
}
|
2021-09-23 16:25:25 -05:00
|
|
|
}
|
2020-01-18 22:19:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Parse XML codes about <configuration_protocol> to an object of ConfigProtocol
|
|
|
|
*******************************************************************/
|
|
|
|
ConfigProtocol read_xml_config_protocol(pugi::xml_node& Node,
|
|
|
|
const pugiutil::loc_data& loc_data) {
|
|
|
|
ConfigProtocol config_protocol;
|
|
|
|
|
|
|
|
/* Parse configuration protocol root node */
|
2022-10-06 19:08:50 -05:00
|
|
|
pugi::xml_node xml_config =
|
|
|
|
get_single_child(Node, "configuration_protocol", loc_data);
|
2020-01-18 22:19:20 -06:00
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
pugi::xml_node xml_config_orgz =
|
|
|
|
get_single_child(xml_config, "organization", loc_data);
|
2020-01-18 22:19:20 -06:00
|
|
|
read_xml_config_organization(xml_config_orgz, loc_data, config_protocol);
|
|
|
|
|
|
|
|
return config_protocol;
|
|
|
|
}
|