blob: 728a9bd0c4d295f143e68cc7ebf6d9cba597163e [file] [log] [blame]
#include "RoutingGraph.hpp"
#include "Chip.hpp"
#include "Tile.hpp"
#include <regex>
#include <iostream>
namespace Trellis {
const Location GlobalLoc(-2, -2);
RoutingGraph::RoutingGraph(const Chip &c) : chip_name(c.info.name), max_row(c.get_max_row()), max_col(c.get_max_col())
{
tiles[GlobalLoc].loc = GlobalLoc;
for (int y = 0; y <= max_row; y++) {
for (int x = 0; x <= max_col; x++) {
Location loc(x, y);
tiles[loc].loc = loc;
}
}
if (chip_name.find("25F") != string::npos || chip_name.find("12F") != string::npos)
chip_prefix = "25K_";
else if (chip_name.find("45F") != string::npos)
chip_prefix = "45K_";
else if (chip_name.find("85F") != string::npos)
chip_prefix = "85K_";
else
assert(false);
}
ident_t IdStore::ident(const std::string &str) const
{
if (str_to_id.find(str) != str_to_id.end()) {
return str_to_id.at(str);
} else {
str_to_id[str] = int(identifiers.size());
identifiers.push_back(str);
return str_to_id.at(str);
}
}
std::string IdStore::to_str(ident_t id) const
{
return identifiers.at(id);
}
RoutingId IdStore::id_at_loc(int16_t x, int16_t y, const std::string &str) const
{
RoutingId rid;
rid.id = ident(str);
rid.loc = Location(x, y);
return rid;
}
RoutingId RoutingGraph::globalise_net(int row, int col, const std::string &db_name)
{
static const std::regex e(R"(^([NS]\d+)?([EW]\d+)?_(.*))", std::regex::optimize);
std::string stripped_name = db_name;
if (db_name.find("25K_") == 0 || db_name.find("45K_") == 0 || db_name.find("85K_") == 0) {
if (db_name.substr(0, 4) == chip_prefix) {
stripped_name = db_name.substr(4);
} else {
return RoutingId();
}
}
// Workaround for PCSA/B sharing tile dbs
if (col >= 69) {
size_t pcsa_pos = stripped_name.find("PCSA");
if (pcsa_pos != std::string::npos)
stripped_name.replace(pcsa_pos + 3, 1, "B");
}
if (stripped_name.find("G_") == 0 || stripped_name.find("L_") == 0 || stripped_name.find("R_") == 0) {
// Global net
// TODO: quadrants and TAP_DRIVE regions
// TAP_DRIVE and SPINE wires go in their respective tiles
// Other globals are placed at a nominal location of (0, 0)
RoutingId id;
if (stripped_name.find("G_") == 0 && stripped_name.find("VPTX") == string::npos &&
stripped_name.find("HPBX") == string::npos && stripped_name.find("HPRX") == string::npos) {
id.loc.x = 0;
id.loc.y = 0;
} else {
id.loc.x = int16_t(col);
id.loc.y = int16_t(row);
}
id.id = ident(stripped_name);
return id;
} else {
RoutingId id;
id.loc.x = int16_t(col);
id.loc.y = int16_t(row);
// Local net, process prefix
smatch m;
if (regex_match(stripped_name, m, e)) {
for (int i = 1; i < int(m.size()) - 1; i++) {
string g = m.str(i);
if (g.empty()) continue;
if (g[0] == 'N') id.loc.y -= std::stoi(g.substr(1));
else if (g[0] == 'S') id.loc.y += std::stoi(g.substr(1));
else if (g[0] == 'W') id.loc.x -= std::stoi(g.substr(1));
else if (g[0] == 'E') id.loc.x += std::stoi(g.substr(1));
else
assert(false);
}
id.id = ident(m.str(m.size() - 1));
} else {
id.id = ident(stripped_name);
}
if (id.loc.x < 0 || id.loc.x > max_col || id.loc.y < 0 || id.loc.y > max_row)
return RoutingId(); // TODO: handle edge nets properly
return id;
}
}
void RoutingGraph::add_arc(Location loc, const RoutingArc &arc)
{
RoutingId arcId;
arcId.loc = loc;
arcId.id = arc.id;
add_wire(arc.source);
add_wire(arc.sink);
tiles[loc].arcs[arc.id] = arc;
tiles[arc.sink.loc].wires.at(arc.sink.id).uphill.push_back(arcId);
tiles[arc.source.loc].wires.at(arc.source.id).downhill.push_back(arcId);
}
void RoutingGraph::add_wire(RoutingId wire)
{
RoutingTileLoc &tile = tiles[wire.loc];
if (tile.wires.find(wire.id) == tile.wires.end()) {
RoutingWire rw;
rw.id = wire.id;
tiles[wire.loc].wires[rw.id] = rw;
}
}
void RoutingGraph::add_bel(RoutingBel &bel)
{
tiles[bel.loc].bels[bel.name] = bel;
}
void RoutingGraph::add_bel_input(RoutingBel &bel, ident_t pin, int wire_x, int wire_y, ident_t wire_name) {
RoutingId wireId, belId;
wireId.id = wire_name;
wireId.loc.x = wire_x;
wireId.loc.y = wire_y;
belId.id = bel.name;
belId.loc = bel.loc;
add_wire(wireId);
bel.pins[pin] = make_pair(wireId, PORT_IN);
tiles[wireId.loc].wires[wireId.id].belsUphill.push_back(make_pair(belId, pin));
}
void RoutingGraph::add_bel_output(RoutingBel &bel, ident_t pin, int wire_x, int wire_y, ident_t wire_name) {
RoutingId wireId, belId;
wireId.id = wire_name;
wireId.loc.x = wire_x;
wireId.loc.y = wire_y;
belId.id = bel.name;
belId.loc = bel.loc;
add_wire(wireId);
bel.pins[pin] = make_pair(wireId, PORT_OUT);
tiles[wireId.loc].wires[wireId.id].belsDownhill.push_back(make_pair(belId, pin));
}
}