blob: 73eec15f8031221bb092891099917a7476826987 [file] [log] [blame]
/*
* yosys -- Yosys Open SYnthesis Suite
*
* Copyright (C) 2020 The Symbiflow Authors
*
* 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 <algorithm>
#include "clocks.h"
#include "kernel/log.h"
#include "kernel/register.h"
#include "kernel/rtlil.h"
#include "propagation.h"
USING_YOSYS_NAMESPACE
PRIVATE_NAMESPACE_BEGIN
struct ReadSdcCmd : public Frontend {
ReadSdcCmd() : Frontend("sdc", "Read SDC file") {}
void help() override {
log("\n");
log(" read_sdc <filename>\n");
log("\n");
log("Read SDC file.\n");
log("\n");
}
void execute(std::istream*& f, std::string filename,
std::vector<std::string> args, RTLIL::Design*) override {
if (args.size() < 2) {
log_cmd_error("Missing script file.\n");
}
log("\nReading clock constraints file(SDC)\n\n");
size_t argidx = 1;
extra_args(f, filename, args, argidx);
std::string content{std::istreambuf_iterator<char>(*f),
std::istreambuf_iterator<char>()};
log("%s\n", content.c_str());
Tcl_Interp* interp = yosys_get_tcl_interp();
if (Tcl_EvalFile(interp, args[argidx].c_str()) != TCL_OK) {
log_cmd_error("TCL interpreter returned an error: %s\n",
Tcl_GetStringResult(interp));
}
}
};
struct WriteSdcCmd : public Backend {
WriteSdcCmd(Clocks& clocks)
: Backend("sdc", "Write SDC file"), clocks_(clocks) {}
void help() override {
log("\n");
log(" write_sdc <filename>\n");
log("\n");
log("Write SDC file.\n");
log("\n");
}
void execute(std::ostream*& f, std::string filename,
std::vector<std::string> args, RTLIL::Design*) override {
if (args.size() < 2) {
log_cmd_error("Missing output file.\n");
}
log("\nWriting out clock constraints file(SDC)\n");
extra_args(f, filename, args, 1);
clocks_.WriteSdc(*f);
}
Clocks& clocks_;
};
struct CreateClockCmd : public Pass {
CreateClockCmd(Clocks& clocks)
: Pass("create_clock", "Create clock object"), clocks_(clocks) {}
void help() override {
log("\n");
log(" create_clock [ -name clock_name ] -period period_value "
"[-waveform <edge_list>] <target>\n");
log("Define a clock.\n");
log("If name is not specified then the name of the first target is "
"selected as the clock's name.\n");
log("Period is expressed in nanoseconds.\n");
log("The waveform option specifies the duty cycle (the rising a "
"falling edges) of the clock.\n");
log("It is specified as a list of two elements/time values: the first "
"rising edge and the next falling edge.\n");
log("\n");
}
void execute(std::vector<std::string> args,
RTLIL::Design* design) override {
size_t argidx;
std::string name;
bool is_waveform_specified(false);
float rising_edge(0);
float falling_edge(0);
float period(0);
if (args.size() < 4) {
log_cmd_error("Incorrect number of arguments\n");
}
for (argidx = 1; argidx < args.size(); argidx++) {
std::string arg = args[argidx];
if (arg == "-name" && argidx + 1 < args.size()) {
name = args[++argidx];
continue;
}
if (arg == "-period" && argidx + 1 < args.size()) {
period = std::stof(args[++argidx]);
continue;
}
if (arg == "-waveform" && argidx + 1 < args.size()) {
std::string edges(args[++argidx]);
std::copy_if(edges.begin(), edges.end(), edges.begin(),
[](char c) { return c != '{' or c != '}'; });
std::stringstream ss(edges);
ss >> rising_edge >> falling_edge;
is_waveform_specified = true;
continue;
}
break;
}
if (period <= 0) {
log_cmd_error("Incorrect period value\n");
}
// Add "w:" prefix to selection arguments to enforce wire object
// selection
AddWirePrefix(args, argidx);
extra_args(args, argidx, design);
// If clock name is not specified then take the name of the first target
std::vector<RTLIL::Wire*> selected_wires;
for (auto module : design->modules()) {
if (!design->selected(module)) {
continue;
}
for (auto wire : module->wires()) {
if (design->selected(module, wire)) {
#ifdef SDC_DEBUG
log("Selected wire %s\n",
RTLIL::unescape_id(wire->name).c_str());
#endif
selected_wires.push_back(wire);
}
}
}
if (selected_wires.size() == 0) {
log_cmd_error("Target selection is empty\n");
}
if (name.empty()) {
name = RTLIL::unescape_id(selected_wires.at(0)->name);
}
if (!is_waveform_specified) {
rising_edge = 0;
falling_edge = period / 2;
}
clocks_.AddClock(name, selected_wires, period, rising_edge,
falling_edge);
log("Created clock %s with period %f, waveform {%f %f}\n", name.c_str(),
period, rising_edge, falling_edge);
}
void AddWirePrefix(std::vector<std::string>& args, size_t argidx) {
auto selection_begin = args.begin() + argidx;
std::transform(selection_begin, args.end(), selection_begin,
[](std::string& w) { return "w:" + w; });
}
Clocks& clocks_;
};
struct GetClocksCmd : public Pass {
GetClocksCmd(Clocks& clocks)
: Pass("get_clocks", "Create clock object"), clocks_(clocks) {}
void help() override {
log("\n");
log(" get_clocks\n");
log("\n");
log("Returns all clocks in the design.\n");
log("\n");
}
void execute(__attribute__((unused)) std::vector<std::string> args,
__attribute__((unused)) RTLIL::Design* design) override {
std::vector<std::string> clock_names(clocks_.GetClockNames());
if (clock_names.size() == 0) {
log_warning("No clocks found in design\n");
}
Tcl_Interp* interp = yosys_get_tcl_interp();
Tcl_Obj* tcl_list = Tcl_NewListObj(0, NULL);
for (auto name : clock_names) {
Tcl_Obj* name_obj = Tcl_NewStringObj(name.c_str(), name.size());
Tcl_ListObjAppendElement(interp, tcl_list, name_obj);
}
Tcl_SetObjResult(interp, tcl_list);
}
Clocks& clocks_;
};
struct PropagateClocksCmd : public Pass {
PropagateClocksCmd(Clocks& clocks)
: Pass("propagate_clocks", "Propagate clock information"),
clocks_(clocks) {}
void help() override {
log("\n");
log(" propagate_clocks\n");
log("\n");
log("Propagate clock information throughout the design.\n");
log("\n");
}
void execute(__attribute__((unused)) std::vector<std::string> args,
RTLIL::Design* design) override {
if (!design->top_module()) {
log_cmd_error("No top module selected\n");
}
std::array<std::unique_ptr<Propagation>, 3> passes{
std::unique_ptr<NaturalPropagation>(
new NaturalPropagation(design, this)),
std::unique_ptr<BufferPropagation>(
new BufferPropagation(design, this)),
std::unique_ptr<ClockDividerPropagation>(
new ClockDividerPropagation(design, this))};
for (auto& pass : passes) {
pass->Run(clocks_);
}
}
Clocks& clocks_;
};
class SdcPlugin {
public:
SdcPlugin()
: write_sdc_cmd_(clocks_),
create_clock_cmd_(clocks_),
get_clocks_cmd_(clocks_),
propagate_clocks_cmd_(clocks_) {
log("Loaded SDC plugin\n");
}
ReadSdcCmd read_sdc_cmd_;
WriteSdcCmd write_sdc_cmd_;
CreateClockCmd create_clock_cmd_;
GetClocksCmd get_clocks_cmd_;
PropagateClocksCmd propagate_clocks_cmd_;
private:
Clocks clocks_;
} SdcPlugin;
PRIVATE_NAMESPACE_END