2012-04-13 03:08:20 +02:00
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
2006-08-26 13:35:28 +02:00
|
|
|
//*************************************************************************
|
|
|
|
|
// DESCRIPTION: Verilator: Removal of named begin blocks
|
|
|
|
|
//
|
2019-11-08 04:33:59 +01:00
|
|
|
// Code available from: https://verilator.org
|
2006-08-26 13:35:28 +02:00
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
//
|
2021-01-01 16:29:54 +01:00
|
|
|
// Copyright 2003-2021 by Wilson Snyder. This program is free software; you
|
2020-03-21 16:24:24 +01:00
|
|
|
// can redistribute it and/or modify it under the terms of either the GNU
|
2009-05-04 23:07:57 +02:00
|
|
|
// Lesser General Public License Version 3 or the Perl Artistic License
|
|
|
|
|
// Version 2.0.
|
2020-03-21 16:24:24 +01:00
|
|
|
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
2006-08-26 13:35:28 +02:00
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
// V3Begin's Transformations:
|
2008-06-10 03:25:10 +02:00
|
|
|
//
|
2006-08-26 13:35:28 +02:00
|
|
|
// Each module:
|
2019-05-19 22:13:13 +02:00
|
|
|
// Look for BEGINs
|
|
|
|
|
// BEGIN(VAR...) -> VAR ... {renamed}
|
|
|
|
|
// FOR -> WHILEs
|
2006-08-26 13:35:28 +02:00
|
|
|
//
|
2009-10-12 02:50:31 +02:00
|
|
|
// There are two scopes; named BEGINs change %m and variable scopes.
|
|
|
|
|
// Unnamed BEGINs change only variable, not $display("%m") scope.
|
|
|
|
|
//
|
2006-08-26 13:35:28 +02:00
|
|
|
//*************************************************************************
|
2019-10-05 02:17:11 +02:00
|
|
|
|
2006-12-18 20:20:45 +01:00
|
|
|
#include "config_build.h"
|
|
|
|
|
#include "verilatedos.h"
|
2006-08-26 13:35:28 +02:00
|
|
|
|
|
|
|
|
#include "V3Global.h"
|
|
|
|
|
#include "V3Begin.h"
|
|
|
|
|
#include "V3Ast.h"
|
|
|
|
|
|
2018-10-14 19:43:24 +02:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
2006-08-26 13:35:28 +02:00
|
|
|
//######################################################################
|
|
|
|
|
|
2020-11-19 03:32:16 +01:00
|
|
|
class BeginState final {
|
2012-02-22 04:23:06 +01:00
|
|
|
private:
|
|
|
|
|
// NODE STATE
|
2020-04-15 13:58:34 +02:00
|
|
|
// Entire netlist:
|
2019-05-19 22:13:13 +02:00
|
|
|
// AstNodeFTask::user1 -> bool, 1=processed
|
2020-04-15 13:58:34 +02:00
|
|
|
AstUser1InUse m_inuser1;
|
2020-08-15 19:11:27 +02:00
|
|
|
bool m_anyFuncInBegin = false;
|
2020-04-15 13:58:34 +02:00
|
|
|
|
2012-02-22 04:23:06 +01:00
|
|
|
public:
|
2020-11-17 01:56:16 +01:00
|
|
|
BeginState() = default;
|
|
|
|
|
~BeginState() = default;
|
2015-11-14 15:06:09 +01:00
|
|
|
void userMarkChanged(AstNode* nodep) {
|
2019-05-19 22:13:13 +02:00
|
|
|
nodep->user1(true);
|
|
|
|
|
m_anyFuncInBegin = true;
|
2012-02-22 04:23:06 +01:00
|
|
|
}
|
|
|
|
|
bool anyFuncInBegin() const { return m_anyFuncInBegin; }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//######################################################################
|
|
|
|
|
|
2020-11-19 03:32:16 +01:00
|
|
|
class BeginVisitor final : public AstNVisitor {
|
2006-08-26 13:35:28 +02:00
|
|
|
private:
|
|
|
|
|
// STATE
|
2020-04-15 13:58:34 +02:00
|
|
|
BeginState* m_statep; // Current global state
|
2020-08-15 19:11:27 +02:00
|
|
|
AstNodeModule* m_modp = nullptr; // Current module
|
|
|
|
|
AstNodeFTask* m_ftaskp = nullptr; // Current function/task
|
2020-04-15 13:58:34 +02:00
|
|
|
string m_namedScope; // Name of begin blocks above us
|
|
|
|
|
string m_unnamedScope; // Name of begin blocks, including unnamed blocks
|
2020-08-15 19:11:27 +02:00
|
|
|
int m_ifDepth = 0; // Current if depth
|
2009-01-21 22:56:50 +01:00
|
|
|
|
|
|
|
|
// METHODS
|
2018-05-14 12:50:47 +02:00
|
|
|
VL_DEBUG_FUNC; // Declare debug()
|
2006-08-26 13:35:28 +02:00
|
|
|
|
2020-10-28 03:35:29 +01:00
|
|
|
string dot(const string& a, const string& b) {
|
|
|
|
|
if (a == "") return b;
|
|
|
|
|
return a + "__DOT__" + b;
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-26 13:35:28 +02:00
|
|
|
// VISITORS
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstNodeModule* nodep) override {
|
2020-08-25 03:10:43 +02:00
|
|
|
VL_RESTORER(m_modp);
|
2020-01-20 19:27:27 +01:00
|
|
|
{
|
|
|
|
|
m_modp = nodep;
|
|
|
|
|
iterateChildren(nodep);
|
|
|
|
|
}
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstNodeFTask* nodep) override {
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
// Rename it
|
|
|
|
|
if (m_unnamedScope != "") {
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->name(dot(m_unnamedScope, nodep->name()));
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " rename to " << nodep->name() << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
m_statep->userMarkChanged(nodep);
|
|
|
|
|
}
|
|
|
|
|
// BEGIN wrapping a function rename that function, but don't affect
|
|
|
|
|
// the inside function's variables. We then restart with empty
|
|
|
|
|
// naming; so that any begin's inside the function will rename
|
|
|
|
|
// inside the function.
|
|
|
|
|
// Process children
|
2020-08-25 03:10:43 +02:00
|
|
|
VL_RESTORER(m_namedScope);
|
|
|
|
|
VL_RESTORER(m_unnamedScope);
|
2019-05-19 22:13:13 +02:00
|
|
|
{
|
|
|
|
|
m_namedScope = "";
|
|
|
|
|
m_unnamedScope = "";
|
|
|
|
|
m_ftaskp = nodep;
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2020-08-15 16:12:55 +02:00
|
|
|
m_ftaskp = nullptr;
|
2019-05-19 22:13:13 +02:00
|
|
|
}
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstBegin* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
// Begin blocks were only useful in variable creation, change names and delete
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " " << nodep << endl);
|
2020-10-28 03:35:29 +01:00
|
|
|
VL_RESTORER(m_namedScope);
|
|
|
|
|
VL_RESTORER(m_unnamedScope);
|
2019-05-19 22:13:13 +02:00
|
|
|
{
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, "nname " << m_namedScope << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
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;
|
2020-04-15 13:58:34 +02:00
|
|
|
while ((pos = dottedname.find("__DOT__")) != string::npos) {
|
2019-05-19 22:13:13 +02:00
|
|
|
string ident = dottedname.substr(0, pos);
|
2020-04-15 13:58:34 +02:00
|
|
|
dottedname = dottedname.substr(pos + strlen("__DOT__"));
|
2020-10-28 03:35:29 +01:00
|
|
|
if (nodep->name() != "") m_namedScope = dot(m_namedScope, ident);
|
|
|
|
|
m_unnamedScope = dot(m_unnamedScope, ident);
|
2019-05-19 22:13:13 +02:00
|
|
|
// Create CellInline for dotted var resolution
|
|
|
|
|
if (!m_ftaskp) {
|
2020-04-16 01:39:03 +02:00
|
|
|
AstCellInline* inlinep = new AstCellInline(
|
|
|
|
|
nodep->fileline(), m_unnamedScope, "__BEGIN__", m_modp->timeunit());
|
2019-05-19 22:13:13 +02:00
|
|
|
m_modp->addInlinesp(inlinep); // Must be parsed before any AstCells
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-08-26 13:35:28 +02:00
|
|
|
|
2019-05-19 22:13:13 +02:00
|
|
|
// Remap var names and replace lower Begins
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateAndNextNull(nodep->stmtsp());
|
2019-07-06 18:57:50 +02:00
|
|
|
UASSERT_OBJ(!nodep->genforp(), nodep, "GENFORs should have been expanded earlier");
|
2011-11-30 00:23:18 +01:00
|
|
|
|
2020-10-28 03:35:29 +01:00
|
|
|
// Cleanup
|
|
|
|
|
AstNode* addsp = nullptr;
|
|
|
|
|
if (AstNode* stmtsp = nodep->stmtsp()) {
|
|
|
|
|
stmtsp->unlinkFrBackWithNext();
|
|
|
|
|
if (addsp) {
|
|
|
|
|
addsp = addsp->addNextNull(stmtsp);
|
|
|
|
|
} else {
|
|
|
|
|
addsp = stmtsp;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-04-15 13:58:34 +02:00
|
|
|
if (addsp) {
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->replaceWith(addsp);
|
2020-04-15 13:58:34 +02:00
|
|
|
} else {
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->unlinkFrBack();
|
2020-04-15 13:58:34 +02:00
|
|
|
}
|
2020-10-28 03:35:29 +01:00
|
|
|
VL_DO_DANGLING(pushDeletep(nodep), nodep);
|
2019-05-19 22:13:13 +02:00
|
|
|
}
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstVar* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
if (m_unnamedScope != "") {
|
|
|
|
|
// Rename it
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->name(dot(m_unnamedScope, nodep->name()));
|
2019-05-19 22:13:13 +02:00
|
|
|
m_statep->userMarkChanged(nodep);
|
|
|
|
|
// Move to module
|
|
|
|
|
nodep->unlinkFrBack();
|
2020-04-15 13:58:34 +02:00
|
|
|
if (m_ftaskp) {
|
|
|
|
|
m_ftaskp->addStmtsp(nodep); // Begins under funcs just move into the func
|
|
|
|
|
} else {
|
|
|
|
|
m_modp->addStmtp(nodep);
|
|
|
|
|
}
|
2019-05-19 22:13:13 +02:00
|
|
|
}
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstTypedef* nodep) override {
|
2020-03-26 23:10:20 +01:00
|
|
|
if (m_unnamedScope != "") {
|
|
|
|
|
// Rename it
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->name(dot(m_unnamedScope, nodep->name()));
|
2020-03-26 23:10:20 +01:00
|
|
|
m_statep->userMarkChanged(nodep);
|
|
|
|
|
// Move to module
|
|
|
|
|
nodep->unlinkFrBack();
|
|
|
|
|
// Begins under funcs just move into the func
|
2020-04-15 13:58:34 +02:00
|
|
|
if (m_ftaskp) {
|
|
|
|
|
m_ftaskp->addStmtsp(nodep);
|
|
|
|
|
} else {
|
|
|
|
|
m_modp->addStmtp(nodep);
|
|
|
|
|
}
|
2020-03-26 23:10:20 +01:00
|
|
|
}
|
|
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstCell* nodep) override {
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " CELL " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
if (m_namedScope != "") {
|
|
|
|
|
m_statep->userMarkChanged(nodep);
|
|
|
|
|
// Rename it
|
2020-10-28 03:35:29 +01:00
|
|
|
nodep->name(dot(m_namedScope, nodep->name()));
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " rename to " << nodep->name() << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
// Move to module
|
|
|
|
|
nodep->unlinkFrBack();
|
|
|
|
|
m_modp->addStmtp(nodep);
|
|
|
|
|
}
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstVarXRef* nodep) override {
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(9, " VARXREF " << nodep << endl);
|
2020-09-27 16:10:44 +02:00
|
|
|
if (m_namedScope != "" && nodep->inlinedDots() == "" && !m_ftaskp) {
|
2019-05-19 22:13:13 +02:00
|
|
|
nodep->inlinedDots(m_namedScope);
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(9, " rescope to " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
}
|
2015-12-06 01:39:40 +01:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstScopeName* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
// If there's a %m in the display text, we add a special node that will contain the name()
|
|
|
|
|
// Similar code in V3Inline
|
|
|
|
|
if (nodep->user1SetOnce()) return; // Don't double-add text's
|
|
|
|
|
if (m_namedScope != "") {
|
|
|
|
|
// To keep correct visual order, must add before other Text's
|
|
|
|
|
AstNode* afterp = nodep->scopeAttrp();
|
|
|
|
|
if (afterp) afterp->unlinkFrBackWithNext();
|
2020-04-15 13:58:34 +02:00
|
|
|
nodep->scopeAttrp(new AstText(nodep->fileline(), string("__DOT__") + m_namedScope));
|
2019-05-19 22:13:13 +02:00
|
|
|
if (afterp) nodep->scopeAttrp(afterp);
|
|
|
|
|
}
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2007-03-06 19:53:24 +01:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstCoverDecl* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
// Don't need to fix path in coverage statements, they're not under
|
|
|
|
|
// any BEGINs, but V3Coverage adds them all under the module itself.
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2008-11-05 16:23:03 +01:00
|
|
|
}
|
2010-12-26 15:31:09 +01:00
|
|
|
// VISITORS - LINT CHECK
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstIf* nodep) override { // not AstNodeIf; other types not covered
|
2019-05-19 22:13:13 +02:00
|
|
|
// Check IFDEPTH warning - could be in other transform files if desire
|
2020-08-25 03:10:43 +02:00
|
|
|
VL_RESTORER(m_ifDepth);
|
2020-04-15 13:58:34 +02:00
|
|
|
if (m_ifDepth == -1 || v3Global.opt.ifDepth() < 1) { // Turned off
|
2019-05-19 22:13:13 +02:00
|
|
|
} else if (nodep->uniquePragma() || nodep->unique0Pragma() || nodep->priorityPragma()) {
|
|
|
|
|
m_ifDepth = -1;
|
|
|
|
|
} else if (++m_ifDepth > v3Global.opt.ifDepth()) {
|
2020-04-15 13:58:34 +02:00
|
|
|
nodep->v3warn(IFDEPTH,
|
|
|
|
|
"Deep 'if' statement; suggest unique/priority to avoid slow logic");
|
2019-05-19 22:13:13 +02:00
|
|
|
nodep->fileline()->modifyWarnOff(V3ErrorCode::IFDEPTH, true); // Warn only once
|
|
|
|
|
m_ifDepth = -1;
|
|
|
|
|
}
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2015-12-06 01:39:40 +01:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstNode* nodep) override { iterateChildren(nodep); }
|
2020-04-04 14:31:14 +02:00
|
|
|
|
2006-08-26 13:35:28 +02:00
|
|
|
public:
|
2019-09-12 13:22:22 +02:00
|
|
|
// CONSTRUCTORS
|
2020-08-15 19:11:27 +02:00
|
|
|
BeginVisitor(AstNetlist* nodep, BeginState* statep)
|
2020-08-16 15:55:36 +02:00
|
|
|
: m_statep{statep} {
|
2018-05-11 02:55:37 +02:00
|
|
|
iterate(nodep);
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|
2020-11-17 01:56:16 +01:00
|
|
|
virtual ~BeginVisitor() override = default;
|
2006-08-26 13:35:28 +02:00
|
|
|
};
|
|
|
|
|
|
2012-02-22 04:23:06 +01:00
|
|
|
//######################################################################
|
|
|
|
|
|
2020-11-19 03:32:16 +01:00
|
|
|
class BeginRelinkVisitor final : public AstNVisitor {
|
2012-02-22 04:23:06 +01:00
|
|
|
// Replace tasks with new pointer
|
|
|
|
|
private:
|
|
|
|
|
// NODE STATE
|
|
|
|
|
// Input:
|
2019-05-19 22:13:13 +02:00
|
|
|
// AstNodeFTask::user1p // Node replaced, rename it
|
2012-02-22 04:23:06 +01:00
|
|
|
|
|
|
|
|
// VISITORS
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstNodeFTaskRef* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
if (nodep->taskp()->user1()) { // It was converted
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(9, " relinkFTask " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
nodep->name(nodep->taskp()->name());
|
|
|
|
|
}
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2012-02-22 04:23:06 +01:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstVarRef* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
if (nodep->varp()->user1()) { // It was converted
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(9, " relinVarRef " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
nodep->name(nodep->varp()->name());
|
|
|
|
|
}
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2015-11-14 15:06:09 +01:00
|
|
|
}
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstIfaceRefDType* nodep) override {
|
2019-05-19 22:13:13 +02:00
|
|
|
// May have changed cell names
|
|
|
|
|
// TypeTable is always after all modules, so names are stable
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " IFACEREFDTYPE " << nodep << endl);
|
2019-05-19 22:13:13 +02:00
|
|
|
if (nodep->cellp()) nodep->cellName(nodep->cellp()->name());
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(8, " rename to " << nodep << endl);
|
2018-05-11 02:55:37 +02:00
|
|
|
iterateChildren(nodep);
|
2015-11-14 15:06:09 +01:00
|
|
|
}
|
2012-02-22 04:23:06 +01:00
|
|
|
//--------------------
|
2020-08-15 16:03:34 +02:00
|
|
|
virtual void visit(AstNode* nodep) override { iterateChildren(nodep); }
|
2020-04-04 14:31:14 +02:00
|
|
|
|
2012-02-22 04:23:06 +01:00
|
|
|
public:
|
2019-09-12 13:22:22 +02:00
|
|
|
// CONSTRUCTORS
|
2020-04-15 13:58:34 +02:00
|
|
|
BeginRelinkVisitor(AstNetlist* nodep, BeginState*) { iterate(nodep); }
|
2020-11-17 01:56:16 +01:00
|
|
|
virtual ~BeginRelinkVisitor() override = default;
|
2012-02-22 04:23:06 +01:00
|
|
|
};
|
|
|
|
|
|
2006-08-26 13:35:28 +02:00
|
|
|
//######################################################################
|
|
|
|
|
// Task class functions
|
|
|
|
|
|
|
|
|
|
void V3Begin::debeginAll(AstNetlist* nodep) {
|
2020-04-15 13:58:34 +02:00
|
|
|
UINFO(2, __FUNCTION__ << ": " << endl);
|
2018-03-10 18:57:50 +01:00
|
|
|
{
|
|
|
|
|
BeginState state;
|
2020-04-15 13:58:34 +02:00
|
|
|
{ BeginVisitor bvisitor(nodep, &state); }
|
|
|
|
|
if (state.anyFuncInBegin()) { BeginRelinkVisitor brvisitor(nodep, &state); }
|
2018-03-10 18:57:50 +01:00
|
|
|
} // Destruct before checking
|
2017-09-18 04:52:57 +02:00
|
|
|
V3Global::dumpCheckGlobalTree("begin", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 3);
|
2006-08-26 13:35:28 +02:00
|
|
|
}
|