forked from github/verilator
Fix processing unused parametrized modules, bug470.
This commit is contained in:
parent
81d83f629f
commit
996f48fcf0
2
Changes
2
Changes
@ -8,6 +8,8 @@ indicates the contributor was also the author of the fix; Thanks!
|
|||||||
|
|
||||||
*** Support += and -= in standard for loops, bug463. [Alex Solomatnikov]
|
*** Support += and -= in standard for loops, bug463. [Alex Solomatnikov]
|
||||||
|
|
||||||
|
*** Fix processing unused parametrized modules, bug470. [Alex Solomatnikov]
|
||||||
|
|
||||||
**** Fix signed array warning, bug456. [Alex Solomatnikov]
|
**** Fix signed array warning, bug456. [Alex Solomatnikov]
|
||||||
|
|
||||||
**** Fix genvar and begin under generate, bug461. [Alex Solomatnikov]
|
**** Fix genvar and begin under generate, bug461. [Alex Solomatnikov]
|
||||||
|
@ -26,6 +26,8 @@
|
|||||||
// Clone module cell calls, renaming with __{par1}_{par2}_...
|
// Clone module cell calls, renaming with __{par1}_{par2}_...
|
||||||
// Substitute constants for cell's module's parameters
|
// Substitute constants for cell's module's parameters
|
||||||
// Relink pins and cell to point to new module
|
// Relink pins and cell to point to new module
|
||||||
|
// Then process all modules called by that cell
|
||||||
|
// (Cells never referenced after parameters expanded must be ignored.)
|
||||||
//
|
//
|
||||||
//*************************************************************************
|
//*************************************************************************
|
||||||
|
|
||||||
@ -36,6 +38,7 @@
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <deque>
|
||||||
|
|
||||||
#include "V3Global.h"
|
#include "V3Global.h"
|
||||||
#include "V3Param.h"
|
#include "V3Param.h"
|
||||||
@ -51,7 +54,8 @@
|
|||||||
class ParamVisitor : public AstNVisitor {
|
class ParamVisitor : public AstNVisitor {
|
||||||
private:
|
private:
|
||||||
// NODE STATE
|
// NODE STATE
|
||||||
// AstNodeModule::user5() // bool True if parameters numbered
|
// AstNodeModule::user5() // bool True if processed
|
||||||
|
// AstVar::user5() // bool True if constant propagated
|
||||||
// AstVar::user4() // int Global parameter number (for naming new module)
|
// AstVar::user4() // int Global parameter number (for naming new module)
|
||||||
// // (0=not processed, 1=iterated, but no number, 65+ parameter numbered)
|
// // (0=not processed, 1=iterated, but no number, 65+ parameter numbered)
|
||||||
AstUser4InUse m_inuser4;
|
AstUser4InUse m_inuser4;
|
||||||
@ -72,6 +76,9 @@ private:
|
|||||||
LongMap m_longMap; // Hash of very long names to unique identity number
|
LongMap m_longMap; // Hash of very long names to unique identity number
|
||||||
int m_longId;
|
int m_longId;
|
||||||
|
|
||||||
|
typedef deque<AstNodeModule*> ModDeque;
|
||||||
|
ModDeque m_todoModps; // Modules left to process
|
||||||
|
|
||||||
// METHODS
|
// METHODS
|
||||||
static int debug() {
|
static int debug() {
|
||||||
static int level = -1;
|
static int level = -1;
|
||||||
@ -116,6 +123,19 @@ private:
|
|||||||
pinp->modVarp(cloneiter->second);
|
pinp->modVarp(cloneiter->second);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
void visitModules() {
|
||||||
|
// Loop on all modules left to process
|
||||||
|
// Hitting a cell adds to the END of this list, so since cells originally exist top->bottom
|
||||||
|
// we process in top->bottom order too.
|
||||||
|
while (!m_todoModps.empty()) {
|
||||||
|
AstNodeModule* nodep = m_todoModps.front(); m_todoModps.pop_front();
|
||||||
|
if (!nodep->user5Inc()) { // Process once; note clone() must clear so we do it again
|
||||||
|
UINFO(4," MOD "<<nodep<<endl);
|
||||||
|
nodep->iterateChildren(*this);
|
||||||
|
// Note this may add to m_todoModps
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// VISITORS
|
// VISITORS
|
||||||
virtual void visit(AstNetlist* nodep, AstNUser*) {
|
virtual void visit(AstNetlist* nodep, AstNUser*) {
|
||||||
@ -123,8 +143,15 @@ private:
|
|||||||
nodep->iterateChildren(*this);
|
nodep->iterateChildren(*this);
|
||||||
}
|
}
|
||||||
virtual void visit(AstNodeModule* nodep, AstNUser*) {
|
virtual void visit(AstNodeModule* nodep, AstNUser*) {
|
||||||
UINFO(4," MOD "<<nodep<<endl);
|
if (nodep->level() <= 2) { // Haven't added top yet, so level 2 is the top
|
||||||
nodep->iterateChildren(*this);
|
// Add request to END of modules left to process
|
||||||
|
m_todoModps.push_back(nodep);
|
||||||
|
visitModules();
|
||||||
|
} else if (nodep->user5()) {
|
||||||
|
UINFO(4," MOD-done "<<nodep<<endl); // Already did it
|
||||||
|
} else {
|
||||||
|
UINFO(4," MOD-dead? "<<nodep<<endl); // Should have been done by now, if not dead
|
||||||
|
}
|
||||||
}
|
}
|
||||||
virtual void visit(AstCell* nodep, AstNUser*);
|
virtual void visit(AstCell* nodep, AstNUser*);
|
||||||
|
|
||||||
@ -325,6 +352,7 @@ void ParamVisitor::visit(AstCell* nodep, AstNUser*) {
|
|||||||
// However links outside the module (like on the upper cells) will not.
|
// However links outside the module (like on the upper cells) will not.
|
||||||
modp = nodep->modp()->cloneTree(false);
|
modp = nodep->modp()->cloneTree(false);
|
||||||
modp->name(newname);
|
modp->name(newname);
|
||||||
|
modp->user5(0); // We need to re-recurse this module once changed
|
||||||
nodep->modp()->addNextHere(modp); // Keep tree sorted by cell occurrences
|
nodep->modp()->addNextHere(modp); // Keep tree sorted by cell occurrences
|
||||||
|
|
||||||
m_modNameMap.insert(make_pair(modp->name(), ModInfo(modp)));
|
m_modNameMap.insert(make_pair(modp->name(), ModInfo(modp)));
|
||||||
@ -377,6 +405,9 @@ void ParamVisitor::visit(AstCell* nodep, AstNUser*) {
|
|||||||
nodep->paramsp()->unlinkFrBackWithNext()->deleteTree();
|
nodep->paramsp()->unlinkFrBackWithNext()->deleteTree();
|
||||||
UINFO(8," Done with "<<nodep<<endl);
|
UINFO(8," Done with "<<nodep<<endl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Now remember to process the child module at the end of the module
|
||||||
|
m_todoModps.push_back(nodep->modp());
|
||||||
}
|
}
|
||||||
|
|
||||||
//######################################################################
|
//######################################################################
|
||||||
|
@ -7,8 +7,6 @@ if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); di
|
|||||||
# Lesser General Public License Version 3 or the Perl Artistic License
|
# Lesser General Public License Version 3 or the Perl Artistic License
|
||||||
# Version 2.0.
|
# Version 2.0.
|
||||||
|
|
||||||
$Self->{vlt} and $Self->unsupported("Verilator unsupported, bug470");
|
|
||||||
|
|
||||||
compile (
|
compile (
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -12,6 +12,11 @@ module t (/*AUTOINST*/);
|
|||||||
|
|
||||||
u_test_inst();
|
u_test_inst();
|
||||||
|
|
||||||
|
initial begin
|
||||||
|
$write("*-* All Finished *-*\n");
|
||||||
|
$finish;
|
||||||
|
end
|
||||||
|
|
||||||
endmodule
|
endmodule
|
||||||
|
|
||||||
module Test ();
|
module Test ();
|
||||||
|
Loading…
Reference in New Issue
Block a user