yosys/passes/silimate/clkmerge.cc

178 lines
5.6 KiB
C++

/*
* yosys -- Yosys Open SYnthesis Suite
*
* Copyright (C) 2026 Silimate Inc.
*
* 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/sigtools.h"
#include "kernel/ff.h"
USING_YOSYS_NAMESPACE
PRIVATE_NAMESPACE_BEGIN
struct ClkMergePass : public Pass {
ClkMergePass() : Pass("clkmerge", "merge multiple clock domains into one") { }
void help() override
{
// |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
log("\n");
log(" clkmerge [-target <signal>] [selection]\n");
log("\n");
log("This command rewrites all flip-flop clock ports and polarities in the\n");
log("selected modules to use a single clock domain, merging both different\n");
log("clock signals and different clock polarities (posedge vs negedge).\n");
log("\n");
log("This is useful for miter-based equivalence checking where gold and gate\n");
log("copies may use different clock signal names or opposite edges for what\n");
log("is logically the same clock. Without merging, ABC's -dff mode will\n");
log("partition them into separate clock domains, creating spurious cross-domain\n");
log("ports that degrade SAT performance.\n");
log("\n");
log(" -target <signal>\n");
log(" use the specified signal as the merged clock. if not given, the\n");
log(" clock signal of the first FF encountered is used. all FFs will\n");
log(" be set to positive-edge triggered on this signal.\n");
log("\n");
}
void execute(std::vector<std::string> args, RTLIL::Design *design) override
{
std::string target_clk_name;
size_t argidx;
for (argidx = 1; argidx < args.size(); argidx++) {
if (args[argidx] == "-target" && argidx + 1 < args.size()) {
target_clk_name = args[++argidx];
continue;
}
break;
}
extra_args(args, argidx, design);
for (auto module : design->selected_modules())
{
SigMap sigmap(module);
struct FfInfo {
Cell *cell;
SigSpec clk;
bool pol_clk;
};
std::vector<FfInfo> ff_list;
for (auto cell : module->selected_cells())
{
if (!cell->is_builtin_ff())
continue;
FfData ff(nullptr, cell);
if (!ff.has_clk)
continue;
ff_list.push_back({cell, sigmap(ff.sig_clk), ff.pol_clk});
}
if (ff_list.empty())
continue;
// Determine target clock signal and polarity
SigSpec target_clk;
bool target_pol = true; // default to posedge
if (!target_clk_name.empty()) {
RTLIL::SigSpec sig;
if (!SigSpec::parse_sel(sig, design, module, target_clk_name))
log_cmd_error("Failed to parse target clock expression '%s'.\n", target_clk_name.c_str());
target_clk = sigmap(sig);
} else {
target_clk = ff_list[0].clk;
target_pol = ff_list[0].pol_clk;
}
// Collect distinct (signal, polarity) pairs
pool<std::pair<SigSpec, bool>> all_domains;
for (auto &info : ff_list)
all_domains.insert({info.clk, info.pol_clk});
if (all_domains.size() <= 1) {
log("Module %s: only one clock domain found, nothing to merge.\n", log_id(module));
continue;
}
log("Module %s: merging %d clock domains into %s%s\n",
log_id(module), GetSize(all_domains),
target_pol ? "" : "!", log_signal(target_clk));
for (auto &dom : all_domains)
log(" found domain: %s%s\n",
dom.second ? "" : "!", log_signal(dom.first));
int clk_rewritten = 0, pol_rewritten = 0;
for (auto &info : ff_list)
{
Cell *cell = info.cell;
bool needs_clk_change = (info.clk != target_clk);
bool needs_pol_change = (info.pol_clk != target_pol);
if (!needs_clk_change && !needs_pol_change)
continue;
if (needs_clk_change) {
if (cell->hasPort(ID::C))
cell->setPort(ID::C, target_clk);
else if (cell->hasPort(ID::CLK))
cell->setPort(ID::CLK, target_clk);
clk_rewritten++;
}
if (needs_pol_change) {
// Coarse-grain FFs ($dff, $dffe, etc.) use CLK_POLARITY param
if (cell->hasParam(ID::CLK_POLARITY)) {
cell->setParam(ID::CLK_POLARITY, target_pol ? State::S1 : State::S0);
pol_rewritten++;
}
// Fine-grain FFs encode polarity in the type name, so
// we need to swap the cell type (e.g. $_DFF_N_ <-> $_DFF_P_)
else {
std::string type = cell->type.str();
// Fine-grain FF types have P/N at position 6 for the clock polarity:
// $_DFF_P_, $_DFF_N_, $_DFFE_PP_, $_DFFE_NP_, etc.
// The clock polarity is always the first letter after "$_DFF_" or "$_DFFE_"
size_t pos = type.find("_DFF_");
if (pos != std::string::npos) {
pos += 5; // skip "_DFF_"
if (pos < type.size()) {
if (type[pos] == 'P' && !target_pol)
type[pos] = 'N';
else if (type[pos] == 'N' && target_pol)
type[pos] = 'P';
cell->type = RTLIL::IdString(type);
pol_rewritten++;
}
}
}
}
}
log(" Rewrote clock signal on %d FFs, polarity on %d FFs.\n",
clk_rewritten, pol_rewritten);
}
}
} ClkMergePass;
PRIVATE_NAMESPACE_END