2020-01-21 21:46:24 -06:00
|
|
|
/*********************************************************************
|
|
|
|
* Member functions for class Shell
|
|
|
|
********************************************************************/
|
|
|
|
#include <fstream>
|
2020-01-22 17:05:14 -06:00
|
|
|
#include <algorithm>
|
2020-01-21 21:46:24 -06:00
|
|
|
|
|
|
|
/* Headers from vtrutil library */
|
|
|
|
#include "vtr_log.h"
|
|
|
|
#include "vtr_assert.h"
|
|
|
|
|
|
|
|
/* Headers from openfpgautil library */
|
|
|
|
#include "openfpga_tokenizer.h"
|
|
|
|
|
|
|
|
/* Headers from readline library */
|
|
|
|
#include <readline/readline.h>
|
|
|
|
#include <readline/history.h>
|
|
|
|
|
|
|
|
/* Headers from openfpgashell library */
|
|
|
|
#include "command_parser.h"
|
|
|
|
#include "command_echo.h"
|
|
|
|
|
2020-01-22 17:49:32 -06:00
|
|
|
/* Begin namespace openfpga */
|
|
|
|
namespace openfpga {
|
2020-01-21 21:46:24 -06:00
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* Public constructors
|
|
|
|
********************************************************************/
|
|
|
|
template<class T>
|
2022-11-23 18:52:22 -06:00
|
|
|
Shell<T>::Shell() {
|
|
|
|
name_ = std::string("shell_no_name");
|
2020-07-02 19:02:33 -05:00
|
|
|
time_start_ = 0;
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* Public Accessors : aggregates
|
|
|
|
***********************************************************************/
|
|
|
|
template<class T>
|
|
|
|
std::string Shell<T>::name() const {
|
|
|
|
return name_;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
std::string Shell<T>::title() const {
|
|
|
|
return title_;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
typename Shell<T>::shell_command_range Shell<T>::commands() const {
|
|
|
|
return vtr::make_range(command_ids_.begin(), command_ids_.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
ShellCommandId Shell<T>::command(const std::string& name) const {
|
|
|
|
/* Ensure that the name is unique in the command list */
|
|
|
|
std::map<std::string, ShellCommandId>::const_iterator name_it = command_name2ids_.find(name);
|
|
|
|
if (name_it == command_name2ids_.end()) {
|
|
|
|
return ShellCommandId::INVALID();
|
|
|
|
}
|
|
|
|
return command_name2ids_.at(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
std::string Shell<T>::command_description(const ShellCommandId& cmd_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
return command_description_[cmd_id];
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template<class T>
|
|
|
|
ShellCommandClassId Shell<T>::command_class(const ShellCommandId& cmd_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
return command_classes_[cmd_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
std::string Shell<T>::command_class_name(const ShellCommandClassId& cmd_class_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_class_id(cmd_class_id));
|
|
|
|
return command_class_names_[cmd_class_id];
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
template<class T>
|
|
|
|
const Command& Shell<T>::command(const ShellCommandId& cmd_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
return commands_[cmd_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
const CommandContext& Shell<T>::command_context(const ShellCommandId& cmd_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
return command_contexts_[cmd_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
std::vector<ShellCommandId> Shell<T>::command_dependency(const ShellCommandId& cmd_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
return command_dependencies_[cmd_id];
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template<class T>
|
|
|
|
std::vector<ShellCommandId> Shell<T>::commands_by_class(const ShellCommandClassId& cmd_class_id) const {
|
|
|
|
VTR_ASSERT(true == valid_command_class_id(cmd_class_id));
|
|
|
|
return commands_by_classes_[cmd_class_id];
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/************************************************************************
|
|
|
|
* Public mutators
|
|
|
|
***********************************************************************/
|
2022-11-23 18:52:22 -06:00
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_name(const char* name) {
|
|
|
|
name_ = std::string(name);
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
template<class T>
|
2020-01-21 23:59:53 -06:00
|
|
|
void Shell<T>::add_title(const char* title) {
|
2020-01-21 21:46:24 -06:00
|
|
|
title_ = std::string(title);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add a command with it description */
|
|
|
|
template<class T>
|
|
|
|
ShellCommandId Shell<T>::add_command(const Command& cmd, const char* descr) {
|
|
|
|
/* Ensure that the name is unique in the command list */
|
|
|
|
std::map<std::string, ShellCommandId>::const_iterator name_it = command_name2ids_.find(std::string(cmd.name()));
|
|
|
|
if (name_it != command_name2ids_.end()) {
|
|
|
|
return ShellCommandId::INVALID();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is a legal name. we can create a new id */
|
|
|
|
ShellCommandId shell_cmd = ShellCommandId(command_ids_.size());
|
|
|
|
command_ids_.push_back(shell_cmd);
|
|
|
|
commands_.emplace_back(cmd);
|
|
|
|
command_contexts_.push_back(CommandContext(cmd));
|
|
|
|
command_description_.push_back(descr);
|
2020-01-22 17:05:14 -06:00
|
|
|
command_classes_.push_back(ShellCommandClassId::INVALID());
|
|
|
|
command_execute_function_types_.emplace_back();
|
2020-01-23 15:50:03 -06:00
|
|
|
command_const_execute_functions_.emplace_back();
|
2020-01-22 17:05:14 -06:00
|
|
|
command_standard_execute_functions_.emplace_back();
|
2020-01-23 15:50:03 -06:00
|
|
|
command_short_const_execute_functions_.emplace_back();
|
2020-01-22 17:05:14 -06:00
|
|
|
command_short_execute_functions_.emplace_back();
|
|
|
|
command_builtin_execute_functions_.emplace_back();
|
2023-01-01 12:10:09 -06:00
|
|
|
command_wrapper_execute_functions_.emplace_back();
|
2020-01-22 20:30:36 -06:00
|
|
|
command_macro_execute_functions_.emplace_back();
|
2020-04-08 17:18:05 -05:00
|
|
|
command_status_.push_back(CMD_EXEC_NONE); /* By default, the command should be marked as fatal error as it has been never executed */
|
2020-01-21 21:46:24 -06:00
|
|
|
command_dependencies_.emplace_back();
|
|
|
|
|
|
|
|
/* Register the name in the name2id map */
|
2020-01-21 23:59:53 -06:00
|
|
|
command_name2ids_[cmd.name()] = shell_cmd;
|
2020-01-21 21:46:24 -06:00
|
|
|
|
|
|
|
return shell_cmd;
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_command_class(const ShellCommandId& cmd_id, const ShellCommandClassId& cmd_class_id) {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
VTR_ASSERT(true == valid_command_class_id(cmd_class_id));
|
|
|
|
command_classes_[cmd_id] = cmd_class_id;
|
|
|
|
/* Update the fast look-up to spot commands in a class */
|
|
|
|
std::vector<ShellCommandId>::iterator it = std::find(commands_by_classes_[cmd_class_id].begin(), commands_by_classes_[cmd_class_id].end(), cmd_id);
|
|
|
|
/* The command does not exist in the class, add it */
|
|
|
|
if (it == commands_by_classes_[cmd_class_id].end()) {
|
|
|
|
commands_by_classes_[cmd_class_id].push_back(cmd_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-23 15:50:03 -06:00
|
|
|
template<class T>
|
2020-01-23 21:58:15 -06:00
|
|
|
void Shell<T>::set_command_const_execute_function(const ShellCommandId& cmd_id,
|
2020-04-08 17:18:05 -05:00
|
|
|
std::function<int(const T&, const Command&, const CommandContext&)> exec_func) {
|
2020-01-23 15:50:03 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = CONST_STANDARD;
|
2020-01-23 21:58:15 -06:00
|
|
|
command_const_execute_functions_[cmd_id] = exec_func;
|
2020-01-23 15:50:03 -06:00
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
template<class T>
|
2020-01-21 23:59:53 -06:00
|
|
|
void Shell<T>::set_command_execute_function(const ShellCommandId& cmd_id,
|
2020-04-08 17:18:05 -05:00
|
|
|
std::function<int(T&, const Command&, const CommandContext&)> exec_func) {
|
2020-01-21 21:46:24 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
2020-01-22 17:05:14 -06:00
|
|
|
command_execute_function_types_[cmd_id] = STANDARD;
|
|
|
|
command_standard_execute_functions_[cmd_id] = exec_func;
|
|
|
|
}
|
|
|
|
|
2020-01-23 15:50:03 -06:00
|
|
|
template<class T>
|
2020-01-23 21:58:15 -06:00
|
|
|
void Shell<T>::set_command_const_execute_function(const ShellCommandId& cmd_id,
|
2020-04-08 17:18:05 -05:00
|
|
|
std::function<int(const T&)> exec_func) {
|
2020-01-23 15:50:03 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = CONST_SHORT;
|
2020-01-23 21:58:15 -06:00
|
|
|
command_short_const_execute_functions_[cmd_id] = exec_func;
|
2020-01-23 15:50:03 -06:00
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_command_execute_function(const ShellCommandId& cmd_id,
|
2020-04-08 17:18:05 -05:00
|
|
|
std::function<int(T&)> exec_func) {
|
2020-01-22 17:05:14 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = SHORT;
|
|
|
|
command_short_execute_functions_[cmd_id] = exec_func;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_command_execute_function(const ShellCommandId& cmd_id,
|
|
|
|
std::function<void()> exec_func) {
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = BUILTIN;
|
|
|
|
command_builtin_execute_functions_[cmd_id] = exec_func;
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
2020-01-22 20:30:36 -06:00
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_command_execute_function(const ShellCommandId& cmd_id,
|
2020-01-23 15:57:53 -06:00
|
|
|
std::function<int(int, char**)> exec_func) {
|
2020-01-22 20:30:36 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = MACRO;
|
|
|
|
command_macro_execute_functions_[cmd_id] = exec_func;
|
|
|
|
}
|
|
|
|
|
2023-01-01 12:10:09 -06:00
|
|
|
template<class T>
|
|
|
|
void Shell<T>::set_command_execute_function(const ShellCommandId& cmd_id,
|
2023-01-01 12:22:47 -06:00
|
|
|
std::function<int(Shell<T>*, T&, const Command&, const CommandContext&)> exec_func) {
|
2023-01-01 12:10:09 -06:00
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
command_execute_function_types_[cmd_id] = WRAPPER;
|
|
|
|
command_wrapper_execute_functions_[cmd_id] = exec_func;
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
template<class T>
|
2020-01-21 23:59:53 -06:00
|
|
|
void Shell<T>::set_command_dependency(const ShellCommandId& cmd_id,
|
2020-01-24 17:46:39 -06:00
|
|
|
const std::vector<ShellCommandId>& dependent_cmds) {
|
2020-01-21 21:46:24 -06:00
|
|
|
/* Validate the command id as well as each of the command dependency */
|
|
|
|
VTR_ASSERT(true == valid_command_id(cmd_id));
|
|
|
|
for (ShellCommandId dependent_cmd : dependent_cmds) {
|
|
|
|
VTR_ASSERT(true == valid_command_id(dependent_cmd));
|
|
|
|
}
|
|
|
|
command_dependencies_[cmd_id] = dependent_cmds;
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
/* Add a command with it description */
|
|
|
|
template<class T>
|
|
|
|
ShellCommandClassId Shell<T>::add_command_class(const char* name) {
|
|
|
|
/* Ensure that the name is unique in the command list */
|
|
|
|
std::map<std::string, ShellCommandClassId>::const_iterator name_it = command_class2ids_.find(std::string(name));
|
|
|
|
if (name_it != command_class2ids_.end()) {
|
|
|
|
return ShellCommandClassId::INVALID();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This is a legal name. we can create a new id */
|
|
|
|
ShellCommandClassId cmd_class = ShellCommandClassId(command_class_ids_.size());
|
|
|
|
command_class_ids_.push_back(cmd_class);
|
|
|
|
command_class_names_.push_back(std::string(name));
|
|
|
|
|
|
|
|
/* Register the name in the name2id map */
|
|
|
|
command_class2ids_[std::string(name)] = cmd_class;
|
|
|
|
|
|
|
|
/* Register in the fast look-up for commands by classes */
|
|
|
|
commands_by_classes_.emplace_back();
|
|
|
|
|
|
|
|
return cmd_class;
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/************************************************************************
|
|
|
|
* Public executors
|
|
|
|
***********************************************************************/
|
|
|
|
template <class T>
|
2020-01-27 18:56:24 -06:00
|
|
|
void Shell<T>::run_interactive_mode(T& context, const bool& quiet_mode) {
|
|
|
|
if (false == quiet_mode) {
|
2020-07-02 19:02:33 -05:00
|
|
|
/* Reset timer since it does not come from another mode */
|
|
|
|
time_start_ = std::clock();
|
|
|
|
|
2020-01-27 18:56:24 -06:00
|
|
|
VTR_LOG("Start interactive mode of %s...\n",
|
|
|
|
name().c_str());
|
|
|
|
|
|
|
|
/* Print the title of the shell */
|
|
|
|
if (!title().empty()) {
|
|
|
|
VTR_LOG("%s\n", title().c_str());
|
|
|
|
}
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for users input and execute the command */
|
|
|
|
char* cmd_line;
|
|
|
|
while ((cmd_line = readline(std::string(name() + std::string("> ")).c_str())) != nullptr) {
|
|
|
|
/* If the line is not empty:
|
|
|
|
* Try to execute the command and
|
|
|
|
* Add to history
|
|
|
|
*/
|
|
|
|
if (strlen(cmd_line) > 0) {
|
|
|
|
execute_command((const char*)cmd_line, context);
|
|
|
|
add_history(cmd_line);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Free the line as readline malloc a new line each time */
|
|
|
|
free(cmd_line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2021-01-24 15:33:58 -06:00
|
|
|
void Shell<T>::run_script_mode(const char* script_file_name,
|
|
|
|
T& context,
|
|
|
|
const bool& batch_mode) {
|
2020-01-21 21:46:24 -06:00
|
|
|
|
2020-07-02 19:02:33 -05:00
|
|
|
time_start_ = std::clock();
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
VTR_LOG("Reading script file %s...\n", script_file_name);
|
|
|
|
|
|
|
|
/* Print the title of the shell */
|
|
|
|
if (!title().empty()) {
|
2020-01-21 23:59:53 -06:00
|
|
|
VTR_LOG("%s\n", title().c_str());
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string line;
|
|
|
|
|
|
|
|
/* Create an input file stream */
|
|
|
|
std::ifstream fp(script_file_name);
|
|
|
|
|
|
|
|
if (!fp.is_open()) {
|
|
|
|
/* Fail to open the file, ask user to check */
|
|
|
|
VTR_LOG("Fail to open the script file: %s! Please check its location\n",
|
|
|
|
script_file_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-07 22:27:51 -05:00
|
|
|
/* Consider that each line may not end due to the continued line charactor
|
|
|
|
* Use cmd_line to conjunct multiple lines
|
|
|
|
*/
|
|
|
|
std::string cmd_line;
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/* Read line by line */
|
|
|
|
while (getline(fp, line)) {
|
2020-06-25 11:18:05 -05:00
|
|
|
/* Skip empty line */
|
|
|
|
if (true == line.empty()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/* If the line that starts with '#', it is commented, we can skip */
|
|
|
|
if ('#' == line.front()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Try to split the line with '#', the string before '#' is the read command we want */
|
|
|
|
std::string cmd_part = line;
|
|
|
|
std::size_t cmd_end_pos = line.find_first_of('#');
|
|
|
|
/* If the full line has '#', we need the part before it */
|
|
|
|
if (cmd_end_pos != std::string::npos) {
|
|
|
|
cmd_part = line.substr(0, cmd_end_pos);
|
|
|
|
}
|
2020-04-07 22:27:51 -05:00
|
|
|
|
|
|
|
/* Remove the space at the end of the line
|
|
|
|
* So that we can check easily if there is a continued line in the end
|
|
|
|
*/
|
2020-04-13 12:08:31 -05:00
|
|
|
StringToken cmd_part_tokenizer(cmd_part);
|
|
|
|
cmd_part_tokenizer.rtrim(std::string(" "));
|
|
|
|
cmd_part = cmd_part_tokenizer.data();
|
2020-04-07 22:27:51 -05:00
|
|
|
|
|
|
|
/* If the line ends with '\', this is a continued line, parse the next until it ends */
|
|
|
|
if ('\\' == cmd_part.back()) {
|
|
|
|
/* Pop up the last charactor and conjunct to cmd_line */
|
|
|
|
cmd_part.pop_back();
|
|
|
|
|
|
|
|
if (!cmd_part.empty()) {
|
|
|
|
cmd_line += cmd_part;
|
|
|
|
}
|
|
|
|
/* Not finished yet. Parse the next line */
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
/* End of this line, if cmd_line is empty,
|
|
|
|
* there is no previous lines, cache the part we have
|
|
|
|
* and then execute the command
|
|
|
|
*/
|
|
|
|
cmd_line += cmd_part;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove the space at the beginning of the line */
|
2020-04-13 12:08:31 -05:00
|
|
|
StringToken cmd_line_tokenizer(cmd_line);
|
|
|
|
cmd_line_tokenizer.ltrim(std::string(" "));
|
|
|
|
cmd_line = cmd_line_tokenizer.data();
|
2020-04-07 22:27:51 -05:00
|
|
|
|
|
|
|
/* Process the command only when the full command line in ended */
|
|
|
|
if (!cmd_line.empty()) {
|
|
|
|
VTR_LOG("\nCommand line to execute: %s\n", cmd_line.c_str());
|
2020-04-08 17:18:05 -05:00
|
|
|
int status = execute_command(cmd_line.c_str(), context);
|
2020-04-07 22:27:51 -05:00
|
|
|
/* Empty the line ready to start a new line */
|
|
|
|
cmd_line.clear();
|
2020-04-08 17:18:05 -05:00
|
|
|
|
2021-01-24 15:33:58 -06:00
|
|
|
/* Check the execution status of the command,
|
|
|
|
* if fatal error happened, we should abort immediately
|
|
|
|
*/
|
2020-04-08 17:18:05 -05:00
|
|
|
if (CMD_EXEC_FATAL_ERROR == status) {
|
2021-01-24 15:33:58 -06:00
|
|
|
VTR_LOG("Fatal error occurred!\n");
|
2021-01-24 18:44:51 -06:00
|
|
|
/* If in the batch mode, we will exit with errors */
|
2022-12-30 13:58:45 -06:00
|
|
|
VTR_LOGV(batch_mode, "%s Abort\n", name_.c_str());
|
2021-01-24 18:44:51 -06:00
|
|
|
if (batch_mode) {
|
|
|
|
exit(CMD_EXEC_FATAL_ERROR);
|
|
|
|
}
|
|
|
|
/* If not in the batch mode, we will got to interactive mode */
|
2021-01-24 15:33:58 -06:00
|
|
|
VTR_LOGV(!batch_mode, "Enter interactive mode\n");
|
2020-04-08 17:18:05 -05:00
|
|
|
break;
|
|
|
|
}
|
2020-01-22 17:05:14 -06:00
|
|
|
}
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
fp.close();
|
2020-01-27 18:56:24 -06:00
|
|
|
|
2021-01-24 15:33:58 -06:00
|
|
|
/* If not in batch mode, switch to interactive mode, stay tuned */
|
|
|
|
if (!batch_mode) {
|
|
|
|
run_interactive_mode(context, true);
|
|
|
|
}
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template <class T>
|
|
|
|
void Shell<T>::print_commands() const {
|
|
|
|
/* Print the commands by their classes */
|
|
|
|
for (const ShellCommandClassId& cmd_class : command_class_ids_) {
|
|
|
|
/* Print the class name */
|
|
|
|
VTR_LOG("%s:\n", command_class_names_[cmd_class].c_str());
|
|
|
|
|
|
|
|
for (const ShellCommandId& cmd : commands_by_classes_[cmd_class]) {
|
|
|
|
/* Print the command names in this class
|
|
|
|
* but limited4 command per line for a clean layout
|
|
|
|
*/
|
2020-03-20 12:07:45 -05:00
|
|
|
VTR_LOG("\t%s\n", commands_[cmd].name().c_str());
|
2020-01-22 17:05:14 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Put a new line in the end as a splitter */
|
|
|
|
VTR_LOG("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put a new line in the end as a splitter */
|
|
|
|
VTR_LOG("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2021-01-24 15:48:27 -06:00
|
|
|
int Shell<T>::exit_code() const {
|
2020-04-08 17:18:05 -05:00
|
|
|
/* Check all the command status, if we see fatal errors or minor errors, we drop an error code */
|
|
|
|
int exit_code = 0;
|
|
|
|
for (const int& status : command_status_) {
|
|
|
|
if ( (status == CMD_EXEC_FATAL_ERROR)
|
|
|
|
|| (status == CMD_EXEC_MINOR_ERROR) ) {
|
|
|
|
exit_code = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-24 15:48:27 -06:00
|
|
|
return exit_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
int Shell<T>::execution_errors() const {
|
2020-04-08 17:18:05 -05:00
|
|
|
/* Show error message if we detect any errors */
|
|
|
|
int num_err = 0;
|
2021-01-24 15:48:27 -06:00
|
|
|
|
|
|
|
for (const ShellCommandId& cmd : commands()) {
|
|
|
|
if (command_status_[cmd] == CMD_EXEC_FATAL_ERROR) {
|
|
|
|
VTR_LOG_ERROR("Command '%s' execution has fatal errors\n",
|
|
|
|
commands_[cmd].name().c_str());
|
|
|
|
num_err++;
|
2020-04-08 17:18:05 -05:00
|
|
|
}
|
2021-01-24 15:48:27 -06:00
|
|
|
|
|
|
|
if (command_status_[cmd] == CMD_EXEC_MINOR_ERROR) {
|
|
|
|
VTR_LOG_ERROR("Command '%s' execution has minor errors\n",
|
|
|
|
commands_[cmd].name().c_str());
|
|
|
|
num_err++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return num_err;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2021-01-24 18:44:51 -06:00
|
|
|
void Shell<T>::exit(const int& init_err) const {
|
2021-01-24 15:48:27 -06:00
|
|
|
/* Check all the command status, if we see fatal errors or minor errors, we drop an error code */
|
2021-01-24 18:44:51 -06:00
|
|
|
int shell_exit_code = exit_code() | init_err;
|
2021-01-24 15:48:27 -06:00
|
|
|
|
|
|
|
/* Show error message if we detect any errors */
|
2021-01-24 18:44:51 -06:00
|
|
|
int num_err = init_err;
|
|
|
|
if (CMD_EXEC_SUCCESS != shell_exit_code) {
|
2021-01-24 15:48:27 -06:00
|
|
|
VTR_LOG("\n");
|
2021-01-24 18:44:51 -06:00
|
|
|
num_err += execution_errors();
|
2020-04-08 17:18:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
VTR_LOG("\nFinish execution with %d errors\n",
|
|
|
|
num_err);
|
|
|
|
|
2022-12-30 13:58:45 -06:00
|
|
|
VTR_LOG("\nThe entire %s flow took %g seconds\n",
|
|
|
|
name_.c_str(),
|
2020-07-02 19:02:33 -05:00
|
|
|
(double)(std::clock() - time_start_) / (double)CLOCKS_PER_SEC);
|
|
|
|
|
2020-01-22 18:11:24 -06:00
|
|
|
VTR_LOG("\nThank you for using %s!\n",
|
2020-01-22 17:05:14 -06:00
|
|
|
name().c_str());
|
2020-04-08 17:18:05 -05:00
|
|
|
|
2021-01-24 15:48:27 -06:00
|
|
|
std::exit(shell_exit_code);
|
2020-01-22 17:05:14 -06:00
|
|
|
}
|
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/************************************************************************
|
|
|
|
* Private executors
|
|
|
|
***********************************************************************/
|
|
|
|
template <class T>
|
2020-04-08 17:18:05 -05:00
|
|
|
int Shell<T>::execute_command(const char* cmd_line,
|
2023-01-01 12:10:09 -06:00
|
|
|
T& common_context) {
|
2020-01-21 21:46:24 -06:00
|
|
|
/* Tokenize the line */
|
|
|
|
openfpga::StringToken tokenizer(cmd_line);
|
|
|
|
std::vector<std::string> tokens = tokenizer.split(" ");
|
|
|
|
|
|
|
|
/* Find if the command name is valid */
|
|
|
|
ShellCommandId cmd_id = command(tokens[0]);
|
|
|
|
if (ShellCommandId::INVALID() == cmd_id) {
|
|
|
|
VTR_LOG("Try to call a command '%s' which is not defined!\n",
|
|
|
|
tokens[0].c_str());
|
2020-04-08 17:18:05 -05:00
|
|
|
return CMD_EXEC_FATAL_ERROR;
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
2020-01-24 17:46:39 -06:00
|
|
|
/* Check the dependency graph to see if all the prequistics have been met */
|
|
|
|
for (const ShellCommandId& dep_cmd : command_dependencies_[cmd_id]) {
|
2020-04-08 17:18:05 -05:00
|
|
|
if ( (CMD_EXEC_NONE == command_status_[dep_cmd])
|
|
|
|
|| (CMD_EXEC_FATAL_ERROR == command_status_[dep_cmd]) ) {
|
2020-01-24 17:46:39 -06:00
|
|
|
VTR_LOG("Command '%s' is required to be executed before command '%s'!\n",
|
|
|
|
commands_[dep_cmd].name().c_str(), commands_[cmd_id].name().c_str());
|
2020-03-09 10:34:21 -05:00
|
|
|
/* Echo the command help desk */
|
|
|
|
print_command_options(commands_[cmd_id]);
|
2021-01-24 18:25:03 -06:00
|
|
|
command_status_[cmd_id] = CMD_EXEC_FATAL_ERROR;
|
2020-04-08 17:18:05 -05:00
|
|
|
return CMD_EXEC_FATAL_ERROR;
|
2020-01-24 17:46:39 -06:00
|
|
|
}
|
|
|
|
}
|
2020-01-22 17:05:14 -06:00
|
|
|
|
2020-01-22 20:30:36 -06:00
|
|
|
/* Find the command! Parse the options
|
|
|
|
* Note:
|
|
|
|
* Macro command will not be parsed! It will be directly executed
|
|
|
|
*/
|
|
|
|
if (MACRO == command_execute_function_types_[cmd_id]) {
|
|
|
|
/* Convert the tokens from string to char */
|
|
|
|
char** argv = (char**)malloc(tokens.size() * sizeof(char*));
|
|
|
|
for (size_t itok = 0; itok < tokens.size(); ++itok) {
|
|
|
|
argv[itok] = (char*)malloc((tokens[itok].length() + 1) * sizeof(char));
|
|
|
|
strcpy(argv[itok], tokens[itok].c_str());
|
|
|
|
}
|
2020-04-08 17:18:05 -05:00
|
|
|
/* Execute the marco function and record the execution status */
|
|
|
|
command_status_[cmd_id] = command_macro_execute_functions_[cmd_id](tokens.size(), argv);
|
2020-01-22 20:30:36 -06:00
|
|
|
/* Free the argv */
|
|
|
|
for (size_t itok = 0; itok < tokens.size(); ++itok) {
|
|
|
|
free(argv[itok]);
|
|
|
|
}
|
|
|
|
free(argv);
|
2020-01-27 18:43:22 -06:00
|
|
|
|
2020-01-22 20:30:36 -06:00
|
|
|
/* Finish for macro command, return */
|
2020-04-08 17:18:05 -05:00
|
|
|
return command_status_[cmd_id];
|
2020-01-22 20:30:36 -06:00
|
|
|
}
|
|
|
|
|
2020-05-06 17:20:04 -05:00
|
|
|
/* Reset the command parse results to initial status
|
|
|
|
* Avoid conflict when calling the same command in the second time
|
|
|
|
*/
|
|
|
|
command_contexts_[cmd_id].reset();
|
2020-01-21 21:46:24 -06:00
|
|
|
if (false == parse_command(tokens, commands_[cmd_id], command_contexts_[cmd_id])) {
|
|
|
|
/* Echo the command */
|
|
|
|
print_command_options(commands_[cmd_id]);
|
2021-01-24 18:25:03 -06:00
|
|
|
command_status_[cmd_id] = CMD_EXEC_FATAL_ERROR;
|
2020-04-08 17:18:05 -05:00
|
|
|
return CMD_EXEC_FATAL_ERROR;
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
2020-01-24 17:46:39 -06:00
|
|
|
|
2020-01-21 21:46:24 -06:00
|
|
|
/* Parse succeed. Let user to confirm selected options */
|
|
|
|
print_command_context(commands_[cmd_id], command_contexts_[cmd_id]);
|
2020-01-24 17:46:39 -06:00
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
/* Execute the command depending on the type of function ! */
|
|
|
|
switch (command_execute_function_types_[cmd_id]) {
|
2023-01-01 12:10:09 -06:00
|
|
|
case WRAPPER:
|
2023-01-01 12:22:47 -06:00
|
|
|
command_status_[cmd_id] = command_wrapper_execute_functions_[cmd_id](this, common_context, commands_[cmd_id], command_contexts_[cmd_id]);
|
2023-01-01 12:10:09 -06:00
|
|
|
break;
|
2020-01-23 15:50:03 -06:00
|
|
|
case CONST_STANDARD:
|
2020-04-08 17:18:05 -05:00
|
|
|
command_status_[cmd_id] = command_const_execute_functions_[cmd_id](common_context, commands_[cmd_id], command_contexts_[cmd_id]);
|
2020-01-23 15:50:03 -06:00
|
|
|
break;
|
2020-01-22 17:05:14 -06:00
|
|
|
case STANDARD:
|
2020-04-08 17:18:05 -05:00
|
|
|
command_status_[cmd_id] = command_standard_execute_functions_[cmd_id](common_context, commands_[cmd_id], command_contexts_[cmd_id]);
|
2020-01-22 17:05:14 -06:00
|
|
|
break;
|
2020-01-23 15:50:03 -06:00
|
|
|
case CONST_SHORT:
|
2020-04-08 17:18:05 -05:00
|
|
|
command_status_[cmd_id] = command_short_const_execute_functions_[cmd_id](common_context);
|
2020-01-23 15:50:03 -06:00
|
|
|
break;
|
2020-01-22 17:05:14 -06:00
|
|
|
case SHORT:
|
2020-04-08 17:18:05 -05:00
|
|
|
command_status_[cmd_id] = command_short_execute_functions_[cmd_id](common_context);
|
2020-01-22 17:05:14 -06:00
|
|
|
break;
|
|
|
|
case BUILTIN:
|
|
|
|
command_builtin_execute_functions_[cmd_id]();
|
2020-04-08 17:18:05 -05:00
|
|
|
/* Built-in execution is always correct */
|
|
|
|
command_status_[cmd_id] = CMD_EXEC_SUCCESS;
|
2020-01-22 17:05:14 -06:00
|
|
|
break;
|
2020-01-22 20:30:36 -06:00
|
|
|
/* MACRO should be executed eariler in this function. It should not appear here */
|
2020-01-22 17:05:14 -06:00
|
|
|
default:
|
|
|
|
/* This is not allowed! Error out */
|
2020-04-08 17:18:05 -05:00
|
|
|
VTR_LOG_ERROR("Invalid type of execute function for command '%s'!\n",
|
|
|
|
commands_[cmd_id].name().c_str());
|
2020-01-22 17:05:14 -06:00
|
|
|
/* Exit the shell using the exit() function inside this class! */
|
2020-04-08 17:18:05 -05:00
|
|
|
return CMD_EXEC_FATAL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Forbid users to return the status CMD_EXEC_NONE */
|
|
|
|
if (CMD_EXEC_NONE == command_status_[cmd_id]) {
|
|
|
|
VTR_LOG_ERROR("It is illegal to return never-executed status for an executed command!\n");
|
|
|
|
return CMD_EXEC_FATAL_ERROR;
|
2020-01-22 17:05:14 -06:00
|
|
|
}
|
2020-01-24 17:46:39 -06:00
|
|
|
|
2020-04-08 17:18:05 -05:00
|
|
|
return command_status_[cmd_id];
|
2020-01-21 21:46:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* Public invalidators/validators
|
|
|
|
***********************************************************************/
|
|
|
|
template<class T>
|
|
|
|
bool Shell<T>::valid_command_id(const ShellCommandId& cmd_id) const {
|
|
|
|
return ( size_t(cmd_id) < command_ids_.size() ) && ( cmd_id == command_ids_[cmd_id] );
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:05:14 -06:00
|
|
|
template<class T>
|
|
|
|
bool Shell<T>::valid_command_class_id(const ShellCommandClassId& cmd_class_id) const {
|
|
|
|
return ( size_t(cmd_class_id) < command_class_ids_.size() ) && ( cmd_class_id == command_class_ids_[cmd_class_id] );
|
|
|
|
}
|
|
|
|
|
2020-01-22 17:49:32 -06:00
|
|
|
} /* End namespace openfpga */
|