forked from github/verilator
Merge branch 'master' into develop-v5
This commit is contained in:
commit
e7ca4a69e3
17
Changes
17
Changes
@ -19,7 +19,14 @@ Verilator 5.001 devel
|
|||||||
clocks are now simulated correctly (#3278, #3384). [Geza Lore, Shunyao CAD]
|
clocks are now simulated correctly (#3278, #3384). [Geza Lore, Shunyao CAD]
|
||||||
|
|
||||||
|
|
||||||
Verilator 4.223 devel
|
Verilator 4.225 devel
|
||||||
|
==========================
|
||||||
|
|
||||||
|
**Minor:**
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Verilator 4.224 2022-06-19
|
||||||
==========================
|
==========================
|
||||||
|
|
||||||
**Major:**
|
**Major:**
|
||||||
@ -30,16 +37,24 @@ Verilator 4.223 devel
|
|||||||
|
|
||||||
* Add -f<optimization> options to replace -O<letter> options (#3436).
|
* Add -f<optimization> options to replace -O<letter> options (#3436).
|
||||||
* Changed --no-merge-const-pool to -fno-merge-const-pool (#3436).
|
* Changed --no-merge-const-pool to -fno-merge-const-pool (#3436).
|
||||||
|
* Changed --no-decoration to remove output whitespace (#3460). [Kamil Rakoczy]
|
||||||
* Support compile time trace signal selection with tracing_on/off (#3323). [Shunyao CAD]
|
* Support compile time trace signal selection with tracing_on/off (#3323). [Shunyao CAD]
|
||||||
* Support non-ANSI interface port declarations (#3439). [Geza Lore, Shunyao CAD]
|
* Support non-ANSI interface port declarations (#3439). [Geza Lore, Shunyao CAD]
|
||||||
* Support concat assignment to packed array (#3446).
|
* Support concat assignment to packed array (#3446).
|
||||||
* Improve conditional merging optimization (#3125). [Geza Lore, Shunyao CAD]
|
* Improve conditional merging optimization (#3125). [Geza Lore, Shunyao CAD]
|
||||||
* Define VM_TRACE_VCD when tracing in VCD format. [Geza Lore, Shunyao CAD]
|
* Define VM_TRACE_VCD when tracing in VCD format. [Geza Lore, Shunyao CAD]
|
||||||
* Add assert when VerilatedContext is mis-deleted (#3121). [Rupert Swarbrick]
|
* Add assert when VerilatedContext is mis-deleted (#3121). [Rupert Swarbrick]
|
||||||
|
* Internal prep work towards timing control. [Krzysztof Bieganski]
|
||||||
* Fix hang with large case statement optimization (#3405). [Mike Urbach]
|
* Fix hang with large case statement optimization (#3405). [Mike Urbach]
|
||||||
|
* Fix UNOPTFLAT warning from initial static var (#3406). [Kamil Rakoczy]
|
||||||
|
* Fix compile error when enable VL_LEAK_CHECKS (#3411). [HungMingWu]
|
||||||
|
* Fix cmake rules to support higher-level targets (#3377) (#3386). [Martin Stadler]
|
||||||
|
* Fix BLKANDNBLK on $readmem/$writemem (#3379). [Alex Solomatnikov]
|
||||||
* Fix 'with' operator with type casting (#3387). [xiak95]
|
* Fix 'with' operator with type casting (#3387). [xiak95]
|
||||||
* Fix incorrect conditional merging (#3409). [Raynard Qiao]
|
* Fix incorrect conditional merging (#3409). [Raynard Qiao]
|
||||||
* Fix passing VL_TRACE_FST_WRITER_THREAD in CMake build. [Geza Lore, Shunyao CAD]
|
* Fix passing VL_TRACE_FST_WRITER_THREAD in CMake build. [Geza Lore, Shunyao CAD]
|
||||||
|
* Fix compile error under strict C++11 mode (#3463). [Kevin Kiningham]
|
||||||
|
* Fix public unpacked input ports (#3465). [Todd Strader]
|
||||||
|
|
||||||
|
|
||||||
Verilator 4.222 2022-05-02
|
Verilator 4.222 2022-05-02
|
||||||
|
@ -5,14 +5,14 @@
|
|||||||
# General Public License Version 3 or the Perl Artistic License Version 2.0.
|
# General Public License Version 3 or the Perl Artistic License Version 2.0.
|
||||||
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
# When releasing, also update header of Changes file
|
||||||
|
# and commit using "devel release" or "Version bump" message
|
||||||
|
# Then 'make maintainer-dist'
|
||||||
#AC_INIT([Verilator],[#.### YYYY-MM-DD])
|
#AC_INIT([Verilator],[#.### YYYY-MM-DD])
|
||||||
#AC_INIT([Verilator],[#.### devel])
|
#AC_INIT([Verilator],[#.### devel])
|
||||||
AC_INIT([Verilator],[5.001 devel],
|
AC_INIT([Verilator],[5.001 devel],
|
||||||
[https://verilator.org],
|
[https://verilator.org],
|
||||||
[verilator],[https://verilator.org])
|
[verilator],[https://verilator.org])
|
||||||
# When releasing, also update header of Changes file
|
|
||||||
# and commit using "devel release" or "Version bump" message
|
|
||||||
# Then 'make maintainer-dist'
|
|
||||||
|
|
||||||
AC_CONFIG_HEADERS(src/config_build.h)
|
AC_CONFIG_HEADERS(src/config_build.h)
|
||||||
AC_CONFIG_FILES(Makefile src/Makefile src/Makefile_obj include/verilated.mk include/verilated_config.h verilator.pc verilator-config.cmake verilator-config-version.cmake)
|
AC_CONFIG_FILES(Makefile src/Makefile src/Makefile_obj include/verilated.mk include/verilated_config.h verilator.pc verilator-config.cmake verilator-config-version.cmake)
|
||||||
|
@ -456,6 +456,8 @@ Summary:
|
|||||||
|
|
||||||
.. option:: -fno-merge-cond
|
.. option:: -fno-merge-cond
|
||||||
|
|
||||||
|
.. option:: -fno-merge-cond-motion
|
||||||
|
|
||||||
.. option:: -fno-merge-const-pool
|
.. option:: -fno-merge-const-pool
|
||||||
|
|
||||||
.. option:: -fno-reloop
|
.. option:: -fno-reloop
|
||||||
|
@ -125,6 +125,7 @@ Those developing Verilator itself may also want these (see internals.rst):
|
|||||||
::
|
::
|
||||||
|
|
||||||
sudo apt-get install gdb graphviz cmake clang clang-format-11 gprof lcov
|
sudo apt-get install gdb graphviz cmake clang clang-format-11 gprof lcov
|
||||||
|
sudo apt-get install yapf3
|
||||||
sudo pip3 install sphinx sphinx_rtd_theme sphinxcontrib-spelling breathe
|
sudo pip3 install sphinx sphinx_rtd_theme sphinxcontrib-spelling breathe
|
||||||
cpan install Pod::Perldoc
|
cpan install Pod::Perldoc
|
||||||
cpan install Parallel::Forker
|
cpan install Parallel::Forker
|
||||||
|
19
src/V3Ast.h
19
src/V3Ast.h
@ -2034,11 +2034,12 @@ private:
|
|||||||
return Default;
|
return Default;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T_Node> constexpr static void checkTypeParameter() {
|
template <typename T_Node> constexpr static bool checkTypeParameter() {
|
||||||
static_assert(!std::is_const<T_Node>::value,
|
static_assert(!std::is_const<T_Node>::value,
|
||||||
"Type parameter 'T_Node' should not be const qualified");
|
"Type parameter 'T_Node' should not be const qualified");
|
||||||
static_assert(std::is_base_of<AstNode, T_Node>::value,
|
static_assert(std::is_base_of<AstNode, T_Node>::value,
|
||||||
"Type parameter 'T_Node' must be a subtype of AstNode");
|
"Type parameter 'T_Node' must be a subtype of AstNode");
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -2048,25 +2049,25 @@ public:
|
|||||||
// operation function in 'foreach' should be completely predictable by branch target caches in
|
// operation function in 'foreach' should be completely predictable by branch target caches in
|
||||||
// modern CPUs, while it is basically unpredictable for VNVisitor.
|
// modern CPUs, while it is basically unpredictable for VNVisitor.
|
||||||
template <typename T_Node> void foreach (std::function<void(T_Node*)> f) {
|
template <typename T_Node> void foreach (std::function<void(T_Node*)> f) {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
foreachImpl<T_Node, /* VisitNext: */ false>(this, f);
|
foreachImpl<T_Node, /* VisitNext: */ false>(this, f);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node> void foreach (std::function<void(const T_Node*)> f) const {
|
template <typename T_Node> void foreach (std::function<void(const T_Node*)> f) const {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
foreachImpl<const T_Node, /* VisitNext: */ false>(this, f);
|
foreachImpl<const T_Node, /* VisitNext: */ false>(this, f);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as 'foreach' but also follows 'this->nextp()'
|
// Same as 'foreach' but also follows 'this->nextp()'
|
||||||
template <typename T_Node> void foreachAndNext(std::function<void(T_Node*)> f) {
|
template <typename T_Node> void foreachAndNext(std::function<void(T_Node*)> f) {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
foreachImpl<T_Node, /* VisitNext: */ true>(this, f);
|
foreachImpl<T_Node, /* VisitNext: */ true>(this, f);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as 'foreach' but also follows 'this->nextp()'
|
// Same as 'foreach' but also follows 'this->nextp()'
|
||||||
template <typename T_Node> void foreachAndNext(std::function<void(const T_Node*)> f) const {
|
template <typename T_Node> void foreachAndNext(std::function<void(const T_Node*)> f) const {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
foreachImpl<const T_Node, /* VisitNext: */ true>(this, f);
|
foreachImpl<const T_Node, /* VisitNext: */ true>(this, f);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2075,13 +2076,13 @@ public:
|
|||||||
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
||||||
// result can be determined.
|
// result can be determined.
|
||||||
template <typename T_Node> bool exists(std::function<bool(T_Node*)> p) {
|
template <typename T_Node> bool exists(std::function<bool(T_Node*)> p) {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
return predicateImpl<T_Node, /* Default: */ false, /* VisitNext: */ false>(this, p);
|
return predicateImpl<T_Node, /* Default: */ false, /* VisitNext: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node> void exists(std::function<bool(const T_Node*)> p) const {
|
template <typename T_Node> void exists(std::function<bool(const T_Node*)> p) const {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
return predicateImpl<const T_Node, /* Default: */ false, /* VisitNext: */ false>(this, p);
|
return predicateImpl<const T_Node, /* Default: */ false, /* VisitNext: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2090,13 +2091,13 @@ public:
|
|||||||
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
||||||
// result can be determined.
|
// result can be determined.
|
||||||
template <typename T_Node> bool forall(std::function<bool(T_Node*)> p) {
|
template <typename T_Node> bool forall(std::function<bool(T_Node*)> p) {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
return predicateImpl<T_Node, /* Default: */ true, /* VisitNext: */ false>(this, p);
|
return predicateImpl<T_Node, /* Default: */ true, /* VisitNext: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node> void forall(std::function<bool(const T_Node*)> p) const {
|
template <typename T_Node> void forall(std::function<bool(const T_Node*)> p) const {
|
||||||
checkTypeParameter<T_Node>();
|
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
||||||
return predicateImpl<const T_Node, /* Default: */ true, /* VisitNext: */ false>(this, p);
|
return predicateImpl<const T_Node, /* Default: */ true, /* VisitNext: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -94,6 +94,7 @@ private:
|
|||||||
bool m_inDly = false; // True in delayed assignments
|
bool m_inDly = false; // True in delayed assignments
|
||||||
bool m_inLoop = false; // True in for loops
|
bool m_inLoop = false; // True in for loops
|
||||||
bool m_inInitial = false; // True in static initializers and initial blocks
|
bool m_inInitial = false; // True in static initializers and initial blocks
|
||||||
|
bool m_ignoreBlkAndNBlk = false; // Suppress delayed assignment BLKANDNBLK
|
||||||
using VarMap = std::map<const std::pair<AstNodeModule*, std::string>, AstVar*>;
|
using VarMap = std::map<const std::pair<AstNodeModule*, std::string>, AstVar*>;
|
||||||
VarMap m_modVarMap; // Table of new var names created under module
|
VarMap m_modVarMap; // Table of new var names created under module
|
||||||
VDouble0 m_statSharedSet; // Statistic tracking
|
VDouble0 m_statSharedSet; // Statistic tracking
|
||||||
@ -105,6 +106,7 @@ private:
|
|||||||
void markVarUsage(AstNodeVarRef* nodep, bool blocking) {
|
void markVarUsage(AstNodeVarRef* nodep, bool blocking) {
|
||||||
// Ignore if warning is disabled on this reference (used by V3Force).
|
// Ignore if warning is disabled on this reference (used by V3Force).
|
||||||
if (nodep->fileline()->warnIsOff(V3ErrorCode::BLKANDNBLK)) return;
|
if (nodep->fileline()->warnIsOff(V3ErrorCode::BLKANDNBLK)) return;
|
||||||
|
if (m_ignoreBlkAndNBlk) return;
|
||||||
if (blocking) nodep->user5(true);
|
if (blocking) nodep->user5(true);
|
||||||
AstVarScope* const vscp = nodep->varScopep();
|
AstVarScope* const vscp = nodep->varScopep();
|
||||||
// UINFO(4, " MVU " << blocking << " " << nodep << endl);
|
// UINFO(4, " MVU " << blocking << " " << nodep << endl);
|
||||||
@ -517,6 +519,13 @@ private:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual void visit(AstNodeReadWriteMem* nodep) override {
|
||||||
|
VL_RESTORER(m_ignoreBlkAndNBlk);
|
||||||
|
m_ignoreBlkAndNBlk = true; // $readmem/$writemem often used in mem models
|
||||||
|
// so we will suppress BLKANDNBLK warnings
|
||||||
|
iterateChildren(nodep);
|
||||||
|
}
|
||||||
|
|
||||||
virtual void visit(AstNodeFor* nodep) override { // LCOV_EXCL_LINE
|
virtual void visit(AstNodeFor* nodep) override { // LCOV_EXCL_LINE
|
||||||
nodep->v3fatalSrc(
|
nodep->v3fatalSrc(
|
||||||
"For statements should have been converted to while statements in V3Begin");
|
"For statements should have been converted to while statements in V3Begin");
|
||||||
|
@ -800,6 +800,10 @@ class EmitVPrefixedFormatter final : public V3OutFormatter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual void putsOutput(const char* strg) override {
|
||||||
|
for (const char* cp = strg; *cp; cp++) putcOutput(*cp);
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void prefixFl(FileLine* fl) { m_prefixFl = fl; }
|
void prefixFl(FileLine* fl) { m_prefixFl = fl; }
|
||||||
FileLine* prefixFl() const { return m_prefixFl; }
|
FileLine* prefixFl() const { return m_prefixFl; }
|
||||||
|
@ -700,6 +700,10 @@ int V3OutFormatter::endLevels(const char* strg) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void V3OutFormatter::puts(const char* strg) {
|
void V3OutFormatter::puts(const char* strg) {
|
||||||
|
if (!v3Global.opt.decoration()) {
|
||||||
|
putsOutput(strg);
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (m_prependIndent && strg[0] != '\n') {
|
if (m_prependIndent && strg[0] != '\n') {
|
||||||
putsNoTracking(indentSpaces(endLevels(strg)));
|
putsNoTracking(indentSpaces(endLevels(strg)));
|
||||||
m_prependIndent = false;
|
m_prependIndent = false;
|
||||||
@ -759,13 +763,8 @@ void V3OutFormatter::puts(const char* strg) {
|
|||||||
break;
|
break;
|
||||||
case '(':
|
case '(':
|
||||||
indentInc();
|
indentInc();
|
||||||
if (v3Global.opt.decoration()) {
|
|
||||||
// Line up continuation with open paren, plus one indent
|
// Line up continuation with open paren, plus one indent
|
||||||
m_parenVec.push(m_column);
|
m_parenVec.push(m_column);
|
||||||
} else {
|
|
||||||
// Line up continuation with block+1
|
|
||||||
m_parenVec.push(m_indentLevel * m_blockIndent);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case ')':
|
case ')':
|
||||||
if (!m_parenVec.empty()) m_parenVec.pop();
|
if (!m_parenVec.empty()) m_parenVec.pop();
|
||||||
@ -806,6 +805,7 @@ void V3OutFormatter::putBreakExpr() {
|
|||||||
|
|
||||||
// Add a line break if too wide
|
// Add a line break if too wide
|
||||||
void V3OutFormatter::putBreak() {
|
void V3OutFormatter::putBreak() {
|
||||||
|
if (!v3Global.opt.decoration()) return;
|
||||||
if (!m_nobreak) {
|
if (!m_nobreak) {
|
||||||
// char s[1000]; sprintf(s, "{%d,%d}", m_column, m_parenVec.top()); putsNoTracking(s);
|
// char s[1000]; sprintf(s, "{%d,%d}", m_column, m_parenVec.top()); putsNoTracking(s);
|
||||||
if (exceededWidth()) {
|
if (exceededWidth()) {
|
||||||
@ -824,11 +824,19 @@ void V3OutFormatter::putsQuoted(const string& strg) {
|
|||||||
putcNoTracking('"');
|
putcNoTracking('"');
|
||||||
}
|
}
|
||||||
void V3OutFormatter::putsNoTracking(const string& strg) {
|
void V3OutFormatter::putsNoTracking(const string& strg) {
|
||||||
|
if (!v3Global.opt.decoration()) {
|
||||||
|
putsOutput(strg.c_str());
|
||||||
|
return;
|
||||||
|
}
|
||||||
// Don't track {}'s, probably because it's a $display format string
|
// Don't track {}'s, probably because it's a $display format string
|
||||||
for (const char c : strg) putcNoTracking(c);
|
for (const char c : strg) putcNoTracking(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
void V3OutFormatter::putcNoTracking(char chr) {
|
void V3OutFormatter::putcNoTracking(char chr) {
|
||||||
|
if (!v3Global.opt.decoration()) {
|
||||||
|
putcOutput(chr);
|
||||||
|
return;
|
||||||
|
}
|
||||||
switch (chr) {
|
switch (chr) {
|
||||||
case '\n':
|
case '\n':
|
||||||
m_lineno++;
|
m_lineno++;
|
||||||
|
@ -176,6 +176,7 @@ public:
|
|||||||
|
|
||||||
// CALLBACKS - MUST OVERRIDE
|
// CALLBACKS - MUST OVERRIDE
|
||||||
virtual void putcOutput(char chr) = 0;
|
virtual void putcOutput(char chr) = 0;
|
||||||
|
virtual void putsOutput(const char* str) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
@ -193,6 +194,7 @@ public:
|
|||||||
private:
|
private:
|
||||||
// CALLBACKS
|
// CALLBACKS
|
||||||
virtual void putcOutput(char chr) override { fputc(chr, m_fp); }
|
virtual void putcOutput(char chr) override { fputc(chr, m_fp); }
|
||||||
|
virtual void putsOutput(const char* str) override { fputs(str, m_fp); }
|
||||||
};
|
};
|
||||||
|
|
||||||
class V3OutCFile VL_NOT_FINAL : public V3OutFile {
|
class V3OutCFile VL_NOT_FINAL : public V3OutFile {
|
||||||
|
@ -311,8 +311,8 @@ private:
|
|||||||
// code will be emitted.
|
// code will be emitted.
|
||||||
UINFO(9, "assign to public and unpacked: " << nodep << endl);
|
UINFO(9, "assign to public and unpacked: " << nodep << endl);
|
||||||
m_modp->addStmtp(
|
m_modp->addStmtp(
|
||||||
new AstAssignW(flp, new AstVarRef(flp, exprvarrefp->varp(), VAccess::WRITE),
|
new AstAssignW{flp, new AstVarRef{flp, nodep, VAccess::WRITE},
|
||||||
new AstVarRef(flp, nodep, VAccess::READ)));
|
new AstVarRef{flp, exprvarrefp->varp(), VAccess::READ}});
|
||||||
} else if (nodep->isIfaceRef()) {
|
} else if (nodep->isIfaceRef()) {
|
||||||
m_modp->addStmtp(
|
m_modp->addStmtp(
|
||||||
new AstAssignVarScope(flp, new AstVarRef(flp, nodep, VAccess::WRITE),
|
new AstAssignVarScope(flp, new AstVarRef(flp, nodep, VAccess::WRITE),
|
||||||
|
@ -218,10 +218,10 @@ public:
|
|||||||
void lifeToAbove() {
|
void lifeToAbove() {
|
||||||
// Any varrefs under a if/else branch affect statements outside and after the if/else
|
// Any varrefs under a if/else branch affect statements outside and after the if/else
|
||||||
if (!m_aboveLifep) v3fatalSrc("Pushing life when already at the top level");
|
if (!m_aboveLifep) v3fatalSrc("Pushing life when already at the top level");
|
||||||
for (LifeMap::iterator it = m_map.begin(); it != m_map.end(); ++it) {
|
for (auto& itr : m_map) {
|
||||||
AstVarScope* const nodep = it->first;
|
AstVarScope* const nodep = itr.first;
|
||||||
m_aboveLifep->complexAssignFind(nodep);
|
m_aboveLifep->complexAssignFind(nodep);
|
||||||
if (it->second.everSet()) {
|
if (itr.second.everSet()) {
|
||||||
// Record there may be an assignment, so we don't constant propagate across the if.
|
// Record there may be an assignment, so we don't constant propagate across the if.
|
||||||
complexAssignFind(nodep);
|
complexAssignFind(nodep);
|
||||||
} else {
|
} else {
|
||||||
@ -235,14 +235,14 @@ public:
|
|||||||
// life1p->lifeDump();
|
// life1p->lifeDump();
|
||||||
// life2p->lifeDump();
|
// life2p->lifeDump();
|
||||||
AstNode::user1ClearTree(); // user1p() used on entire tree
|
AstNode::user1ClearTree(); // user1p() used on entire tree
|
||||||
for (LifeMap::iterator it = life1p->m_map.begin(); it != life1p->m_map.end(); ++it) {
|
for (auto& itr : life1p->m_map) {
|
||||||
// When the if branch sets a var before it's used, mark that variable
|
// When the if branch sets a var before it's used, mark that variable
|
||||||
if (it->second.setBeforeUse()) it->first->user1(1);
|
if (itr.second.setBeforeUse()) itr.first->user1(1);
|
||||||
}
|
}
|
||||||
for (LifeMap::iterator it = life2p->m_map.begin(); it != life2p->m_map.end(); ++it) {
|
for (auto& itr : life2p->m_map) {
|
||||||
// When the else branch sets a var before it's used
|
// When the else branch sets a var before it's used
|
||||||
AstVarScope* const nodep = it->first;
|
AstVarScope* const nodep = itr.first;
|
||||||
if (it->second.setBeforeUse() && nodep->user1()) {
|
if (itr.second.setBeforeUse() && nodep->user1()) {
|
||||||
// Both branches set the var, we can remove the assignment before the IF.
|
// Both branches set the var, we can remove the assignment before the IF.
|
||||||
UINFO(4, "DUALBRANCH " << nodep << endl);
|
UINFO(4, "DUALBRANCH " << nodep << endl);
|
||||||
const auto itab = m_map.find(nodep);
|
const auto itab = m_map.find(nodep);
|
||||||
@ -251,14 +251,15 @@ public:
|
|||||||
}
|
}
|
||||||
// this->lifeDump();
|
// this->lifeDump();
|
||||||
}
|
}
|
||||||
|
void clear() { m_map.clear(); }
|
||||||
// DEBUG
|
// DEBUG
|
||||||
void lifeDump() {
|
void lifeDump() {
|
||||||
UINFO(5, " LifeMap:" << endl);
|
UINFO(5, " LifeMap:" << endl);
|
||||||
for (LifeMap::iterator it = m_map.begin(); it != m_map.end(); ++it) {
|
for (const auto& itr : m_map) {
|
||||||
UINFO(5, " Ent: " << (it->second.setBeforeUse() ? "[F] " : " ") << it->first
|
UINFO(5, " Ent: " << (itr.second.setBeforeUse() ? "[F] " : " ") << itr.first
|
||||||
<< endl);
|
<< endl);
|
||||||
if (it->second.assignp()) { //
|
if (itr.second.assignp()) { //
|
||||||
UINFO(5, " Ass: " << it->second.assignp() << endl);
|
UINFO(5, " Ass: " << itr.second.assignp() << endl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -283,6 +284,10 @@ private:
|
|||||||
|
|
||||||
// METHODS
|
// METHODS
|
||||||
VL_DEBUG_FUNC; // Declare debug()
|
VL_DEBUG_FUNC; // Declare debug()
|
||||||
|
void setNoopt() {
|
||||||
|
m_noopt = true;
|
||||||
|
m_lifep->clear();
|
||||||
|
}
|
||||||
|
|
||||||
// VISITORS
|
// VISITORS
|
||||||
virtual void visit(AstVarRef* nodep) override {
|
virtual void visit(AstVarRef* nodep) override {
|
||||||
@ -386,13 +391,12 @@ private:
|
|||||||
// Just don't optimize blocks with labels; they're rare - so far.
|
// Just don't optimize blocks with labels; they're rare - so far.
|
||||||
LifeBlock* const prevLifep = m_lifep;
|
LifeBlock* const prevLifep = m_lifep;
|
||||||
LifeBlock* const bodyLifep = new LifeBlock(prevLifep, m_statep);
|
LifeBlock* const bodyLifep = new LifeBlock(prevLifep, m_statep);
|
||||||
const bool prev_noopt = m_noopt;
|
|
||||||
{
|
{
|
||||||
|
VL_RESTORER(m_noopt);
|
||||||
m_lifep = bodyLifep;
|
m_lifep = bodyLifep;
|
||||||
m_noopt = true;
|
setNoopt();
|
||||||
iterateAndNextNull(nodep->stmtsp());
|
iterateAndNextNull(nodep->stmtsp());
|
||||||
m_lifep = prevLifep;
|
m_lifep = prevLifep;
|
||||||
m_noopt = prev_noopt;
|
|
||||||
}
|
}
|
||||||
UINFO(4, " joinjump" << endl);
|
UINFO(4, " joinjump" << endl);
|
||||||
// For the next assignments, clear any variables that were read or written in the block
|
// For the next assignments, clear any variables that were read or written in the block
|
||||||
|
@ -477,10 +477,12 @@ private:
|
|||||||
AstNode* currp = m_workQueuep->front();
|
AstNode* currp = m_workQueuep->front();
|
||||||
m_workQueuep->pop();
|
m_workQueuep->pop();
|
||||||
|
|
||||||
// Analyse sub-tree list for code motion
|
// Analyse sub-tree list for code motion and conditional merging
|
||||||
CodeMotionAnalysisVisitor::analyze(currp, stmtProperties);
|
CodeMotionAnalysisVisitor::analyze(currp, stmtProperties);
|
||||||
// Perform the code motion within the whole sub-tree list
|
// Perform the code motion within the whole sub-tree list
|
||||||
|
if (v3Global.opt.fMergeCondMotion()) {
|
||||||
currp = CodeMotionOptimizeVisitor::optimize(currp, stmtProperties);
|
currp = CodeMotionOptimizeVisitor::optimize(currp, stmtProperties);
|
||||||
|
}
|
||||||
|
|
||||||
// Merge conditionals in the whole sub-tree list (this might create new work items)
|
// Merge conditionals in the whole sub-tree list (this might create new work items)
|
||||||
iterateAndNextNull(currp);
|
iterateAndNextNull(currp);
|
||||||
|
@ -1097,6 +1097,7 @@ void V3Options::parseOptsList(FileLine* fl, const string& optdir, int argc, char
|
|||||||
DECL_OPTION("-flife-post", FOnOff, &m_fLifePost);
|
DECL_OPTION("-flife-post", FOnOff, &m_fLifePost);
|
||||||
DECL_OPTION("-flocalize", FOnOff, &m_fLocalize);
|
DECL_OPTION("-flocalize", FOnOff, &m_fLocalize);
|
||||||
DECL_OPTION("-fmerge-cond", FOnOff, &m_fMergeCond);
|
DECL_OPTION("-fmerge-cond", FOnOff, &m_fMergeCond);
|
||||||
|
DECL_OPTION("-fmerge-cond-motion", FOnOff, &m_fMergeCondMotion);
|
||||||
DECL_OPTION("-fmerge-const-pool", FOnOff, &m_fMergeConstPool);
|
DECL_OPTION("-fmerge-const-pool", FOnOff, &m_fMergeConstPool);
|
||||||
DECL_OPTION("-freloop", FOnOff, &m_fReloop);
|
DECL_OPTION("-freloop", FOnOff, &m_fReloop);
|
||||||
DECL_OPTION("-freorder", FOnOff, &m_fReorder);
|
DECL_OPTION("-freorder", FOnOff, &m_fReorder);
|
||||||
|
@ -353,7 +353,8 @@ private:
|
|||||||
bool m_fLifePost; // main switch: -fno-life-post: delayed assignment elimination
|
bool m_fLifePost; // main switch: -fno-life-post: delayed assignment elimination
|
||||||
bool m_fLocalize; // main switch: -fno-localize: convert temps to local variables
|
bool m_fLocalize; // main switch: -fno-localize: convert temps to local variables
|
||||||
bool m_fMergeCond; // main switch: -fno-merge-cond: merge conditionals
|
bool m_fMergeCond; // main switch: -fno-merge-cond: merge conditionals
|
||||||
bool m_fMergeConstPool = true; // main switch: --fmerge-const-pool
|
bool m_fMergeCondMotion = true; // main switch: -fno-merge-cond-motion: perform code motion
|
||||||
|
bool m_fMergeConstPool = true; // main switch: -fno-merge-const-pool
|
||||||
bool m_fReloop; // main switch: -fno-reloop: reform loops
|
bool m_fReloop; // main switch: -fno-reloop: reform loops
|
||||||
bool m_fReorder; // main switch: -fno-reorder: reorder assignments in blocks
|
bool m_fReorder; // main switch: -fno-reorder: reorder assignments in blocks
|
||||||
bool m_fSplit; // main switch: -fno-split: always assignment splitting
|
bool m_fSplit; // main switch: -fno-split: always assignment splitting
|
||||||
@ -585,6 +586,7 @@ public:
|
|||||||
bool fLifePost() const { return m_fLifePost; }
|
bool fLifePost() const { return m_fLifePost; }
|
||||||
bool fLocalize() const { return m_fLocalize; }
|
bool fLocalize() const { return m_fLocalize; }
|
||||||
bool fMergeCond() const { return m_fMergeCond; }
|
bool fMergeCond() const { return m_fMergeCond; }
|
||||||
|
bool fMergeCondMotion() const { return m_fMergeCondMotion; }
|
||||||
bool fMergeConstPool() const { return m_fMergeConstPool; }
|
bool fMergeConstPool() const { return m_fMergeConstPool; }
|
||||||
bool fReloop() const { return m_fReloop; }
|
bool fReloop() const { return m_fReloop; }
|
||||||
bool fReorder() const { return m_fReorder; }
|
bool fReorder() const { return m_fReorder; }
|
||||||
|
34
test_regress/t/t_merge_cond_no_motion.pl
Executable file
34
test_regress/t/t_merge_cond_no_motion.pl
Executable file
@ -0,0 +1,34 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2022 by Geza Lore. This program is free software; you
|
||||||
|
# can redistribute it and/or modify it under the terms of either the GNU
|
||||||
|
# Lesser General Public License Version 3 or the Perl Artistic License
|
||||||
|
# Version 2.0.
|
||||||
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
scenarios(vlt_all => 1);
|
||||||
|
|
||||||
|
top_filename("t/t_merge_cond.v");
|
||||||
|
|
||||||
|
compile(
|
||||||
|
verilator_flags2 => ["-unroll-count 64", "--stats", "-fno-merge-cond-motion"],
|
||||||
|
);
|
||||||
|
|
||||||
|
execute(
|
||||||
|
check_finished => 1,
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($Self->{vlt}) {
|
||||||
|
# Note, with vltmt this might be split differently, so only checking vlt
|
||||||
|
file_grep($Self->{stats}, qr/Optimizations, MergeCond merges\s+(\d+)/i,
|
||||||
|
10);
|
||||||
|
file_grep($Self->{stats}, qr/Optimizations, MergeCond merged items\s+(\d+)/i,
|
||||||
|
580);
|
||||||
|
file_grep($Self->{stats}, qr/Optimizations, MergeCond longest merge\s+(\d+)/i,
|
||||||
|
64);
|
||||||
|
}
|
||||||
|
|
||||||
|
ok(1);
|
||||||
|
1;
|
18
test_regress/t/t_pub_unpacked_port.pl
Executable file
18
test_regress/t/t_pub_unpacked_port.pl
Executable file
@ -0,0 +1,18 @@
|
|||||||
|
#!/usr/bin/env perl
|
||||||
|
if (!$::Driver) { use FindBin; exec("$FindBin::Bin/bootstrap.pl", @ARGV, $0); die; }
|
||||||
|
# DESCRIPTION: Verilator: Verilog Test driver/expect definition
|
||||||
|
#
|
||||||
|
# Copyright 2022 by Todd Strader. This program is free software; you
|
||||||
|
# can redistribute it and/or modify it under the terms of either the GNU
|
||||||
|
# Lesser General Public License Version 3 or the Perl Artistic License
|
||||||
|
# Version 2.0.
|
||||||
|
# SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
||||||
|
|
||||||
|
scenarios(
|
||||||
|
vlt => 1,
|
||||||
|
);
|
||||||
|
|
||||||
|
compile();
|
||||||
|
execute();
|
||||||
|
ok(1);
|
||||||
|
1;
|
55
test_regress/t/t_pub_unpacked_port.v
Normal file
55
test_regress/t/t_pub_unpacked_port.v
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
// DESCRIPTION: Verilator: Verilog Test module
|
||||||
|
// This file ONLY is placed into the Public Domain, for any use,
|
||||||
|
// without warranty, 2022 by Todd Strader.
|
||||||
|
// SPDX-License-Identifier: CC0-1.0
|
||||||
|
|
||||||
|
module sub (
|
||||||
|
output logic [31:0] sub_s1up_out[0:0] /* verilator public_flat_rw */,
|
||||||
|
input logic sub_clk,
|
||||||
|
input logic [31:0] sub_s1up_in[0:0] /* verilator public_flat_rw */
|
||||||
|
);
|
||||||
|
|
||||||
|
// Evaluate clock edges
|
||||||
|
always @(posedge sub_clk) begin
|
||||||
|
sub_s1up_out <= sub_s1up_in;
|
||||||
|
end
|
||||||
|
|
||||||
|
endmodule
|
||||||
|
|
||||||
|
|
||||||
|
module t (/*AUTOARG*/
|
||||||
|
// Inputs
|
||||||
|
clk
|
||||||
|
);
|
||||||
|
input clk;
|
||||||
|
|
||||||
|
integer cyc = 0;
|
||||||
|
logic [31:0] s1up_in[1];
|
||||||
|
logic [31:0] s1up_out[1];
|
||||||
|
|
||||||
|
sub the_sub (
|
||||||
|
.sub_s1up_in (s1up_in),
|
||||||
|
.sub_s1up_out (s1up_out),
|
||||||
|
.sub_clk (clk));
|
||||||
|
|
||||||
|
always_comb s1up_in[0] = cyc;
|
||||||
|
|
||||||
|
always @(posedge clk) begin
|
||||||
|
cyc <= cyc + 1;
|
||||||
|
|
||||||
|
if (cyc == 10) begin
|
||||||
|
if (s1up_out[0] != 9) begin
|
||||||
|
$display("%%Error: got %0d instead of 9", s1up_out);
|
||||||
|
$stop;
|
||||||
|
end
|
||||||
|
if (the_sub.sub_s1up_in[0] != 10) begin
|
||||||
|
$display("%%Error: the_sub.sub_s1up_in was %0d instead of 10", the_sub.sub_s1up_in[0]);
|
||||||
|
$stop;
|
||||||
|
end
|
||||||
|
$display("final cycle = %0d", cyc);
|
||||||
|
$write("*-* All Finished *-*\n");
|
||||||
|
$finish;
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
endmodule
|
@ -6,22 +6,37 @@
|
|||||||
|
|
||||||
`define STRINGIFY(x) `"x`"
|
`define STRINGIFY(x) `"x`"
|
||||||
|
|
||||||
module t;
|
module t(/*AUTOARG*/
|
||||||
|
// Inputs
|
||||||
|
clk
|
||||||
|
);
|
||||||
|
|
||||||
|
input clk;
|
||||||
|
int cyc;
|
||||||
|
|
||||||
reg [5:0] assoc_c[int];
|
reg [5:0] assoc_c[int];
|
||||||
reg [95:0] assoc_w[int];
|
reg [95:0] assoc_w[int];
|
||||||
|
|
||||||
initial begin
|
always_ff @ (posedge clk) begin
|
||||||
assoc_c[300] = 10; // See if clearing must happen first
|
cyc <= cyc + 1;
|
||||||
|
if (cyc == 1) begin
|
||||||
|
assoc_c[300] <= 10; // See if clearing must happen first
|
||||||
|
// Also checks no BLKANDNBLK due to readmem/writemem
|
||||||
|
end
|
||||||
|
else if (cyc == 2) begin
|
||||||
$readmemb("t/t_sys_readmem_b.mem", assoc_c);
|
$readmemb("t/t_sys_readmem_b.mem", assoc_c);
|
||||||
$display("assoc_c=%p", assoc_c);
|
$display("assoc_c=%p", assoc_c);
|
||||||
$writememh({`STRINGIFY(`TEST_OBJ_DIR),"/t_sys_writemem_c_b.mem"}, assoc_c);
|
$writememh({`STRINGIFY(`TEST_OBJ_DIR),"/t_sys_writemem_c_b.mem"}, assoc_c);
|
||||||
|
end
|
||||||
|
else if (cyc == 3) begin
|
||||||
$readmemb("t/t_sys_readmem_b.mem", assoc_w);
|
$readmemb("t/t_sys_readmem_b.mem", assoc_w);
|
||||||
// Not conditional with TEST_VERBOSE as found bug with wide display
|
// Not conditional with TEST_VERBOSE as found bug with wide display
|
||||||
$display("assoc_w=%p", assoc_w);
|
$display("assoc_w=%p", assoc_w);
|
||||||
$writememh({`STRINGIFY(`TEST_OBJ_DIR),"/t_sys_writemem_w_h.mem"}, assoc_w);
|
$writememh({`STRINGIFY(`TEST_OBJ_DIR),"/t_sys_writemem_w_h.mem"}, assoc_w);
|
||||||
|
end
|
||||||
|
else if (cyc == 4) begin
|
||||||
$write("*-* All Finished *-*\n");
|
$write("*-* All Finished *-*\n");
|
||||||
$finish;
|
$finish;
|
||||||
end
|
end
|
||||||
|
end
|
||||||
endmodule
|
endmodule
|
||||||
|
Loading…
Reference in New Issue
Block a user