check slews speedup with multiple corners

Signed-off-by: James Cherry <cherry@parallaxsw.com>
This commit is contained in:
James Cherry 2024-12-08 14:49:58 -08:00
parent a82361ce3d
commit c312d5b6cd
2 changed files with 173 additions and 174 deletions

View File

@ -87,6 +87,34 @@ CheckSlewLimits::CheckSlewLimits(const StaState *sta) :
{
}
void
CheckSlewLimits::checkSlewLimits(const Pin *pin,
bool violators,
const Corner *corner,
const MinMax *min_max,
PinSeq &slew_pins,
float &min_slack)
{
const Corner *corner1;
const RiseFall *rf;
Slew slew;
float limit, slack;
checkSlew(pin, corner, min_max, true, corner1, rf, slew, limit, slack);
if (!fuzzyInf(slack)) {
if (violators) {
if (slack < 0.0)
slew_pins.push_back(pin);
}
else {
if (slew_pins.empty()
|| slack < min_slack) {
slew_pins.push_back(pin);
min_slack = slack;
}
}
}
}
void
CheckSlewLimits::checkSlew(const Pin *pin,
const Corner *corner,
@ -94,86 +122,121 @@ CheckSlewLimits::checkSlew(const Pin *pin,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf,
Slew &slew,
float &limit,
float &slack) const
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const
{
corner1 = nullptr;
rf = nullptr;
slew = 0.0;
limit = 0.0;
slack = MinMax::min()->initValue();
if (corner)
checkSlews1(pin, corner, min_max, check_clks,
corner1, rf, slew, limit, slack);
else {
for (auto corner : *sta_->corners()) {
checkSlews1(pin, corner, min_max, check_clks,
corner1, rf, slew, limit, slack);
rf1 = nullptr;
slew1 = 0.0;
limit1 = 0.0;
slack1 = MinMax::min()->initValue();
Vertex *vertex, *bidirect_drvr_vertex;
sta_->graph()->pinVertices(pin, vertex, bidirect_drvr_vertex);
if (vertex)
checkSlew1(pin, vertex, corner, min_max, check_clks,
corner1, rf1, slew1, limit1, slack1);
if (bidirect_drvr_vertex)
checkSlew1(pin, bidirect_drvr_vertex, corner, min_max, check_clks,
corner1, rf1, slew1, limit1, slack1);
}
void
CheckSlewLimits::checkSlew1(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const
{
if (!vertex->isDisabledConstraint()
&& !vertex->isConstant()
&& !sta_->clkNetwork()->isIdealClock(pin)) {
ClockSet clks;
if (check_clks)
clks = clockDomains(vertex);
if (corner)
checkSlew2(pin, vertex, corner, min_max, clks,
corner1, rf1, slew1, limit1, slack1);
else {
for (auto corner : *sta_->corners()) {
checkSlew2(pin, vertex, corner, min_max, clks,
corner1, rf1, slew1, limit1, slack1);
}
}
}
}
void
CheckSlewLimits::checkSlews1(const Pin *pin,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const
CheckSlewLimits::checkSlew2(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
const ClockSet &clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const
{
Vertex *vertex, *bidirect_drvr_vertex;
sta_->graph()->pinVertices(pin, vertex, bidirect_drvr_vertex);
if (vertex)
checkSlews1(vertex, corner, min_max, check_clks,
corner1, rf1, slew1, limit1, slack1);
if (bidirect_drvr_vertex)
checkSlews1(bidirect_drvr_vertex, corner, min_max, check_clks,
corner1, rf1, slew1, limit1, slack1);
}
void
CheckSlewLimits::checkSlews1(Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const
{
const Pin *pin = vertex->pin();
if (!vertex->isDisabledConstraint()
&& !vertex->isConstant()
&& !sta_->clkNetwork()->isIdealClock(pin)) {
for (auto rf : RiseFall::range()) {
float limit;
bool exists;
findLimit(pin, vertex, corner, rf, min_max, check_clks,
limit, exists);
if (exists) {
checkSlew(vertex, corner, rf, min_max, limit,
corner1, rf1, slew1, slack1, limit1);
}
for (const RiseFall *rf : RiseFall::range()) {
float limit;
bool exists;
findLimit(pin, corner, rf, min_max, clks,
limit, exists);
if (exists) {
checkSlew3(vertex, corner, rf, min_max, limit,
corner1, rf1, slew1, slack1, limit1);
}
}
}
void
CheckSlewLimits::checkSlew3(const Vertex *vertex,
const Corner *corner,
const RiseFall *rf,
const MinMax *min_max,
float limit,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &slack1,
float &limit1) const
{
const DcalcAnalysisPt *dcalc_ap = corner->findDcalcAnalysisPt(min_max);
Slew slew = sta_->graph()->slew(vertex, rf, dcalc_ap->index());
float slew2 = delayAsFloat(slew);
float slack = (min_max == MinMax::max())
? limit - slew2 : slew2 - limit;
if (corner1 == nullptr
|| (slack < slack1
// Break ties for the sake of regression stability.
|| (fuzzyEqual(slack, slack1)
&& rf->index() < rf1->index()))) {
corner1 = corner;
rf1 = rf;
slew1 = slew;
slack1 = slack;
limit1 = limit;
}
}
// Return the tightest limit.
void
CheckSlewLimits::findLimit(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const RiseFall *rf,
const MinMax *min_max,
bool check_clks,
const ClockSet &clks,
// Return values.
float &limit,
bool &exists) const
@ -186,14 +249,10 @@ CheckSlewLimits::findLimit(const Pin *pin,
float limit1;
bool exists1;
if (check_clks) {
if (!clks.empty()) {
// Look for clock slew limits.
bool is_clk = sta_->clkNetwork()->isIdealClock(pin);
ClockSet clks;
clockDomains(vertex, clks);
ClockSet::Iterator clk_iter(clks);
while (clk_iter.hasNext()) {
Clock *clk = clk_iter.next();
for (Clock *clk : clks) {
PathClkOrData clk_data = is_clk ? PathClkOrData::clk : PathClkOrData::data;
sdc->slewLimit(clk, rf, clk_data, min_max,
limit1, exists1);
@ -284,11 +343,10 @@ CheckSlewLimits::findLimit(const LibertyPort *port,
}
}
void
CheckSlewLimits::clockDomains(const Vertex *vertex,
// Return value.
ClockSet &clks) const
ClockSet
CheckSlewLimits::clockDomains(const Vertex *vertex) const
{
ClockSet clks;
VertexPathIterator path_iter(const_cast<Vertex*>(vertex), sta_);
while (path_iter.hasNext()) {
Path *path = path_iter.next();
@ -296,37 +354,7 @@ CheckSlewLimits::clockDomains(const Vertex *vertex,
if (clk)
clks.insert(const_cast<Clock*>(clk));
}
}
void
CheckSlewLimits::checkSlew(Vertex *vertex,
const Corner *corner,
const RiseFall *rf,
const MinMax *min_max,
float limit,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &slack1,
float &limit1) const
{
const DcalcAnalysisPt *dcalc_ap = corner->findDcalcAnalysisPt(min_max);
Slew slew = sta_->graph()->slew(vertex, rf, dcalc_ap->index());
float slew2 = delayAsFloat(slew);
float slack = (min_max == MinMax::max())
? limit - slew2 : slew2 - limit;
if (corner1 == nullptr
|| (slack < slack1
// Break ties for the sake of regression stability.
|| (fuzzyEqual(slack, slack1)
&& rf->index() < rf1->index()))) {
corner1 = corner;
rf1 = rf;
slew1 = slew;
slack1 = slack;
limit1 = limit;
}
return clks;
}
////////////////////////////////////////////////////////////////
@ -383,32 +411,4 @@ CheckSlewLimits::checkSlewLimits(const Instance *inst,
delete pin_iter;
}
void
CheckSlewLimits::checkSlewLimits(const Pin *pin,
bool violators,
const Corner *corner,
const MinMax *min_max,
PinSeq &slew_pins,
float &min_slack)
{
const Corner *corner1;
const RiseFall *rf;
Slew slew;
float limit, slack;
checkSlew(pin, corner, min_max, true, corner1, rf, slew, limit, slack);
if (!fuzzyInf(slack)) {
if (violators) {
if (slack < 0.0)
slew_pins.push_back(pin);
}
else {
if (slew_pins.empty()
|| slack < min_slack) {
slew_pins.push_back(pin);
min_slack = slack;
}
}
}
}
} // namespace

View File

@ -33,6 +33,13 @@ class CheckSlewLimits
{
public:
CheckSlewLimits(const StaState *sta);
// Return pins with the min/max slew limit slack.
// net=null check all nets
// corner=nullptr checks all corners.
PinSeq checkSlewLimits(const Net *net,
bool violators,
const Corner *corner,
const MinMax *min_max);
// corner=nullptr checks all corners.
void checkSlew(const Pin *pin,
const Corner *corner,
@ -45,13 +52,6 @@ public:
Slew &slew,
float &limit,
float &slack) const;
// Return pins with the min/max slew limit slack.
// net=null check all nets
// corner=nullptr checks all corners.
PinSeq checkSlewLimits(const Net *net,
bool violators,
const Corner *corner,
const MinMax *min_max);
void findLimit(const LibertyPort *port,
const Corner *corner,
const MinMax *min_max,
@ -60,43 +60,44 @@ public:
bool &exists) const;
protected:
void checkSlews1(const Pin *pin,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const;
void checkSlews1(Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const;
void checkSlew(Vertex *vertex,
const Corner *corner1,
const RiseFall *rf1,
const MinMax *min_max,
float limit1,
// Return values.
const Corner *&corner,
const RiseFall *&rf,
Slew &slew,
float &slack,
float &limit) const;
void checkSlew1(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
bool check_clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const;
void checkSlew2(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const MinMax *min_max,
const ClockSet &clks,
// Return values.
const Corner *&corner1,
const RiseFall *&rf1,
Slew &slew1,
float &limit1,
float &slack1) const;
void checkSlew3(const Vertex *vertex,
const Corner *corner1,
const RiseFall *rf1,
const MinMax *min_max,
float limit1,
// Return values.
const Corner *&corner,
const RiseFall *&rf,
Slew &slew,
float &slack,
float &limit) const;
void findLimit(const Pin *pin,
const Vertex *vertex,
const Corner *corner,
const RiseFall *rf,
const MinMax *min_max,
bool check_clks,
const ClockSet &clks,
// Return values.
float &limit,
bool &limit_exists) const;
@ -112,9 +113,7 @@ protected:
const MinMax *min_max,
PinSeq &slew_pins,
float &min_slack);
void clockDomains(const Vertex *vertex,
// Return value.
ClockSet &clks) const;
ClockSet clockDomains(const Vertex *vertex) const;
const StaState *sta_;
};