2018-07-26 19:05:15 +08:00
|
|
|
/*
|
|
|
|
* nextpnr -- Next Generation Place and Route
|
|
|
|
*
|
|
|
|
* Copyright (C) 2018 David Shah <david@symbioticeda.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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-09-30 01:37:17 +08:00
|
|
|
#include "globals.h"
|
2018-07-29 16:07:58 +08:00
|
|
|
#include <algorithm>
|
2018-07-29 16:35:05 +08:00
|
|
|
#include <iomanip>
|
|
|
|
#include <queue>
|
2018-08-07 18:16:51 +08:00
|
|
|
#include "cells.h"
|
2018-07-29 16:35:05 +08:00
|
|
|
#include "log.h"
|
2018-09-30 01:37:17 +08:00
|
|
|
#include "nextpnr.h"
|
2018-10-14 20:22:47 +08:00
|
|
|
#include "place_common.h"
|
2018-11-01 00:22:34 +08:00
|
|
|
#include "util.h"
|
2018-07-29 16:35:05 +08:00
|
|
|
#define fmt_str(x) (static_cast<const std::ostringstream &>(std::ostringstream() << x).str())
|
2018-07-26 19:05:15 +08:00
|
|
|
|
|
|
|
NEXTPNR_NAMESPACE_BEGIN
|
|
|
|
|
2018-07-31 20:03:35 +08:00
|
|
|
static std::string get_quad_name(GlobalQuadrant quad)
|
|
|
|
{
|
|
|
|
switch (quad) {
|
|
|
|
case QUAD_UL:
|
|
|
|
return "UL";
|
|
|
|
case QUAD_UR:
|
|
|
|
return "UR";
|
|
|
|
case QUAD_LL:
|
|
|
|
return "LL";
|
|
|
|
case QUAD_LR:
|
|
|
|
return "LR";
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2018-07-26 19:05:15 +08:00
|
|
|
class Ecp5GlobalRouter
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Ecp5GlobalRouter(Context *ctx) : ctx(ctx){};
|
2018-07-29 16:07:58 +08:00
|
|
|
|
2018-07-29 16:35:05 +08:00
|
|
|
private:
|
|
|
|
bool is_clock_port(const PortRef &user)
|
|
|
|
{
|
2018-10-02 00:45:35 +08:00
|
|
|
if (user.cell->type == id_TRELLIS_SLICE && (user.port == id_CLK || user.port == id_WCK))
|
2018-07-29 16:07:58 +08:00
|
|
|
return true;
|
2018-11-11 18:22:57 +08:00
|
|
|
if (user.cell->type == id_DCUA && (user.port == id_CH0_FF_RXI_CLK || user.port == id_CH1_FF_RXI_CLK ||
|
|
|
|
user.port == id_CH0_FF_TXI_CLK || user.port == id_CH1_FF_TXI_CLK))
|
|
|
|
return true;
|
2018-12-15 00:40:38 +08:00
|
|
|
if ((user.cell->type == id_IOLOGIC || user.cell->type == id_SIOLOGIC) && user.port == id_CLK)
|
|
|
|
return true;
|
2018-07-29 16:07:58 +08:00
|
|
|
return false;
|
|
|
|
}
|
2019-11-19 22:08:35 +08:00
|
|
|
bool is_lsr_port(const PortRef &user)
|
|
|
|
{
|
|
|
|
if (user.cell->type == id_TRELLIS_SLICE && (user.port == id_LSR))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2018-07-29 16:35:05 +08:00
|
|
|
std::vector<NetInfo *> get_clocks()
|
2018-07-29 16:07:58 +08:00
|
|
|
{
|
|
|
|
std::unordered_map<IdString, int> clockCount;
|
|
|
|
for (auto &net : ctx->nets) {
|
|
|
|
NetInfo *ni = net.second.get();
|
|
|
|
clockCount[ni->name] = 0;
|
|
|
|
for (const auto &user : ni->users) {
|
2018-11-11 18:22:57 +08:00
|
|
|
if (is_clock_port(user)) {
|
2018-07-29 16:07:58 +08:00
|
|
|
clockCount[ni->name]++;
|
2018-11-11 18:22:57 +08:00
|
|
|
if (user.cell->type == id_DCUA)
|
|
|
|
clockCount[ni->name] += 100;
|
|
|
|
}
|
2018-07-29 16:07:58 +08:00
|
|
|
}
|
2018-09-30 01:37:17 +08:00
|
|
|
// log_info("clkcount %s: %d\n", ni->name.c_str(ctx),clockCount[ni->name]);
|
2018-07-29 16:07:58 +08:00
|
|
|
}
|
2019-08-31 17:45:12 +08:00
|
|
|
// DCCAs must always drive globals
|
2018-07-29 16:35:05 +08:00
|
|
|
std::vector<NetInfo *> clocks;
|
2019-08-31 17:45:12 +08:00
|
|
|
for (auto &cell : sorted(ctx->cells)) {
|
|
|
|
CellInfo *ci = cell.second;
|
|
|
|
if (ci->type == id_DCCA) {
|
|
|
|
NetInfo *glb = ci->ports.at(id_CLKO).net;
|
|
|
|
if (glb != nullptr) {
|
|
|
|
clocks.push_back(glb);
|
|
|
|
clockCount.erase(glb->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-29 16:07:58 +08:00
|
|
|
while (clocks.size() < 16) {
|
2018-07-29 16:35:05 +08:00
|
|
|
auto max = std::max_element(clockCount.begin(), clockCount.end(),
|
|
|
|
[](const decltype(clockCount)::value_type &a,
|
|
|
|
const decltype(clockCount)::value_type &b) { return a.second < b.second; });
|
2018-09-29 23:37:18 +08:00
|
|
|
if (max == clockCount.end() || max->second < 5)
|
2018-07-29 16:07:58 +08:00
|
|
|
break;
|
|
|
|
clocks.push_back(ctx->nets.at(max->first).get());
|
|
|
|
clockCount.erase(max->first);
|
|
|
|
}
|
|
|
|
return clocks;
|
|
|
|
}
|
2018-07-26 19:05:15 +08:00
|
|
|
|
2019-11-19 22:08:35 +08:00
|
|
|
std::vector<NetInfo *> get_globalworthy_lsrs()
|
|
|
|
{
|
|
|
|
// Fanout threshold for LSRs is high (300), and we only promote up to two of them, in the nature of being
|
|
|
|
// conservative given the cost of a global
|
|
|
|
std::unordered_map<IdString, int> lsrCount;
|
|
|
|
for (auto &net : ctx->nets) {
|
|
|
|
NetInfo *ni = net.second.get();
|
|
|
|
lsrCount[ni->name] = 0;
|
|
|
|
for (const auto &user : ni->users)
|
|
|
|
if (is_lsr_port(user))
|
|
|
|
lsrCount[ni->name]++;
|
|
|
|
}
|
|
|
|
// DCCAs must always drive globals
|
|
|
|
std::vector<NetInfo *> lsrs;
|
|
|
|
while (lsrs.size() < 2) {
|
|
|
|
auto max = std::max_element(lsrCount.begin(), lsrCount.end(),
|
|
|
|
[](const decltype(lsrCount)::value_type &a,
|
|
|
|
const decltype(lsrCount)::value_type &b) { return a.second < b.second; });
|
|
|
|
if (max == lsrCount.end() || max->second < 300)
|
|
|
|
break;
|
|
|
|
lsrs.push_back(ctx->nets.at(max->first).get());
|
|
|
|
lsrCount.erase(max->first);
|
|
|
|
}
|
|
|
|
return lsrs;
|
|
|
|
}
|
|
|
|
|
2018-07-26 19:05:15 +08:00
|
|
|
PipId find_tap_pip(WireId tile_glb)
|
|
|
|
{
|
|
|
|
std::string wireName = ctx->getWireBasename(tile_glb).str(ctx);
|
|
|
|
std::string glbName = wireName.substr(2);
|
|
|
|
TapDirection td = ctx->globalInfoAtLoc(tile_glb.location).tap_dir;
|
|
|
|
WireId tap_wire;
|
|
|
|
Location tap_loc;
|
|
|
|
tap_loc.x = ctx->globalInfoAtLoc(tile_glb.location).tap_col;
|
|
|
|
tap_loc.y = tile_glb.location.y;
|
|
|
|
if (td == TAP_DIR_LEFT) {
|
|
|
|
tap_wire = ctx->getWireByLocAndBasename(tap_loc, "L_" + glbName);
|
|
|
|
} else {
|
|
|
|
tap_wire = ctx->getWireByLocAndBasename(tap_loc, "R_" + glbName);
|
|
|
|
}
|
2018-09-30 00:13:50 +08:00
|
|
|
NPNR_ASSERT(tap_wire != WireId());
|
2018-07-26 19:05:15 +08:00
|
|
|
return *(ctx->getPipsUphill(tap_wire).begin());
|
|
|
|
}
|
|
|
|
|
2018-08-07 21:53:41 +08:00
|
|
|
PipId find_spine_pip(WireId tap_wire)
|
|
|
|
{
|
|
|
|
std::string wireName = ctx->getWireBasename(tap_wire).str(ctx);
|
|
|
|
Location spine_loc;
|
|
|
|
spine_loc.x = ctx->globalInfoAtLoc(tap_wire.location).spine_col;
|
|
|
|
spine_loc.y = ctx->globalInfoAtLoc(tap_wire.location).spine_row;
|
|
|
|
WireId spine_wire = ctx->getWireByLocAndBasename(spine_loc, wireName);
|
|
|
|
return *(ctx->getPipsUphill(spine_wire).begin());
|
|
|
|
}
|
|
|
|
|
2018-08-06 18:24:41 +08:00
|
|
|
void route_logic_tile_global(NetInfo *net, int global_index, PortRef user)
|
2018-07-29 16:35:05 +08:00
|
|
|
{
|
2018-09-29 23:15:17 +08:00
|
|
|
WireId userWire = ctx->getBelPinWire(user.cell->bel, user.port);
|
2018-07-29 16:35:05 +08:00
|
|
|
WireId globalWire;
|
|
|
|
IdString global_name = ctx->id(fmt_str("G_HPBX" << std::setw(2) << std::setfill('0') << global_index << "00"));
|
|
|
|
std::queue<WireId> upstream;
|
|
|
|
std::unordered_map<WireId, PipId> backtrace;
|
|
|
|
upstream.push(userWire);
|
|
|
|
bool already_routed = false;
|
2018-09-30 00:13:50 +08:00
|
|
|
WireId next;
|
2019-11-19 22:08:35 +08:00
|
|
|
|
|
|
|
IdString id_lsr1 = ctx->id("LSR1");
|
|
|
|
|
2018-07-29 16:35:05 +08:00
|
|
|
// Search back from the pin until we reach the global network
|
|
|
|
while (true) {
|
2018-09-30 00:13:50 +08:00
|
|
|
next = upstream.front();
|
2018-07-29 16:35:05 +08:00
|
|
|
upstream.pop();
|
|
|
|
|
|
|
|
if (ctx->getBoundWireNet(next) == net) {
|
|
|
|
already_routed = true;
|
|
|
|
globalWire = next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->getWireBasename(next) == global_name) {
|
|
|
|
globalWire = next;
|
|
|
|
break;
|
|
|
|
}
|
2019-11-19 22:08:35 +08:00
|
|
|
|
|
|
|
// Forbid to use LSR1 for non-WRE LSRs; as this is reserved for WRE
|
|
|
|
if (ctx->getWireBasename(next) == id_lsr1 && user.port != id_WRE)
|
|
|
|
continue;
|
|
|
|
|
2018-07-29 16:35:05 +08:00
|
|
|
if (ctx->checkWireAvail(next)) {
|
|
|
|
for (auto pip : ctx->getPipsUphill(next)) {
|
|
|
|
WireId src = ctx->getPipSrcWire(pip);
|
|
|
|
backtrace[src] = pip;
|
|
|
|
upstream.push(src);
|
|
|
|
}
|
|
|
|
}
|
2018-09-30 01:06:08 +08:00
|
|
|
if (upstream.size() > 30000) {
|
2018-07-29 16:35:05 +08:00
|
|
|
log_error("failed to route HPBX%02d00 to %s.%s\n", global_index,
|
|
|
|
ctx->getBelName(user.cell->bel).c_str(ctx), user.port.c_str(ctx));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Set all the pips we found along the way
|
2018-09-30 00:13:50 +08:00
|
|
|
WireId cursor = next;
|
2018-07-29 16:35:05 +08:00
|
|
|
while (true) {
|
|
|
|
auto fnd = backtrace.find(cursor);
|
|
|
|
if (fnd == backtrace.end())
|
|
|
|
break;
|
|
|
|
ctx->bindPip(fnd->second, net, STRENGTH_LOCKED);
|
2018-09-30 00:13:50 +08:00
|
|
|
cursor = ctx->getPipDstWire(fnd->second);
|
2018-07-29 16:35:05 +08:00
|
|
|
}
|
|
|
|
// If the global network inside the tile isn't already set up,
|
|
|
|
// we also need to bind the buffers along the way
|
|
|
|
if (!already_routed) {
|
2018-09-30 00:13:50 +08:00
|
|
|
ctx->bindWire(next, net, STRENGTH_LOCKED);
|
|
|
|
PipId tap_pip = find_tap_pip(next);
|
2018-08-06 18:24:41 +08:00
|
|
|
NetInfo *tap_net = ctx->getBoundPipNet(tap_pip);
|
|
|
|
if (tap_net == nullptr) {
|
2018-07-29 16:35:05 +08:00
|
|
|
ctx->bindPip(tap_pip, net, STRENGTH_LOCKED);
|
2018-08-07 21:53:41 +08:00
|
|
|
PipId spine_pip = find_spine_pip(ctx->getPipSrcWire(tap_pip));
|
|
|
|
NetInfo *spine_net = ctx->getBoundPipNet(spine_pip);
|
|
|
|
if (spine_net == nullptr) {
|
|
|
|
ctx->bindPip(spine_pip, net, STRENGTH_LOCKED);
|
|
|
|
} else {
|
|
|
|
NPNR_ASSERT(spine_net == net);
|
|
|
|
}
|
2018-07-29 16:35:05 +08:00
|
|
|
} else {
|
|
|
|
NPNR_ASSERT(tap_net == net);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-31 20:03:35 +08:00
|
|
|
bool is_global_io(CellInfo *io, std::string &glb_name)
|
|
|
|
{
|
|
|
|
std::string func_name = ctx->getPioFunctionName(io->bel);
|
|
|
|
if (func_name.substr(0, 5) == "PCLKT") {
|
|
|
|
func_name.erase(func_name.find('_'), 1);
|
|
|
|
glb_name = "G_" + func_name;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
WireId get_global_wire(GlobalQuadrant quad, int network)
|
|
|
|
{
|
2018-09-30 01:37:17 +08:00
|
|
|
return ctx->getWireByLocAndBasename(Location(0, 0),
|
|
|
|
"G_" + get_quad_name(quad) + "PCLK" + std::to_string(network));
|
2018-07-31 20:03:35 +08:00
|
|
|
}
|
|
|
|
|
2018-09-29 23:37:18 +08:00
|
|
|
bool simple_router(NetInfo *net, WireId src, WireId dst, bool allow_fail = false)
|
2018-07-31 20:03:35 +08:00
|
|
|
{
|
|
|
|
std::queue<WireId> visit;
|
|
|
|
std::unordered_map<WireId, PipId> backtrace;
|
|
|
|
visit.push(src);
|
|
|
|
WireId cursor;
|
|
|
|
while (true) {
|
2018-09-30 00:01:19 +08:00
|
|
|
|
2018-07-31 20:03:35 +08:00
|
|
|
if (visit.empty() || visit.size() > 50000) {
|
2018-09-29 23:37:18 +08:00
|
|
|
if (allow_fail)
|
|
|
|
return false;
|
2018-07-31 20:03:35 +08:00
|
|
|
log_error("cannot route global from %s to %s.\n", ctx->getWireName(src).c_str(ctx),
|
|
|
|
ctx->getWireName(dst).c_str(ctx));
|
|
|
|
}
|
2018-09-30 00:01:19 +08:00
|
|
|
cursor = visit.front();
|
2018-07-31 20:03:35 +08:00
|
|
|
visit.pop();
|
2018-08-06 18:24:41 +08:00
|
|
|
NetInfo *bound = ctx->getBoundWireNet(cursor);
|
2018-07-31 20:03:35 +08:00
|
|
|
if (bound == net) {
|
2018-08-06 18:24:41 +08:00
|
|
|
} else if (bound != nullptr) {
|
2018-07-31 20:03:35 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (cursor == dst)
|
|
|
|
break;
|
|
|
|
for (auto dh : ctx->getPipsDownhill(cursor)) {
|
|
|
|
WireId pipDst = ctx->getPipDstWire(dh);
|
|
|
|
if (backtrace.count(pipDst))
|
|
|
|
continue;
|
|
|
|
backtrace[pipDst] = dh;
|
|
|
|
visit.push(pipDst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (true) {
|
|
|
|
auto fnd = backtrace.find(cursor);
|
|
|
|
if (fnd == backtrace.end())
|
|
|
|
break;
|
2018-09-30 01:37:17 +08:00
|
|
|
NetInfo *bound = ctx->getBoundWireNet(cursor);
|
2018-09-30 00:01:19 +08:00
|
|
|
if (bound != nullptr) {
|
|
|
|
NPNR_ASSERT(bound == net);
|
|
|
|
break;
|
|
|
|
}
|
2018-07-31 20:03:35 +08:00
|
|
|
ctx->bindPip(fnd->second, net, STRENGTH_LOCKED);
|
|
|
|
cursor = ctx->getPipSrcWire(fnd->second);
|
2018-09-30 01:37:17 +08:00
|
|
|
}
|
2018-09-30 00:01:19 +08:00
|
|
|
if (ctx->getBoundWireNet(src) == nullptr)
|
|
|
|
ctx->bindWire(src, net, STRENGTH_LOCKED);
|
2018-09-29 23:37:18 +08:00
|
|
|
return true;
|
2018-07-31 20:03:35 +08:00
|
|
|
}
|
|
|
|
|
2018-09-29 23:37:18 +08:00
|
|
|
bool route_onto_global(NetInfo *net, int network)
|
2018-07-31 20:03:35 +08:00
|
|
|
{
|
|
|
|
WireId glb_src;
|
2018-09-29 23:37:18 +08:00
|
|
|
NPNR_ASSERT(net->driver.cell->type == id_DCCA);
|
|
|
|
glb_src = ctx->getNetinfoSourceWire(net);
|
2018-07-31 20:03:35 +08:00
|
|
|
for (int quad = QUAD_UL; quad < QUAD_LR + 1; quad++) {
|
|
|
|
WireId glb_dst = get_global_wire(GlobalQuadrant(quad), network);
|
2018-09-29 23:49:29 +08:00
|
|
|
NPNR_ASSERT(glb_dst != WireId());
|
2018-09-29 23:37:18 +08:00
|
|
|
bool routed = simple_router(net, glb_src, glb_dst);
|
|
|
|
if (!routed)
|
|
|
|
return false;
|
2018-07-31 20:03:35 +08:00
|
|
|
}
|
2018-09-29 23:37:18 +08:00
|
|
|
return true;
|
2018-07-31 20:03:35 +08:00
|
|
|
}
|
|
|
|
|
2018-11-01 00:22:34 +08:00
|
|
|
// Get DCC wirelength based on source
|
|
|
|
wirelen_t get_dcc_wirelen(CellInfo *dcc)
|
|
|
|
{
|
|
|
|
NetInfo *clki = dcc->ports.at(id_CLKI).net;
|
|
|
|
BelId drv_bel;
|
|
|
|
const PortRef &drv = clki->driver;
|
|
|
|
if (drv.cell == nullptr) {
|
|
|
|
return 0;
|
|
|
|
} else if (drv.cell->attrs.count(ctx->id("BEL"))) {
|
2019-08-05 23:31:35 +08:00
|
|
|
drv_bel = ctx->getBelByName(ctx->id(drv.cell->attrs.at(ctx->id("BEL")).as_string()));
|
2018-11-01 00:22:34 +08:00
|
|
|
} else {
|
|
|
|
// Check if driver is a singleton
|
|
|
|
BelId last_bel;
|
|
|
|
bool singleton = true;
|
|
|
|
for (auto bel : ctx->getBels()) {
|
|
|
|
if (ctx->getBelType(bel) == drv.cell->type) {
|
|
|
|
if (last_bel != BelId()) {
|
|
|
|
singleton = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
last_bel = bel;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (singleton && last_bel != BelId()) {
|
|
|
|
drv_bel = last_bel;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (drv_bel == BelId()) {
|
|
|
|
// Driver is not locked. Use standard metric
|
|
|
|
float tns;
|
|
|
|
return get_net_metric(ctx, clki, MetricType::WIRELENGTH, tns);
|
|
|
|
} else {
|
2018-11-07 22:36:42 +08:00
|
|
|
// Check for dedicated routing
|
|
|
|
if (has_short_route(ctx->getBelPinWire(drv_bel, drv.port), ctx->getBelPinWire(dcc->bel, id_CLKI))) {
|
2018-12-01 00:09:56 +08:00
|
|
|
// log_info("dedicated route %s -> %s\n", ctx->getWireName(ctx->getBelPinWire(drv_bel,
|
|
|
|
// drv.port)).c_str(ctx), ctx->getBelName(dcc->bel).c_str(ctx));
|
2018-11-07 22:36:42 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2018-11-01 00:22:34 +08:00
|
|
|
// Driver is locked
|
|
|
|
Loc dcc_loc = ctx->getBelLocation(dcc->bel);
|
|
|
|
Loc drv_loc = ctx->getBelLocation(drv_bel);
|
|
|
|
return std::abs(dcc_loc.x - drv_loc.x) + std::abs(dcc_loc.y - drv_loc.y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-07 22:36:42 +08:00
|
|
|
// Return true if a short (<5) route exists between two wires
|
2018-12-01 00:09:56 +08:00
|
|
|
bool has_short_route(WireId src, WireId dst, int thresh = 7)
|
2018-11-07 23:08:47 +08:00
|
|
|
{
|
2018-11-07 22:36:42 +08:00
|
|
|
std::queue<WireId> visit;
|
|
|
|
std::unordered_map<WireId, PipId> backtrace;
|
|
|
|
visit.push(src);
|
|
|
|
WireId cursor;
|
|
|
|
while (true) {
|
|
|
|
|
2018-12-01 00:09:56 +08:00
|
|
|
if (visit.empty() || visit.size() > 10000) {
|
2018-11-07 23:08:47 +08:00
|
|
|
// log_info ("dist %s -> %s = inf\n", ctx->getWireName(src).c_str(ctx),
|
|
|
|
// ctx->getWireName(dst).c_str(ctx));
|
2018-11-07 22:36:42 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
cursor = visit.front();
|
|
|
|
visit.pop();
|
|
|
|
|
|
|
|
if (cursor == dst)
|
|
|
|
break;
|
|
|
|
for (auto dh : ctx->getPipsDownhill(cursor)) {
|
|
|
|
WireId pipDst = ctx->getPipDstWire(dh);
|
|
|
|
if (backtrace.count(pipDst))
|
|
|
|
continue;
|
|
|
|
backtrace[pipDst] = dh;
|
|
|
|
visit.push(pipDst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int length = 0;
|
|
|
|
while (true) {
|
|
|
|
auto fnd = backtrace.find(cursor);
|
|
|
|
if (fnd == backtrace.end())
|
|
|
|
break;
|
|
|
|
cursor = ctx->getPipSrcWire(fnd->second);
|
|
|
|
length++;
|
|
|
|
}
|
2018-11-07 23:08:47 +08:00
|
|
|
// log_info ("dist %s -> %s = %d\n", ctx->getWireName(src).c_str(ctx), ctx->getWireName(dst).c_str(ctx),
|
|
|
|
// length);
|
2018-11-07 22:36:42 +08:00
|
|
|
return length < thresh;
|
|
|
|
}
|
|
|
|
|
2018-09-29 23:37:18 +08:00
|
|
|
// Attempt to place a DCC
|
2018-09-30 01:37:17 +08:00
|
|
|
void place_dcc(CellInfo *dcc)
|
|
|
|
{
|
2018-10-14 20:22:47 +08:00
|
|
|
BelId best_bel;
|
2019-08-31 17:45:12 +08:00
|
|
|
bool using_ce = get_net_or_empty(dcc, ctx->id("CE")) != nullptr;
|
2018-10-14 20:22:47 +08:00
|
|
|
wirelen_t best_wirelen = 9999999;
|
2018-09-29 23:37:18 +08:00
|
|
|
for (auto bel : ctx->getBels()) {
|
|
|
|
if (ctx->getBelType(bel) == id_DCCA && ctx->checkBelAvail(bel)) {
|
|
|
|
if (ctx->isValidBelForCell(dcc, bel)) {
|
2019-08-31 17:45:12 +08:00
|
|
|
std::string belname = ctx->locInfo(bel)->bel_data[bel.index].name.get();
|
|
|
|
if (belname.at(0) == 'D' && using_ce)
|
|
|
|
continue; // don't allow DCCs with CE at center
|
2018-09-29 23:37:18 +08:00
|
|
|
ctx->bindBel(bel, dcc, STRENGTH_LOCKED);
|
2018-11-01 00:22:34 +08:00
|
|
|
wirelen_t wirelen = get_dcc_wirelen(dcc);
|
2018-10-14 20:22:47 +08:00
|
|
|
if (wirelen < best_wirelen) {
|
|
|
|
best_bel = bel;
|
|
|
|
best_wirelen = wirelen;
|
|
|
|
}
|
|
|
|
ctx->unbindBel(bel);
|
2018-09-29 23:37:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-10-14 20:22:47 +08:00
|
|
|
NPNR_ASSERT(best_bel != BelId());
|
|
|
|
ctx->bindBel(best_bel, dcc, STRENGTH_LOCKED);
|
2018-09-29 23:37:18 +08:00
|
|
|
}
|
|
|
|
|
2018-08-07 18:16:51 +08:00
|
|
|
// Insert a DCC into a net to promote it to a global
|
|
|
|
NetInfo *insert_dcc(NetInfo *net)
|
|
|
|
{
|
2019-08-31 17:45:12 +08:00
|
|
|
NetInfo *glbptr = nullptr;
|
|
|
|
CellInfo *dccptr = nullptr;
|
|
|
|
if (net->driver.cell != nullptr && net->driver.cell->type == id_DCCA) {
|
|
|
|
// Already have a DCC (such as clock gating)
|
|
|
|
glbptr = net;
|
|
|
|
dccptr = net->driver.cell;
|
|
|
|
} else {
|
|
|
|
auto dcc = create_ecp5_cell(ctx, id_DCCA, "$gbuf$" + net->name.str(ctx));
|
|
|
|
std::unique_ptr<NetInfo> glbnet = std::unique_ptr<NetInfo>(new NetInfo);
|
|
|
|
glbnet->name = ctx->id("$glbnet$" + net->name.str(ctx));
|
|
|
|
glbnet->driver.cell = dcc.get();
|
|
|
|
glbnet->driver.port = id_CLKO;
|
|
|
|
dcc->ports[id_CLKO].net = glbnet.get();
|
|
|
|
std::vector<PortRef> keep_users;
|
|
|
|
for (auto user : net->users) {
|
|
|
|
if (user.port == id_CLKFB) {
|
|
|
|
keep_users.push_back(user);
|
|
|
|
} else if (net->driver.cell->type == id_EXTREFB && user.cell->type == id_DCUA) {
|
|
|
|
keep_users.push_back(user);
|
|
|
|
} else {
|
|
|
|
glbnet->users.push_back(user);
|
|
|
|
user.cell->ports.at(user.port).net = glbnet.get();
|
|
|
|
}
|
2018-11-01 03:52:41 +08:00
|
|
|
}
|
2019-08-31 17:45:12 +08:00
|
|
|
net->users = keep_users;
|
2018-08-07 18:16:51 +08:00
|
|
|
|
2019-08-31 17:45:12 +08:00
|
|
|
dcc->ports[id_CLKI].net = net;
|
|
|
|
PortRef clki_pr;
|
|
|
|
clki_pr.port = id_CLKI;
|
|
|
|
clki_pr.cell = dcc.get();
|
|
|
|
net->users.push_back(clki_pr);
|
|
|
|
if (net->clkconstr) {
|
|
|
|
glbnet->clkconstr = std::unique_ptr<ClockConstraint>(new ClockConstraint());
|
|
|
|
glbnet->clkconstr->low = net->clkconstr->low;
|
|
|
|
glbnet->clkconstr->high = net->clkconstr->high;
|
|
|
|
glbnet->clkconstr->period = net->clkconstr->period;
|
|
|
|
}
|
|
|
|
glbptr = glbnet.get();
|
|
|
|
ctx->nets[glbnet->name] = std::move(glbnet);
|
|
|
|
dccptr = dcc.get();
|
|
|
|
ctx->cells[dcc->name] = std::move(dcc);
|
2018-11-12 21:59:09 +08:00
|
|
|
}
|
2019-08-31 17:45:12 +08:00
|
|
|
glbptr->attrs[ctx->id("ECP5_IS_GLOBAL")] = 1;
|
|
|
|
if (str_or_default(dccptr->attrs, ctx->id("BEL"), "") == "")
|
|
|
|
place_dcc(dccptr);
|
2018-08-07 18:16:51 +08:00
|
|
|
return glbptr;
|
|
|
|
}
|
2018-10-02 01:45:14 +08:00
|
|
|
|
|
|
|
int global_route_priority(const PortRef &load)
|
|
|
|
{
|
|
|
|
if (load.port == id_WCK || load.port == id_WRE)
|
|
|
|
return 90;
|
|
|
|
return 99;
|
|
|
|
}
|
|
|
|
|
2018-07-26 19:05:15 +08:00
|
|
|
Context *ctx;
|
2018-09-29 23:37:18 +08:00
|
|
|
|
2018-09-30 01:37:17 +08:00
|
|
|
public:
|
2018-11-01 00:22:34 +08:00
|
|
|
void promote_globals()
|
2018-09-30 01:37:17 +08:00
|
|
|
{
|
2019-06-22 23:57:00 +08:00
|
|
|
bool is_ooc = bool_or_default(ctx->settings, ctx->id("arch.ooc"));
|
2018-11-01 00:22:34 +08:00
|
|
|
log_info("Promoting globals...\n");
|
2018-09-29 23:37:18 +08:00
|
|
|
auto clocks = get_clocks();
|
2018-11-01 00:22:34 +08:00
|
|
|
for (auto clock : clocks) {
|
2019-08-25 00:09:42 +08:00
|
|
|
bool is_noglobal = bool_or_default(clock->attrs, ctx->id("noglobal"), false);
|
|
|
|
if (is_noglobal)
|
2019-08-22 06:43:03 +08:00
|
|
|
continue;
|
2018-11-01 00:22:34 +08:00
|
|
|
log_info(" promoting clock net %s to global network\n", clock->name.c_str(ctx));
|
2019-06-22 23:57:00 +08:00
|
|
|
if (is_ooc) // Don't actually do anything in OOC mode, global routing will be done in the full design
|
|
|
|
clock->is_global = true;
|
|
|
|
else
|
|
|
|
insert_dcc(clock);
|
2018-11-01 00:22:34 +08:00
|
|
|
}
|
2019-11-19 22:08:35 +08:00
|
|
|
if (clocks.size() < 12 && !bool_or_default(ctx->settings, ctx->id("arch.nolsrglobal"), false)) {
|
|
|
|
// Promote up to two set/resets to global
|
|
|
|
auto lsrs = get_globalworthy_lsrs();
|
|
|
|
for (auto lsr : lsrs) {
|
|
|
|
bool is_noglobal = bool_or_default(lsr->attrs, ctx->id("noglobal"), false);
|
|
|
|
if (is_noglobal)
|
|
|
|
continue;
|
|
|
|
log_info(" promoting LSR net %s to global network\n", lsr->name.c_str(ctx));
|
|
|
|
if (is_ooc) // Don't actually do anything in OOC mode, global routing will be done in the full design
|
|
|
|
lsr->is_global = true;
|
|
|
|
else
|
|
|
|
insert_dcc(lsr);
|
|
|
|
}
|
|
|
|
}
|
2018-11-01 00:22:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void route_globals()
|
|
|
|
{
|
|
|
|
log_info("Routing globals...\n");
|
2019-11-19 22:08:35 +08:00
|
|
|
std::set<int> all_globals, fab_globals, lsr_globals;
|
2018-09-30 01:06:08 +08:00
|
|
|
for (int i = 0; i < 16; i++) {
|
|
|
|
all_globals.insert(i);
|
2019-11-19 22:08:35 +08:00
|
|
|
if (i < 8) {
|
|
|
|
if (i >= 4)
|
|
|
|
lsr_globals.insert(i);
|
2018-09-30 01:06:08 +08:00
|
|
|
fab_globals.insert(i);
|
2019-11-19 22:08:35 +08:00
|
|
|
}
|
2018-09-30 01:06:08 +08:00
|
|
|
}
|
2019-02-25 18:54:24 +08:00
|
|
|
std::vector<std::pair<PortRef *, int>> toroute;
|
|
|
|
std::unordered_map<int, NetInfo *> clocks;
|
2018-11-01 00:22:34 +08:00
|
|
|
for (auto cell : sorted(ctx->cells)) {
|
|
|
|
CellInfo *ci = cell.second;
|
|
|
|
if (ci->type == id_DCCA) {
|
|
|
|
NetInfo *clock = ci->ports.at(id_CLKO).net;
|
|
|
|
NPNR_ASSERT(clock != nullptr);
|
|
|
|
bool drives_fabric = std::any_of(clock->users.begin(), clock->users.end(),
|
|
|
|
[this](const PortRef &port) { return !is_clock_port(port); });
|
2019-11-19 22:08:35 +08:00
|
|
|
bool drives_lsr = std::any_of(clock->users.begin(), clock->users.end(),
|
|
|
|
[this](const PortRef &port) { return is_lsr_port(port); });
|
2018-11-01 00:22:34 +08:00
|
|
|
int glbid;
|
2019-11-19 22:08:35 +08:00
|
|
|
if (drives_lsr) {
|
|
|
|
if (lsr_globals.empty()) {
|
|
|
|
// Route LSR through fabric (suboptimal)
|
|
|
|
NPNR_ASSERT(!fab_globals.empty());
|
|
|
|
glbid = *(fab_globals.begin());
|
|
|
|
} else {
|
|
|
|
glbid = *(lsr_globals.begin());
|
|
|
|
}
|
|
|
|
} else if (drives_fabric) {
|
2018-11-01 00:22:34 +08:00
|
|
|
if (fab_globals.empty())
|
|
|
|
continue;
|
|
|
|
glbid = *(fab_globals.begin());
|
|
|
|
} else {
|
|
|
|
glbid = *(all_globals.begin());
|
|
|
|
}
|
2018-11-01 03:52:41 +08:00
|
|
|
all_globals.erase(glbid);
|
|
|
|
fab_globals.erase(glbid);
|
2019-11-19 22:08:35 +08:00
|
|
|
lsr_globals.erase(glbid);
|
2018-11-01 00:22:34 +08:00
|
|
|
|
|
|
|
log_info(" routing clock net %s using global %d\n", clock->name.c_str(ctx), glbid);
|
|
|
|
bool routed = route_onto_global(clock, glbid);
|
|
|
|
NPNR_ASSERT(routed);
|
|
|
|
|
|
|
|
// WCK must have routing priority
|
2019-02-25 18:54:24 +08:00
|
|
|
for (auto &user : clock->users)
|
|
|
|
toroute.emplace_back(&user, glbid);
|
|
|
|
clocks[glbid] = clock;
|
2018-09-29 23:37:18 +08:00
|
|
|
}
|
|
|
|
}
|
2019-02-25 18:54:24 +08:00
|
|
|
std::sort(toroute.begin(), toroute.end(),
|
|
|
|
[this](const std::pair<PortRef *, int> &a, const std::pair<PortRef *, int> &b) {
|
|
|
|
return global_route_priority(*a.first) < global_route_priority(*b.first);
|
|
|
|
});
|
|
|
|
for (const auto &user : toroute) {
|
|
|
|
route_logic_tile_global(clocks.at(user.second), user.second, *user.first);
|
|
|
|
}
|
2018-09-29 23:37:18 +08:00
|
|
|
}
|
2018-07-26 19:05:15 +08:00
|
|
|
};
|
2018-11-01 00:22:34 +08:00
|
|
|
void promote_ecp5_globals(Context *ctx) { Ecp5GlobalRouter(ctx).promote_globals(); }
|
|
|
|
void route_ecp5_globals(Context *ctx) { Ecp5GlobalRouter(ctx).route_globals(); }
|
2018-07-26 19:05:15 +08:00
|
|
|
|
|
|
|
NEXTPNR_NAMESPACE_END
|