2012-04-13 01:08:20 +00:00
|
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//*************************************************************************
|
|
|
|
|
// DESCRIPTION: Verilator: Add temporaries, such as for unroll nodes
|
|
|
|
|
//
|
2008-04-25 12:14:27 +00:00
|
|
|
|
// Code available from: http://www.veripool.org/verilator
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
//
|
2018-01-02 23:05:06 +00:00
|
|
|
|
// Copyright 2003-2018 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.
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
// V3Unroll's Transformations:
|
|
|
|
|
// Note is called twice. Once on modules for GenFor unrolling,
|
|
|
|
|
// Again after V3Scope for normal for loop unrolling.
|
2008-06-10 01:25:10 +00:00
|
|
|
|
//
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Each module:
|
|
|
|
|
// Look for "FOR" loops and unroll them if <= 32 loops.
|
|
|
|
|
// (Eventually, a better way would be to simulate the entire loop; ala V3Table.)
|
|
|
|
|
// Convert remaining FORs to WHILEs
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
|
2006-12-18 19:20:45 +00:00
|
|
|
|
#include "config_build.h"
|
|
|
|
|
#include "verilatedos.h"
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
|
|
|
|
#include "V3Global.h"
|
|
|
|
|
#include "V3Unroll.h"
|
|
|
|
|
#include "V3Stats.h"
|
|
|
|
|
#include "V3Const.h"
|
|
|
|
|
#include "V3Ast.h"
|
2016-01-22 00:00:19 +00:00
|
|
|
|
#include "V3Simulate.h"
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
2018-10-14 17:43:24 +00:00
|
|
|
|
#include <algorithm>
|
|
|
|
|
#include <cstdarg>
|
|
|
|
|
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//######################################################################
|
|
|
|
|
// Unroll state, as a visitor of each AstNode
|
|
|
|
|
|
|
|
|
|
class UnrollVisitor : public AstNVisitor {
|
|
|
|
|
private:
|
|
|
|
|
// STATE
|
|
|
|
|
AstVar* m_forVarp; // Iterator variable
|
|
|
|
|
AstVarScope* m_forVscp; // Iterator variable scope (NULL for generate pass)
|
|
|
|
|
AstConst* m_varValuep; // Current value of loop
|
2006-09-05 20:06:23 +00:00
|
|
|
|
AstNode* m_ignoreIncp; // Increment node to ignore
|
2006-08-26 11:35:28 +00:00
|
|
|
|
bool m_varModeCheck; // Just checking RHS assignments
|
|
|
|
|
bool m_varModeReplace; // Replacing varrefs
|
|
|
|
|
bool m_varAssignHit; // Assign var hit
|
|
|
|
|
bool m_generate; // Expand single generate For loop
|
2012-07-21 21:12:42 +00:00
|
|
|
|
string m_beginName; // What name to give begin iterations
|
2006-08-26 11:35:28 +00:00
|
|
|
|
V3Double0 m_statLoops; // Statistic tracking
|
|
|
|
|
V3Double0 m_statIters; // Statistic tracking
|
|
|
|
|
|
2009-01-21 21:56:50 +00:00
|
|
|
|
// METHODS
|
2018-05-14 10:50:47 +00:00
|
|
|
|
VL_DEBUG_FUNC; // Declare debug()
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
|
|
|
|
// VISITORS
|
2006-09-05 20:06:23 +00:00
|
|
|
|
bool cantUnroll(AstNode* nodep, const char* reason) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (m_generate) {
|
|
|
|
|
nodep->v3error("Unsupported: Can't unroll generate for; "<<reason);
|
|
|
|
|
}
|
|
|
|
|
UINFO(3," Can't Unroll: "<<reason<<" :"<<nodep<<endl);
|
2014-03-09 21:12:52 +00:00
|
|
|
|
//if (debug()>=9) nodep->dumpTree(cout,"-cant-");
|
2006-08-26 11:35:28 +00:00
|
|
|
|
V3Stats::addStatSum(string("Unrolling gave up, ")+reason, 1);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-06 13:59:22 +00:00
|
|
|
|
int unrollCount() {
|
|
|
|
|
return m_generate ? v3Global.opt.unrollCount()*16
|
|
|
|
|
: v3Global.opt.unrollCount();
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-17 12:01:22 +00:00
|
|
|
|
bool bodySizeOverRecurse(AstNode* nodep, int& bodySize, int bodyLimit) {
|
|
|
|
|
if (!nodep) return false;
|
|
|
|
|
bodySize++;
|
|
|
|
|
// Exit once exceeds limits, rather than always total
|
|
|
|
|
// so don't go O(n^2) when can't unroll
|
|
|
|
|
if (bodySize > bodyLimit) return true;
|
|
|
|
|
if (bodySizeOverRecurse(nodep->op1p(), bodySize, bodyLimit)) return true;
|
|
|
|
|
if (bodySizeOverRecurse(nodep->op2p(), bodySize, bodyLimit)) return true;
|
|
|
|
|
if (bodySizeOverRecurse(nodep->op3p(), bodySize, bodyLimit)) return true;
|
|
|
|
|
if (bodySizeOverRecurse(nodep->op4p(), bodySize, bodyLimit)) return true;
|
|
|
|
|
// Tail recurse.
|
|
|
|
|
return bodySizeOverRecurse(nodep->nextp(), bodySize, bodyLimit);
|
|
|
|
|
}
|
|
|
|
|
|
2006-09-05 20:06:23 +00:00
|
|
|
|
bool forUnrollCheck(AstNode* nodep,
|
|
|
|
|
AstNode* initp, // Maybe under nodep (no nextp), or standalone (ignore nextp)
|
|
|
|
|
AstNode* precondsp, AstNode* condp,
|
|
|
|
|
AstNode* incp, // Maybe under nodep or in bodysp
|
|
|
|
|
AstNode* bodysp) {
|
|
|
|
|
// To keep the IF levels low, we return as each test fails.
|
|
|
|
|
UINFO(4, " FOR Check "<<nodep<<endl);
|
|
|
|
|
if (initp) UINFO(6, " Init "<<initp<<endl);
|
|
|
|
|
if (precondsp) UINFO(6, " Pcon "<<precondsp<<endl);
|
|
|
|
|
if (condp) UINFO(6, " Cond "<<condp<<endl);
|
|
|
|
|
if (incp) UINFO(6, " Inc "<<incp<<endl);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// Initial value check
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstAssign* initAssp = VN_CAST(initp, Assign);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (!initAssp) return cantUnroll(nodep, "no initial assignment");
|
|
|
|
|
if (initp->nextp() && initp->nextp()!=nodep) nodep->v3fatalSrc("initial assignment shouldn't be a list");
|
2018-02-02 02:32:58 +00:00
|
|
|
|
if (!VN_IS(initAssp->lhsp(), VarRef)) return cantUnroll(nodep, "no initial assignment to simple variable");
|
2016-01-22 00:00:19 +00:00
|
|
|
|
//
|
|
|
|
|
// Condition check
|
|
|
|
|
if (condp->nextp()) nodep->v3fatalSrc("conditional shouldn't be a list");
|
|
|
|
|
//
|
|
|
|
|
// Assignment of next value check
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstAssign* incAssp = VN_CAST(incp, Assign);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
if (!incAssp) return cantUnroll(nodep, "no increment assignment");
|
|
|
|
|
if (incAssp->nextp()) nodep->v3fatalSrc("increment shouldn't be a list");
|
|
|
|
|
|
2018-02-02 02:32:58 +00:00
|
|
|
|
m_forVarp = VN_CAST(initAssp->lhsp(), VarRef)->varp();
|
|
|
|
|
m_forVscp = VN_CAST(initAssp->lhsp(), VarRef)->varScopep();
|
|
|
|
|
if (VN_IS(nodep, GenFor) && !m_forVarp->isGenVar()) {
|
2013-03-12 11:27:17 +00:00
|
|
|
|
nodep->v3error("Non-genvar used in generate for: "<<m_forVarp->prettyName()<<endl);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
2009-10-15 00:13:04 +00:00
|
|
|
|
if (m_generate) V3Const::constifyParamsEdit(initAssp->rhsp()); // rhsp may change
|
2016-01-22 00:00:19 +00:00
|
|
|
|
|
|
|
|
|
// This check shouldn't be needed when using V3Simulate
|
|
|
|
|
// however, for repeat loops, the loop variable is auto-generated
|
|
|
|
|
// and the initp statements will reference a variable outside of the initp scope
|
|
|
|
|
// alas, failing to simulate.
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstConst* constInitp = VN_CAST(initAssp->rhsp(), Const);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (!constInitp) return cantUnroll(nodep, "non-constant initializer");
|
2016-01-22 00:00:19 +00:00
|
|
|
|
|
2006-09-05 20:06:23 +00:00
|
|
|
|
//
|
2016-01-22 00:00:19 +00:00
|
|
|
|
// Now, make sure there's no assignment to this variable in the loop
|
|
|
|
|
m_varModeCheck = true;
|
|
|
|
|
m_varAssignHit = false;
|
|
|
|
|
m_ignoreIncp = incp;
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateAndNextNull(precondsp);
|
|
|
|
|
iterateAndNextNull(bodysp);
|
|
|
|
|
iterateAndNextNull(incp);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
m_varModeCheck = false;
|
|
|
|
|
m_ignoreIncp = NULL;
|
|
|
|
|
if (m_varAssignHit) return cantUnroll(nodep, "genvar assigned *inside* loop");
|
|
|
|
|
|
2006-09-05 20:06:23 +00:00
|
|
|
|
//
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (m_forVscp) { UINFO(8, " Loop Variable: "<<m_forVscp<<endl); }
|
|
|
|
|
else { UINFO(8, " Loop Variable: "<<m_forVarp<<endl); }
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (debug()>=9) nodep->dumpTree(cout,"- for: ");
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
2014-10-16 01:29:37 +00:00
|
|
|
|
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (!m_generate) {
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstAssign *incpAssign = VN_CAST(incp, Assign);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
if (!canSimulate(incpAssign->rhsp())) return cantUnroll(incp, "Unable to simulate increment");
|
|
|
|
|
if (!canSimulate(condp)) return cantUnroll(condp, "Unable to simulate condition");
|
|
|
|
|
|
|
|
|
|
// Check whether to we actually want to try and unroll.
|
|
|
|
|
int loops;
|
|
|
|
|
if (!countLoops(initAssp, condp, incp, unrollCount(), loops))
|
|
|
|
|
return cantUnroll(nodep, "Unable to simulate loop");
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
2007-11-30 22:12:53 +00:00
|
|
|
|
// Less than 10 statements in the body?
|
2006-08-26 11:35:28 +00:00
|
|
|
|
int bodySize = 0;
|
2010-04-17 12:01:22 +00:00
|
|
|
|
int bodyLimit = v3Global.opt.unrollStmts();
|
|
|
|
|
if (loops>0) bodyLimit = v3Global.opt.unrollStmts() / loops;
|
|
|
|
|
if (bodySizeOverRecurse(precondsp, bodySize/*ref*/, bodyLimit)
|
|
|
|
|
|| bodySizeOverRecurse(bodysp, bodySize/*ref*/, bodyLimit)
|
|
|
|
|
|| bodySizeOverRecurse(incp, bodySize/*ref*/, bodyLimit)) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
return cantUnroll(nodep, "too many statements");
|
2010-04-17 12:01:22 +00:00
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
// Finally, we can do it
|
2016-01-22 00:00:19 +00:00
|
|
|
|
if (!forUnroller(nodep, initAssp, condp, precondsp, incp, bodysp)) {
|
|
|
|
|
return cantUnroll(nodep, "Unable to unroll loop");
|
|
|
|
|
}
|
|
|
|
|
VL_DANGLING(nodep);
|
2007-11-02 11:23:03 +00:00
|
|
|
|
// Cleanup
|
2006-08-26 11:35:28 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-22 00:00:19 +00:00
|
|
|
|
bool canSimulate(AstNode *nodep) {
|
2018-03-10 19:10:41 +00:00
|
|
|
|
SimulateVisitor simvis;
|
|
|
|
|
AstNode* clonep = nodep->cloneTree(true);
|
|
|
|
|
simvis.mainCheckTree(clonep);
|
|
|
|
|
pushDeletep(clonep); clonep = NULL;
|
|
|
|
|
return simvis.optimizable();
|
2016-01-22 00:00:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool simulateTree(AstNode *nodep, const V3Number *loopValue, AstNode *dtypep, V3Number &outNum) {
|
|
|
|
|
AstNode* clone = nodep->cloneTree(true);
|
|
|
|
|
if (!clone) {
|
|
|
|
|
nodep->v3fatalSrc("Failed to clone tree");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (loopValue) {
|
|
|
|
|
m_varValuep = new AstConst (nodep->fileline(), *loopValue);
|
|
|
|
|
// Iteration requires a back, so put under temporary node
|
|
|
|
|
AstBegin* tempp = new AstBegin (nodep->fileline(), "[EditWrapper]", clone);
|
|
|
|
|
m_varModeReplace = true;
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateAndNextNull(tempp->stmtsp());
|
2016-01-22 00:00:19 +00:00
|
|
|
|
m_varModeReplace = false;
|
|
|
|
|
clone = tempp->stmtsp()->unlinkFrBackWithNext();
|
|
|
|
|
tempp->deleteTree();
|
|
|
|
|
tempp = NULL;
|
|
|
|
|
pushDeletep(m_varValuep); m_varValuep = NULL;
|
|
|
|
|
}
|
|
|
|
|
SimulateVisitor simvis;
|
|
|
|
|
simvis.mainParamEmulate(clone);
|
|
|
|
|
if (!simvis.optimizable()) {
|
|
|
|
|
UINFO(3, "Unable to simulate" << endl);
|
|
|
|
|
if (debug()>=9) nodep->dumpTree(cout,"- _simtree: ");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Fetch the result
|
|
|
|
|
V3Number* res = simvis.fetchNumberNull(clone);
|
|
|
|
|
if (!res) {
|
|
|
|
|
UINFO(3, "No number returned from simulation" << endl);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Patch up datatype
|
|
|
|
|
if (dtypep) {
|
|
|
|
|
AstConst new_con (clone->fileline(), *res);
|
|
|
|
|
new_con.dtypeFrom(dtypep);
|
|
|
|
|
outNum = new_con.num();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
outNum = *res;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool countLoops(AstAssign *initp, AstNode *condp, AstNode *incp, int max, int &outLoopsr) {
|
|
|
|
|
outLoopsr = 0;
|
|
|
|
|
V3Number loopValue = V3Number(initp->fileline());
|
|
|
|
|
if (!simulateTree(initp->rhsp(), NULL, initp, loopValue)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
while (1) {
|
|
|
|
|
V3Number res = V3Number(initp->fileline());
|
|
|
|
|
if (!simulateTree(condp, &loopValue, NULL, res)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
if (!res.isEqOne()) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
outLoopsr++;
|
|
|
|
|
|
|
|
|
|
// Run inc
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstAssign* incpass = VN_CAST(incp, Assign);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
V3Number newLoopValue = V3Number(initp->fileline());
|
|
|
|
|
if (!simulateTree(incpass->rhsp(), &loopValue, incpass, newLoopValue)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
loopValue.opAssign(newLoopValue);
|
|
|
|
|
if (outLoopsr > max) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool forUnroller(AstNode* nodep,
|
|
|
|
|
AstAssign* initp,
|
|
|
|
|
AstNode* condp,
|
2014-10-16 01:29:37 +00:00
|
|
|
|
AstNode* precondsp,
|
2016-01-22 00:00:19 +00:00
|
|
|
|
AstNode* incp, AstNode* bodysp) {
|
2016-11-30 01:40:58 +00:00
|
|
|
|
UINFO(9, "forUnroller "<<nodep<<endl);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
V3Number loopValue = V3Number(nodep->fileline());
|
|
|
|
|
if (!simulateTree(initp->rhsp(), NULL, initp, loopValue)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2006-09-05 20:06:23 +00:00
|
|
|
|
AstNode* stmtsp = NULL;
|
|
|
|
|
if (initp) {
|
|
|
|
|
initp->unlinkFrBack(); // Always a single statement; nextp() may be nodep
|
|
|
|
|
// Don't add to list, we do it once, and setting loop index isn't needed as we're constant propagating it
|
|
|
|
|
}
|
|
|
|
|
if (precondsp) {
|
|
|
|
|
precondsp->unlinkFrBackWithNext();
|
2016-02-09 03:15:44 +00:00
|
|
|
|
stmtsp = AstNode::addNextNull(stmtsp, precondsp);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
}
|
|
|
|
|
if (bodysp) {
|
|
|
|
|
bodysp->unlinkFrBackWithNext();
|
2016-02-09 03:15:44 +00:00
|
|
|
|
stmtsp = AstNode::addNextNull(stmtsp, bodysp); // Maybe null if no body
|
2006-09-05 20:06:23 +00:00
|
|
|
|
}
|
2018-02-02 02:32:58 +00:00
|
|
|
|
if (incp && !VN_IS(nodep, GenFor)) { // Generates don't need to increment loop index
|
2010-02-14 15:01:21 +00:00
|
|
|
|
incp->unlinkFrBackWithNext();
|
2016-02-09 03:15:44 +00:00
|
|
|
|
stmtsp = AstNode::addNextNull(stmtsp, incp); // Maybe null if no body
|
2010-02-14 15:01:21 +00:00
|
|
|
|
}
|
2011-01-01 01:18:21 +00:00
|
|
|
|
// Mark variable to disable some later warnings
|
|
|
|
|
m_forVarp->usedLoopIdx(true);
|
|
|
|
|
|
2006-09-05 20:06:23 +00:00
|
|
|
|
AstNode* newbodysp = NULL;
|
2011-08-05 01:15:24 +00:00
|
|
|
|
++m_statLoops;
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (stmtsp) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
int times = 0;
|
|
|
|
|
while (1) {
|
|
|
|
|
UINFO(8," Looping "<<loopValue<<endl);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
V3Number res = V3Number(nodep->fileline());
|
|
|
|
|
if (!simulateTree(condp, &loopValue, NULL, res)) {
|
|
|
|
|
nodep->v3error("Loop unrolling failed.");
|
|
|
|
|
return false;
|
2014-10-16 01:29:37 +00:00
|
|
|
|
}
|
2016-01-22 00:00:19 +00:00
|
|
|
|
if (!res.isEqOne()) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
break; // Done with the loop
|
2016-01-22 00:00:19 +00:00
|
|
|
|
}
|
|
|
|
|
else {
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// Replace iterator values with constant.
|
|
|
|
|
AstNode* oneloopp = stmtsp->cloneTree(true);
|
|
|
|
|
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_varValuep = new AstConst(nodep->fileline(), loopValue);
|
2007-11-02 11:23:03 +00:00
|
|
|
|
|
2009-10-04 21:01:35 +00:00
|
|
|
|
// Iteration requires a back, so put under temporary node
|
2012-03-20 20:01:53 +00:00
|
|
|
|
if (oneloopp) {
|
2009-10-04 21:01:35 +00:00
|
|
|
|
AstBegin* tempp = new AstBegin(oneloopp->fileline(),"[EditWrapper]",oneloopp);
|
|
|
|
|
m_varModeReplace = true;
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateAndNextNull(tempp->stmtsp());
|
2009-10-04 21:01:35 +00:00
|
|
|
|
m_varModeReplace = false;
|
2015-10-04 17:16:35 +00:00
|
|
|
|
oneloopp = tempp->stmtsp()->unlinkFrBackWithNext(); tempp->deleteTree(); VL_DANGLING(tempp);
|
2012-07-21 21:12:42 +00:00
|
|
|
|
}
|
|
|
|
|
if (m_generate) {
|
|
|
|
|
string index = AstNode::encodeNumber(m_varValuep->toSInt());
|
|
|
|
|
string nname = m_beginName + "__BRA__" + index + "__KET__";
|
|
|
|
|
oneloopp = new AstBegin(oneloopp->fileline(),nname,oneloopp,true);
|
2009-10-04 21:01:35 +00:00
|
|
|
|
}
|
2016-01-22 00:00:19 +00:00
|
|
|
|
pushDeletep(m_varValuep); m_varValuep=NULL;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (newbodysp) newbodysp->addNext(oneloopp);
|
|
|
|
|
else newbodysp = oneloopp;
|
|
|
|
|
|
2011-08-05 01:15:24 +00:00
|
|
|
|
++m_statIters;
|
2008-10-06 13:59:22 +00:00
|
|
|
|
if (++times > unrollCount()*3) {
|
2018-03-10 21:32:04 +00:00
|
|
|
|
nodep->v3error("Loop unrolling took too long;"
|
|
|
|
|
" probably this is an infinite loop, or set --unroll-count above "
|
|
|
|
|
<<unrollCount());
|
2006-08-26 11:35:28 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-22 00:00:19 +00:00
|
|
|
|
// loopValue += valInc
|
2018-02-02 02:32:58 +00:00
|
|
|
|
AstAssign *incpass = VN_CAST(incp, Assign);
|
2016-01-22 00:00:19 +00:00
|
|
|
|
V3Number newLoopValue = V3Number(nodep->fileline());
|
|
|
|
|
if (!simulateTree(incpass->rhsp(), &loopValue, incpass, newLoopValue)) {
|
|
|
|
|
nodep->v3error("Loop unrolling failed");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
loopValue.opAssign(newLoopValue);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Replace the FOR()
|
|
|
|
|
if (newbodysp) nodep->replaceWith(newbodysp);
|
|
|
|
|
else nodep->unlinkFrBack();
|
2015-10-04 17:16:35 +00:00
|
|
|
|
if (bodysp) { pushDeletep(bodysp); VL_DANGLING(bodysp); }
|
|
|
|
|
if (precondsp) { pushDeletep(precondsp); VL_DANGLING(precondsp); }
|
|
|
|
|
if (initp) { pushDeletep(initp); VL_DANGLING(initp); }
|
|
|
|
|
if (incp && !incp->backp()) { pushDeletep(incp); VL_DANGLING(incp); }
|
2016-11-30 01:40:58 +00:00
|
|
|
|
if (debug()>=9 && newbodysp) newbodysp->dumpTree(cout,"- _new: ");
|
2016-01-22 00:00:19 +00:00
|
|
|
|
return true;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-27 13:11:38 +00:00
|
|
|
|
virtual void visit(AstWhile* nodep) {
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateChildren(nodep);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (m_varModeCheck || m_varModeReplace) {
|
|
|
|
|
} else {
|
|
|
|
|
// Constify before unroll call, as it may change what is underneath.
|
2009-10-15 00:13:04 +00:00
|
|
|
|
if (nodep->precondsp()) V3Const::constifyEdit(nodep->precondsp()); // precondsp may change
|
2016-01-22 00:00:19 +00:00
|
|
|
|
if (nodep->condp()) V3Const::constifyEdit(nodep->condp()); // condp may change
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// Grab initial value
|
|
|
|
|
AstNode* initp = NULL; // Should be statement before the while.
|
|
|
|
|
if (nodep->backp()->nextp() == nodep) initp=nodep->backp();
|
2015-10-04 17:16:35 +00:00
|
|
|
|
if (initp) { V3Const::constifyEdit(initp); VL_DANGLING(initp); }
|
2009-10-15 00:13:04 +00:00
|
|
|
|
if (nodep->backp()->nextp() == nodep) initp=nodep->backp();
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// Grab assignment
|
|
|
|
|
AstNode* incp = NULL; // Should be last statement
|
2010-02-14 15:01:21 +00:00
|
|
|
|
if (nodep->incsp()) V3Const::constifyEdit(nodep->incsp());
|
|
|
|
|
if (nodep->incsp()) incp = nodep->incsp();
|
|
|
|
|
else {
|
|
|
|
|
for (incp = nodep->bodysp(); incp && incp->nextp(); incp = incp->nextp()) {}
|
2015-10-04 17:16:35 +00:00
|
|
|
|
if (incp) { V3Const::constifyEdit(incp); VL_DANGLING(incp); }
|
2010-02-14 15:01:21 +00:00
|
|
|
|
for (incp = nodep->bodysp(); incp && incp->nextp(); incp = incp->nextp()) {} // Again, as may have changed
|
|
|
|
|
}
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// And check it
|
|
|
|
|
if (forUnrollCheck(nodep, initp,
|
|
|
|
|
nodep->precondsp(), nodep->condp(),
|
|
|
|
|
incp, nodep->bodysp())) {
|
2015-10-04 17:16:35 +00:00
|
|
|
|
pushDeletep(nodep); VL_DANGLING(nodep); // Did replacement
|
2006-09-05 20:06:23 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-11-27 13:11:38 +00:00
|
|
|
|
virtual void visit(AstGenFor* nodep) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (!m_generate || m_varModeReplace) {
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateChildren(nodep);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
} // else V3Param will recursively call each for loop to be unrolled for us
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (m_varModeCheck || m_varModeReplace) {
|
|
|
|
|
} else {
|
2006-09-05 20:06:23 +00:00
|
|
|
|
// Constify before unroll call, as it may change what is underneath.
|
2009-10-15 00:13:04 +00:00
|
|
|
|
if (nodep->initsp()) V3Const::constifyEdit(nodep->initsp()); // initsp may change
|
|
|
|
|
if (nodep->condp()) V3Const::constifyEdit(nodep->condp()); // condp may change
|
|
|
|
|
if (nodep->incsp()) V3Const::constifyEdit(nodep->incsp()); // incsp may change
|
2007-11-05 14:09:22 +00:00
|
|
|
|
if (nodep->condp()->isZero()) {
|
|
|
|
|
// We don't need to do any loops. Remove the GenFor,
|
|
|
|
|
// Genvar's don't care about any initial assignments.
|
|
|
|
|
//
|
|
|
|
|
// Note normal For's can't do exactly this deletion, as
|
|
|
|
|
// we'd need to initialize the variable to the initial
|
|
|
|
|
// condition, but they'll become while's which can be
|
|
|
|
|
// deleted by V3Const.
|
2015-10-04 17:16:35 +00:00
|
|
|
|
nodep->unlinkFrBack()->deleteTree(); VL_DANGLING(nodep);
|
2007-11-05 14:09:22 +00:00
|
|
|
|
} else if (forUnrollCheck(nodep, nodep->initsp(),
|
|
|
|
|
NULL, nodep->condp(),
|
|
|
|
|
nodep->incsp(), nodep->bodysp())) {
|
2015-10-04 17:16:35 +00:00
|
|
|
|
pushDeletep(nodep); VL_DANGLING(nodep); // Did replacement
|
2006-08-26 11:35:28 +00:00
|
|
|
|
} else {
|
2010-04-09 23:45:46 +00:00
|
|
|
|
nodep->v3error("For loop doesn't have genvar index, or is malformed");
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-11-27 13:11:38 +00:00
|
|
|
|
virtual void visit(AstNodeFor* nodep) {
|
2008-10-29 01:38:01 +00:00
|
|
|
|
if (m_generate) { // Ignore for's when expanding genfor's
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateChildren(nodep);
|
2008-10-29 01:38:01 +00:00
|
|
|
|
} else {
|
2009-02-26 03:06:59 +00:00
|
|
|
|
nodep->v3error("V3Begin should have removed standard FORs");
|
2008-10-29 01:38:01 +00:00
|
|
|
|
}
|
2006-09-05 20:06:23 +00:00
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
|
2016-11-27 13:11:38 +00:00
|
|
|
|
virtual void visit(AstVarRef* nodep) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (m_varModeCheck
|
|
|
|
|
&& nodep->varp() == m_forVarp
|
|
|
|
|
&& nodep->varScopep() == m_forVscp
|
|
|
|
|
&& nodep->lvalue()) {
|
2008-06-10 01:25:10 +00:00
|
|
|
|
UINFO(8," Itervar assigned to: "<<nodep<<endl);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_varAssignHit = true;
|
|
|
|
|
}
|
2015-11-23 02:16:13 +00:00
|
|
|
|
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (m_varModeReplace
|
|
|
|
|
&& nodep->varp() == m_forVarp
|
2006-09-05 20:06:23 +00:00
|
|
|
|
&& nodep->varScopep() == m_forVscp
|
2015-11-23 02:16:13 +00:00
|
|
|
|
&& !nodep->lvalue()) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
AstNode* newconstp = m_varValuep->cloneTree(false);
|
|
|
|
|
nodep->replaceWith(newconstp);
|
2007-11-02 11:23:03 +00:00
|
|
|
|
pushDeletep(nodep);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//--------------------
|
|
|
|
|
// Default: Just iterate
|
2016-11-27 13:11:38 +00:00
|
|
|
|
virtual void visit(AstNode* nodep) {
|
2006-09-05 20:06:23 +00:00
|
|
|
|
if (m_varModeCheck && nodep == m_ignoreIncp) {
|
|
|
|
|
// Ignore subtree that is the increment
|
|
|
|
|
} else {
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterateChildren(nodep);
|
2006-09-05 20:06:23 +00:00
|
|
|
|
}
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
// CONSTUCTORS
|
2018-02-08 00:31:21 +00:00
|
|
|
|
UnrollVisitor(AstNode* nodep, bool generate, const string& beginName) {
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_forVarp = NULL;
|
|
|
|
|
m_forVscp = NULL;
|
2018-06-14 22:59:24 +00:00
|
|
|
|
m_varValuep = NULL;
|
2006-09-05 20:06:23 +00:00
|
|
|
|
m_ignoreIncp = NULL;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_varModeCheck = false;
|
|
|
|
|
m_varModeReplace = false;
|
2018-06-14 22:59:24 +00:00
|
|
|
|
m_varAssignHit = false;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
m_generate = generate;
|
2012-07-21 21:12:42 +00:00
|
|
|
|
m_beginName = beginName;
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
2018-05-11 00:55:37 +00:00
|
|
|
|
iterate(nodep);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
virtual ~UnrollVisitor() {
|
2016-01-07 01:53:48 +00:00
|
|
|
|
V3Stats::addStatSum("Optimizations, Unrolled Loops", m_statLoops);
|
|
|
|
|
V3Stats::addStatSum("Optimizations, Unrolled Iterations", m_statIters);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//######################################################################
|
|
|
|
|
// Unroll class functions
|
|
|
|
|
|
|
|
|
|
void V3Unroll::unrollAll(AstNetlist* nodep) {
|
|
|
|
|
UINFO(2,__FUNCTION__<<": "<<endl);
|
2018-03-10 17:57:50 +00:00
|
|
|
|
{
|
|
|
|
|
UnrollVisitor visitor (nodep, false, "");
|
|
|
|
|
} // Destruct before checking
|
2017-09-18 02:52:57 +00:00
|
|
|
|
V3Global::dumpCheckGlobalTree("unroll", 0, v3Global.opt.dumpTreeLevel(__FILE__) >= 3);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-08 00:31:21 +00:00
|
|
|
|
void V3Unroll::unrollGen(AstNodeFor* nodep, const string& beginName) {
|
2018-04-13 02:00:34 +00:00
|
|
|
|
UINFO(5,__FUNCTION__<<": "<<endl);
|
2012-07-21 21:12:42 +00:00
|
|
|
|
UnrollVisitor visitor (nodep, true, beginName);
|
2006-08-26 11:35:28 +00:00
|
|
|
|
}
|