2023-08-26 14:54:12 -05:00
|
|
|
/********************************************************************
|
|
|
|
* Unit test functions to validate the correctness of
|
|
|
|
* 1. parser of data structures
|
|
|
|
* 2. writer of data structures
|
|
|
|
*******************************************************************/
|
|
|
|
/* Headers from vtrutils */
|
|
|
|
#include "vtr_assert.h"
|
|
|
|
#include "vtr_log.h"
|
|
|
|
|
|
|
|
/* Headers from fabric key */
|
2023-08-26 15:19:18 -05:00
|
|
|
#include "check_fabric_key.h"
|
2023-08-26 14:54:12 -05:00
|
|
|
#include "command_echo.h"
|
2023-08-26 15:19:18 -05:00
|
|
|
#include "command_exit_codes.h"
|
2023-08-26 14:54:12 -05:00
|
|
|
#include "command_parser.h"
|
|
|
|
#include "read_xml_fabric_key.h"
|
|
|
|
#include "write_xml_fabric_key.h"
|
|
|
|
|
2023-08-26 15:19:18 -05:00
|
|
|
/** @brief Initialize the options from command-line inputs and organize in the
|
|
|
|
* format that is ready for parsing */
|
|
|
|
static std::vector<std::string> format_argv(const std::string& cmd_name,
|
|
|
|
int argc, const char** argv) {
|
2023-08-26 14:54:12 -05:00
|
|
|
std::vector<std::string> cmd_opts;
|
|
|
|
cmd_opts.push_back(cmd_name);
|
|
|
|
for (int iarg = 1; iarg < argc; ++iarg) {
|
|
|
|
cmd_opts.push_back(std::string(argv[iarg]));
|
|
|
|
}
|
|
|
|
return cmd_opts;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @brief Checks to be done:
|
|
|
|
* - Each alias of reference key can be found in the input key
|
|
|
|
*/
|
2023-08-26 15:19:18 -05:00
|
|
|
static int check_input_and_ref_key_alias_match(
|
2023-08-26 20:15:30 -05:00
|
|
|
const openfpga::FabricKey& input_key, const openfpga::FabricKey& ref_key,
|
|
|
|
const bool& verbose) {
|
2023-08-26 14:54:12 -05:00
|
|
|
size_t num_errors = 0;
|
2024-08-02 14:31:55 -05:00
|
|
|
size_t num_ref_keys_checked = 0;
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG(
|
|
|
|
"Checking key alias matching between reference key and input keys...\n");
|
|
|
|
for (openfpga::FabricKeyId key_id : ref_key.keys()) {
|
2023-08-26 14:54:12 -05:00
|
|
|
/* Note that this is slow. May consider to build a map first */
|
|
|
|
std::string curr_alias = ref_key.key_alias(key_id);
|
2023-08-26 15:19:18 -05:00
|
|
|
std::vector<openfpga::FabricKeyId> input_found_keys =
|
|
|
|
input_key.find_key_by_alias(curr_alias);
|
2024-08-02 14:31:55 -05:00
|
|
|
float progress = static_cast<float>(num_ref_keys_checked) /
|
2023-08-26 15:19:18 -05:00
|
|
|
static_cast<float>(ref_key.num_keys()) * 100.0;
|
2023-08-26 20:07:08 -05:00
|
|
|
VTR_LOGV(verbose, "[%lu%] Checking key alias '%s'\r", size_t(progress),
|
2023-08-26 20:15:30 -05:00
|
|
|
curr_alias.c_str());
|
2023-08-26 14:54:12 -05:00
|
|
|
if (input_found_keys.empty()) {
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG_ERROR(
|
2023-08-26 20:15:30 -05:00
|
|
|
"Invalid alias '%s' in the reference key (id='%lu'), which does not "
|
2023-08-26 15:19:18 -05:00
|
|
|
"exist in the input key!\n",
|
|
|
|
curr_alias.c_str(), size_t(key_id));
|
2023-08-26 14:54:12 -05:00
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
if (input_found_keys.size() > 1) {
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG_ERROR(
|
2023-08-26 20:15:30 -05:00
|
|
|
"Invalid alias '%s' in the input key (id='%lu'), which have been "
|
2023-08-26 15:19:18 -05:00
|
|
|
"found %lu times!\n",
|
2023-08-26 20:12:25 -05:00
|
|
|
curr_alias.c_str(), size_t(key_id), input_found_keys.size());
|
2023-08-26 14:54:12 -05:00
|
|
|
num_errors++;
|
|
|
|
}
|
2024-08-02 14:31:55 -05:00
|
|
|
num_ref_keys_checked++;
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG(
|
|
|
|
"Checking key alias matching between reference key and input keys... %s\n",
|
|
|
|
num_errors ? "[Fail]" : "[Pass]");
|
2024-08-02 14:31:55 -05:00
|
|
|
/* If failed, provide a detailed diff on the key alias */
|
|
|
|
if (num_errors) {
|
|
|
|
size_t num_input_keys_checked = 0;
|
|
|
|
for (openfpga::FabricKeyId key_id : input_key.keys()) {
|
|
|
|
/* Note that this is slow. May consider to build a map first */
|
|
|
|
std::string curr_alias = input_key.key_alias(key_id);
|
|
|
|
std::vector<openfpga::FabricKeyId> ref_found_keys =
|
|
|
|
ref_key.find_key_by_alias(curr_alias);
|
|
|
|
float progress = static_cast<float>(num_input_keys_checked) /
|
|
|
|
static_cast<float>(input_key.num_keys()) * 100.0;
|
|
|
|
VTR_LOGV(verbose, "[%lu%] Checking key alias '%s'\r", size_t(progress),
|
|
|
|
curr_alias.c_str());
|
|
|
|
if (ref_found_keys.empty()) {
|
|
|
|
VTR_LOG_ERROR(
|
|
|
|
"Invalid alias '%s' in the input key (id='%lu'), which does not "
|
|
|
|
"exist in the reference key!\n",
|
|
|
|
curr_alias.c_str(), size_t(key_id));
|
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
if (ref_found_keys.size() > 1) {
|
|
|
|
VTR_LOG_ERROR(
|
|
|
|
"Invalid alias '%s' in the reference key (id='%lu'), which have been "
|
|
|
|
"found %lu times!\n",
|
|
|
|
curr_alias.c_str(), size_t(key_id), ref_found_keys.size());
|
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
num_input_keys_checked++;
|
|
|
|
}
|
|
|
|
}
|
2023-08-26 15:19:18 -05:00
|
|
|
return num_errors ? openfpga::CMD_EXEC_FATAL_ERROR
|
|
|
|
: openfpga::CMD_EXEC_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @brief Checks to be done:
|
|
|
|
* - Number of configuration regions match
|
|
|
|
* - Number of keys match
|
|
|
|
*/
|
|
|
|
static int check_input_key(const openfpga::FabricKey& input_key,
|
2023-08-26 20:07:08 -05:00
|
|
|
const openfpga::FabricKey& ref_key,
|
|
|
|
const bool& verbose) {
|
2023-08-26 15:19:18 -05:00
|
|
|
if (ref_key.num_regions() != input_key.num_regions()) {
|
|
|
|
VTR_LOG_ERROR(
|
|
|
|
"Different number of configuration regions between reference key "
|
|
|
|
"(='%lu') and input key ('=%lu')!\n",
|
|
|
|
ref_key.num_regions(), input_key.num_regions());
|
|
|
|
return openfpga::CMD_EXEC_FATAL_ERROR;
|
|
|
|
}
|
|
|
|
if (ref_key.num_keys() != input_key.num_keys()) {
|
|
|
|
VTR_LOG_ERROR(
|
|
|
|
"Different number of keys between reference key (='%lu') and input key "
|
|
|
|
"('=%lu')!\n",
|
|
|
|
ref_key.num_keys(), input_key.num_keys());
|
|
|
|
return openfpga::CMD_EXEC_FATAL_ERROR;
|
|
|
|
}
|
|
|
|
size_t num_errors = 0;
|
|
|
|
size_t curr_num_err = 0;
|
|
|
|
VTR_LOG("Checking key alias in reference key...\n");
|
2023-08-26 20:07:08 -05:00
|
|
|
curr_num_err = openfpga::check_fabric_key_alias(ref_key, verbose);
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG("Checking key alias in reference key... %s\n",
|
|
|
|
curr_num_err ? "[Fail]" : "[Pass]");
|
|
|
|
VTR_LOG("Checking key names and values in reference key...\n");
|
2023-08-26 20:07:08 -05:00
|
|
|
curr_num_err = openfpga::check_fabric_key_names_and_values(ref_key, verbose);
|
2023-08-26 15:19:18 -05:00
|
|
|
num_errors += curr_num_err;
|
|
|
|
VTR_LOG("Checking key names and valus in reference key... %s\n",
|
|
|
|
curr_num_err ? "[Fail]" : "[Pass]");
|
|
|
|
VTR_LOG("Checking key alias in input key...\n");
|
2023-08-26 20:07:08 -05:00
|
|
|
curr_num_err = openfpga::check_fabric_key_alias(input_key, verbose);
|
2023-08-26 15:19:18 -05:00
|
|
|
num_errors += curr_num_err;
|
|
|
|
VTR_LOG("Checking key alias in input key... %s\n",
|
|
|
|
curr_num_err ? "[Fail]" : "[Pass]");
|
2023-08-26 20:15:30 -05:00
|
|
|
num_errors +=
|
|
|
|
check_input_and_ref_key_alias_match(input_key, ref_key, verbose);
|
2023-08-26 15:19:18 -05:00
|
|
|
return num_errors ? openfpga::CMD_EXEC_FATAL_ERROR
|
|
|
|
: openfpga::CMD_EXEC_SUCCESS;
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @brief Checks to be done:
|
|
|
|
* - Each alias of input key can be found in the reference key
|
2023-08-26 15:19:18 -05:00
|
|
|
* - Update input key with pair of name and value which matches the alias from
|
|
|
|
* the reference key
|
2023-08-26 14:54:12 -05:00
|
|
|
*/
|
2023-08-26 15:19:18 -05:00
|
|
|
static int update_input_key(openfpga::FabricKey& input_key,
|
2023-08-26 20:07:08 -05:00
|
|
|
const openfpga::FabricKey& ref_key,
|
|
|
|
const bool& verbose) {
|
2023-08-26 14:54:12 -05:00
|
|
|
size_t num_errors = 0;
|
|
|
|
size_t num_keys_checked = 0;
|
|
|
|
float progress = 0.;
|
|
|
|
VTR_LOG("Pairing key alias between reference key and input keys...\n");
|
2023-08-26 15:19:18 -05:00
|
|
|
for (openfpga::FabricKeyId key_id : input_key.keys()) {
|
2023-08-26 14:54:12 -05:00
|
|
|
/* Note that this is slow. May consider to build a map first */
|
|
|
|
std::string curr_alias = input_key.key_alias(key_id);
|
2023-08-26 15:19:18 -05:00
|
|
|
std::vector<openfpga::FabricKeyId> ref_found_keys =
|
|
|
|
ref_key.find_key_by_alias(curr_alias);
|
|
|
|
progress = static_cast<float>(num_keys_checked) /
|
|
|
|
static_cast<float>(input_key.num_keys()) * 100.0;
|
2023-08-26 20:07:08 -05:00
|
|
|
VTR_LOGV(verbose, "[%lu%] Pairing key alias '%s'\r", size_t(progress),
|
2023-08-26 20:15:30 -05:00
|
|
|
curr_alias.c_str());
|
2023-08-26 14:54:12 -05:00
|
|
|
if (ref_found_keys.empty()) {
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG_ERROR(
|
2023-08-26 20:15:30 -05:00
|
|
|
"Invalid alias '%s' in the input key (id='%lu'), which does not "
|
2023-08-26 15:19:18 -05:00
|
|
|
"exist in the reference key!\n",
|
|
|
|
curr_alias.c_str(), size_t(key_id));
|
2023-08-26 14:54:12 -05:00
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
if (ref_found_keys.size() > 1) {
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG_ERROR(
|
2023-08-26 20:15:30 -05:00
|
|
|
"Invalid alias '%s' in the reference key (id='%lu'), which have been "
|
2023-08-26 15:19:18 -05:00
|
|
|
"found %lu times!\n",
|
|
|
|
curr_alias.c_str(), size_t(key_id));
|
2023-08-26 14:54:12 -05:00
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
/* Now we have a key, get the name and value, and update input key */
|
|
|
|
input_key.set_key_name(key_id, ref_key.key_name(ref_found_keys[0]));
|
|
|
|
input_key.set_key_value(key_id, ref_key.key_value(ref_found_keys[0]));
|
2023-08-26 20:15:30 -05:00
|
|
|
VTR_LOGV(verbose, "[%lu%] Pairing key alias '%s' -> ('%s', %lu)\r",
|
|
|
|
size_t(progress), curr_alias.c_str(),
|
|
|
|
input_key.key_name(key_id).c_str(), input_key.key_value(key_id));
|
2023-08-26 14:54:12 -05:00
|
|
|
num_keys_checked++;
|
|
|
|
}
|
2023-08-26 15:19:18 -05:00
|
|
|
return num_errors ? openfpga::CMD_EXEC_FATAL_ERROR
|
|
|
|
: openfpga::CMD_EXEC_SUCCESS;
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @brief Checks to be done:
|
|
|
|
* - Number of configuration regions match
|
|
|
|
* - Number of keys match
|
|
|
|
* - Each alias can be found in the reference key
|
|
|
|
*/
|
2023-08-26 15:19:18 -05:00
|
|
|
static int check_and_update_input_key(openfpga::FabricKey& input_key,
|
2023-08-26 20:07:08 -05:00
|
|
|
const openfpga::FabricKey& ref_key,
|
|
|
|
const bool& verbose) {
|
2023-08-26 15:19:18 -05:00
|
|
|
int status = openfpga::CMD_EXEC_SUCCESS;
|
2023-08-26 20:07:08 -05:00
|
|
|
status = check_input_key(input_key, ref_key, verbose);
|
2023-08-26 15:19:18 -05:00
|
|
|
if (status != openfpga::CMD_EXEC_SUCCESS) {
|
|
|
|
return openfpga::CMD_EXEC_FATAL_ERROR;
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|
2023-08-26 20:07:08 -05:00
|
|
|
return update_input_key(input_key, ref_key, verbose);
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, const char** argv) {
|
2023-08-26 15:19:18 -05:00
|
|
|
/* Create a new command and Initialize the options available in the user
|
|
|
|
* interface */
|
2023-08-26 14:54:12 -05:00
|
|
|
openfpga::Command cmd("fabric_key_assistant");
|
2023-08-26 15:19:18 -05:00
|
|
|
openfpga::CommandOptionId opt_ref =
|
|
|
|
cmd.add_option("reference", true, "Specify the reference fabric key file");
|
|
|
|
cmd.set_option_require_value(opt_ref, openfpga::OPT_STRING);
|
|
|
|
openfpga::CommandOptionId opt_input =
|
|
|
|
cmd.add_option("input", true, "Specify the hand-crafted fabric key file");
|
|
|
|
cmd.set_option_require_value(opt_input, openfpga::OPT_STRING);
|
|
|
|
openfpga::CommandOptionId opt_output = cmd.add_option(
|
|
|
|
"output", true, "Specify the final fabric key file to be outputted");
|
|
|
|
cmd.set_option_require_value(opt_output, openfpga::OPT_STRING);
|
2023-08-26 20:07:08 -05:00
|
|
|
openfpga::CommandOptionId opt_verbose =
|
|
|
|
cmd.add_option("verbose", false, "Show verbose outputs");
|
2023-08-26 15:19:18 -05:00
|
|
|
openfpga::CommandOptionId opt_help =
|
|
|
|
cmd.add_option("help", false, "Show help desk");
|
2023-08-26 14:54:12 -05:00
|
|
|
|
|
|
|
/* Parse the option, to avoid issues, we use the command name to replace the
|
|
|
|
* argv[0] */
|
|
|
|
std::vector<std::string> cmd_opts = format_argv(cmd.name(), argc, argv);
|
|
|
|
|
2023-08-26 15:19:18 -05:00
|
|
|
openfpga::CommandContext cmd_ctx(cmd);
|
|
|
|
if (false == parse_command(cmd_opts, cmd, cmd_ctx) ||
|
|
|
|
cmd_ctx.option_enable(cmd, opt_help)) {
|
2023-08-26 14:54:12 -05:00
|
|
|
/* Echo the command */
|
|
|
|
print_command_options(cmd);
|
2023-08-26 15:19:18 -05:00
|
|
|
return openfpga::CMD_EXEC_FATAL_ERROR;
|
2023-08-26 14:54:12 -05:00
|
|
|
} else {
|
|
|
|
/* Let user to confirm selected options */
|
|
|
|
print_command_context(cmd, cmd_ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse the fabric key from an XML file */
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG("Read the reference fabric key from an XML file: %s.\n",
|
|
|
|
cmd_ctx.option_value(cmd, opt_ref).c_str());
|
|
|
|
openfpga::FabricKey ref_key =
|
|
|
|
openfpga::read_xml_fabric_key(cmd_ctx.option_value(cmd, opt_ref).c_str());
|
2023-08-26 14:54:12 -05:00
|
|
|
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG("Read the hand-crafted fabric key from an XML file: %s.\n",
|
|
|
|
cmd_ctx.option_value(cmd, opt_input).c_str());
|
|
|
|
openfpga::FabricKey input_key =
|
|
|
|
openfpga::read_xml_fabric_key(cmd_ctx.option_value(cmd, opt_input).c_str());
|
2023-08-26 14:54:12 -05:00
|
|
|
|
|
|
|
/* Check the input key */
|
2023-08-26 20:15:30 -05:00
|
|
|
if (check_and_update_input_key(input_key, ref_key,
|
|
|
|
cmd_ctx.option_enable(cmd, opt_verbose))) {
|
2023-08-26 14:54:12 -05:00
|
|
|
return openfpga::CMD_EXEC_FATAL_ERROR;
|
|
|
|
}
|
|
|
|
|
2023-08-26 15:19:18 -05:00
|
|
|
VTR_LOG("Write the final fabric key to an XML file: %s.\n",
|
|
|
|
cmd_ctx.option_value(cmd, opt_output).c_str());
|
|
|
|
return openfpga::write_xml_fabric_key(
|
|
|
|
cmd_ctx.option_value(cmd, opt_output).c_str(), input_key);
|
2023-08-26 14:54:12 -05:00
|
|
|
}
|