nextpnr/himbaechel/uarch/xilinx/pack_clocking.cc
gatecat 5bfe0dd1b1 himbaechel: Adding a xilinx uarch for xc7 with prjxray
Signed-off-by: gatecat <gatecat@ds0.me>
2023-11-14 17:12:09 +01:00

351 lines
13 KiB
C++

/*
* nextpnr -- Next Generation Place and Route
*
* Copyright (C) 2019-2023 Myrtle Shah <gatecat@ds0.me>
*
* 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 <boost/optional.hpp>
#include <iterator>
#include <queue>
#include <unordered_set>
#include "chain_utils.h"
#include "design_utils.h"
#include "extra_data.h"
#include "log.h"
#include "nextpnr.h"
#include "pack.h"
#include "pins.h"
#define HIMBAECHEL_CONSTIDS "uarch/xilinx/constids.inc"
#include "himbaechel_constids.h"
NEXTPNR_NAMESPACE_BEGIN
BelId XilinxPacker::find_bel_with_short_route(WireId source, IdString beltype, IdString belpin)
{
if (source == WireId())
return BelId();
const size_t max_visit = 50000; // effort/runtime tradeoff
pool<WireId> visited;
std::queue<WireId> visit;
visit.push(source);
while (!visit.empty() && visited.size() < max_visit) {
WireId cursor = visit.front();
visit.pop();
for (auto bp : ctx->getWireBelPins(cursor))
if (bp.pin == belpin && ctx->getBelType(bp.bel) == beltype && ctx->checkBelAvail(bp.bel))
return bp.bel;
for (auto pip : ctx->getPipsDownhill(cursor)) {
WireId dst = ctx->getPipDstWire(pip);
if (visited.count(dst))
continue;
visit.push(dst);
visited.insert(dst);
}
}
return BelId();
}
void XilinxPacker::try_preplace(CellInfo *cell, IdString port)
{
if (cell->attrs.count(id_BEL) || cell->bel != BelId())
return;
NetInfo *n = cell->getPort(port);
if (n == nullptr || n->driver.cell == nullptr)
return;
CellInfo *drv = n->driver.cell;
BelId drv_bel = drv->bel;
if (drv_bel == BelId())
return;
WireId drv_wire = ctx->getBelPinWire(drv_bel, n->driver.port);
if (drv_wire == WireId())
return;
BelId tgt = find_bel_with_short_route(drv_wire, cell->type, port);
if (tgt != BelId()) {
ctx->bindBel(tgt, cell, STRENGTH_LOCKED);
log_info(" Constrained %s '%s' to bel '%s' based on dedicated routing\n", cell->type.c_str(ctx),
ctx->nameOf(cell), ctx->nameOfBel(tgt));
}
}
void XilinxPacker::preplace_unique(CellInfo *cell)
{
if (cell->attrs.count(id_BEL) || cell->bel != BelId())
return;
for (auto bel : ctx->getBels()) {
if (ctx->checkBelAvail(bel) && ctx->getBelType(bel) == cell->type) {
ctx->bindBel(bel, cell, STRENGTH_LOCKED);
return;
}
}
}
void XC7Packer::prepare_clocking()
{
log_info("Preparing clocking...\n");
dict<IdString, IdString> upgrade;
upgrade[id_MMCME2_BASE] = id_MMCME2_ADV;
upgrade[id_PLLE2_BASE] = id_PLLE2_ADV;
for (auto &cell : ctx->cells) {
CellInfo *ci = cell.second.get();
if (upgrade.count(ci->type)) {
IdString new_type = upgrade.at(ci->type);
ci->type = new_type;
} else if (ci->type == id_BUFG) {
ci->type = id_BUFGCTRL;
ci->renamePort(id_I, id_I0);
tie_port(ci, "CE0", true, true);
tie_port(ci, "S0", true, true);
tie_port(ci, "S1", false, true);
tie_port(ci, "IGNORE0", true, true);
} else if (ci->type == id_BUFGCE) {
ci->type = id_BUFGCTRL;
ci->renamePort(id_I, id_I0);
ci->renamePort(id_CE, id_CE0);
tie_port(ci, "S0", true, true);
tie_port(ci, "S1", false, true);
tie_port(ci, "IGNORE0", true, true);
}
}
}
void XC7Packer::pack_plls()
{
log_info("Packing PLLs...\n");
auto set_default = [](CellInfo *ci, IdString param, const Property &value) {
if (!ci->params.count(param))
ci->params[param] = value;
};
dict<IdString, XFormRule> pll_rules;
pll_rules[id_MMCME2_ADV].new_type = id_MMCME2_ADV_MMCME2_ADV;
pll_rules[id_PLLE2_ADV].new_type = id_PLLE2_ADV_PLLE2_ADV;
generic_xform(pll_rules);
for (auto &cell : ctx->cells) {
CellInfo *ci = cell.second.get();
// Preplace PLLs to make use of dedicated/short routing paths
if (ci->type.in(id_MMCM_MMCM_TOP, id_PLL_PLL_TOP))
try_preplace(ci, id_CLKIN1);
if (ci->type == id_MMCM_MMCM_TOP) {
// Fixup parameters
for (int i = 1; i <= 2; i++)
set_default(ci, ctx->idf("CLKIN%d_PERIOD", i), Property("0.0"));
for (int i = 0; i <= 6; i++) {
set_default(ci, ctx->idf("CLKOUT%d_CASCADE", i), Property("FALSE"));
set_default(ci, ctx->idf("CLKOUT%d_DIVIDE", i), Property(1));
set_default(ci, ctx->idf("CLKOUT%d_DUTY_CYCLE", i), Property("0.5"));
set_default(ci, ctx->idf("CLKOUT%d_PHASE", i), Property(0));
set_default(ci, ctx->idf("CLKOUT%d_USE_FINE_PS", i), Property("FALSE"));
}
set_default(ci, id_COMPENSATION, Property("INTERNAL"));
// Fixup routing
if (str_or_default(ci->params, id_COMPENSATION, "INTERNAL") == "INTERNAL") {
ci->disconnectPort(id_CLKFBIN);
ci->connectPort(id_CLKFBIN, ctx->nets.at(ctx->id("$PACKER_VCC_NET")).get());
}
}
}
}
void XC7Packer::pack_gbs()
{
log_info("Packing global buffers...\n");
dict<IdString, XFormRule> gb_rules;
gb_rules[id_BUFGCTRL].new_type = id_BUFGCTRL;
gb_rules[id_BUFGCTRL].new_type = id_BUFGCTRL;
generic_xform(gb_rules);
// Make sure prerequisites are set up first
for (auto &cell : ctx->cells) {
CellInfo *ci = cell.second.get();
if (ci->type == id_PS7_PS7)
preplace_unique(ci);
if (ci->type.in(id_PSEUDO_GND, id_PSEUDO_VCC))
preplace_unique(ci);
}
// Preplace global buffers to make use of dedicated/short routing
for (auto &cell : ctx->cells) {
CellInfo *ci = cell.second.get();
if (ci->type == id_BUFGCTRL)
try_preplace(ci, id_I0);
if (ci->type == id_BUFG_BUFG)
try_preplace(ci, id_I);
}
}
void XC7Packer::pack_clocking()
{
pack_plls();
pack_gbs();
}
void XilinxImpl::route_clocks()
{
log_info("Routing global clocks...\n");
// Special pass for faster routing of global clock psuedo-net
for (auto &net : ctx->nets) {
NetInfo *clk_net = net.second.get();
if (!clk_net->driver.cell)
continue;
// check if we have a global clock net, skip otherwise
bool is_global = false;
if ((clk_net->driver.cell->type.in(id_BUFGCTRL, id_BUFCE_BUFG_PS, id_BUFCE_BUFCE, id_BUFGCE_DIV_BUFGCE_DIV)) &&
clk_net->driver.port == id_O)
is_global = true;
else if (clk_net->driver.cell->type == id_PLLE2_ADV_PLLE2_ADV && clk_net->users.entries() == 1 &&
((*clk_net->users.begin()).cell->type.in(id_BUFGCTRL, id_BUFCE_BUFCE, id_BUFGCE_DIV_BUFGCE_DIV)))
is_global = true;
else if (clk_net->users.entries() == 1 && (*clk_net->users.begin()).cell->type == id_PLLE2_ADV_PLLE2_ADV &&
(*clk_net->users.begin()).port == id_CLKIN1)
is_global = true;
if (!is_global)
continue;
log_info(" routing clock '%s'\n", clk_net->name.c_str(ctx));
ctx->bindWire(ctx->getNetinfoSourceWire(clk_net), clk_net, STRENGTH_LOCKED);
for (auto &usr : clk_net->users) {
std::queue<WireId> visit;
dict<WireId, PipId> backtrace;
WireId dest = WireId();
auto sink_wire = ctx->getNetinfoSinkWire(clk_net, usr, 0);
if (ctx->debug) {
auto sink_wire_name = "(uninitialized)";
if (sink_wire != WireId())
sink_wire_name = ctx->nameOfWire(sink_wire);
log_info(" routing arc to %s.%s (wire %s):\n", usr.cell->name.c_str(ctx), usr.port.c_str(ctx),
sink_wire_name);
}
visit.push(sink_wire);
while (!visit.empty()) {
WireId curr = visit.front();
visit.pop();
if (ctx->getBoundWireNet(curr) == clk_net) {
dest = curr;
break;
}
for (auto uh : ctx->getPipsUphill(curr)) {
if (!ctx->checkPipAvail(uh))
continue;
WireId src = ctx->getPipSrcWire(uh);
if (backtrace.count(src))
continue;
IdString intent = ctx->getWireType(src);
if (intent.in(id_NODE_DOUBLE, id_NODE_HLONG, id_NODE_HQUAD, id_NODE_VLONG, id_NODE_VQUAD,
id_NODE_SINGLE, id_NODE_CLE_OUTPUT, id_NODE_OPTDELAY, id_BENTQUAD, id_DOUBLE,
id_HLONG, id_HQUAD, id_OPTDELAY, id_SINGLE, id_VLONG, id_VLONG12, id_VQUAD,
id_PINBOUNCE))
continue;
if (!ctx->checkWireAvail(src) && ctx->getBoundWireNet(src) != clk_net)
continue;
backtrace[src] = uh;
visit.push(src);
}
}
if (dest == WireId()) {
log_info(" failed to find a route using dedicated resources.\n");
if (clk_net->users.entries() == 1 && (*clk_net->users.begin()).cell->type == id_PLLE2_ADV_PLLE2_ADV &&
(*clk_net->users.begin()).port == id_CLKIN1) {
// Due to some missing pips, currently special case more lenient solution
std::queue<WireId> empty;
std::swap(visit, empty);
backtrace.clear();
visit.push(sink_wire);
while (!visit.empty()) {
WireId curr = visit.front();
visit.pop();
if (ctx->getBoundWireNet(curr) == clk_net) {
dest = curr;
break;
}
for (auto uh : ctx->getPipsUphill(curr)) {
if (!ctx->checkPipAvail(uh))
continue;
WireId src = ctx->getPipSrcWire(uh);
if (backtrace.count(src))
continue;
if (!ctx->checkWireAvail(src) && ctx->getBoundWireNet(src) != clk_net)
continue;
backtrace[src] = uh;
visit.push(src);
}
}
if (dest == WireId())
continue;
} else {
continue;
}
}
while (backtrace.count(dest)) {
auto uh = backtrace[dest];
dest = ctx->getPipDstWire(uh);
if (ctx->getBoundWireNet(dest) == clk_net) {
NPNR_ASSERT(clk_net->wires.at(dest).pip == uh);
break;
}
if (ctx->debug)
log_info(" bind pip %s --> %s\n", ctx->nameOfPip(uh), ctx->nameOfWire(dest));
ctx->bindPip(uh, clk_net, STRENGTH_LOCKED);
}
}
}
#if 0
for (auto& net : nets) {
NetInfo *ni = net.second.get();
for (auto &usr : ni->users) {
if (usr.cell->type != id_BUFGCTRL || usr.port != id_I0)
continue;
WireId dst = getCtx()->getNetinfoSinkWire(ni, usr, 0);
std::queue<WireId> visit;
visit.push(dst);
int i = 0;
while(!visit.empty() && i < 5000) {
WireId curr = visit.front();
visit.pop();
log(" %s\n", nameOfWire(curr));
for (auto pip : getPipsUphill(curr)) {
auto &pd = locInfo(pip).pip_data[pip.index];
log_info(" p %s sr %s (t %d s %d sv %d)\n", nameOfPip(pip), nameOfWire(getPipSrcWire(pip)), pd.flags, pd.site, pd.site_variant);
if (!checkPipAvail(pip)) {
log(" p unavail\n");
continue;
}
WireId src = getPipSrcWire(pip);
if (!checkWireAvail(src)) {
log(" w unavail (%s)\n", nameOf(getBoundWireNet(src)));
continue;
}
log_info(" p %s s %s\n", nameOfPip(pip), nameOfWire(src));
visit.push(src);
}
++i;
}
}
}
#endif
}
NEXTPNR_NAMESPACE_END