2006-08-26 11:35:28 +00:00
|
|
|
|
//*************************************************************************
|
|
|
|
|
// DESCRIPTION: Verilator: Removal of named begin blocks
|
|
|
|
|
//
|
2008-04-25 12:14:27 +00:00
|
|
|
|
// Code available from: http://www.veripool.org/verilator
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
|
|
|
|
// AUTHORS: Wilson Snyder with Paul Wasson, Duane Gabli
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
//
|
2010-01-06 02:15:06 +00:00
|
|
|
|
// Copyright 2003-2010 by Wilson Snyder. This program is free software; you can
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// redistribute it and/or modify it under the terms of either the GNU
|
2009-05-04 21:07:57 +00:00
|
|
|
|
// Lesser General Public License Version 3 or the Perl Artistic License
|
|
|
|
|
// Version 2.0.
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
|
|
|
|
// Verilator is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
// V3Begin's Transformations:
|
2008-06-10 01:25:10 +00:00
|
|
|
|
//
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Each module:
|
|
|
|
|
// Look for BEGINs
|
|
|
|
|
// BEGIN(VAR...) -> VAR ... {renamed}
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// FOR -> WHILEs
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
2009-10-12 00:50:31 +00:00
|
|
|
|
// There are two scopes; named BEGINs change %m and variable scopes.
|
|
|
|
|
// Unnamed BEGINs change only variable, not $display("%m") scope.
|
|
|
|
|
//
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//*************************************************************************
|
|
|
|
|
|
2006-12-18 19:20:45 +00:00
|
|
|
|
#include "config_build.h"
|
|
|
|
|
#include "verilatedos.h"
|
2008-06-30 17:11:25 +00:00
|
|
|
|
#include <cstdio>
|
|
|
|
|
#include <cstdarg>
|
2006-08-26 11:35:28 +00:00
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
#include "V3Global.h"
|
|
|
|
|
#include "V3Begin.h"
|
|
|
|
|
#include "V3Inst.h"
|
|
|
|
|
#include "V3Ast.h"
|
|
|
|
|
|
|
|
|
|
//######################################################################
|
|
|
|
|
|
|
|
|
|
class BeginVisitor : public AstNVisitor {
|
|
|
|
|
private:
|
|
|
|
|
// STATE
|
2009-11-07 11:20:20 +00:00
|
|
|
|
AstNodeModule* m_modp; // Current module
|
2006-08-26 11:35:28 +00:00
|
|
|
|
AstNodeFTask* m_ftaskp; // Current function/task
|
2009-10-12 00:50:31 +00:00
|
|
|
|
string m_namedScope; // Name of begin blocks above us
|
|
|
|
|
string m_unnamedScope; // Name of begin blocks, including unnamed blocks
|
2009-02-26 03:06:59 +00:00
|
|
|
|
int m_repeatNum; // Repeat counter
|
2010-12-26 14:31:09 +00:00
|
|
|
|
int m_ifDepth; // Current if depth
|
2009-01-21 21:56:50 +00:00
|
|
|
|
|
|
|
|
|
// METHODS
|
|
|
|
|
static int debug() {
|
|
|
|
|
static int level = -1;
|
|
|
|
|
if (VL_UNLIKELY(level < 0)) level = v3Global.opt.debugSrcLevel(__FILE__);
|
|
|
|
|
return level;
|
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
|
|
|
|
// VISITORS
|
2009-11-07 11:20:20 +00:00
|
|
|
|
virtual void visit(AstNodeModule* nodep, AstNUser*) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_modp = nodep;
|
2009-02-26 03:06:59 +00:00
|
|
|
|
m_repeatNum = 0;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
m_modp = NULL;
|
|
|
|
|
}
|
|
|
|
|
virtual void visit(AstNodeFTask* nodep, AstNUser*) {
|
|
|
|
|
m_ftaskp = nodep;
|
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
m_ftaskp = NULL;
|
|
|
|
|
}
|
|
|
|
|
virtual void visit(AstBegin* nodep, AstNUser*) {
|
|
|
|
|
// Begin blocks were only useful in variable creation, change names and delete
|
|
|
|
|
UINFO(8," "<<nodep<<endl);
|
2009-10-12 00:50:31 +00:00
|
|
|
|
string oldScope = m_namedScope;
|
|
|
|
|
string oldUnnamed = m_unnamedScope;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
{
|
2009-10-12 00:50:31 +00:00
|
|
|
|
//UINFO(8,"nname "<<m_namedScope<<endl);
|
|
|
|
|
if (nodep->name() != "") { // Else unneeded unnamed block
|
|
|
|
|
// Create data for dotted variable resolution
|
|
|
|
|
string dottedname = nodep->name() + "__DOT__"; // So always found
|
|
|
|
|
string::size_type pos;
|
|
|
|
|
while ((pos=dottedname.find("__DOT__")) != string::npos) {
|
|
|
|
|
string ident = dottedname.substr(0,pos);
|
|
|
|
|
dottedname = dottedname.substr(pos+strlen("__DOT__"));
|
|
|
|
|
if (!nodep->unnamed()) {
|
|
|
|
|
if (m_namedScope=="") m_namedScope = ident;
|
|
|
|
|
else m_namedScope = m_namedScope + "__DOT__"+ident;
|
|
|
|
|
}
|
|
|
|
|
if (m_unnamedScope=="") m_unnamedScope = ident;
|
|
|
|
|
else m_unnamedScope = m_unnamedScope + "__DOT__"+ident;
|
|
|
|
|
// Create CellInline for dotted var resolution
|
|
|
|
|
AstCellInline* inlinep = new AstCellInline(nodep->fileline(),
|
|
|
|
|
m_unnamedScope, "__BEGIN__");
|
|
|
|
|
m_modp->addInlinesp(inlinep); // Must be parsed before any AstCells
|
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-10-12 00:50:31 +00:00
|
|
|
|
// Remap var names and replace lower Begins
|
2006-08-26 11:35:28 +00:00
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
|
|
|
|
|
if (AstNode* stmtsp = nodep->stmtsp()) {
|
|
|
|
|
stmtsp->unlinkFrBackWithNext();
|
|
|
|
|
nodep->replaceWith(stmtsp);
|
|
|
|
|
} else {
|
|
|
|
|
nodep->unlinkFrBack();
|
|
|
|
|
}
|
|
|
|
|
pushDeletep(nodep); nodep=NULL;
|
|
|
|
|
}
|
2009-10-12 00:50:31 +00:00
|
|
|
|
m_namedScope = oldScope;
|
|
|
|
|
m_unnamedScope = oldUnnamed;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
virtual void visit(AstVar* nodep, AstNUser*) {
|
2009-10-12 00:50:31 +00:00
|
|
|
|
if (m_unnamedScope != "") {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Rename it
|
2009-10-12 00:50:31 +00:00
|
|
|
|
nodep->name(m_unnamedScope+"__DOT__"+nodep->name());
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Move to module
|
|
|
|
|
nodep->unlinkFrBack();
|
|
|
|
|
if (m_ftaskp) m_ftaskp->addStmtsp(nodep); // Begins under funcs just move into the func
|
|
|
|
|
else m_modp->addStmtp(nodep);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
virtual void visit(AstCell* nodep, AstNUser*) {
|
|
|
|
|
UINFO(8," CELL "<<nodep<<endl);
|
2009-10-12 00:50:31 +00:00
|
|
|
|
if (m_namedScope != "") {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Rename it
|
2009-10-12 00:50:31 +00:00
|
|
|
|
nodep->name(m_namedScope+"__DOT__"+nodep->name());
|
2006-08-26 11:35:28 +00:00
|
|
|
|
UINFO(8," rename to "<<nodep->name()<<endl);
|
|
|
|
|
// Move to module
|
|
|
|
|
nodep->unlinkFrBack();
|
|
|
|
|
m_modp->addStmtp(nodep);
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-06-14 16:41:32 +00:00
|
|
|
|
virtual void visit(AstScopeName* nodep, AstNUser*) {
|
2007-03-06 18:53:24 +00:00
|
|
|
|
// If there's a %m in the display text, we add a special node that will contain the name()
|
|
|
|
|
// Similar code in V3Inline
|
2009-10-12 00:50:31 +00:00
|
|
|
|
if (m_namedScope != "") {
|
2007-03-06 18:53:24 +00:00
|
|
|
|
// To keep correct visual order, must add before other Text's
|
|
|
|
|
AstNode* afterp = nodep->scopeAttrp();
|
|
|
|
|
if (afterp) afterp->unlinkFrBackWithNext();
|
2009-12-05 15:38:49 +00:00
|
|
|
|
nodep->scopeAttrp(new AstText(nodep->fileline(), (string)"__DOT__"+m_namedScope));
|
2007-03-06 18:53:24 +00:00
|
|
|
|
if (afterp) nodep->scopeAttrp(afterp);
|
|
|
|
|
}
|
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
}
|
2008-11-05 15:23:03 +00:00
|
|
|
|
virtual void visit(AstCoverDecl* nodep, AstNUser*) {
|
|
|
|
|
// Don't need to fix path in coverage statements, they're not under
|
|
|
|
|
// any BEGINs, but V3Coverage adds them all under the module itself.
|
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
}
|
2010-12-26 14:31:09 +00:00
|
|
|
|
// VISITORS - LINT CHECK
|
|
|
|
|
virtual void visit(AstIf* nodep, AstNUser*) { // Note not AstNodeIf; other types don't get covered
|
|
|
|
|
// Check IFDEPTH warning - could be in other transform files if desire
|
|
|
|
|
int prevIfDepth = m_ifDepth;
|
|
|
|
|
if (m_ifDepth == -1 || v3Global.opt.ifDepth()<1) { // Turned off
|
|
|
|
|
} else if (nodep->uniquePragma() || nodep->unique0Pragma() || nodep->priorityPragma()) {
|
|
|
|
|
m_ifDepth = -1;
|
|
|
|
|
} else if (++m_ifDepth > v3Global.opt.ifDepth()) {
|
|
|
|
|
nodep->v3warn(IFDEPTH,"Deep 'if' statement; suggest unique/priority to avoid slow logic");
|
|
|
|
|
nodep->fileline()->warnOn(V3ErrorCode::IFDEPTH, false); // Warn only once
|
|
|
|
|
m_ifDepth = -1;
|
|
|
|
|
}
|
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
m_ifDepth = prevIfDepth;
|
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
virtual void visit(AstNode* nodep, AstNUser*) {
|
|
|
|
|
nodep->iterateChildren(*this);
|
|
|
|
|
}
|
|
|
|
|
public:
|
|
|
|
|
// CONSTUCTORS
|
|
|
|
|
BeginVisitor(AstNetlist* nodep) {
|
|
|
|
|
m_modp = NULL;
|
|
|
|
|
m_ftaskp = NULL;
|
2009-02-26 03:06:59 +00:00
|
|
|
|
m_repeatNum = 0;
|
2010-12-26 14:31:09 +00:00
|
|
|
|
m_ifDepth = 0;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
nodep->accept(*this);
|
|
|
|
|
}
|
|
|
|
|
virtual ~BeginVisitor() {}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//######################################################################
|
|
|
|
|
// Task class functions
|
|
|
|
|
|
|
|
|
|
void V3Begin::debeginAll(AstNetlist* nodep) {
|
|
|
|
|
UINFO(2,__FUNCTION__<<": "<<endl);
|
|
|
|
|
BeginVisitor bvisitor (nodep);
|
|
|
|
|
}
|