aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Shah <davey1576@gmail.com>2018-11-26 18:11:16 +0000
committerGitHub <noreply@github.com>2018-11-26 18:11:16 +0000
commit86108bfd395352af2342177eba6453cd8a58bdfa (patch)
tree510f698705c96b2def9f7b20732ceaf4eabe09b4
parent5a1190ade259259fa31cf6a9f35ec3794bbaeb08 (diff)
parent584e8c58a61748b88639336ad5defb41d5389c3a (diff)
downloadnextpnr-86108bfd395352af2342177eba6453cd8a58bdfa.tar.gz
nextpnr-86108bfd395352af2342177eba6453cd8a58bdfa.tar.bz2
nextpnr-86108bfd395352af2342177eba6453cd8a58bdfa.zip
Merge pull request #149 from smunaut/issue_148
Fixes for global promotion
-rw-r--r--common/placer1.cc5
-rw-r--r--ice40/arch.h7
-rw-r--r--ice40/arch_place.cc3
-rw-r--r--ice40/pack.cc38
4 files changed, 43 insertions, 10 deletions
diff --git a/common/placer1.cc b/common/placer1.cc
index 0db7ce00..b42ef2ff 100644
--- a/common/placer1.cc
+++ b/common/placer1.cc
@@ -337,9 +337,10 @@ class SAPlacer
}
} else {
uint64_t score = ctx->rng64();
- if (score <= best_ripup_score) {
+ CellInfo *bound_cell = ctx->getBoundBelCell(bel);
+ if (score <= best_ripup_score && bound_cell->belStrength < STRENGTH_STRONG) {
best_ripup_score = score;
- ripup_target = ctx->getBoundBelCell(bel);
+ ripup_target = bound_cell;
ripup_bel = bel;
}
}
diff --git a/ice40/arch.h b/ice40/arch.h
index e8c597c9..10255dbe 100644
--- a/ice40/arch.h
+++ b/ice40/arch.h
@@ -884,6 +884,13 @@ struct Arch : BaseCtx
}
NPNR_ASSERT_FALSE("Expected PLL pin to share an output with an SB_IO D_IN_{0,1}");
}
+
+ int getDrivenGlobalNetwork(BelId bel) const
+ {
+ NPNR_ASSERT(getBelType(bel) == id_SB_GB);
+ IdString glb_net = getWireName(getBelPinWire(bel, id_GLOBAL_BUFFER_OUTPUT));
+ return std::stoi(std::string("") + glb_net.str(this).back());
+ }
};
void ice40DelayFuzzerMain(Context *ctx);
diff --git a/ice40/arch_place.cc b/ice40/arch_place.cc
index 41f9b640..90bb62b9 100644
--- a/ice40/arch_place.cc
+++ b/ice40/arch_place.cc
@@ -165,8 +165,7 @@ bool Arch::isValidBelForCell(CellInfo *cell, BelId bel) const
return true;
NPNR_ASSERT(cell->ports.at(id_GLOBAL_BUFFER_OUTPUT).net != nullptr);
const NetInfo *net = cell->ports.at(id_GLOBAL_BUFFER_OUTPUT).net;
- IdString glb_net = getWireName(getBelPinWire(bel, id_GLOBAL_BUFFER_OUTPUT));
- int glb_id = std::stoi(std::string("") + glb_net.str(this).back());
+ int glb_id = getDrivenGlobalNetwork(bel);
if (net->is_reset && net->is_enable)
return false;
else if (net->is_reset)
diff --git a/ice40/pack.cc b/ice40/pack.cc
index 682baadd..fc28121e 100644
--- a/ice40/pack.cc
+++ b/ice40/pack.cc
@@ -587,10 +587,36 @@ static void promote_globals(Context *ctx)
}
}
int prom_globals = 0, prom_resets = 0, prom_cens = 0, prom_logics = 0;
- int gbs_available = 8;
+ int gbs_available = 8, resets_available = 4, cens_available = 4;
for (auto &cell : ctx->cells)
- if (is_gbuf(ctx, cell.second.get()))
+ if (is_gbuf(ctx, cell.second.get())) {
+ /* One less buffer available */
--gbs_available;
+
+ /* And possibly limits what we can promote */
+ if (cell.second->attrs.find(ctx->id("BEL")) != cell.second->attrs.end()) {
+ /* If the SB_GB is locked, doesn't matter what it drives */
+ BelId bel = ctx->getBelByName(ctx->id(cell.second->attrs[ctx->id("BEL")]));
+ int glb_id = ctx->getDrivenGlobalNetwork(bel);
+ if ((glb_id % 2) == 0)
+ resets_available--;
+ else if ((glb_id % 2) == 1)
+ cens_available--;
+ } else {
+ /* If it's free to move around, then look at what it drives */
+ NetInfo *ni = cell.second->ports[id_GLOBAL_BUFFER_OUTPUT].net;
+
+ for (auto user : ni->users) {
+ if (is_reset_port(ctx, user)) {
+ resets_available--;
+ break;
+ } else if (is_enable_port(ctx, user)) {
+ cens_available--;
+ break;
+ }
+ }
+ }
+ }
while (prom_globals < gbs_available) {
auto global_clock = std::max_element(clock_count.begin(), clock_count.end(),
[](const std::pair<IdString, int> &a, const std::pair<IdString, int> &b) {
@@ -610,8 +636,8 @@ static void promote_globals(Context *ctx)
return a.second < b.second;
});
if (global_clock->second == 0 && prom_logics < 4 && global_logic->second > logic_fanout_thresh &&
- (global_logic->second > global_cen->second || prom_cens >= 4) &&
- (global_logic->second > global_reset->second || prom_resets >= 4)) {
+ (global_logic->second > global_cen->second || prom_cens >= cens_available) &&
+ (global_logic->second > global_reset->second || prom_resets >= resets_available)) {
NetInfo *logicnet = ctx->nets[global_logic->first].get();
insert_global(ctx, logicnet, false, false, true);
++prom_globals;
@@ -620,7 +646,7 @@ static void promote_globals(Context *ctx)
reset_count.erase(logicnet->name);
cen_count.erase(logicnet->name);
logic_count.erase(logicnet->name);
- } else if (global_reset->second > global_clock->second && prom_resets < 4) {
+ } else if (global_reset->second > global_clock->second && prom_resets < resets_available) {
NetInfo *rstnet = ctx->nets[global_reset->first].get();
insert_global(ctx, rstnet, true, false, false);
++prom_globals;
@@ -629,7 +655,7 @@ static void promote_globals(Context *ctx)
reset_count.erase(rstnet->name);
cen_count.erase(rstnet->name);
logic_count.erase(rstnet->name);
- } else if (global_cen->second > global_clock->second && prom_cens < 4 &&
+ } else if (global_cen->second > global_clock->second && prom_cens < cens_available &&
global_cen->second > enable_fanout_thresh) {
NetInfo *cennet = ctx->nets[global_cen->first].get();
insert_global(ctx, cennet, false, true, false);