mirror of
https://github.com/YosysHQ/yosys
synced 2025-04-05 09:04:08 +00:00
Allows for more expressive code when constructing help messages for cells. Will also move extra logic in parsing help strings into the initial python parse instead of doing it in the C++ at export time.
1116 lines
32 KiB
C++
1116 lines
32 KiB
C++
/*
|
|
* yosys -- Yosys Open SYnthesis Suite
|
|
*
|
|
* Copyright (C) 2012 Claire Xenia Wolf <claire@yosyshq.com>
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
#include "kernel/yosys.h"
|
|
#include "kernel/satgen.h"
|
|
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
|
|
#ifdef YOSYS_ENABLE_ZLIB
|
|
#include <zlib.h>
|
|
|
|
PRIVATE_NAMESPACE_BEGIN
|
|
#define GZ_BUFFER_SIZE 8192
|
|
void decompress_gzip(const std::string &filename, std::stringstream &out)
|
|
{
|
|
char buffer[GZ_BUFFER_SIZE];
|
|
int bytes_read;
|
|
gzFile gzf = gzopen(filename.c_str(), "rb");
|
|
while(!gzeof(gzf)) {
|
|
bytes_read = gzread(gzf, reinterpret_cast<void *>(buffer), GZ_BUFFER_SIZE);
|
|
out.write(buffer, bytes_read);
|
|
}
|
|
gzclose(gzf);
|
|
}
|
|
|
|
/*
|
|
An output stream that uses a stringbuf to buffer data internally,
|
|
using zlib to write gzip-compressed data every time the stream is flushed.
|
|
*/
|
|
class gzip_ostream : public std::ostream {
|
|
public:
|
|
gzip_ostream() : std::ostream(nullptr)
|
|
{
|
|
rdbuf(&outbuf);
|
|
}
|
|
bool open(const std::string &filename)
|
|
{
|
|
return outbuf.open(filename);
|
|
}
|
|
private:
|
|
class gzip_streambuf : public std::stringbuf {
|
|
public:
|
|
gzip_streambuf() { };
|
|
bool open(const std::string &filename)
|
|
{
|
|
gzf = gzopen(filename.c_str(), "wb");
|
|
return gzf != nullptr;
|
|
}
|
|
virtual int sync() override
|
|
{
|
|
gzwrite(gzf, reinterpret_cast<const void *>(str().c_str()), unsigned(str().size()));
|
|
str("");
|
|
return 0;
|
|
}
|
|
virtual ~gzip_streambuf()
|
|
{
|
|
sync();
|
|
gzclose(gzf);
|
|
}
|
|
private:
|
|
gzFile gzf = nullptr;
|
|
} outbuf;
|
|
};
|
|
PRIVATE_NAMESPACE_END
|
|
|
|
#endif
|
|
|
|
YOSYS_NAMESPACE_BEGIN
|
|
|
|
#define MAX_REG_COUNT 1000
|
|
|
|
bool echo_mode = false;
|
|
Pass *first_queued_pass;
|
|
Pass *current_pass;
|
|
|
|
std::map<std::string, Frontend*> frontend_register;
|
|
std::map<std::string, Pass*> pass_register;
|
|
std::map<std::string, Backend*> backend_register;
|
|
|
|
std::vector<std::string> Frontend::next_args;
|
|
|
|
Pass::Pass(std::string name, std::string short_help) : pass_name(name), short_help(short_help)
|
|
{
|
|
next_queued_pass = first_queued_pass;
|
|
first_queued_pass = this;
|
|
call_counter = 0;
|
|
runtime_ns = 0;
|
|
}
|
|
|
|
void Pass::run_register()
|
|
{
|
|
if (pass_register.count(pass_name) && !replace_existing_pass())
|
|
log_error("Unable to register pass '%s', pass already exists!\n", pass_name.c_str());
|
|
pass_register[pass_name] = this;
|
|
}
|
|
|
|
void Pass::init_register()
|
|
{
|
|
vector<Pass*> added_passes;
|
|
while (first_queued_pass) {
|
|
added_passes.push_back(first_queued_pass);
|
|
first_queued_pass->run_register();
|
|
first_queued_pass = first_queued_pass->next_queued_pass;
|
|
}
|
|
for (auto added_pass : added_passes)
|
|
added_pass->on_register();
|
|
}
|
|
|
|
void Pass::done_register()
|
|
{
|
|
for (auto &it : pass_register)
|
|
it.second->on_shutdown();
|
|
|
|
frontend_register.clear();
|
|
pass_register.clear();
|
|
backend_register.clear();
|
|
log_assert(first_queued_pass == NULL);
|
|
}
|
|
|
|
void Pass::on_register()
|
|
{
|
|
}
|
|
|
|
void Pass::on_shutdown()
|
|
{
|
|
}
|
|
|
|
Pass::~Pass()
|
|
{
|
|
}
|
|
|
|
Pass::pre_post_exec_state_t Pass::pre_execute()
|
|
{
|
|
pre_post_exec_state_t state;
|
|
call_counter++;
|
|
state.begin_ns = PerformanceTimer::query();
|
|
state.parent_pass = current_pass;
|
|
current_pass = this;
|
|
clear_flags();
|
|
return state;
|
|
}
|
|
|
|
void Pass::post_execute(Pass::pre_post_exec_state_t state)
|
|
{
|
|
IdString::checkpoint();
|
|
log_suppressed();
|
|
|
|
int64_t time_ns = PerformanceTimer::query() - state.begin_ns;
|
|
runtime_ns += time_ns;
|
|
current_pass = state.parent_pass;
|
|
if (current_pass)
|
|
current_pass->runtime_ns -= time_ns;
|
|
}
|
|
|
|
void Pass::help()
|
|
{
|
|
log("\n");
|
|
log("No help message for command `%s'.\n", pass_name.c_str());
|
|
log("\n");
|
|
}
|
|
|
|
void Pass::clear_flags()
|
|
{
|
|
}
|
|
|
|
void Pass::cmd_log_args(const std::vector<std::string> &args)
|
|
{
|
|
if (args.size() <= 1)
|
|
return;
|
|
log("Full command line:");
|
|
for (size_t i = 0; i < args.size(); i++)
|
|
log(" %s", args[i].c_str());
|
|
log("\n");
|
|
}
|
|
|
|
void Pass::cmd_error(const std::vector<std::string> &args, size_t argidx, std::string msg)
|
|
{
|
|
std::string command_text;
|
|
int error_pos = 0;
|
|
|
|
for (size_t i = 0; i < args.size(); i++) {
|
|
if (i < argidx)
|
|
error_pos += args[i].size() + 1;
|
|
command_text = command_text + (command_text.empty() ? "" : " ") + args[i];
|
|
}
|
|
|
|
log("\nSyntax error in command `%s':\n", command_text.c_str());
|
|
help();
|
|
|
|
log_cmd_error("Command syntax error: %s\n> %s\n> %*s^\n",
|
|
msg.c_str(), command_text.c_str(), error_pos, "");
|
|
}
|
|
|
|
void Pass::extra_args(std::vector<std::string> args, size_t argidx, RTLIL::Design *design, bool select)
|
|
{
|
|
for (; argidx < args.size(); argidx++)
|
|
{
|
|
std::string arg = args[argidx];
|
|
|
|
if (arg.compare(0, 1, "-") == 0)
|
|
cmd_error(args, argidx, "Unknown option or option in arguments.");
|
|
|
|
if (!select)
|
|
cmd_error(args, argidx, "Extra argument.");
|
|
|
|
handle_extra_select_args(this, args, argidx, args.size(), design);
|
|
break;
|
|
}
|
|
// cmd_log_args(args);
|
|
}
|
|
|
|
void Pass::call(RTLIL::Design *design, std::string command)
|
|
{
|
|
std::vector<std::string> args;
|
|
|
|
std::string cmd_buf = command;
|
|
std::string tok = next_token(cmd_buf, " \t\r\n", true);
|
|
|
|
if (tok.empty())
|
|
return;
|
|
|
|
if (tok[0] == '!') {
|
|
#if !defined(YOSYS_DISABLE_SPAWN)
|
|
cmd_buf = command.substr(command.find('!') + 1);
|
|
while (!cmd_buf.empty() && (cmd_buf.back() == ' ' || cmd_buf.back() == '\t' ||
|
|
cmd_buf.back() == '\r' || cmd_buf.back() == '\n'))
|
|
cmd_buf.resize(cmd_buf.size()-1);
|
|
log_header(design, "Shell command: %s\n", cmd_buf.c_str());
|
|
int retCode = run_command(cmd_buf);
|
|
if (retCode != 0)
|
|
log_cmd_error("Shell command returned error code %d.\n", retCode);
|
|
return;
|
|
#else
|
|
log_cmd_error("Shell is not available.\n");
|
|
#endif
|
|
}
|
|
|
|
while (!tok.empty()) {
|
|
if (tok[0] == '#') {
|
|
int stop;
|
|
for (stop = 0; stop < GetSize(cmd_buf); stop++)
|
|
if (cmd_buf[stop] == '\r' || cmd_buf[stop] == '\n')
|
|
break;
|
|
cmd_buf = cmd_buf.substr(stop);
|
|
} else
|
|
if (tok.back() == ';') {
|
|
int num_semikolon = 0;
|
|
while (!tok.empty() && tok.back() == ';')
|
|
tok.resize(tok.size()-1), num_semikolon++;
|
|
if (!tok.empty())
|
|
args.push_back(tok);
|
|
call(design, args);
|
|
args.clear();
|
|
if (num_semikolon == 2)
|
|
call(design, "clean");
|
|
if (num_semikolon == 3)
|
|
call(design, "clean -purge");
|
|
} else
|
|
args.push_back(tok);
|
|
bool found_nl = false;
|
|
for (auto c : cmd_buf) {
|
|
if (c == ' ' || c == '\t')
|
|
continue;
|
|
if (c == '\r' || c == '\n')
|
|
found_nl = true;
|
|
break;
|
|
}
|
|
if (found_nl) {
|
|
call(design, args);
|
|
args.clear();
|
|
}
|
|
tok = next_token(cmd_buf, " \t\r\n", true);
|
|
}
|
|
|
|
call(design, args);
|
|
}
|
|
|
|
void Pass::call(RTLIL::Design *design, std::vector<std::string> args)
|
|
{
|
|
if (args.size() == 0 || args[0][0] == '#' || args[0][0] == ':')
|
|
return;
|
|
|
|
if (echo_mode) {
|
|
log("%s", create_prompt(design, 0));
|
|
for (size_t i = 0; i < args.size(); i++)
|
|
log("%s%s", i ? " " : "", args[i].c_str());
|
|
log("\n");
|
|
}
|
|
|
|
if (pass_register.count(args[0]) == 0)
|
|
log_cmd_error("No such command: %s (type 'help' for a command overview)\n", args[0].c_str());
|
|
|
|
if (pass_register[args[0]]->experimental_flag)
|
|
log_experimental("%s", args[0].c_str());
|
|
|
|
size_t orig_sel_stack_pos = design->selection_stack.size();
|
|
auto state = pass_register[args[0]]->pre_execute();
|
|
pass_register[args[0]]->execute(args, design);
|
|
pass_register[args[0]]->post_execute(state);
|
|
while (design->selection_stack.size() > orig_sel_stack_pos)
|
|
design->selection_stack.pop_back();
|
|
}
|
|
|
|
void Pass::call_on_selection(RTLIL::Design *design, const RTLIL::Selection &selection, std::string command)
|
|
{
|
|
std::string backup_selected_active_module = design->selected_active_module;
|
|
design->selected_active_module.clear();
|
|
design->selection_stack.push_back(selection);
|
|
|
|
Pass::call(design, command);
|
|
|
|
design->selection_stack.pop_back();
|
|
design->selected_active_module = backup_selected_active_module;
|
|
}
|
|
|
|
void Pass::call_on_selection(RTLIL::Design *design, const RTLIL::Selection &selection, std::vector<std::string> args)
|
|
{
|
|
std::string backup_selected_active_module = design->selected_active_module;
|
|
design->selected_active_module.clear();
|
|
design->selection_stack.push_back(selection);
|
|
|
|
Pass::call(design, args);
|
|
|
|
design->selection_stack.pop_back();
|
|
design->selected_active_module = backup_selected_active_module;
|
|
}
|
|
|
|
void Pass::call_on_module(RTLIL::Design *design, RTLIL::Module *module, std::string command)
|
|
{
|
|
std::string backup_selected_active_module = design->selected_active_module;
|
|
design->selected_active_module = module->name.str();
|
|
design->selection_stack.push_back(RTLIL::Selection(false));
|
|
design->selection_stack.back().select(module);
|
|
|
|
Pass::call(design, command);
|
|
|
|
design->selection_stack.pop_back();
|
|
design->selected_active_module = backup_selected_active_module;
|
|
}
|
|
|
|
void Pass::call_on_module(RTLIL::Design *design, RTLIL::Module *module, std::vector<std::string> args)
|
|
{
|
|
std::string backup_selected_active_module = design->selected_active_module;
|
|
design->selected_active_module = module->name.str();
|
|
design->selection_stack.push_back(RTLIL::Selection(false));
|
|
design->selection_stack.back().select(module);
|
|
|
|
Pass::call(design, args);
|
|
|
|
design->selection_stack.pop_back();
|
|
design->selected_active_module = backup_selected_active_module;
|
|
}
|
|
|
|
bool ScriptPass::check_label(std::string label, std::string info)
|
|
{
|
|
if (active_design == nullptr) {
|
|
log("\n");
|
|
if (info.empty())
|
|
log(" %s:\n", label.c_str());
|
|
else
|
|
log(" %s: %s\n", label.c_str(), info.c_str());
|
|
return true;
|
|
} else {
|
|
if (!active_run_from.empty() && active_run_from == active_run_to) {
|
|
block_active = (label == active_run_from);
|
|
} else {
|
|
if (label == active_run_from)
|
|
block_active = true;
|
|
if (label == active_run_to)
|
|
block_active = false;
|
|
}
|
|
return block_active;
|
|
}
|
|
}
|
|
|
|
void ScriptPass::run(std::string command, std::string info)
|
|
{
|
|
if (active_design == nullptr) {
|
|
if (info.empty())
|
|
log(" %s\n", command.c_str());
|
|
else
|
|
log(" %s %s\n", command.c_str(), info.c_str());
|
|
} else {
|
|
Pass::call(active_design, command);
|
|
active_design->check();
|
|
}
|
|
}
|
|
|
|
void ScriptPass::run_nocheck(std::string command, std::string info)
|
|
{
|
|
if (active_design == nullptr) {
|
|
if (info.empty())
|
|
log(" %s\n", command.c_str());
|
|
else
|
|
log(" %s %s\n", command.c_str(), info.c_str());
|
|
} else {
|
|
Pass::call(active_design, command);
|
|
}
|
|
}
|
|
|
|
void ScriptPass::run_script(RTLIL::Design *design, std::string run_from, std::string run_to)
|
|
{
|
|
help_mode = false;
|
|
active_design = design;
|
|
block_active = run_from.empty();
|
|
active_run_from = run_from;
|
|
active_run_to = run_to;
|
|
script();
|
|
}
|
|
|
|
void ScriptPass::help_script()
|
|
{
|
|
clear_flags();
|
|
help_mode = true;
|
|
active_design = nullptr;
|
|
block_active = true;
|
|
active_run_from.clear();
|
|
active_run_to.clear();
|
|
script();
|
|
}
|
|
|
|
Frontend::Frontend(std::string name, std::string short_help) :
|
|
Pass(name.rfind("=", 0) == 0 ? name.substr(1) : "read_" + name, short_help),
|
|
frontend_name(name.rfind("=", 0) == 0 ? name.substr(1) : name)
|
|
{
|
|
}
|
|
|
|
void Frontend::run_register()
|
|
{
|
|
if (pass_register.count(pass_name) && !replace_existing_pass())
|
|
log_error("Unable to register pass '%s', pass already exists!\n", pass_name.c_str());
|
|
pass_register[pass_name] = this;
|
|
|
|
if (frontend_register.count(frontend_name) && !replace_existing_pass())
|
|
log_error("Unable to register frontend '%s', frontend already exists!\n", frontend_name.c_str());
|
|
frontend_register[frontend_name] = this;
|
|
}
|
|
|
|
Frontend::~Frontend()
|
|
{
|
|
}
|
|
|
|
void Frontend::execute(std::vector<std::string> args, RTLIL::Design *design)
|
|
{
|
|
log_assert(next_args.empty());
|
|
do {
|
|
std::istream *f = NULL;
|
|
next_args.clear();
|
|
auto state = pre_execute();
|
|
execute(f, std::string(), args, design);
|
|
post_execute(state);
|
|
args = next_args;
|
|
delete f;
|
|
} while (!args.empty());
|
|
}
|
|
|
|
FILE *Frontend::current_script_file = NULL;
|
|
std::string Frontend::last_here_document;
|
|
|
|
void Frontend::extra_args(std::istream *&f, std::string &filename, std::vector<std::string> args, size_t argidx, bool bin_input)
|
|
{
|
|
bool called_with_fp = f != NULL;
|
|
|
|
next_args.clear();
|
|
|
|
if (argidx < args.size())
|
|
{
|
|
std::string arg = args[argidx];
|
|
|
|
if (arg.compare(0, 1, "-") == 0)
|
|
cmd_error(args, argidx, "Unknown option or option in arguments.");
|
|
if (f != NULL)
|
|
cmd_error(args, argidx, "Extra filename argument in direct file mode.");
|
|
|
|
filename = arg;
|
|
//Accommodate heredocs with EOT marker spaced out from "<<", e.g. "<< EOT" vs. "<<EOT"
|
|
if (filename == "<<" && argidx+1 < args.size())
|
|
filename += args[++argidx];
|
|
if (filename.compare(0, 2, "<<") == 0) {
|
|
if (filename.size() <= 2)
|
|
log_error("Missing EOT marker in here document!\n");
|
|
std::string eot_marker = filename.substr(2);
|
|
if (Frontend::current_script_file == nullptr)
|
|
filename = "<stdin>";
|
|
last_here_document.clear();
|
|
while (1) {
|
|
std::string buffer;
|
|
char block[4096];
|
|
while (1) {
|
|
if (fgets(block, 4096, Frontend::current_script_file == nullptr? stdin : Frontend::current_script_file) == nullptr)
|
|
log_error("Unexpected end of file in here document '%s'!\n", filename.c_str());
|
|
buffer += block;
|
|
if (buffer.size() > 0 && (buffer[buffer.size() - 1] == '\n' || buffer[buffer.size() - 1] == '\r'))
|
|
break;
|
|
}
|
|
size_t indent = buffer.find_first_not_of(" \t\r\n");
|
|
if (indent != std::string::npos && buffer.compare(indent, eot_marker.size(), eot_marker) == 0)
|
|
break;
|
|
last_here_document += buffer;
|
|
}
|
|
f = new std::istringstream(last_here_document);
|
|
} else {
|
|
rewrite_filename(filename);
|
|
vector<string> filenames = glob_filename(filename);
|
|
filename = filenames.front();
|
|
if (GetSize(filenames) > 1) {
|
|
next_args.insert(next_args.end(), args.begin(), args.begin()+argidx);
|
|
next_args.insert(next_args.end(), filenames.begin()+1, filenames.end());
|
|
}
|
|
std::ifstream *ff = new std::ifstream;
|
|
ff->open(filename.c_str(), bin_input ? std::ifstream::binary : std::ifstream::in);
|
|
yosys_input_files.insert(filename);
|
|
if (ff->fail()) {
|
|
delete ff;
|
|
ff = nullptr;
|
|
}
|
|
f = ff;
|
|
if (f != NULL) {
|
|
// Check for gzip magic
|
|
unsigned char magic[3];
|
|
int n = 0;
|
|
while (n < 3)
|
|
{
|
|
int c = ff->get();
|
|
if (c != EOF) {
|
|
magic[n] = (unsigned char) c;
|
|
}
|
|
n++;
|
|
}
|
|
if (n == 3 && magic[0] == 0x1f && magic[1] == 0x8b) {
|
|
#ifdef YOSYS_ENABLE_ZLIB
|
|
log("Found gzip magic in file `%s', decompressing using zlib.\n", filename.c_str());
|
|
if (magic[2] != 8)
|
|
log_cmd_error("gzip file `%s' uses unsupported compression type %02x\n",
|
|
filename.c_str(), unsigned(magic[2]));
|
|
delete ff;
|
|
std::stringstream *df = new std::stringstream();
|
|
decompress_gzip(filename, *df);
|
|
f = df;
|
|
#else
|
|
log_cmd_error("File `%s' is a gzip file, but Yosys is compiled without zlib.\n", filename.c_str());
|
|
#endif
|
|
} else {
|
|
ff->clear();
|
|
ff->seekg(0, std::ios::beg);
|
|
}
|
|
}
|
|
}
|
|
if (f == NULL)
|
|
log_cmd_error("Can't open input file `%s' for reading: %s\n", filename.c_str(), strerror(errno));
|
|
|
|
for (size_t i = argidx+1; i < args.size(); i++)
|
|
if (args[i].compare(0, 1, "-") == 0)
|
|
cmd_error(args, i, "Found option, expected arguments.");
|
|
|
|
if (argidx+1 < args.size()) {
|
|
if (next_args.empty())
|
|
next_args.insert(next_args.end(), args.begin(), args.begin()+argidx);
|
|
next_args.insert(next_args.end(), args.begin()+argidx+1, args.end());
|
|
args.erase(args.begin()+argidx+1, args.end());
|
|
}
|
|
}
|
|
|
|
if (f == NULL)
|
|
cmd_error(args, argidx, "No filename given.");
|
|
|
|
if (called_with_fp)
|
|
args.push_back(filename);
|
|
args[0] = pass_name;
|
|
// cmd_log_args(args);
|
|
}
|
|
|
|
void Frontend::frontend_call(RTLIL::Design *design, std::istream *f, std::string filename, std::string command)
|
|
{
|
|
std::vector<std::string> args;
|
|
char *s = strdup(command.c_str());
|
|
for (char *p = strtok(s, " \t\r\n"); p; p = strtok(NULL, " \t\r\n"))
|
|
args.push_back(p);
|
|
free(s);
|
|
frontend_call(design, f, filename, args);
|
|
}
|
|
|
|
void Frontend::frontend_call(RTLIL::Design *design, std::istream *f, std::string filename, std::vector<std::string> args)
|
|
{
|
|
if (args.size() == 0)
|
|
return;
|
|
if (frontend_register.count(args[0]) == 0)
|
|
log_cmd_error("No such frontend: %s\n", args[0].c_str());
|
|
|
|
if (f != NULL) {
|
|
auto state = frontend_register[args[0]]->pre_execute();
|
|
frontend_register[args[0]]->execute(f, filename, args, design);
|
|
frontend_register[args[0]]->post_execute(state);
|
|
} else if (filename == "-") {
|
|
std::istream *f_cin = &std::cin;
|
|
auto state = frontend_register[args[0]]->pre_execute();
|
|
frontend_register[args[0]]->execute(f_cin, "<stdin>", args, design);
|
|
frontend_register[args[0]]->post_execute(state);
|
|
} else {
|
|
if (!filename.empty())
|
|
args.push_back(filename);
|
|
frontend_register[args[0]]->execute(args, design);
|
|
}
|
|
}
|
|
|
|
Backend::Backend(std::string name, std::string short_help) :
|
|
Pass(name.rfind("=", 0) == 0 ? name.substr(1) : "write_" + name, short_help),
|
|
backend_name(name.rfind("=", 0) == 0 ? name.substr(1) : name)
|
|
{
|
|
}
|
|
|
|
void Backend::run_register()
|
|
{
|
|
if (pass_register.count(pass_name))
|
|
log_error("Unable to register pass '%s', pass already exists!\n", pass_name.c_str());
|
|
pass_register[pass_name] = this;
|
|
|
|
if (backend_register.count(backend_name))
|
|
log_error("Unable to register backend '%s', backend already exists!\n", backend_name.c_str());
|
|
backend_register[backend_name] = this;
|
|
}
|
|
|
|
Backend::~Backend()
|
|
{
|
|
}
|
|
|
|
void Backend::execute(std::vector<std::string> args, RTLIL::Design *design)
|
|
{
|
|
std::ostream *f = NULL;
|
|
auto state = pre_execute();
|
|
execute(f, std::string(), args, design);
|
|
post_execute(state);
|
|
if (f != &std::cout)
|
|
delete f;
|
|
}
|
|
|
|
void Backend::extra_args(std::ostream *&f, std::string &filename, std::vector<std::string> args, size_t argidx, bool bin_output)
|
|
{
|
|
bool called_with_fp = f != NULL;
|
|
|
|
for (; argidx < args.size(); argidx++)
|
|
{
|
|
std::string arg = args[argidx];
|
|
|
|
if (arg.compare(0, 1, "-") == 0 && arg != "-")
|
|
cmd_error(args, argidx, "Unknown option or option in arguments.");
|
|
if (f != NULL)
|
|
cmd_error(args, argidx, "Extra filename argument in direct file mode.");
|
|
|
|
if (arg == "-") {
|
|
filename = "<stdout>";
|
|
f = &std::cout;
|
|
continue;
|
|
}
|
|
|
|
filename = arg;
|
|
rewrite_filename(filename);
|
|
if (filename.size() > 3 && filename.compare(filename.size()-3, std::string::npos, ".gz") == 0) {
|
|
#ifdef YOSYS_ENABLE_ZLIB
|
|
gzip_ostream *gf = new gzip_ostream;
|
|
if (!gf->open(filename)) {
|
|
delete gf;
|
|
log_cmd_error("Can't open output file `%s' for writing: %s\n", filename.c_str(), strerror(errno));
|
|
}
|
|
yosys_output_files.insert(filename);
|
|
f = gf;
|
|
#else
|
|
log_cmd_error("Yosys is compiled without zlib support, unable to write gzip output.\n");
|
|
#endif
|
|
} else {
|
|
std::ofstream *ff = new std::ofstream;
|
|
ff->open(filename.c_str(), bin_output ? (std::ofstream::trunc | std::ofstream::binary) : std::ofstream::trunc);
|
|
yosys_output_files.insert(filename);
|
|
if (ff->fail()) {
|
|
delete ff;
|
|
log_cmd_error("Can't open output file `%s' for writing: %s\n", filename.c_str(), strerror(errno));
|
|
}
|
|
f = ff;
|
|
}
|
|
}
|
|
|
|
if (called_with_fp)
|
|
args.push_back(filename);
|
|
args[0] = pass_name;
|
|
// cmd_log_args(args);
|
|
|
|
if (f == NULL) {
|
|
filename = "<stdout>";
|
|
f = &std::cout;
|
|
}
|
|
}
|
|
|
|
void Backend::backend_call(RTLIL::Design *design, std::ostream *f, std::string filename, std::string command)
|
|
{
|
|
std::vector<std::string> args;
|
|
char *s = strdup(command.c_str());
|
|
for (char *p = strtok(s, " \t\r\n"); p; p = strtok(NULL, " \t\r\n"))
|
|
args.push_back(p);
|
|
free(s);
|
|
backend_call(design, f, filename, args);
|
|
}
|
|
|
|
void Backend::backend_call(RTLIL::Design *design, std::ostream *f, std::string filename, std::vector<std::string> args)
|
|
{
|
|
if (args.size() == 0)
|
|
return;
|
|
if (backend_register.count(args[0]) == 0)
|
|
log_cmd_error("No such backend: %s\n", args[0].c_str());
|
|
|
|
size_t orig_sel_stack_pos = design->selection_stack.size();
|
|
|
|
if (f != NULL) {
|
|
auto state = backend_register[args[0]]->pre_execute();
|
|
backend_register[args[0]]->execute(f, filename, args, design);
|
|
backend_register[args[0]]->post_execute(state);
|
|
} else if (filename == "-") {
|
|
std::ostream *f_cout = &std::cout;
|
|
auto state = backend_register[args[0]]->pre_execute();
|
|
backend_register[args[0]]->execute(f_cout, "<stdout>", args, design);
|
|
backend_register[args[0]]->post_execute(state);
|
|
} else {
|
|
if (!filename.empty())
|
|
args.push_back(filename);
|
|
backend_register[args[0]]->execute(args, design);
|
|
}
|
|
|
|
while (design->selection_stack.size() > orig_sel_stack_pos)
|
|
design->selection_stack.pop_back();
|
|
}
|
|
|
|
struct SimHelper {
|
|
string name;
|
|
inline string filesafe_name() {
|
|
if (name.at(0) == '$')
|
|
return name.substr(1);
|
|
else
|
|
return name;
|
|
}
|
|
string short_desc;
|
|
string ports;
|
|
string desc;
|
|
string code;
|
|
string ver;
|
|
};
|
|
|
|
static struct CellHelpMessages {
|
|
dict<string, SimHelper> cell_help, cell_code;
|
|
CellHelpMessages() {
|
|
SimHelper tempCell;
|
|
#include "techlibs/common/simlib_help.inc"
|
|
#include "techlibs/common/simcells_help.inc"
|
|
cell_help.sort();
|
|
cell_code.sort();
|
|
}
|
|
} cell_help_messages;
|
|
|
|
struct HelpPass : public Pass {
|
|
HelpPass() : Pass("help", "display help messages") { }
|
|
void help() override
|
|
{
|
|
log("\n");
|
|
log(" help ................ list all commands\n");
|
|
log(" help <command> ...... print help message for given command\n");
|
|
log(" help -all ........... print complete command reference\n");
|
|
log("\n");
|
|
log(" help -cells .......... list all cell types\n");
|
|
log(" help <celltype> ..... print help message for given cell type\n");
|
|
log(" help <celltype>+ .... print verilog code for given cell type\n");
|
|
log("\n");
|
|
}
|
|
void write_cmd_rst(std::string cmd, std::string title, std::string text)
|
|
{
|
|
FILE *f = fopen(stringf("docs/source/cmd/%s.rst", cmd.c_str()).c_str(), "wt");
|
|
// make header
|
|
size_t char_len = cmd.length() + 3 + title.length();
|
|
std::string title_line = "\n";
|
|
title_line.insert(0, char_len, '=');
|
|
fprintf(f, "%s", title_line.c_str());
|
|
fprintf(f, "%s - %s\n", cmd.c_str(), title.c_str());
|
|
fprintf(f, "%s\n", title_line.c_str());
|
|
fprintf(f, ".. raw:: latex\n\n \\begin{comment}\n\n");
|
|
|
|
// render html
|
|
fprintf(f, ".. cmd:def:: %s\n", cmd.c_str());
|
|
fprintf(f, " :title: %s\n\n", title.c_str());
|
|
std::stringstream ss;
|
|
std::string textcp = text;
|
|
ss << text;
|
|
bool IsUsage = true;
|
|
int blank_count = 0;
|
|
size_t def_strip_count = 0;
|
|
bool WasDefinition = false;
|
|
for (std::string line; std::getline(ss, line, '\n');) {
|
|
// find position of first non space character
|
|
std::size_t first_pos = line.find_first_not_of(" \t");
|
|
std::size_t last_pos = line.find_last_not_of(" \t");
|
|
if (first_pos == std::string::npos) {
|
|
// skip formatting empty lines
|
|
if (!WasDefinition)
|
|
fputc('\n', f);
|
|
blank_count += 1;
|
|
continue;
|
|
}
|
|
|
|
// strip leading and trailing whitespace
|
|
std::string stripped_line = line.substr(first_pos, last_pos - first_pos +1);
|
|
bool IsDefinition = stripped_line[0] == '-';
|
|
IsDefinition &= stripped_line[1] != ' ' && stripped_line[1] != '>';
|
|
bool IsDedent = def_strip_count && first_pos <= def_strip_count;
|
|
bool IsIndent = first_pos == 2 || first_pos == 4;
|
|
if (cmd.compare(0, 7, "verific") == 0)
|
|
// verific.cc has strange and different formatting from the rest
|
|
IsIndent = false;
|
|
|
|
// another usage block
|
|
bool NewUsage = stripped_line.find(cmd) == 0;
|
|
|
|
if (IsUsage) {
|
|
if (stripped_line.compare(0, 4, "See ") == 0) {
|
|
// description refers to another function
|
|
fprintf(f, "\n %s\n", stripped_line.c_str());
|
|
} else {
|
|
// usage should be the first line of help output
|
|
fprintf(f, "\n .. code:: yoscrypt\n\n %s\n\n ", stripped_line.c_str());
|
|
WasDefinition = true;
|
|
}
|
|
IsUsage = false;
|
|
} else if (IsIndent && NewUsage && (blank_count >= 2 || WasDefinition)) {
|
|
// another usage block
|
|
fprintf(f, "\n .. code:: yoscrypt\n\n %s\n\n ", stripped_line.c_str());
|
|
WasDefinition = true;
|
|
def_strip_count = 0;
|
|
} else if (IsIndent && IsDefinition && (blank_count || WasDefinition)) {
|
|
// format definition term
|
|
fprintf(f, "\n\n .. code:: yoscrypt\n\n %s\n\n ", stripped_line.c_str());
|
|
WasDefinition = true;
|
|
def_strip_count = first_pos;
|
|
} else {
|
|
if (IsDedent) {
|
|
fprintf(f, "\n\n ::\n");
|
|
def_strip_count = first_pos;
|
|
} else if (WasDefinition) {
|
|
fprintf(f, " ::\n");
|
|
WasDefinition = false;
|
|
}
|
|
fprintf(f, "\n %s", line.substr(def_strip_count, std::string::npos).c_str());
|
|
}
|
|
|
|
blank_count = 0;
|
|
}
|
|
fputc('\n', f);
|
|
|
|
// render latex
|
|
fprintf(f, ".. raw:: latex\n\n \\end{comment}\n\n");
|
|
fprintf(f, ".. only:: latex\n\n");
|
|
fprintf(f, " ::\n\n");
|
|
std::stringstream ss2;
|
|
ss2 << textcp;
|
|
for (std::string line; std::getline(ss2, line, '\n');) {
|
|
fprintf(f, " %s\n", line.c_str());
|
|
}
|
|
fclose(f);
|
|
}
|
|
void write_cell_rst(std::string cell, std::string help, std::string code)
|
|
{
|
|
// open
|
|
string safe_name = cell.substr(1);
|
|
FILE *f = fopen(stringf("docs/source/cell/%s.rst", safe_name.c_str()).c_str(), "wt");
|
|
|
|
// make header
|
|
string short_help = "";
|
|
string long_help = "";
|
|
string title_line = cell;
|
|
for (auto line : split_tokens(help, "\n")) {
|
|
if (short_help == "") {
|
|
size_t first_pos = line.find_first_not_of(" \t");
|
|
// skip empty lines
|
|
if (first_pos == string::npos) continue;
|
|
if (line.find(cell.c_str()) == string::npos) {
|
|
// strip leading/trailing space
|
|
size_t last_pos = line.find_last_not_of(" \t");
|
|
short_help = line.substr(first_pos, last_pos - first_pos +1);
|
|
// skip missing help message
|
|
if (short_help == "No help message for this cell type found.") {
|
|
long_help += line;
|
|
long_help += "\n";
|
|
} else {
|
|
title_line = stringf("%s - %s", cell.c_str(), short_help.c_str());
|
|
}
|
|
}
|
|
} else {
|
|
long_help += line;
|
|
long_help += "\n";
|
|
}
|
|
}
|
|
string underline = "\n";
|
|
underline.insert(0, title_line.length(), '=');
|
|
fprintf(f, "%s\n", title_line.c_str());
|
|
fprintf(f, "%s\n", underline.c_str());
|
|
|
|
// help text
|
|
fprintf(f, "%s\n", long_help.c_str());
|
|
|
|
// source code
|
|
fprintf(f, ".. code:: verilog\n\n");
|
|
std::stringstream ss;
|
|
ss << code;
|
|
for (std::string line; std::getline(ss, line, '\n');) {
|
|
fprintf(f, "\t%s\n", line.c_str());
|
|
}
|
|
|
|
// close
|
|
fclose(f);
|
|
}
|
|
void execute(std::vector<std::string> args, RTLIL::Design*) override
|
|
{
|
|
if (args.size() == 1) {
|
|
log("\n");
|
|
for (auto &it : pass_register)
|
|
log(" %-20s %s\n", it.first.c_str(), it.second->short_help.c_str());
|
|
log("\n");
|
|
log("Type 'help <command>' for more information on a command.\n");
|
|
log("Type 'help -cells' for a list of all cell types.\n");
|
|
log("\n");
|
|
return;
|
|
}
|
|
|
|
if (args.size() == 2) {
|
|
if (args[1] == "-all") {
|
|
for (auto &it : pass_register) {
|
|
log("\n\n");
|
|
log("%s -- %s\n", it.first.c_str(), it.second->short_help.c_str());
|
|
for (size_t i = 0; i < it.first.size() + it.second->short_help.size() + 6; i++)
|
|
log("=");
|
|
log("\n");
|
|
it.second->help();
|
|
if (it.second->experimental_flag) {
|
|
log("\n");
|
|
log("WARNING: THE '%s' COMMAND IS EXPERIMENTAL.\n", it.first.c_str());
|
|
log("\n");
|
|
}
|
|
}
|
|
}
|
|
else if (args[1] == "-cells") {
|
|
log("\n");
|
|
for (auto &it : cell_help_messages.cell_help) {
|
|
SimHelper help_cell = it.second;
|
|
log(" %-15s %s\n", help_cell.name.c_str(), help_cell.ports.c_str());
|
|
}
|
|
log("\n");
|
|
log("Type 'help <cell_type>' for more information on a cell type.\n");
|
|
log("\n");
|
|
return;
|
|
}
|
|
// this option is undocumented as it is for internal use only
|
|
else if (args[1] == "-write-rst-command-reference-manual") {
|
|
for (auto &it : pass_register) {
|
|
std::ostringstream buf;
|
|
log_streams.push_back(&buf);
|
|
it.second->help();
|
|
if (it.second->experimental_flag) {
|
|
log("\n");
|
|
log("WARNING: THE '%s' COMMAND IS EXPERIMENTAL.\n", it.first.c_str());
|
|
log("\n");
|
|
}
|
|
log_streams.pop_back();
|
|
write_cmd_rst(it.first, it.second->short_help, buf.str());
|
|
}
|
|
}
|
|
else if (args[1] == "-write-rst-cells-manual") {
|
|
for (auto &it : cell_help_messages.cell_help) {
|
|
write_cell_rst(it.first, it.second.desc, it.second.code);
|
|
}
|
|
}
|
|
else if (pass_register.count(args[1])) {
|
|
pass_register.at(args[1])->help();
|
|
if (pass_register.at(args[1])->experimental_flag) {
|
|
log("\n");
|
|
log("WARNING: THE '%s' COMMAND IS EXPERIMENTAL.\n", args[1].c_str());
|
|
log("\n");
|
|
}
|
|
}
|
|
else if (cell_help_messages.cell_help.count(args[1])) {
|
|
SimHelper help_cell = cell_help_messages.cell_help.at(args[1]);
|
|
if (help_cell.ver == "2") {
|
|
log("\n %s %s\n", help_cell.name.c_str(), help_cell.ports.c_str());
|
|
log("\n%s\n", help_cell.short_desc.c_str());
|
|
log("%s\n", help_cell.desc.c_str());
|
|
log("Run 'help %s+' to display the Verilog model for this cell type.\n", args[1].c_str());
|
|
log("\n");
|
|
} else {
|
|
log("%s\n", help_cell.desc.c_str());
|
|
log("Run 'help %s+' to display the Verilog model for this cell type.\n", args[1].c_str());
|
|
log("\n");
|
|
}
|
|
}
|
|
else if (cell_help_messages.cell_code.count(args[1])) {
|
|
SimHelper help_cell = cell_help_messages.cell_code.at(args[1]);
|
|
log("\n");
|
|
log("%s\n", help_cell.code.c_str());
|
|
}
|
|
else
|
|
log("No such command or cell type: %s\n", args[1].c_str());
|
|
return;
|
|
}
|
|
|
|
help();
|
|
}
|
|
} HelpPass;
|
|
|
|
struct EchoPass : public Pass {
|
|
EchoPass() : Pass("echo", "turning echoing back of commands on and off") { }
|
|
void help() override
|
|
{
|
|
log("\n");
|
|
log(" echo on\n");
|
|
log("\n");
|
|
log("Print all commands to log before executing them.\n");
|
|
log("\n");
|
|
log("\n");
|
|
log(" echo off\n");
|
|
log("\n");
|
|
log("Do not print all commands to log before executing them. (default)\n");
|
|
log("\n");
|
|
}
|
|
void execute(std::vector<std::string> args, RTLIL::Design*) override
|
|
{
|
|
if (args.size() > 2)
|
|
cmd_error(args, 2, "Unexpected argument.");
|
|
|
|
if (args.size() == 2) {
|
|
if (args[1] == "on")
|
|
echo_mode = true;
|
|
else if (args[1] == "off")
|
|
echo_mode = false;
|
|
else
|
|
cmd_error(args, 1, "Unexpected argument.");
|
|
}
|
|
|
|
log("echo %s\n", echo_mode ? "on" : "off");
|
|
}
|
|
} EchoPass;
|
|
|
|
SatSolver *yosys_satsolver_list;
|
|
SatSolver *yosys_satsolver;
|
|
|
|
struct MinisatSatSolver : public SatSolver {
|
|
MinisatSatSolver() : SatSolver("minisat") {
|
|
yosys_satsolver = this;
|
|
}
|
|
ezSAT *create() override {
|
|
return new ezMiniSAT();
|
|
}
|
|
} MinisatSatSolver;
|
|
|
|
struct LicensePass : public Pass {
|
|
LicensePass() : Pass("license", "print license terms") { }
|
|
void help() override
|
|
{
|
|
log("\n");
|
|
log(" license\n");
|
|
log("\n");
|
|
log("This command produces the following notice.\n");
|
|
notice();
|
|
}
|
|
void execute(std::vector<std::string>, RTLIL::Design*) override
|
|
{
|
|
notice();
|
|
}
|
|
void notice()
|
|
{
|
|
log("\n");
|
|
log(" /----------------------------------------------------------------------------\\\n");
|
|
log(" | |\n");
|
|
log(" | yosys -- Yosys Open SYnthesis Suite |\n");
|
|
log(" | |\n");
|
|
log(" | Copyright (C) 2012 - 2024 Claire Xenia Wolf <claire@yosyshq.com> |\n");
|
|
log(" | |\n");
|
|
log(" | Permission to use, copy, modify, and/or distribute this software for any |\n");
|
|
log(" | purpose with or without fee is hereby granted, provided that the above |\n");
|
|
log(" | copyright notice and this permission notice appear in all copies. |\n");
|
|
log(" | |\n");
|
|
log(" | THE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |\n");
|
|
log(" | WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |\n");
|
|
log(" | MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR |\n");
|
|
log(" | ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |\n");
|
|
log(" | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN |\n");
|
|
log(" | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF |\n");
|
|
log(" | OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |\n");
|
|
log(" | |\n");
|
|
log(" \\----------------------------------------------------------------------------/\n");
|
|
log("\n");
|
|
}
|
|
} LicensePass;
|
|
|
|
YOSYS_NAMESPACE_END
|