forked from github/verilator
Improve V3Ast function usage ergonomics (#3650)
Signed-off-by: HungMingWu <u9089000@gmail.com>
This commit is contained in:
parent
627a144b83
commit
196f3292d5
@ -56,7 +56,6 @@ DisableFormat: false
|
|||||||
ExperimentalAutoDetectBinPacking: false
|
ExperimentalAutoDetectBinPacking: false
|
||||||
FixNamespaceComments: true
|
FixNamespaceComments: true
|
||||||
ForEachMacros:
|
ForEachMacros:
|
||||||
- foreach
|
|
||||||
- Q_FOREACH
|
- Q_FOREACH
|
||||||
- BOOST_FOREACH
|
- BOOST_FOREACH
|
||||||
|
|
||||||
|
@ -385,7 +385,7 @@ private:
|
|||||||
// Blocking assignments are always OK in combinational (and initial/final) processes
|
// Blocking assignments are always OK in combinational (and initial/final) processes
|
||||||
if (m_check != CT_SEQ) return;
|
if (m_check != CT_SEQ) return;
|
||||||
|
|
||||||
const bool ignore = nodep->lhsp()->forall<AstVarRef>([&](const AstVarRef* refp) {
|
const bool ignore = nodep->lhsp()->forall([&](const AstVarRef* refp) {
|
||||||
// Ignore reads (e.g.: index expressions)
|
// Ignore reads (e.g.: index expressions)
|
||||||
if (refp->access().isReadOnly()) return true;
|
if (refp->access().isReadOnly()) return true;
|
||||||
const AstVar* const varp = refp->varp();
|
const AstVar* const varp = refp->varp();
|
||||||
@ -500,7 +500,7 @@ private:
|
|||||||
|
|
||||||
void visitSenItems(AstNode* nodep) {
|
void visitSenItems(AstNode* nodep) {
|
||||||
if (v3Global.opt.timing().isSetTrue()) {
|
if (v3Global.opt.timing().isSetTrue()) {
|
||||||
nodep->foreach<AstSenItem>([this](AstSenItem* senItemp) { visit(senItemp); });
|
nodep->foreach([this](AstSenItem* senItemp) { visit(senItemp); });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -561,7 +561,7 @@ private:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
nodep->sensp()->foreach<AstVarRef>([](const AstVarRef* refp) {
|
nodep->sensp()->foreach([](const AstVarRef* refp) {
|
||||||
refp->varp()->usedClock(true);
|
refp->varp()->usedClock(true);
|
||||||
refp->varScopep()->user1(true);
|
refp->varScopep()->user1(true);
|
||||||
});
|
});
|
||||||
|
@ -47,7 +47,7 @@ class ActiveTopVisitor final : public VNVisitor {
|
|||||||
static bool isInitial(AstNode* nodep) {
|
static bool isInitial(AstNode* nodep) {
|
||||||
const VNUser1InUse user1InUse;
|
const VNUser1InUse user1InUse;
|
||||||
// Return true if no variables that read.
|
// Return true if no variables that read.
|
||||||
return nodep->forall<AstVarRef>([&](const AstVarRef* refp) -> bool {
|
return nodep->forall([&](const AstVarRef* refp) -> bool {
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
// Note: Use same heuristic as ordering does to ignore written variables
|
// Note: Use same heuristic as ordering does to ignore written variables
|
||||||
// TODO: Use live variable analysis.
|
// TODO: Use live variable analysis.
|
||||||
|
161
src/V3Ast.h
161
src/V3Ast.h
@ -23,8 +23,10 @@
|
|||||||
#include "V3Broken.h"
|
#include "V3Broken.h"
|
||||||
#include "V3Error.h"
|
#include "V3Error.h"
|
||||||
#include "V3FileLine.h"
|
#include "V3FileLine.h"
|
||||||
|
#include "V3FunctionTraits.h"
|
||||||
#include "V3Global.h"
|
#include "V3Global.h"
|
||||||
#include "V3Number.h"
|
#include "V3Number.h"
|
||||||
|
#include "V3StdFuture.h"
|
||||||
|
|
||||||
#include "V3Ast__gen_forward_class_decls.h" // From ./astgen
|
#include "V3Ast__gen_forward_class_decls.h" // From ./astgen
|
||||||
|
|
||||||
@ -2084,95 +2086,127 @@ private:
|
|||||||
using ConstCorrectAstNode =
|
using ConstCorrectAstNode =
|
||||||
typename std::conditional<std::is_const<T_Arg>::value, const AstNode, AstNode>::type;
|
typename std::conditional<std::is_const<T_Arg>::value, const AstNode, AstNode>::type;
|
||||||
|
|
||||||
template <typename T_Arg>
|
template <typename T_Arg, typename Callable>
|
||||||
inline static void foreachImpl(ConstCorrectAstNode<T_Arg>* nodep,
|
inline static void foreachImpl(ConstCorrectAstNode<T_Arg>* nodep, const Callable& f,
|
||||||
const std::function<void(T_Arg*)>& f, bool visitNext);
|
bool visitNext);
|
||||||
|
|
||||||
template <typename T_Arg, bool Default>
|
template <typename T_Arg, bool Default, typename Callable>
|
||||||
inline static bool predicateImpl(ConstCorrectAstNode<T_Arg>* nodep,
|
inline static bool predicateImpl(ConstCorrectAstNode<T_Arg>* nodep, const Callable& p);
|
||||||
const std::function<bool(T_Arg*)>& p);
|
|
||||||
|
|
||||||
template <typename T_Node>
|
template <typename T_Callable>
|
||||||
constexpr static bool checkTypeParameter() {
|
struct Arg0NoPointerNoCV final {
|
||||||
static_assert(!std::is_const<T_Node>::value,
|
using Traits = FunctionTraits<T_Callable>;
|
||||||
"Type parameter 'T_Node' should not be const qualified");
|
using T_Arg0 = typename Traits::template arg<0>::type;
|
||||||
static_assert(std::is_base_of<AstNode, T_Node>::value,
|
using T_Arg0NoPtr = typename std::remove_pointer<T_Arg0>::type;
|
||||||
"Type parameter 'T_Node' must be a subtype of AstNode");
|
using type = typename std::remove_cv<T_Arg0NoPtr>::type;
|
||||||
return true;
|
};
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Traverse subtree and call given function 'f' in pre-order on each node that has type
|
// Given a callable 'f' that takes a single argument of some AstNode subtype 'T_Node', traverse
|
||||||
// 'T_Node'. The node passd to the function 'f' can be removed or replaced, but other editing
|
// the tree rooted at this node, and call 'f' in pre-order on each node that is of type
|
||||||
|
// 'T_Node'. The node passed to the callable 'f' can be removed or replaced, but other editing
|
||||||
// of the iterated tree is not safe. Prefer 'foreach' over simple VNVisitor that only needs to
|
// of the iterated tree is not safe. Prefer 'foreach' over simple VNVisitor that only needs to
|
||||||
// handle a single (or a few) node types, as it's easier to write, but more importantly, the
|
// handle a single (or a few) node types, as it's easier to write, but more importantly, the
|
||||||
// dispatch to the operation function in 'foreach' should be completely predictable by branch
|
// dispatch to the callable in 'foreach' should be completely predictable by branch target
|
||||||
// target caches in modern CPUs, while it is basically unpredictable for VNVisitor.
|
// caches in modern CPUs, while it is basically unpredictable for VNVisitor.
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
void foreach (std::function<void(T_Node*)> f) {
|
void foreach(Callable&& f) {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable<Callable, T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Callable 'f' must have a signature compatible with 'void(T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
foreachImpl<T_Node>(this, f, /* visitNext: */ false);
|
foreachImpl<T_Node>(this, f, /* visitNext: */ false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
void foreach (std::function<void(const T_Node*)> f) const {
|
void foreach(Callable&& f) const {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable<Callable, const T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Callable 'f' must have a signature compatible with 'void(const T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
foreachImpl<const T_Node>(this, f, /* visitNext: */ false);
|
foreachImpl<const T_Node>(this, f, /* visitNext: */ false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as 'foreach' but also follows 'this->nextp()'
|
// Same as 'foreach' but also traverses 'this->nextp()' transitively
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
void foreachAndNext(std::function<void(T_Node*)> f) {
|
void foreachAndNext(Callable&& f) {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable<Callable, T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Callable 'f' must have a signature compatible with 'void(T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
foreachImpl<T_Node>(this, f, /* visitNext: */ true);
|
foreachImpl<T_Node>(this, f, /* visitNext: */ true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as 'foreach' but also follows 'this->nextp()'
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
void foreachAndNext(std::function<void(const T_Node*)> f) const {
|
void foreachAndNext(Callable&& f) const {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable<Callable, const T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Callable 'f' must have a signature compatible with 'void(const T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
foreachImpl<const T_Node>(this, f, /* visitNext: */ true);
|
foreachImpl<const T_Node>(this, f, /* visitNext: */ true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Given a predicate function 'p' return true if and only if there exists a node of type
|
// Given a predicate 'p' that takes a single argument of some AstNode subtype 'T_Node', return
|
||||||
// 'T_Node' that satisfies the predicate 'p'. Returns false if no node of type 'T_Node' is
|
// true if and only if there exists a node of type 'T_Node' in the tree rooted at this node,
|
||||||
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
// that satisfies the predicate 'p'. Returns false if no node of type 'T_Node' is present.
|
||||||
// result can be determined.
|
// Traversal is performed in some arbitrary order and is terminated as soon as the result can
|
||||||
template <typename T_Node>
|
// be determined.
|
||||||
bool exists(std::function<bool(T_Node*)> p) {
|
template <typename Callable>
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
bool exists(Callable&& p) {
|
||||||
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable_r<bool, Callable, T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Predicate 'p' must have a signature compatible with 'bool(T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
return predicateImpl<T_Node, /* Default: */ false>(this, p);
|
return predicateImpl<T_Node, /* Default: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
bool exists(std::function<bool(const T_Node*)> p) const {
|
bool exists(Callable&& p) const {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable_r<bool, Callable, const T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Predicate 'p' must have a signature compatible with 'bool(const T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
return predicateImpl<const T_Node, /* Default: */ false>(this, p);
|
return predicateImpl<const T_Node, /* Default: */ false>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Given a predicate function 'p' return true if and only if all nodes of type
|
// Given a predicate 'p' that takes a single argument of some AstNode subtype 'T_Node', return
|
||||||
// 'T_Node' satisfy the predicate 'p'. Returns true if no node of type 'T_Node' is
|
// true if and only if all nodes of type 'T_Node' in the tree rooted at this node satisfy the
|
||||||
// present. Traversal is performed in some arbitrary order and is terminated as soon as the
|
// predicate 'p'. Returns true if no node of type 'T_Node' is present. Traversal is performed
|
||||||
// result can be determined.
|
// in some arbitrary order and is terminated as soon as the result can be determined.
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
bool forall(std::function<bool(T_Node*)> p) {
|
bool forall(Callable&& p) {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable_r<bool, Callable, T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Predicate 'p' must have a signature compatible with 'bool(T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
return predicateImpl<T_Node, /* Default: */ true>(this, p);
|
return predicateImpl<T_Node, /* Default: */ true>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as above, but for 'const' nodes
|
// Same as above, but for 'const' nodes
|
||||||
template <typename T_Node>
|
template <typename Callable>
|
||||||
bool forall(std::function<bool(const T_Node*)> p) const {
|
bool forall(Callable&& p) const {
|
||||||
static_assert(checkTypeParameter<T_Node>(), "Invalid type parameter 'T_Node'");
|
using T_Node = typename Arg0NoPointerNoCV<Callable>::type;
|
||||||
|
static_assert(vlstd::is_invocable_r<bool, Callable, const T_Node*>::value
|
||||||
|
&& std::is_base_of<AstNode, T_Node>::value,
|
||||||
|
"Predicate 'p' must have a signature compatible with 'bool(const T_Node*)', "
|
||||||
|
"with 'T_Node' being a subtype of 'AstNode'");
|
||||||
return predicateImpl<const T_Node, /* Default: */ true>(this, p);
|
return predicateImpl<const T_Node, /* Default: */ true>(this, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
int nodeCount() const {
|
int nodeCount() const {
|
||||||
// TODO: this should really return size_t, but need to fix use sites
|
// TODO: this should really return size_t, but need to fix use sites
|
||||||
int count = 0;
|
int count = 0;
|
||||||
this->foreach<AstNode>([&count](const AstNode*) { ++count; });
|
this->foreach([&count](const AstNode*) { ++count; });
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -2243,13 +2277,8 @@ constexpr bool AstNode::isLeaf<AstVarXRef>() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// foreach implementation
|
// foreach implementation
|
||||||
template <typename T_Arg>
|
template <typename T_Arg, typename Callable>
|
||||||
void AstNode::foreachImpl(ConstCorrectAstNode<T_Arg>* nodep, const std::function<void(T_Arg*)>& f,
|
void AstNode::foreachImpl(ConstCorrectAstNode<T_Arg>* nodep, const Callable& f, bool visitNext) {
|
||||||
bool visitNext) {
|
|
||||||
// Checking the function is bound up front eliminates this check from the loop at invocation
|
|
||||||
if (!f) {
|
|
||||||
nodep->v3fatal("AstNode::foreach called with unbound function"); // LCOV_EXCL_LINE
|
|
||||||
} else {
|
|
||||||
// Pre-order traversal implemented directly (without recursion) for speed reasons. The very
|
// Pre-order traversal implemented directly (without recursion) for speed reasons. The very
|
||||||
// first iteration (the one that operates on the input nodep) is special, as we might or
|
// first iteration (the one that operates on the input nodep) is special, as we might or
|
||||||
// might not need to enqueue nodep->nextp() depending on VisitNext, while in all other
|
// might not need to enqueue nodep->nextp() depending on VisitNext, while in all other
|
||||||
@ -2326,17 +2355,12 @@ void AstNode::foreachImpl(ConstCorrectAstNode<T_Arg>* nodep, const std::function
|
|||||||
// Visit the head node
|
// Visit the head node
|
||||||
visit(headp);
|
visit(headp);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// predicate implementation
|
// predicate implementation
|
||||||
template <typename T_Arg, bool Default>
|
template <typename T_Arg, bool Default, typename Callable>
|
||||||
bool AstNode::predicateImpl(ConstCorrectAstNode<T_Arg>* nodep,
|
bool AstNode::predicateImpl(ConstCorrectAstNode<T_Arg>* nodep, const Callable& p) {
|
||||||
const std::function<bool(T_Arg*)>& p) {
|
// Implementation similar to foreach, but abort traversal as soon as result is determined
|
||||||
// Implementation similar to foreach, but abort traversal as soon as result is determined.
|
|
||||||
if (VL_UNCOVERABLE(!p)) {
|
|
||||||
nodep->v3fatal("AstNode::foreach called with unbound function"); // LCOV_EXCL_LINE
|
|
||||||
} else {
|
|
||||||
using T_Arg_NonConst = typename std::remove_const<T_Arg>::type;
|
using T_Arg_NonConst = typename std::remove_const<T_Arg>::type;
|
||||||
using Node = ConstCorrectAstNode<T_Arg>;
|
using Node = ConstCorrectAstNode<T_Arg>;
|
||||||
|
|
||||||
@ -2408,7 +2432,6 @@ bool AstNode::predicateImpl(ConstCorrectAstNode<T_Arg>* nodep,
|
|||||||
}
|
}
|
||||||
|
|
||||||
return Default;
|
return Default;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline std::ostream& operator<<(std::ostream& os, const AstNode* rhs) {
|
inline std::ostream& operator<<(std::ostream& os, const AstNode* rhs) {
|
||||||
|
@ -3833,8 +3833,7 @@ public:
|
|||||||
return new AstAssignW{fileline(), lhsp, rhsp, controlp};
|
return new AstAssignW{fileline(), lhsp, rhsp, controlp};
|
||||||
}
|
}
|
||||||
bool isTimingControl() const override {
|
bool isTimingControl() const override {
|
||||||
return timingControlp()
|
return timingControlp() || lhsp()->exists([](const AstNodeVarRef* refp) {
|
||||||
|| lhsp()->exists<AstNodeVarRef>([](const AstNodeVarRef* const refp) {
|
|
||||||
return refp->access().isWriteOrRW() && refp->varp()->delayp();
|
return refp->access().isWriteOrRW() && refp->varp()->delayp();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -329,7 +329,7 @@ void V3Broken::brokenAll(AstNetlist* nodep) {
|
|||||||
|
|
||||||
// Mark every node in the tree
|
// Mark every node in the tree
|
||||||
const uint8_t brokenCntCurrent = s_brokenCntGlobal.get();
|
const uint8_t brokenCntCurrent = s_brokenCntGlobal.get();
|
||||||
nodep->foreach<AstNode>([brokenCntCurrent](AstNode* nodep) {
|
nodep->foreach([brokenCntCurrent](AstNode* nodep) {
|
||||||
#ifdef VL_LEAK_CHECKS
|
#ifdef VL_LEAK_CHECKS
|
||||||
UASSERT_OBJ(s_allocTable.isAllocated(nodep), nodep,
|
UASSERT_OBJ(s_allocTable.isAllocated(nodep), nodep,
|
||||||
"AstNode is in tree, but not allocated");
|
"AstNode is in tree, but not allocated");
|
||||||
|
@ -2038,10 +2038,10 @@ private:
|
|||||||
// Note only do this (need user4) when m_warn, which is
|
// Note only do this (need user4) when m_warn, which is
|
||||||
// done as unique visitor
|
// done as unique visitor
|
||||||
const VNUser4InUse m_inuser4;
|
const VNUser4InUse m_inuser4;
|
||||||
nodep->lhsp()->foreach<AstVarRef>([](const AstVarRef* nodep) {
|
nodep->lhsp()->foreach([](const AstVarRef* nodep) {
|
||||||
if (nodep->varp()) nodep->varp()->user4(1);
|
if (nodep->varp()) nodep->varp()->user4(1);
|
||||||
});
|
});
|
||||||
nodep->rhsp()->foreach<AstVarRef>([&need_temp](const AstVarRef* nodep) {
|
nodep->rhsp()->foreach([&need_temp](const AstVarRef* nodep) {
|
||||||
if (nodep->varp() && nodep->varp()->user4()) need_temp = true;
|
if (nodep->varp() && nodep->varp()->user4()) need_temp = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -322,7 +322,7 @@ private:
|
|||||||
// And its children may now be killable too; correct counts
|
// And its children may now be killable too; correct counts
|
||||||
// Recurse, as cells may not be directly under the module but in a generate
|
// Recurse, as cells may not be directly under the module but in a generate
|
||||||
if (!modp->dead()) { // If was dead didn't increment user1's
|
if (!modp->dead()) { // If was dead didn't increment user1's
|
||||||
modp->foreach<AstCell>([](const AstCell* cellp) { //
|
modp->foreach([](const AstCell* cellp) { //
|
||||||
cellp->modp()->user1Inc(-1);
|
cellp->modp()->user1Inc(-1);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -91,7 +91,7 @@ class AstToDfgVisitor final : public VNVisitor {
|
|||||||
|
|
||||||
// METHODS
|
// METHODS
|
||||||
void markReferenced(AstNode* nodep) {
|
void markReferenced(AstNode* nodep) {
|
||||||
nodep->foreach<AstVarRef>([this](const AstVarRef* refp) {
|
nodep->foreach([this](const AstVarRef* refp) {
|
||||||
// No need to (and in fact cannot) mark variables with unsupported dtypes
|
// No need to (and in fact cannot) mark variables with unsupported dtypes
|
||||||
if (!DfgVertex::isSupportedDType(refp->varp()->dtypep())) return;
|
if (!DfgVertex::isSupportedDType(refp->varp()->dtypep())) return;
|
||||||
getNet(refp->varp())->setHasModRefs();
|
getNet(refp->varp())->setHasModRefs();
|
||||||
|
@ -425,7 +425,7 @@ class DfgToAstVisitor final : DfgVisitor {
|
|||||||
|
|
||||||
// Remap all references to point to the canonical variables, if one exists
|
// Remap all references to point to the canonical variables, if one exists
|
||||||
VNDeleter deleter;
|
VNDeleter deleter;
|
||||||
m_modp->foreach<AstVarRef>([&](AstVarRef* refp) {
|
m_modp->foreach([&](AstVarRef* refp) {
|
||||||
// Any variable that is written outside the DFG will have itself as the canonical
|
// Any variable that is written outside the DFG will have itself as the canonical
|
||||||
// var, so need not be replaced, furthermore, if a variable is traced, we don't
|
// var, so need not be replaced, furthermore, if a variable is traced, we don't
|
||||||
// want to update the write ref we just created above, so we only replace read only
|
// want to update the write ref we just created above, so we only replace read only
|
||||||
|
@ -249,7 +249,7 @@ void V3DfgOptimizer::optimize(AstNetlist* netlistp, const string& label) {
|
|||||||
const VNUser2InUse user2InUse;
|
const VNUser2InUse user2InUse;
|
||||||
|
|
||||||
// Mark cross-referenced variables
|
// Mark cross-referenced variables
|
||||||
netlistp->foreach<AstVarXRef>([](const AstVarXRef* xrefp) { xrefp->varp()->user2(true); });
|
netlistp->foreach([](const AstVarXRef* xrefp) { xrefp->varp()->user2(true); });
|
||||||
|
|
||||||
V3DfgOptimizationContext ctx{label};
|
V3DfgOptimizationContext ctx{label};
|
||||||
|
|
||||||
|
@ -786,8 +786,7 @@ void EmitCSyms::emitSymImp() {
|
|||||||
if (v3Global.opt.profPgo()) {
|
if (v3Global.opt.profPgo()) {
|
||||||
puts("// Configure profiling for PGO\n");
|
puts("// Configure profiling for PGO\n");
|
||||||
if (v3Global.opt.mtasks()) {
|
if (v3Global.opt.mtasks()) {
|
||||||
v3Global.rootp()->topModulep()->foreach<AstExecGraph>(
|
v3Global.rootp()->topModulep()->foreach([&](const AstExecGraph* execGraphp) {
|
||||||
[&](const AstExecGraph* execGraphp) {
|
|
||||||
for (const V3GraphVertex* vxp = execGraphp->depGraphp()->verticesBeginp(); vxp;
|
for (const V3GraphVertex* vxp = execGraphp->depGraphp()->verticesBeginp(); vxp;
|
||||||
vxp = vxp->verticesNextp()) {
|
vxp = vxp->verticesNextp()) {
|
||||||
const ExecMTask* const mtp = static_cast<const ExecMTask*>(vxp);
|
const ExecMTask* const mtp = static_cast<const ExecMTask*>(vxp);
|
||||||
|
@ -153,7 +153,7 @@ class ForceConvertVisitor final : public VNVisitor {
|
|||||||
// referenced AstVarScope with the given function.
|
// referenced AstVarScope with the given function.
|
||||||
void transformWritenVarScopes(AstNode* nodep, std::function<AstVarScope*(AstVarScope*)> f) {
|
void transformWritenVarScopes(AstNode* nodep, std::function<AstVarScope*(AstVarScope*)> f) {
|
||||||
UASSERT_OBJ(nodep->backp(), nodep, "Must have backp, otherwise will be lost if replaced");
|
UASSERT_OBJ(nodep->backp(), nodep, "Must have backp, otherwise will be lost if replaced");
|
||||||
nodep->foreach<AstNodeVarRef>([&f](AstNodeVarRef* refp) {
|
nodep->foreach([&f](AstNodeVarRef* refp) {
|
||||||
if (refp->access() != VAccess::WRITE) return;
|
if (refp->access() != VAccess::WRITE) return;
|
||||||
// TODO: this is not strictly speaking safe for some complicated lvalues, eg.:
|
// TODO: this is not strictly speaking safe for some complicated lvalues, eg.:
|
||||||
// 'force foo[a(cnt)] = 1;', where 'cnt' is an out parameter, but it will
|
// 'force foo[a(cnt)] = 1;', where 'cnt' is an out parameter, but it will
|
||||||
@ -230,7 +230,7 @@ class ForceConvertVisitor final : public VNVisitor {
|
|||||||
AstAssign* const resetRdp
|
AstAssign* const resetRdp
|
||||||
= new AstAssign{fl_nowarn, lhsp->cloneTree(false), lhsp->unlinkFrBack()};
|
= new AstAssign{fl_nowarn, lhsp->cloneTree(false), lhsp->unlinkFrBack()};
|
||||||
// Replace write refs on the LHS
|
// Replace write refs on the LHS
|
||||||
resetRdp->lhsp()->foreach<AstNodeVarRef>([this](AstNodeVarRef* refp) {
|
resetRdp->lhsp()->foreach([this](AstNodeVarRef* refp) {
|
||||||
if (refp->access() != VAccess::WRITE) return;
|
if (refp->access() != VAccess::WRITE) return;
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
AstVarScope* const newVscp
|
AstVarScope* const newVscp
|
||||||
@ -243,7 +243,7 @@ class ForceConvertVisitor final : public VNVisitor {
|
|||||||
VL_DO_DANGLING(refp->deleteTree(), refp);
|
VL_DO_DANGLING(refp->deleteTree(), refp);
|
||||||
});
|
});
|
||||||
// Replace write refs on RHS
|
// Replace write refs on RHS
|
||||||
resetRdp->rhsp()->foreach<AstNodeVarRef>([this](AstNodeVarRef* refp) {
|
resetRdp->rhsp()->foreach([this](AstNodeVarRef* refp) {
|
||||||
if (refp->access() != VAccess::WRITE) return;
|
if (refp->access() != VAccess::WRITE) return;
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
AstVarScope* const newVscp
|
AstVarScope* const newVscp
|
||||||
@ -273,7 +273,7 @@ class ForceConvertVisitor final : public VNVisitor {
|
|||||||
iterateAndNextNull(nodep->modulesp());
|
iterateAndNextNull(nodep->modulesp());
|
||||||
|
|
||||||
// Replace references to forced signals
|
// Replace references to forced signals
|
||||||
nodep->modulesp()->foreachAndNext<AstVarRef>([this](AstVarRef* nodep) {
|
nodep->modulesp()->foreachAndNext([this](AstVarRef* nodep) {
|
||||||
if (ForceComponentsVarScope* const fcp
|
if (ForceComponentsVarScope* const fcp
|
||||||
= m_forceComponentsVarScope.tryGet(nodep->varScopep())) {
|
= m_forceComponentsVarScope.tryGet(nodep->varScopep())) {
|
||||||
switch (nodep->access()) {
|
switch (nodep->access()) {
|
||||||
|
50
src/V3FunctionTraits.h
Normal file
50
src/V3FunctionTraits.h
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
||||||
|
//*************************************************************************
|
||||||
|
// DESCRIPTION: Verilator: Function traits for metaprogramming
|
||||||
|
//
|
||||||
|
// Code available from: https://verilator.org
|
||||||
|
//
|
||||||
|
//*************************************************************************
|
||||||
|
//
|
||||||
|
// Copyright 2003-2022 by Wilson Snyder. 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
|
||||||
|
//
|
||||||
|
//*************************************************************************
|
||||||
|
|
||||||
|
#ifndef VERILATOR_V3FUNCTIONTRAITS_H_
|
||||||
|
#define VERILATOR_V3FUNCTIONTRAITS_H_
|
||||||
|
|
||||||
|
#include "verilatedos.h"
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <tuple>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct FunctionTraits;
|
||||||
|
|
||||||
|
// For generic types, directly use the result of the signature of its 'operator()'
|
||||||
|
template <typename T>
|
||||||
|
struct FunctionTraits final
|
||||||
|
: public FunctionTraits<decltype(&std::remove_reference<T>::type::operator())> {};
|
||||||
|
|
||||||
|
// Specialization for pointers to member function
|
||||||
|
template <typename ClassType, typename ReturnType, typename... Args>
|
||||||
|
struct FunctionTraits<ReturnType (ClassType::*)(Args...) const> VL_NOT_FINAL {
|
||||||
|
// Number of arguments
|
||||||
|
static constexpr size_t arity = sizeof...(Args);
|
||||||
|
|
||||||
|
// Type of result
|
||||||
|
using result_type = ReturnType;
|
||||||
|
|
||||||
|
// Type of arguments
|
||||||
|
template <std::size_t I>
|
||||||
|
struct arg {
|
||||||
|
using type = typename std::tuple_element<I, std::tuple<Args...>>::type;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -1022,13 +1022,13 @@ static void eliminate(AstNode* logicp,
|
|||||||
nodep->replaceWith(newp);
|
nodep->replaceWith(newp);
|
||||||
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
VL_DO_DANGLING(nodep->deleteTree(), nodep);
|
||||||
// Recursively substitute the new tree
|
// Recursively substitute the new tree
|
||||||
newp->foreach<AstNodeVarRef>(visit);
|
newp->foreach(visit);
|
||||||
|
|
||||||
// Remove from recursion filter
|
// Remove from recursion filter
|
||||||
replaced.erase(vscp);
|
replaced.erase(vscp);
|
||||||
};
|
};
|
||||||
|
|
||||||
logicp->foreach<AstNodeVarRef>(visit);
|
logicp->foreach(visit);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ######################################################################
|
// ######################################################################
|
||||||
|
@ -513,7 +513,7 @@ private:
|
|||||||
newmodp = nodep->modp();
|
newmodp = nodep->modp();
|
||||||
}
|
}
|
||||||
// Find cell cross-references
|
// Find cell cross-references
|
||||||
nodep->modp()->foreach<AstCell>([](AstCell* cellp) {
|
nodep->modp()->foreach([](AstCell* cellp) {
|
||||||
// clonep is nullptr when inlining the last instance, if so the use original node
|
// clonep is nullptr when inlining the last instance, if so the use original node
|
||||||
cellp->user4p(cellp->clonep() ? cellp->clonep() : cellp);
|
cellp->user4p(cellp->clonep() ? cellp->clonep() : cellp);
|
||||||
});
|
});
|
||||||
|
@ -744,7 +744,7 @@ private:
|
|||||||
if (!m_mgFirstp) {
|
if (!m_mgFirstp) {
|
||||||
UASSERT_OBJ(condp, nodep, "Cannot start new list without condition");
|
UASSERT_OBJ(condp, nodep, "Cannot start new list without condition");
|
||||||
// Mark variable references in the condition
|
// Mark variable references in the condition
|
||||||
condp->foreach<AstVarRef>([](const AstVarRef* nodep) { nodep->varp()->user1(1); });
|
condp->foreach([](const AstVarRef* nodep) { nodep->varp()->user1(1); });
|
||||||
// Now check again if mergeable. We need this to pick up assignments to conditions,
|
// Now check again if mergeable. We need this to pick up assignments to conditions,
|
||||||
// e.g.: 'c = c ? a : b' at the beginning of the list, which is in fact not mergeable
|
// e.g.: 'c = c ? a : b' at the beginning of the list, which is in fact not mergeable
|
||||||
// because it updates the condition. We simply bail on these.
|
// because it updates the condition. We simply bail on these.
|
||||||
|
@ -230,7 +230,7 @@ class OrderBuildVisitor final : public VNVisitor {
|
|||||||
m_hybridp = nodep->sensesp();
|
m_hybridp = nodep->sensesp();
|
||||||
// Mark AstVarScopes that are explicit sensitivities
|
// Mark AstVarScopes that are explicit sensitivities
|
||||||
AstNode::user3ClearTree();
|
AstNode::user3ClearTree();
|
||||||
senTreep->foreach<AstVarRef>([](const AstVarRef* refp) { //
|
senTreep->foreach([](const AstVarRef* refp) { //
|
||||||
refp->varScopep()->user3(true);
|
refp->varScopep()->user3(true);
|
||||||
});
|
});
|
||||||
m_readTriggersCombLogic = [](const AstVarScope* vscp) { return !vscp->user3(); };
|
m_readTriggersCombLogic = [](const AstVarScope* vscp) { return !vscp->user3(); };
|
||||||
|
@ -3215,7 +3215,7 @@ static void implementExecGraph(AstExecGraph* const execGraphp) {
|
|||||||
|
|
||||||
void V3Partition::finalize(AstNetlist* netlistp) {
|
void V3Partition::finalize(AstNetlist* netlistp) {
|
||||||
// Called by Verilator top stage
|
// Called by Verilator top stage
|
||||||
netlistp->topModulep()->foreach<AstExecGraph>([&](AstExecGraph* execGraphp) {
|
netlistp->topModulep()->foreach([&](AstExecGraph* execGraphp) {
|
||||||
// Back in V3Order, we partitioned mtasks using provisional cost
|
// Back in V3Order, we partitioned mtasks using provisional cost
|
||||||
// estimates. However, V3Order precedes some optimizations (notably
|
// estimates. However, V3Order precedes some optimizations (notably
|
||||||
// V3LifePost) that can change the cost of logic within each mtask.
|
// V3LifePost) that can change the cost of logic within each mtask.
|
||||||
|
@ -186,10 +186,10 @@ private:
|
|||||||
bool noopt = false;
|
bool noopt = false;
|
||||||
{
|
{
|
||||||
const VNUser3InUse user3InUse;
|
const VNUser3InUse user3InUse;
|
||||||
nodep->lhsp()->foreach<AstVarRef>([](const AstVarRef* refp) {
|
nodep->lhsp()->foreach([](const AstVarRef* refp) {
|
||||||
if (refp->access().isWriteOrRW()) refp->varp()->user3(true);
|
if (refp->access().isWriteOrRW()) refp->varp()->user3(true);
|
||||||
});
|
});
|
||||||
nodep->rhsp()->foreach<AstVarRef>([&noopt](const AstVarRef* refp) {
|
nodep->rhsp()->foreach([&noopt](const AstVarRef* refp) {
|
||||||
if (refp->access().isReadOnly() && refp->varp()->user3()) noopt = true;
|
if (refp->access().isReadOnly() && refp->varp()->user3()) noopt = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -161,10 +161,10 @@ void splitCheck(AstCFunc* ofuncp) {
|
|||||||
LogicClasses gatherLogicClasses(AstNetlist* netlistp) {
|
LogicClasses gatherLogicClasses(AstNetlist* netlistp) {
|
||||||
LogicClasses result;
|
LogicClasses result;
|
||||||
|
|
||||||
netlistp->foreach<AstScope>([&](AstScope* scopep) {
|
netlistp->foreach([&](AstScope* scopep) {
|
||||||
std::vector<AstActive*> empty;
|
std::vector<AstActive*> empty;
|
||||||
|
|
||||||
scopep->foreach<AstActive>([&](AstActive* activep) {
|
scopep->foreach([&](AstActive* activep) {
|
||||||
AstSenTree* const senTreep = activep->sensesp();
|
AstSenTree* const senTreep = activep->sensesp();
|
||||||
if (!activep->stmtsp()) {
|
if (!activep->stmtsp()) {
|
||||||
// Some AstActives might be empty due to previous optimizations
|
// Some AstActives might be empty due to previous optimizations
|
||||||
@ -671,7 +671,7 @@ AstNode* createInputCombLoop(AstNetlist* netlistp, SenExprBuilder& senExprBuilde
|
|||||||
// so we can make them sc_sensitive
|
// so we can make them sc_sensitive
|
||||||
if (v3Global.opt.systemC()) {
|
if (v3Global.opt.systemC()) {
|
||||||
logic.foreachLogic([](AstNode* logicp) {
|
logic.foreachLogic([](AstNode* logicp) {
|
||||||
logicp->foreach<AstVarRef>([](AstVarRef* refp) {
|
logicp->foreach([](AstVarRef* refp) {
|
||||||
if (refp->access().isWriteOnly()) return;
|
if (refp->access().isWriteOnly()) return;
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
if (vscp->scopep()->isTop() && vscp->varp()->isNonOutput()) {
|
if (vscp->scopep()->isTop() && vscp->varp()->isNonOutput()) {
|
||||||
@ -769,13 +769,13 @@ void createEval(AstNetlist* netlistp, //
|
|||||||
AstCFunc* const nbaDumpp = actTrig.m_dumpp->cloneTree(false);
|
AstCFunc* const nbaDumpp = actTrig.m_dumpp->cloneTree(false);
|
||||||
actTrig.m_dumpp->addNextHere(nbaDumpp);
|
actTrig.m_dumpp->addNextHere(nbaDumpp);
|
||||||
nbaDumpp->name("_dump_triggers__nba");
|
nbaDumpp->name("_dump_triggers__nba");
|
||||||
nbaDumpp->foreach<AstVarRef>([&](AstVarRef* refp) {
|
nbaDumpp->foreach([&](AstVarRef* refp) {
|
||||||
UASSERT_OBJ(refp->access().isReadOnly(), refp, "Should only read state");
|
UASSERT_OBJ(refp->access().isReadOnly(), refp, "Should only read state");
|
||||||
if (refp->varScopep() == actTrig.m_vscp) {
|
if (refp->varScopep() == actTrig.m_vscp) {
|
||||||
refp->replaceWith(new AstVarRef{refp->fileline(), nbaTrigsp, VAccess::READ});
|
refp->replaceWith(new AstVarRef{refp->fileline(), nbaTrigsp, VAccess::READ});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
nbaDumpp->foreach<AstText>([&](AstText* textp) { //
|
nbaDumpp->foreach([&](AstText* textp) { //
|
||||||
textp->text(VString::replaceWord(textp->text(), "act", "nba"));
|
textp->text(VString::replaceWord(textp->text(), "act", "nba"));
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -970,7 +970,7 @@ void schedule(AstNetlist* netlistp) {
|
|||||||
// Replace references in each mapped value with a reference to the given vscp
|
// Replace references in each mapped value with a reference to the given vscp
|
||||||
for (auto& pair : newMap) {
|
for (auto& pair : newMap) {
|
||||||
pair.second = pair.second->cloneTree(false);
|
pair.second = pair.second->cloneTree(false);
|
||||||
pair.second->foreach<AstVarRef>([&](AstVarRef* refp) {
|
pair.second->foreach([&](AstVarRef* refp) {
|
||||||
UASSERT_OBJ(refp->varScopep() == actTrigVscp, refp, "Unexpected reference");
|
UASSERT_OBJ(refp->varScopep() == actTrigVscp, refp, "Unexpected reference");
|
||||||
UASSERT_OBJ(refp->access() == VAccess::READ, refp, "Should be read ref");
|
UASSERT_OBJ(refp->access() == VAccess::READ, refp, "Should be read ref");
|
||||||
refp->replaceWith(new AstVarRef{refp->fileline(), vscp, VAccess::READ});
|
refp->replaceWith(new AstVarRef{refp->fileline(), vscp, VAccess::READ});
|
||||||
|
@ -140,7 +140,7 @@ std::unique_ptr<Graph> buildGraph(const LogicByScope& lbs) {
|
|||||||
const VNUser2InUse user2InUse;
|
const VNUser2InUse user2InUse;
|
||||||
const VNUser3InUse user3InUse;
|
const VNUser3InUse user3InUse;
|
||||||
|
|
||||||
nodep->foreach<AstVarRef>([&](AstVarRef* refp) {
|
nodep->foreach([&](AstVarRef* refp) {
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
VarVertex* const vvtxp = getVarVertex(vscp);
|
VarVertex* const vvtxp = getVarVertex(vscp);
|
||||||
// We want to cut the narrowest signals
|
// We want to cut the narrowest signals
|
||||||
|
@ -164,7 +164,7 @@ class SchedGraphBuilder final : public VNVisitor {
|
|||||||
SchedSenVertex* const vtxp = new SchedSenVertex{m_graphp, senItemp};
|
SchedSenVertex* const vtxp = new SchedSenVertex{m_graphp, senItemp};
|
||||||
|
|
||||||
// Connect up the variable references
|
// Connect up the variable references
|
||||||
senItemp->sensp()->foreach<AstVarRef>([&](AstVarRef* refp) {
|
senItemp->sensp()->foreach([&](AstVarRef* refp) {
|
||||||
new V3GraphEdge{m_graphp, getVarVertex(refp->varScopep()), vtxp, 1};
|
new V3GraphEdge{m_graphp, getVarVertex(refp->varScopep()), vtxp, 1};
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -186,7 +186,7 @@ class SchedGraphBuilder final : public VNVisitor {
|
|||||||
|
|
||||||
// Clocked or hybrid logic has explicit sensitivity, so add edge from sensitivity vertex
|
// Clocked or hybrid logic has explicit sensitivity, so add edge from sensitivity vertex
|
||||||
if (!m_senTreep->hasCombo()) {
|
if (!m_senTreep->hasCombo()) {
|
||||||
m_senTreep->foreach<AstSenItem>([=](AstSenItem* senItemp) {
|
m_senTreep->foreach([=](AstSenItem* senItemp) {
|
||||||
if (senItemp->isIllegal()) return;
|
if (senItemp->isIllegal()) return;
|
||||||
UASSERT_OBJ(senItemp->isClocked() || senItemp->isHybrid(), nodep,
|
UASSERT_OBJ(senItemp->isClocked() || senItemp->isHybrid(), nodep,
|
||||||
"Non-clocked SenItem under clocked SenTree");
|
"Non-clocked SenItem under clocked SenTree");
|
||||||
@ -196,7 +196,7 @@ class SchedGraphBuilder final : public VNVisitor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Add edges based on references
|
// Add edges based on references
|
||||||
nodep->foreach<AstVarRef>([=](const AstVarRef* vrefp) {
|
nodep->foreach([=](const AstVarRef* vrefp) {
|
||||||
AstVarScope* const vscp = vrefp->varScopep();
|
AstVarScope* const vscp = vrefp->varScopep();
|
||||||
if (vrefp->access().isReadOrRW() && m_readTriggersThisLogic(vscp)) {
|
if (vrefp->access().isReadOrRW() && m_readTriggersThisLogic(vscp)) {
|
||||||
new V3GraphEdge{m_graphp, getVarVertex(vscp), logicVtxp, 10};
|
new V3GraphEdge{m_graphp, getVarVertex(vscp), logicVtxp, 10};
|
||||||
@ -208,7 +208,7 @@ class SchedGraphBuilder final : public VNVisitor {
|
|||||||
|
|
||||||
// If the logic calls a 'context' DPI import, it might fire the DPI Export trigger
|
// If the logic calls a 'context' DPI import, it might fire the DPI Export trigger
|
||||||
if (m_dpiExportTriggerp) {
|
if (m_dpiExportTriggerp) {
|
||||||
nodep->foreach<AstCCall>([=](const AstCCall* callp) {
|
nodep->foreach([=](const AstCCall* callp) {
|
||||||
if (!callp->funcp()->dpiImportWrapper()) return;
|
if (!callp->funcp()->dpiImportWrapper()) return;
|
||||||
if (!callp->funcp()->dpiContext()) return;
|
if (!callp->funcp()->dpiContext()) return;
|
||||||
new V3GraphEdge{m_graphp, logicVtxp, getVarVertex(m_dpiExportTriggerp), 10};
|
new V3GraphEdge{m_graphp, logicVtxp, getVarVertex(m_dpiExportTriggerp), 10};
|
||||||
@ -226,7 +226,7 @@ class SchedGraphBuilder final : public VNVisitor {
|
|||||||
// Mark explicit sensitivities as not triggering these blocks
|
// Mark explicit sensitivities as not triggering these blocks
|
||||||
if (senTreep->hasHybrid()) {
|
if (senTreep->hasHybrid()) {
|
||||||
AstNode::user2ClearTree();
|
AstNode::user2ClearTree();
|
||||||
senTreep->foreach<AstVarRef>([](const AstVarRef* refp) { //
|
senTreep->foreach([](const AstVarRef* refp) { //
|
||||||
refp->varScopep()->user2(true);
|
refp->varScopep()->user2(true);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -364,7 +364,7 @@ LogicRegions partition(LogicByScope& clockedLogic, LogicByScope& combinationalLo
|
|||||||
const VNUser2InUse user2InUse; // AstVarScope::user2() -> bool: writen in Active region
|
const VNUser2InUse user2InUse; // AstVarScope::user2() -> bool: writen in Active region
|
||||||
|
|
||||||
const auto markVars = [](AstNode* nodep) {
|
const auto markVars = [](AstNode* nodep) {
|
||||||
nodep->foreach<AstNodeVarRef>([](const AstNodeVarRef* vrefp) {
|
nodep->foreach([](const AstNodeVarRef* vrefp) {
|
||||||
AstVarScope* const vscp = vrefp->varScopep();
|
AstVarScope* const vscp = vrefp->varScopep();
|
||||||
if (vrefp->access().isReadOrRW()) vscp->user1(true);
|
if (vrefp->access().isReadOrRW()) vscp->user1(true);
|
||||||
if (vrefp->access().isWriteOrRW()) vscp->user2(true);
|
if (vrefp->access().isWriteOrRW()) vscp->user2(true);
|
||||||
@ -386,7 +386,7 @@ LogicRegions partition(LogicByScope& clockedLogic, LogicByScope& combinationalLo
|
|||||||
nextp = nodep->nextp();
|
nextp = nodep->nextp();
|
||||||
if (AstAssignPre* const logicp = VN_CAST(nodep, AssignPre)) {
|
if (AstAssignPre* const logicp = VN_CAST(nodep, AssignPre)) {
|
||||||
bool toActiveRegion = false;
|
bool toActiveRegion = false;
|
||||||
logicp->foreach<AstNodeVarRef>([&](const AstNodeVarRef* vrefp) {
|
logicp->foreach([&](const AstNodeVarRef* vrefp) {
|
||||||
AstVarScope* const vscp = vrefp->varScopep();
|
AstVarScope* const vscp = vrefp->varScopep();
|
||||||
if (vrefp->access().isReadOnly()) {
|
if (vrefp->access().isReadOnly()) {
|
||||||
// Variable only read in Pre, and is written in active region
|
// Variable only read in Pre, and is written in active region
|
||||||
|
@ -176,7 +176,7 @@ std::unique_ptr<Graph> buildGraph(const LogicRegions& logicRegions) {
|
|||||||
// Hybrid logic is triggered by all reads, except for reads of the explicit
|
// Hybrid logic is triggered by all reads, except for reads of the explicit
|
||||||
// sensitivities
|
// sensitivities
|
||||||
readTriggersThisLogic = [](AstVarScope* vscp) { return !vscp->user4(); };
|
readTriggersThisLogic = [](AstVarScope* vscp) { return !vscp->user4(); };
|
||||||
senTreep->foreach<AstVarRef>([](const AstVarRef* refp) { //
|
senTreep->foreach([](const AstVarRef* refp) { //
|
||||||
refp->varScopep()->user4(true);
|
refp->varScopep()->user4(true);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -187,7 +187,7 @@ std::unique_ptr<Graph> buildGraph(const LogicRegions& logicRegions) {
|
|||||||
const VNUser2InUse user2InUse;
|
const VNUser2InUse user2InUse;
|
||||||
const VNUser3InUse user3InUse;
|
const VNUser3InUse user3InUse;
|
||||||
|
|
||||||
nodep->foreach<AstVarRef>([&](AstVarRef* refp) {
|
nodep->foreach([&](AstVarRef* refp) {
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
VarVertex* const vvtxp = getVarVertex(vscp);
|
VarVertex* const vvtxp = getVarVertex(vscp);
|
||||||
|
|
||||||
|
@ -256,7 +256,7 @@ void transformForks(AstNetlist* const netlistp) {
|
|||||||
// flow analysis framework which we don't have at the moment
|
// flow analysis framework which we don't have at the moment
|
||||||
void remapLocals(AstCFunc* const funcp, AstCCall* const callp) {
|
void remapLocals(AstCFunc* const funcp, AstCCall* const callp) {
|
||||||
const VNUser2InUse user2InUse; // AstVarScope -> AstVarScope: var to remap to
|
const VNUser2InUse user2InUse; // AstVarScope -> AstVarScope: var to remap to
|
||||||
funcp->foreach<AstNodeVarRef>([&](AstNodeVarRef* refp) {
|
funcp->foreach([&](AstNodeVarRef* refp) {
|
||||||
AstVar* const varp = refp->varp();
|
AstVar* const varp = refp->varp();
|
||||||
AstBasicDType* const dtypep = varp->dtypep()->basicp();
|
AstBasicDType* const dtypep = varp->dtypep()->basicp();
|
||||||
// If it a fork sync or an intra-assignment variable, pass it by value
|
// If it a fork sync or an intra-assignment variable, pass it by value
|
||||||
|
@ -56,7 +56,7 @@ class SenExprBuilder final {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static bool isSimpleExpr(const AstNode* const exprp) {
|
static bool isSimpleExpr(const AstNode* const exprp) {
|
||||||
return exprp->forall<AstNode>([](const AstNode* const nodep) {
|
return exprp->forall([](const AstNode* const nodep) {
|
||||||
return VN_IS(nodep, Const) || VN_IS(nodep, NodeVarRef) || VN_IS(nodep, Sel)
|
return VN_IS(nodep, Const) || VN_IS(nodep, NodeVarRef) || VN_IS(nodep, Sel)
|
||||||
|| VN_IS(nodep, NodeSel) || VN_IS(nodep, MemberSel)
|
|| VN_IS(nodep, NodeSel) || VN_IS(nodep, MemberSel)
|
||||||
|| VN_IS(nodep, CMethodHard);
|
|| VN_IS(nodep, CMethodHard);
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
#ifndef VERILATOR_V3STDFUTURE_H_
|
#ifndef VERILATOR_V3STDFUTURE_H_
|
||||||
#define VERILATOR_V3STDFUTURE_H_
|
#define VERILATOR_V3STDFUTURE_H_
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
namespace vlstd {
|
namespace vlstd {
|
||||||
|
|
||||||
// constexpr std::max with arguments passed by value (required by constexpr before C++14)
|
// constexpr std::max with arguments passed by value (required by constexpr before C++14)
|
||||||
@ -25,6 +27,17 @@ constexpr T max(T a, T b) {
|
|||||||
return a > b ? a : b;
|
return a > b ? a : b;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// C++17 is_invocable
|
||||||
|
template <typename F, typename... Args>
|
||||||
|
struct is_invocable
|
||||||
|
: std::is_constructible<std::function<void(Args...)>,
|
||||||
|
std::reference_wrapper<typename std::remove_reference<F>::type>> {};
|
||||||
|
|
||||||
|
// C++17 is_invocable_r
|
||||||
|
template <typename R, typename F, typename... Args>
|
||||||
|
struct is_invocable_r
|
||||||
|
: std::is_constructible<std::function<R(Args...)>,
|
||||||
|
std::reference_wrapper<typename std::remove_reference<F>::type>> {};
|
||||||
}; // namespace vlstd
|
}; // namespace vlstd
|
||||||
|
|
||||||
#endif // Guard
|
#endif // Guard
|
||||||
|
@ -408,7 +408,7 @@ private:
|
|||||||
|
|
||||||
// Replace varrefs with new var pointer
|
// Replace varrefs with new var pointer
|
||||||
void relink(AstNode* nodep) {
|
void relink(AstNode* nodep) {
|
||||||
nodep->foreachAndNext<AstVarRef>([](AstVarRef* refp) {
|
nodep->foreachAndNext([](AstVarRef* refp) {
|
||||||
if (refp->varp()->user2p()) { // It's being converted to an alias.
|
if (refp->varp()->user2p()) { // It's being converted to an alias.
|
||||||
AstVarScope* const newvscp = VN_AS(refp->varp()->user2p(), VarScope);
|
AstVarScope* const newvscp = VN_AS(refp->varp()->user2p(), VarScope);
|
||||||
refp->varScopep(newvscp);
|
refp->varScopep(newvscp);
|
||||||
@ -1276,7 +1276,7 @@ private:
|
|||||||
|
|
||||||
// Mark non-local variables written by the exported function
|
// Mark non-local variables written by the exported function
|
||||||
bool writesNonLocals = false;
|
bool writesNonLocals = false;
|
||||||
cfuncp->foreach<AstVarRef>([&writesNonLocals](AstVarRef* refp) {
|
cfuncp->foreach([&writesNonLocals](AstVarRef* refp) {
|
||||||
if (refp->access().isReadOnly()) return; // Ignore read reference
|
if (refp->access().isReadOnly()) return; // Ignore read reference
|
||||||
AstVar* const varp = refp->varScopep()->varp();
|
AstVar* const varp = refp->varScopep()->varp();
|
||||||
// We are ignoring function locals as they should not be referenced anywhere
|
// We are ignoring function locals as they should not be referenced anywhere
|
||||||
|
@ -138,7 +138,7 @@ private:
|
|||||||
AstDelay* getLhsNetDelay(AstNodeAssign* nodep) const {
|
AstDelay* getLhsNetDelay(AstNodeAssign* nodep) const {
|
||||||
bool foundWrite = false;
|
bool foundWrite = false;
|
||||||
AstDelay* delayp = nullptr;
|
AstDelay* delayp = nullptr;
|
||||||
nodep->lhsp()->foreach<AstNodeVarRef>([&](const AstNodeVarRef* const refp) {
|
nodep->lhsp()->foreach([&](const AstNodeVarRef* const refp) {
|
||||||
if (!refp->access().isWriteOrRW()) return;
|
if (!refp->access().isWriteOrRW()) return;
|
||||||
UASSERT_OBJ(!foundWrite, nodep, "Should only be one variable written to on the LHS");
|
UASSERT_OBJ(!foundWrite, nodep, "Should only be one variable written to on the LHS");
|
||||||
foundWrite = true;
|
foundWrite = true;
|
||||||
@ -190,7 +190,7 @@ private:
|
|||||||
AstSenItem* varRefpsToSenItemsp(AstNode* const nodep) const {
|
AstSenItem* varRefpsToSenItemsp(AstNode* const nodep) const {
|
||||||
AstNode* senItemsp = nullptr;
|
AstNode* senItemsp = nullptr;
|
||||||
const VNUser4InUse user4InUse;
|
const VNUser4InUse user4InUse;
|
||||||
nodep->foreach<AstNodeVarRef>([&](AstNodeVarRef* refp) {
|
nodep->foreach([&](AstNodeVarRef* refp) {
|
||||||
if (refp->access().isWriteOnly()) return;
|
if (refp->access().isWriteOnly()) return;
|
||||||
AstVarScope* const vscp = refp->varScopep();
|
AstVarScope* const vscp = refp->varScopep();
|
||||||
if (vscp->user4SetOnce()) return;
|
if (vscp->user4SetOnce()) return;
|
||||||
@ -530,12 +530,12 @@ private:
|
|||||||
// we want as only the top level selects are LValues. As an example,
|
// we want as only the top level selects are LValues. As an example,
|
||||||
// this transforms 'x[a[i]][b[j]] = y'
|
// this transforms 'x[a[i]][b[j]] = y'
|
||||||
// into 't1 = a[i]; t0 = b[j]; x[t1][t0] = y'.
|
// into 't1 = a[i]; t0 = b[j]; x[t1][t0] = y'.
|
||||||
nodep->lhsp()->foreach<AstSel>([&](AstSel* selp) {
|
nodep->lhsp()->foreach([&](AstSel* selp) {
|
||||||
if (VN_IS(selp->lsbp(), Const)) return;
|
if (VN_IS(selp->lsbp(), Const)) return;
|
||||||
replaceWithIntermediate(selp->lsbp(), m_intraLsbNames.get(nodep));
|
replaceWithIntermediate(selp->lsbp(), m_intraLsbNames.get(nodep));
|
||||||
// widthp should be const
|
// widthp should be const
|
||||||
});
|
});
|
||||||
nodep->lhsp()->foreach<AstNodeSel>([&](AstNodeSel* selp) {
|
nodep->lhsp()->foreach([&](AstNodeSel* selp) {
|
||||||
if (VN_IS(selp->bitp(), Const)) return;
|
if (VN_IS(selp->bitp(), Const)) return;
|
||||||
replaceWithIntermediate(selp->bitp(), m_intraIndexNames.get(nodep));
|
replaceWithIntermediate(selp->bitp(), m_intraIndexNames.get(nodep));
|
||||||
});
|
});
|
||||||
|
@ -434,7 +434,7 @@ private:
|
|||||||
} else if (AstCFunc* const funcp = VN_CAST(insertp, CFunc)) {
|
} else if (AstCFunc* const funcp = VN_CAST(insertp, CFunc)) {
|
||||||
// If there are awaits, insert the setter after each await
|
// If there are awaits, insert the setter after each await
|
||||||
if (funcp->isCoroutine() && funcp->stmtsp()) {
|
if (funcp->isCoroutine() && funcp->stmtsp()) {
|
||||||
funcp->stmtsp()->foreachAndNext<AstCAwait>([&](AstCAwait* awaitp) {
|
funcp->stmtsp()->foreachAndNext([&](AstCAwait* awaitp) {
|
||||||
if (awaitp->nextp()) awaitp->addNextHere(setterp->cloneTree(false));
|
if (awaitp->nextp()) awaitp->addNextHere(setterp->cloneTree(false));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user