Internals: Some V3Tristate cleanups. No functional change.

This commit is contained in:
Wilson Snyder 2021-12-19 16:06:45 -05:00
parent ff425369ac
commit 4135867e53

View File

@ -51,7 +51,10 @@
//
// Note 1800-2012 adds user defined resolution functions. This suggests
// long term this code should be scoped-based and resolve all nodes at once
// rather than hierarchically.
// rather than hierarchically. If/when that is done, make sure to avoid
// duplicating vars and logic that is common between each instance of a
// module.
//
//*************************************************************************
#include "config_build.h"
@ -235,10 +238,10 @@ public:
}
if (debug() >= 9) m_graph.dumpDotFilePrefixed("tri_pos__" + nodep->name());
}
void setTristate(AstNode* nodep) { makeVertex(nodep)->isTristate(true); }
void associate(AstNode* fromp, AstNode* top) {
new V3GraphEdge(&m_graph, makeVertex(fromp), makeVertex(top), 1);
}
void setTristate(AstNode* nodep) { makeVertex(nodep)->isTristate(true); }
bool isTristate(AstNode* nodep) {
const TristateVertex* const vertexp = reinterpret_cast<TristateVertex*>(nodep->user5p());
return vertexp && vertexp->isTristate();
@ -347,7 +350,7 @@ class TristateVisitor final : public TristateBaseVisitor {
//
AstNodeModule* m_modp = nullptr; // Current module
AstCell* m_cellp = nullptr; // current cell
VarMap m_lhsmap; // LHS driver map
VarMap m_lhsmap; // Tristate left-hand-side driver map
int m_unique = 0;
bool m_alhs = false; // On LHS of assignment
const AstNode* m_logicp = nullptr; // Current logic being built
@ -362,67 +365,57 @@ class TristateVisitor final : public TristateBaseVisitor {
if (m_alhs) o += "alhs ";
return o;
}
void modAddStmtp(AstNode* nodep, AstNode* newp) {
if (!m_modp) {
nodep->v3warn(E_UNSUPPORTED,
"Unsupported: Creating tristate signal not underneath a module: "
<< nodep->prettyNameQ());
} else {
m_modp->addStmtp(newp);
}
}
void associateLogic(AstNode* fromp, AstNode* top) {
if (m_logicp) m_tgraph.associate(fromp, top);
}
AstNode* getEnp(AstNode* nodep) {
// checks if user1p() is null, and if so, adds a constant output
// enable driver of all 1's. Otherwise returns the user1p() data.
if (!nodep->user1p()) {
// There's no select being built yet, so add what will become a
// constant output enable driver of all 1's
V3Number num(nodep, nodep->width());
num.setAllBits1();
AstNode* const enp = new AstConst(nodep->fileline(), num);
nodep->user1p(enp);
}
// Otherwise return the previous output enable
return nodep->user1p();
}
AstVar* getCreateEnVarp(AstVar* invarp) {
// Return the master __en for the specified input variable
if (!invarp->user1p()) {
AstVar* const newp = new AstVar(invarp->fileline(), AstVarType::MODULETEMP,
invarp->name() + "__en", invarp);
UINFO(9, " newenv " << newp << endl);
if (!m_modp) {
invarp->v3warn(E_UNSUPPORTED,
"Unsupported: Creating tristate signal not underneath a module: "
<< invarp->prettyNameQ());
} else {
m_modp->addStmtp(newp);
}
modAddStmtp(invarp, newp);
invarp->user1p(newp); // find envar given invarp
}
return VN_AS(invarp->user1p(), Var);
}
AstVar* getCreateOutVarp(AstVar* invarp) {
// Return the master __out for the specified input variable
if (!invarp->user4p()) {
AstVar* const newp = new AstVar(invarp->fileline(), AstVarType::MODULETEMP,
invarp->name() + "__out", invarp);
UINFO(9, " newout " << newp << endl);
if (!m_modp) {
invarp->v3warn(E_UNSUPPORTED,
"Unsupported: Creating tristate signal not underneath a module: "
<< invarp->prettyNameQ());
} else {
m_modp->addStmtp(newp);
}
modAddStmtp(invarp, newp);
invarp->user4p(newp); // find outvar given invarp
}
return VN_AS(invarp->user4p(), Var);
}
AstVar* getCreateUnconnVarp(AstNode* fromp, AstNodeDType* dtypep) {
AstVar* const newp = new AstVar(fromp->fileline(), AstVarType::MODULETEMP,
"__Vtriunconn" + cvtToStr(m_unique++), dtypep);
UINFO(9, " newunc " << newp << endl);
if (!m_modp) {
newp->v3warn(E_UNSUPPORTED,
"Unsupported: Creating tristate signal not underneath a module");
} else {
m_modp->addStmtp(newp);
}
modAddStmtp(newp, newp);
return newp;
}
@ -490,8 +483,8 @@ class TristateVisitor final : public TristateBaseVisitor {
if (m_tgraph.isTristate(varp)) {
const auto it = m_lhsmap.find(varp);
if (it == m_lhsmap.end()) {
// set output enable to always be off on this assign
// statement so that this var is floating
// This variable is floating, set output enable to
// always be off on this assign
UINFO(8, " Adding driver to var " << varp << endl);
AstConst* const constp = new AstConst(
varp->fileline(), AstConst::WidthedValue(), varp->width(), 0);
@ -516,19 +509,23 @@ class TristateVisitor final : public TristateBaseVisitor {
++nextit;
AstVar* const invarp = it->first;
const RefVec* const refsp = it->second;
// Figure out if this var needs tristate expanded.
if (!m_tgraph.isTristate(invarp)) {
// This var has no tristate logic, so we leave it alone.
if (m_tgraph.isTristate(invarp)) {
insertTristatesSignal(nodep, invarp, refsp);
} else {
UINFO(8, " NO TRISTATE ON:" << invarp << endl);
}
// Delete the map and vector list now that we have expanded it.
m_lhsmap.erase(invarp);
VL_DO_DANGLING(delete refsp, refsp);
continue;
}
}
void insertTristatesSignal(AstNodeModule* nodep, AstVar* const invarp,
const RefVec* const refsp) {
UINFO(8, " TRISTATE EXPANDING:" << invarp << endl);
++m_statTriSigs;
m_tgraph.didProcess(invarp);
UINFO(8, " TRISTATE EXPANDING:" << invarp << endl);
// If the lhs var is a port, then we need to create ports for
// the output (__out) and output enable (__en) signals. The
@ -610,7 +607,7 @@ class TristateVisitor final : public TristateBaseVisitor {
undrivenp = new AstConst(invarp->fileline(), ones);
}
if (!outvarp) {
// This is the final resolution of the tristate, so we apply
// This is the final pre-forced resolution of the tristate, so we apply
// the pull direction to any undriven pins.
V3Number pull(invarp, lhsp->width());
const AstPull* const pullp = static_cast<AstPull*>(lhsp->user3p());
@ -628,8 +625,7 @@ class TristateVisitor final : public TristateBaseVisitor {
}
if (envarp) {
nodep->addStmtp(new AstAssignW(
enp->fileline(), new AstVarRef(envarp->fileline(), envarp, VAccess::WRITE),
enp));
enp->fileline(), new AstVarRef(envarp->fileline(), envarp, VAccess::WRITE), enp));
}
// __out (child) or <in> (parent) = drive-value expression
AstNode* const assp = new AstAssignW(
@ -637,10 +633,6 @@ class TristateVisitor final : public TristateBaseVisitor {
assp->user2(U2_BOTH); // Don't process further; already resolved
if (debug() >= 9) assp->dumpTree(cout, "-lhsp-eqn: ");
nodep->addStmtp(assp);
// Delete the map and vector list now that we have expanded it.
m_lhsmap.erase(invarp);
VL_DO_DANGLING(delete refsp, refsp);
}
}
// VISITORS
@ -897,15 +889,15 @@ class TristateVisitor final : public TristateBaseVisitor {
void visitAssign(AstNodeAssign* nodep) {
if (m_graphing) {
if (nodep->user2() & U2_GRAPHING) return;
nodep->user2(U2_GRAPHING);
VL_RESTORER(m_logicp);
m_logicp = nodep;
nodep->user2(U2_GRAPHING);
iterateAndNextNull(nodep->rhsp());
m_alhs = true;
iterateAndNextNull(nodep->lhsp());
m_alhs = false;
associateLogic(nodep->rhsp(), nodep);
associateLogic(nodep, nodep->lhsp());
m_logicp = nullptr;
} else {
if (nodep->user2() & U2_NONGRAPH) {
return; // Iterated here, or created assignment to ignore
@ -1063,11 +1055,11 @@ class TristateVisitor final : public TristateBaseVisitor {
nodep->v3warn(E_UNSUPPORTED, "Unsupported pullup/down (weak driver) construct.");
} else {
if (m_graphing) {
varrefp->access(VAccess::WRITE);
VL_RESTORER(m_logicp);
m_logicp = nodep;
varrefp->access(VAccess::WRITE);
m_tgraph.setTristate(nodep);
associateLogic(nodep, varrefp->varp());
m_logicp = nullptr;
} else {
// Replace any pullup/pulldowns with assignw logic and set the
// direction of the pull in the user3() data on the var. Given
@ -1089,13 +1081,13 @@ class TristateVisitor final : public TristateBaseVisitor {
void iteratePinGuts(AstPin* nodep) {
if (m_graphing) {
VL_RESTORER(m_logicp);
m_logicp = nodep;
if (nodep->exprp()) {
associateLogic(nodep->exprp(), nodep);
associateLogic(nodep, nodep->exprp());
}
iterateChildren(nodep);
m_logicp = nullptr;
} else {
// All heavy lifting completed in graph visitor.
if (nodep->exprp()) m_tgraph.didProcess(nodep);
@ -1399,10 +1391,10 @@ class TristateVisitor final : public TristateBaseVisitor {
}
virtual void visit(AstCell* nodep) override {
VL_RESTORER(m_cellp);
m_cellp = nodep;
m_alhs = false;
iterateChildren(nodep);
m_cellp = nullptr;
}
virtual void visit(AstNetlist* nodep) override { iterateChildrenBackwards(nodep); }