diff --git a/Changes b/Changes index 2ed4937f2..83bfa4136 100644 --- a/Changes +++ b/Changes @@ -27,6 +27,8 @@ The contributors that suggested a given feature are shown in []. Thanks! **** Fix genblk naming to match IEEE (#2686). [tinshark] +**** Fix vpi_release_handle to be called implicitly per IEEE (#2706). + * Verilator 4.106 2020-12-02 diff --git a/include/verilated_vpi.cpp b/include/verilated_vpi.cpp index 522afd094..3ea2afa39 100644 --- a/include/verilated_vpi.cpp +++ b/include/verilated_vpi.cpp @@ -93,8 +93,12 @@ public: "vpi_release_handle() called on same object twice, or on non-Verilator " "VPI object"); } +#ifdef VL_VPI_IMMEDIATE_FREE // Define to aid in finding leaky handles + ::operator delete(oldp); +#else *(reinterpret_cast(oldp)) = t_freeHead; t_freeHead = oldp; +#endif } // MEMBERS static VerilatedVpio* castp(vpiHandle h) { @@ -109,32 +113,44 @@ public: virtual vluint32_t size() const { return 0; } virtual const VerilatedRange* rangep() const { return nullptr; } virtual vpiHandle dovpi_scan() { return nullptr; } + virtual PLI_INT32 dovpi_remove_cb() { return 0; } }; -typedef PLI_INT32 (*VerilatedPliCb)(struct t_cb_data*); - -class VerilatedVpioCb final : public VerilatedVpio { - t_cb_data m_cbData; - s_vpi_value m_value; +class VerilatedVpioTimedCb final : public VerilatedVpio { + // A handle to a timed callback created with vpi_register_cb + // User can call vpi_remove_cb or vpi_release_handle on it + vluint64_t m_id; // Unique id/sequence number to find schedule's event QData m_time; public: - // cppcheck-suppress uninitVar // m_value - VerilatedVpioCb(const t_cb_data* cbDatap, QData time) - : m_cbData(*cbDatap) - , m_time(time) { // Need () or GCC 4.8 false warning - m_value.format = cbDatap->value ? cbDatap->value->format : vpiSuppressVal; - m_cbData.value = &m_value; - } - virtual ~VerilatedVpioCb() override = default; - static VerilatedVpioCb* castp(vpiHandle h) { - return dynamic_cast(reinterpret_cast(h)); + VerilatedVpioTimedCb(vluint64_t id, QData time) + : m_id(id) + , m_time{time} {} + virtual ~VerilatedVpioTimedCb() override = default; + static VerilatedVpioTimedCb* castp(vpiHandle h) { + return dynamic_cast(reinterpret_cast(h)); } virtual vluint32_t type() const override { return vpiCallback; } - vluint32_t reason() const { return m_cbData.reason; } - VerilatedPliCb cb_rtnp() const { return m_cbData.cb_rtn; } - t_cb_data* cb_datap() { return &(m_cbData); } - QData time() const { return m_time; } + virtual PLI_INT32 dovpi_remove_cb() override; +}; + +class VerilatedVpioReasonCb final : public VerilatedVpio { + // A handle to a non-timed callback created with vpi_register_cb + // User can call vpi_remove_cb or vpi_release_handle on it + vluint64_t m_id; // Unique id/sequence number to find schedule's event + PLI_INT32 m_reason; // VPI callback reason code + +public: + // cppcheck-suppress uninitVar // m_value + VerilatedVpioReasonCb(vluint64_t id, PLI_INT32 reason) + : m_id(id) + , m_reason{reason} {} + virtual ~VerilatedVpioReasonCb() override = default; + static VerilatedVpioReasonCb* castp(vpiHandle h) { + return dynamic_cast(reinterpret_cast(h)); + } + virtual vluint32_t type() const override { return vpiCallback; } + virtual PLI_INT32 dovpi_remove_cb() override; }; class VerilatedVpioConst final : public VerilatedVpio { @@ -159,7 +175,6 @@ public: VerilatedVpioParam(const VerilatedVar* varp, const VerilatedScope* scopep) : m_varp{varp} , m_scopep{scopep} {} - virtual ~VerilatedVpioParam() override = default; static VerilatedVpioParam* castp(vpiHandle h) { @@ -206,7 +221,10 @@ public: } virtual vluint32_t type() const override { return vpiIterator; } virtual vpiHandle dovpi_scan() override { - if (m_done) return nullptr; + if (VL_UNLIKELY(m_done)) { + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + return nullptr; + } m_done = true; return ((new VerilatedVpioRange(m_range))->castVpiHandle()); } @@ -230,16 +248,16 @@ public: }; class VerilatedVpioVar VL_NOT_FINAL : public VerilatedVpio { - const VerilatedVar* m_varp; - const VerilatedScope* m_scopep; + const VerilatedVar* m_varp = nullptr; + const VerilatedScope* m_scopep = nullptr; vluint8_t* m_prevDatap = nullptr; // Previous value of data, for cbValueChange union { vluint8_t u8[4]; vluint32_t u32; } m_mask; // memoized variable mask - vluint32_t m_entSize; // memoized variable size + vluint32_t m_entSize = 0; // memoized variable size protected: - void* m_varDatap; // varp()->datap() adjusted for array entries + void* m_varDatap = nullptr; // varp()->datap() adjusted for array entries vlsint32_t m_index = 0; const VerilatedRange& get_range() const { // Determine number of dimensions and return outermost @@ -254,6 +272,19 @@ public: m_entSize = varp->entSize(); m_varDatap = varp->datap(); } + VerilatedVpioVar(const VerilatedVpioVar* varp) { + if (varp) { + m_varp = varp->m_varp; + m_scopep = varp->m_scopep; + m_mask.u32 = varp->m_mask.u32; + m_entSize = varp->m_entSize; + m_varDatap = varp->m_varDatap; + m_index = varp->m_index; + // Not copying m_prevDatap, must be nullptr + } else { + m_mask.u32 = 0; + } + } virtual ~VerilatedVpioVar() override { if (m_prevDatap) VL_DO_CLEAR(delete[] m_prevDatap, m_prevDatap = nullptr); } @@ -331,13 +362,18 @@ public: m_it = varsp->begin(); m_started = true; } else if (VL_UNLIKELY(m_it == varsp->end())) { + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle return nullptr; } else { ++m_it; } - if (m_it == varsp->end()) return nullptr; + if (VL_UNLIKELY(m_it == varsp->end())) { + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + return nullptr; + } return ((new VerilatedVpioVar(&(m_it->second), m_scopep))->castVpiHandle()); } + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle return nullptr; // End of list - only one deep } }; @@ -364,9 +400,11 @@ public: if (!(m_done = (m_iteration == m_varp->unpacked().left()))) m_iteration += m_direction; } virtual vpiHandle dovpi_scan() override { - vpiHandle result; - if (m_done) return nullptr; - result = vpi_handle_by_index(m_handle, m_iteration); + if (VL_UNLIKELY(m_done)) { + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + return nullptr; + } + vpiHandle result = vpi_handle_by_index(m_handle, m_iteration); iterationInc(); return result; } @@ -406,7 +444,10 @@ public: } virtual vluint32_t type() const override { return vpiIterator; } virtual vpiHandle dovpi_scan() override { - if (m_it == m_vec->end()) return nullptr; + if (m_it == m_vec->end()) { + delete this; // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + return nullptr; + } const VerilatedScope* modp = *m_it++; return (new VerilatedVpioModule(modp))->castVpiHandle(); } @@ -414,10 +455,42 @@ public: //====================================================================== +typedef PLI_INT32 (*VerilatedPliCb)(struct t_cb_data*); + +class VerilatedVpiCbHolder final { + // Holds information needed to call a callback + vluint64_t m_id; + s_cb_data m_cbData; + s_vpi_value m_value; + VerilatedVpioVar m_varo; // If a cbValueChange callback, the object we will return + +public: + // cppcheck-suppress uninitVar // m_value + VerilatedVpiCbHolder(vluint64_t id, const s_cb_data* cbDatap, const VerilatedVpioVar* varop) + : m_id(id) + , m_cbData(*cbDatap) + , m_varo(varop) { + m_value.format = cbDatap->value ? cbDatap->value->format : vpiSuppressVal; + m_cbData.value = &m_value; + if (varop) { + m_cbData.obj = m_varo.castVpiHandle(); + m_varo.createPrevDatap(); + } else { + m_cbData.obj = NULL; + } + } + ~VerilatedVpiCbHolder() = default; + VerilatedPliCb cb_rtnp() const { return m_cbData.cb_rtn; } + s_cb_data* cb_datap() { return &m_cbData; } + vluint64_t id() const { return m_id; } + bool invalid() const { return !m_id; } + void invalidate() { m_id = 0; } +}; + struct VerilatedVpiTimedCbsCmp { - /// Ordering sets keyed by time, then callback descriptor - bool operator()(const std::pair& a, - const std::pair& b) const { + /// Ordering sets keyed by time, then callback unique id + bool operator()(const std::pair& a, + const std::pair& b) const { if (a.first < b.first) return true; if (a.first > b.first) return false; return a.second < b.second; @@ -428,55 +501,70 @@ class VerilatedVpiError; class VerilatedVpiImp final { enum { CB_ENUM_MAX_VALUE = cbAtEndOfSimTime + 1 }; // Maxium callback reason - typedef std::list VpioCbList; - typedef std::set, VerilatedVpiTimedCbsCmp> VpioTimedCbs; + typedef std::list VpioCbList; + typedef std::map, VerilatedVpiCbHolder> VpioTimedCbs; VpioCbList m_cbObjLists[CB_ENUM_MAX_VALUE]; // Callbacks for each supported reason VpioTimedCbs m_timedCbs; // Time based callbacks VerilatedVpiError* m_errorInfop = nullptr; // Container for vpi error info VerilatedAssertOneThread m_assertOne; ///< Assert only called from single thread + vluint64_t m_nextCallbackId = 1; // Id to identify callback static VerilatedVpiImp s_s; // Singleton public: - VerilatedVpiImp() = default; - ~VerilatedVpiImp() = default; static void assertOneCheck() { s_s.m_assertOne.check(); } - static void cbReasonAdd(VerilatedVpioCb* vop) { - if (vop->reason() == cbValueChange) { - if (VerilatedVpioVar* varop = VerilatedVpioVar::castp(vop->cb_datap()->obj)) { - varop->createPrevDatap(); - } - } - if (VL_UNCOVERABLE(vop->reason() >= CB_ENUM_MAX_VALUE)) { + static vluint64_t nextCallbackId() { return ++s_s.m_nextCallbackId; } + + static void cbReasonAdd(vluint64_t id, const s_cb_data* cb_data_p) { + // The passed cb_data_p was property of the user, so need to recreate + if (VL_UNCOVERABLE(cb_data_p->reason >= CB_ENUM_MAX_VALUE)) { VL_FATAL_MT(__FILE__, __LINE__, "", "vpi bb reason too large"); } - s_s.m_cbObjLists[vop->reason()].push_back(vop); + VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: vpi_register_cb reason=%d id=%" VL_PRI64 "d obj=%p\n", + cb_data_p->reason, id, cb_data_p->obj);); + VerilatedVpioVar* varop = nullptr; + if (cb_data_p->reason == cbValueChange) varop = VerilatedVpioVar::castp(cb_data_p->obj); + s_s.m_cbObjLists[cb_data_p->reason].emplace_back(id, cb_data_p, varop); } - static void cbTimedAdd(VerilatedVpioCb* vop) { s_s.m_timedCbs.emplace(vop->time(), vop); } - static void cbReasonRemove(VerilatedVpioCb* cbp) { - VpioCbList& cbObjList = s_s.m_cbObjLists[cbp->reason()]; + static void cbTimedAdd(vluint64_t id, const s_cb_data* cb_data_p, QData time) { + // The passed cb_data_p was property of the user, so need to recreate + VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: vpi_register_cb reason=%d id=%" VL_PRI64 + "d delay=%" VL_PRI64 "u\n", + cb_data_p->reason, id, time);); + s_s.m_timedCbs.emplace(std::piecewise_construct, + std::forward_as_tuple(std::make_pair(time, id)), + std::forward_as_tuple(id, cb_data_p, nullptr)); + } + static void cbReasonRemove(vluint64_t id, vluint32_t reason) { + // Id might no longer exist, if already removed due to call after event, or teardown + VpioCbList& cbObjList = s_s.m_cbObjLists[reason]; // We do not remove it now as we may be iterating the list, // instead set to nullptr and will cleanup later for (auto& ir : cbObjList) { - if (ir == cbp) ir = nullptr; + if (ir.id() == id) ir.invalidate(); } } - static void cbTimedRemove(VerilatedVpioCb* cbp) { - const auto it = s_s.m_timedCbs.find(std::make_pair(cbp->time(), cbp)); - if (VL_LIKELY(it != s_s.m_timedCbs.end())) s_s.m_timedCbs.erase(it); + static void cbTimedRemove(vluint64_t id, QData time) { + // Id might no longer exist, if already removed due to call after event, or teardown + const auto it = s_s.m_timedCbs.find(std::make_pair(time, id)); + if (VL_LIKELY(it != s_s.m_timedCbs.end())) it->second.invalidate(); } static void callTimedCbs() VL_MT_UNSAFE_ONE { assertOneCheck(); QData time = VL_TIME_Q(); for (auto it = s_s.m_timedCbs.begin(); it != s_s.m_timedCbs.end();) { - if (VL_UNLIKELY(it->first <= time)) { - VerilatedVpioCb* vop = it->second; + if (VL_UNLIKELY(it->first.first <= time)) { + VerilatedVpiCbHolder& ho = it->second; const auto last_it = it; - ++it; // Timed callbacks are one-shot + ++it; + if (VL_UNLIKELY(!ho.invalid())) { + VL_DEBUG_IF_PLI( + VL_DBG_MSGF("- vpi: timed_callback id=%" VL_PRI64 "d\n", ho.id());); + ho.invalidate(); // Timed callbacks are one-shot + (ho.cb_rtnp())(ho.cb_datap()); + } s_s.m_timedCbs.erase(last_it); - VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: timed_callback %p\n", vop);); - (vop->cb_rtnp())(vop->cb_datap()); } else { ++it; } @@ -484,7 +572,7 @@ public: } static QData cbNextDeadline() { const auto it = s_s.m_timedCbs.cbegin(); - if (VL_LIKELY(it != s_s.m_timedCbs.cend())) return it->first; + if (VL_LIKELY(it != s_s.m_timedCbs.cend())) return it->first.first; return ~0ULL; // maxquad } static bool callCbs(vluint32_t reason) VL_MT_UNSAFE_ONE { @@ -495,16 +583,18 @@ public: for (auto it = cbObjList.begin(); true;) { // cbReasonRemove sets to nullptr, so we know on removal the old end() will still exist bool was_last = it == last; - if (VL_UNLIKELY(!*it)) { // Deleted earlier, cleanup + if (VL_UNLIKELY(it->invalid())) { // Deleted earlier, cleanup it = cbObjList.erase(it); if (was_last) break; continue; } - VerilatedVpioCb* vop = *it++; - VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: reason_callback %d %p\n", reason, vop);); - (vop->cb_rtnp())(vop->cb_datap()); + VerilatedVpiCbHolder& ho = *it; + VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: reason_callback reason=%d id=%" VL_PRI64 "d\n", + reason, ho.id());); + (ho.cb_rtnp())(ho.cb_datap()); called = true; if (was_last) break; + ++it; } return called; } @@ -519,24 +609,25 @@ public: for (auto it = cbObjList.begin(); true;) { // cbReasonRemove sets to nullptr, so we know on removal the old end() will still exist bool was_last = it == last; - if (VL_UNLIKELY(!*it)) { // Deleted earlier, cleanup + if (VL_UNLIKELY(it->invalid())) { // Deleted earlier, cleanup it = cbObjList.erase(it); if (was_last) break; continue; } - VerilatedVpioCb* vop = *it++; - if (VerilatedVpioVar* varop = VerilatedVpioVar::castp(vop->cb_datap()->obj)) { + VerilatedVpiCbHolder& ho = *it++; + if (VerilatedVpioVar* varop = VerilatedVpioVar::castp(ho.cb_datap()->obj)) { void* newDatap = varop->varDatap(); void* prevDatap = varop->prevDatap(); // Was malloced when we added the callback VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: value_test %s v[0]=%d/%d %p %p\n", varop->fullname(), *((CData*)newDatap), *((CData*)prevDatap), newDatap, prevDatap);); if (memcmp(prevDatap, newDatap, varop->entSize()) != 0) { - VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: value_callback %p %s v[0]=%d\n", vop, - varop->fullname(), *((CData*)newDatap));); + VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: value_callback %" VL_PRI64 + "d %s v[0]=%d\n", + ho.id(), varop->fullname(), *((CData*)newDatap));); update.insert(varop); - vpi_get_value(vop->cb_datap()->obj, vop->cb_datap()->value); - (vop->cb_rtnp())(vop->cb_datap()); + vpi_get_value(ho.cb_datap()->obj, ho.cb_datap()->value); + (ho.cb_rtnp())(ho.cb_datap()); called = true; } } @@ -635,6 +726,17 @@ bool VerilatedVpi::callCbs(vluint32_t reason) VL_MT_UNSAFE_ONE { QData VerilatedVpi::cbNextDeadline() VL_MT_UNSAFE_ONE { return VerilatedVpiImp::cbNextDeadline(); } +PLI_INT32 VerilatedVpioTimedCb::dovpi_remove_cb() { + VerilatedVpiImp::cbTimedRemove(m_id, m_time); + delete this; // IEEE 37.2.2 a vpi_remove_cb does a vpi_release_handle + return 1; +} +PLI_INT32 VerilatedVpioReasonCb::dovpi_remove_cb() { + VerilatedVpiImp::cbReasonRemove(m_id, m_reason); + delete this; // IEEE 37.2.2 a vpi_remove_cb does a vpi_release_handle + return 1; +} + //====================================================================== // VerilatedVpiImp implementation @@ -1037,6 +1139,9 @@ void VerilatedVpiError::selfTest() VL_MT_UNSAFE_ONE { // callback related vpiHandle vpi_register_cb(p_cb_data cb_data_p) { + // Returns handle so user can remove the callback, user must vpi_release_handle it + // Don't confuse with the callback-activated t_cb_data object handle + // which is the object causing the callback rather than the callback itself VerilatedVpiImp::assertOneCheck(); _VL_VPI_ERROR_RESET(); // cppcheck-suppress nullPointer @@ -1048,10 +1153,10 @@ vpiHandle vpi_register_cb(p_cb_data cb_data_p) { case cbAfterDelay: { QData time = 0; if (cb_data_p->time) time = _VL_SET_QII(cb_data_p->time->high, cb_data_p->time->low); - VerilatedVpioCb* vop = new VerilatedVpioCb(cb_data_p, VL_TIME_Q() + time); - VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: vpi_register_cb %d %p delay=%" VL_PRI64 "u\n", - cb_data_p->reason, vop, time);); - VerilatedVpiImp::cbTimedAdd(vop); + QData abstime = VL_TIME_Q() + time; + vluint64_t id = VerilatedVpiImp::nextCallbackId(); + VerilatedVpioTimedCb* vop = new VerilatedVpioTimedCb{id, abstime}; + VerilatedVpiImp::cbTimedAdd(id, cb_data_p, abstime); return vop->castVpiHandle(); } case cbReadWriteSynch: // FALLTHRU // Supported via vlt_main.cpp @@ -1064,9 +1169,9 @@ vpiHandle vpi_register_cb(p_cb_data cb_data_p) { case cbEnterInteractive: // FALLTHRU // NOP, but need to return handle, so make object case cbExitInteractive: // FALLTHRU // NOP, but need to return handle, so make object case cbInteractiveScopeChange: { // FALLTHRU // NOP, but need to return handle, so make object - VerilatedVpioCb* vop = new VerilatedVpioCb(cb_data_p, 0); - VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: vpi_register_cb %d %p\n", cb_data_p->reason, vop);); - VerilatedVpiImp::cbReasonAdd(vop); + vluint64_t id = VerilatedVpiImp::nextCallbackId(); + VerilatedVpioReasonCb* vop = new VerilatedVpioReasonCb{id, cb_data_p->reason}; + VerilatedVpiImp::cbReasonAdd(id, cb_data_p); return vop->castVpiHandle(); } default: @@ -1079,15 +1184,10 @@ vpiHandle vpi_register_cb(p_cb_data cb_data_p) { PLI_INT32 vpi_remove_cb(vpiHandle cb_obj) { VL_DEBUG_IF_PLI(VL_DBG_MSGF("- vpi: vpi_remove_cb %p\n", cb_obj);); VerilatedVpiImp::assertOneCheck(); - VerilatedVpioCb* vop = VerilatedVpioCb::castp(cb_obj); _VL_VPI_ERROR_RESET(); + VerilatedVpio* vop = VerilatedVpio::castp(cb_obj); if (VL_UNLIKELY(!vop)) return 0; - if (vop->cb_datap()->reason == cbAfterDelay) { - VerilatedVpiImp::cbTimedRemove(vop); - } else { - VerilatedVpiImp::cbReasonRemove(vop); - } - return 1; + return vop->dovpi_remove_cb(); } void vpi_get_cb_info(vpiHandle /*object*/, p_cb_data /*cb_data_p*/) { _VL_VPI_UNIMP(); } @@ -2023,8 +2123,6 @@ PLI_INT32 vpi_release_handle(vpiHandle object) { VerilatedVpio* vop = VerilatedVpio::castp(object); _VL_VPI_ERROR_RESET(); if (VL_UNLIKELY(!vop)) return 0; - - vpi_remove_cb(object); // May not be a callback, but that's ok VL_DO_DANGLING(delete vop, vop); return 1; } diff --git a/test_regress/t/TestVpi.h b/test_regress/t/TestVpi.h index d4fc51d4c..2dd400b11 100644 --- a/test_regress/t/TestVpi.h +++ b/test_regress/t/TestVpi.h @@ -15,33 +15,30 @@ class TestVpiHandle { /// For testing, etc, wrap vpiHandle in an auto-releasing class - vpiHandle m_handle; - bool m_free; + vpiHandle m_handle = NULL; + bool m_freeit = true; public: - TestVpiHandle() - : m_handle(NULL) - , m_free(true) {} + TestVpiHandle() {} TestVpiHandle(vpiHandle h) - : m_handle(h) - , m_free(true) {} - ~TestVpiHandle() { - if (m_handle && m_free) { - // Below not VL_DO_DANGLING so is portable - { - vpi_release_handle(m_handle); - m_handle = NULL; - } - } - } + : m_handle(h) {} + ~TestVpiHandle() { release(); } operator vpiHandle() const { return m_handle; } inline TestVpiHandle& operator=(vpiHandle h) { + release(); m_handle = h; return *this; } + void release() { + if (m_handle && m_freeit) { + // Below not VL_DO_DANGLING so is portable + vpi_release_handle(m_handle); + m_handle = NULL; + } + } // Freed by another action e.g. vpi_scan; so empty and don't free again void freed() { m_handle = NULL; - m_free = false; + m_freeit = false; } }; diff --git a/test_regress/t/t_vpi_cb_iter.cpp b/test_regress/t/t_vpi_cb_iter.cpp index f9dd322d4..a4a6641cb 100644 --- a/test_regress/t/t_vpi_cb_iter.cpp +++ b/test_regress/t/t_vpi_cb_iter.cpp @@ -26,8 +26,8 @@ bool got_error = false; -vpiHandle vh_value_cb = 0; -vpiHandle vh_rw_cb = 0; +TestVpiHandle vh_value_cb; +TestVpiHandle vh_rw_cb; unsigned int last_value_cb_time = 0; unsigned int last_rw_cb_time = 0; @@ -78,6 +78,7 @@ static void reregister_value_cb() { if (vh_value_cb) { if (verbose) { vpi_printf(const_cast("- Removing cbValueChange callback\n")); } int ret = vpi_remove_cb(vh_value_cb); + vh_value_cb.freed(); CHECK_RESULT(ret, 1); if (verbose) { @@ -93,7 +94,7 @@ static void reregister_value_cb() { cb_data_testcase.cb_rtn = the_value_callback; cb_data_testcase.reason = cbValueChange; - vpiHandle vh1 = VPI_HANDLE("count"); + TestVpiHandle vh1 = VPI_HANDLE("count"); CHECK_RESULT_NZ(vh1); s_vpi_value v; @@ -110,6 +111,7 @@ static void reregister_rw_cb() { if (vh_rw_cb) { if (verbose) { vpi_printf(const_cast("- Removing cbReadWriteSynch callback\n")); } int ret = vpi_remove_cb(vh_rw_cb); + vh_rw_cb.freed(); CHECK_RESULT(ret, 1); if (verbose) { @@ -140,8 +142,8 @@ static void register_filler_cb() { cb_data_1.cb_rtn = the_filler_callback; cb_data_1.reason = cbReadWriteSynch; - vpiHandle vh1 = vpi_register_cb(&cb_data_1); - CHECK_RESULT_NZ(vh1); + TestVpiHandle cb_data_1_h = vpi_register_cb(&cb_data_1); + CHECK_RESULT_NZ(cb_data_1_h); if (verbose) { vpi_printf(const_cast("- Registering filler cbValueChange callback\n")); @@ -151,7 +153,7 @@ static void register_filler_cb() { cb_data_2.cb_rtn = the_filler_callback; cb_data_2.reason = cbValueChange; - vpiHandle vh2 = VPI_HANDLE("count"); + TestVpiHandle vh2 = VPI_HANDLE("count"); CHECK_RESULT_NZ(vh2); s_vpi_value v; @@ -160,8 +162,8 @@ static void register_filler_cb() { cb_data_2.obj = vh2; cb_data_2.value = &v; - vpiHandle vh3 = vpi_register_cb(&cb_data_2); - CHECK_RESULT_NZ(vh3); + TestVpiHandle cb_data_2_h = vpi_register_cb(&cb_data_2); + CHECK_RESULT_NZ(cb_data_2_h); } double sc_time_stamp() { return main_time; } diff --git a/test_regress/t/t_vpi_cbs_called.cpp b/test_regress/t/t_vpi_cbs_called.cpp index c5ac031b6..8a8b929d3 100644 --- a/test_regress/t/t_vpi_cbs_called.cpp +++ b/test_regress/t/t_vpi_cbs_called.cpp @@ -32,7 +32,7 @@ const std::vector cb_states{PRE_REGISTER, ACTIVE, ACTIVE_AGAIN, R POST_REMOVE}; #define CB_COUNT cbAtEndOfSimTime + 1 -vpiHandle vh_registered_cbs[CB_COUNT] = {0}; +TestVpiHandle vh_registered_cbs[CB_COUNT] = {0}; unsigned int callback_counts[CB_COUNT] = {0}; unsigned int callback_expected_counts[CB_COUNT] = {0}; @@ -43,7 +43,6 @@ bool callbacks_expected_called[CB_COUNT] = {false}; std::vector::const_iterator cb_iter; std::vector::const_iterator state_iter; -vpiHandle vh_test_cb = 0; unsigned int main_time = 0; bool got_error = false; @@ -94,18 +93,17 @@ static int the_callback(p_cb_data cb_data) { static int register_cb(const int next_state) { int cb = *cb_iter; t_cb_data cb_data_testcase; + s_vpi_value v; // Needed in this scope as is in cb_data bzero(&cb_data_testcase, sizeof(cb_data_testcase)); cb_data_testcase.cb_rtn = the_callback; cb_data_testcase.reason = cb; + TestVpiHandle count_h = VPI_HANDLE("count"); // Needed in this scope as is in cb_data + CHECK_RESULT_NZ(count_h); if (cb == cbValueChange) { - vpiHandle vh1 = VPI_HANDLE("count"); - CHECK_RESULT_NZ(vh1); - - s_vpi_value v; v.format = vpiSuppressVal; - cb_data_testcase.obj = vh1; + cb_data_testcase.obj = count_h; cb_data_testcase.value = &v; } @@ -117,6 +115,7 @@ static int register_cb(const int next_state) { vpi_printf(const_cast(" - Registering callback %s\n"), cb_reason_to_string(cb)); } + vh_registered_cbs[cb].release(); vh_registered_cbs[cb] = vpi_register_cb(&cb_data_testcase); break; } @@ -126,6 +125,7 @@ static int register_cb(const int next_state) { cb_reason_to_string(cb)); } int ret = vpi_remove_cb(vh_registered_cbs[cb]); + vh_registered_cbs[cb].freed(); CHECK_RESULT(ret, 1); vh_registered_cbs[cb] = vpi_register_cb(&cb_data_testcase); break; @@ -136,6 +136,7 @@ static int register_cb(const int next_state) { cb_reason_to_string(cb)); } int ret = vpi_remove_cb(vh_registered_cbs[cb]); + vh_registered_cbs[cb].freed(); CHECK_RESULT(ret, 1); break; } @@ -215,7 +216,7 @@ static int test_callbacks(p_cb_data cb_data) { t1.low = 1; cb_data_n.time = &t1; cb_data_n.cb_rtn = test_callbacks; - vh_test_cb = vpi_register_cb(&cb_data_n); + TestVpiHandle vh_test_cb = vpi_register_cb(&cb_data_n); CHECK_RESULT_NZ(vh_test_cb); } @@ -235,7 +236,7 @@ static int register_test_callback() { t1.low = 1; cb_data.time = &t1; cb_data.cb_rtn = test_callbacks; - vh_test_cb = vpi_register_cb(&cb_data); + TestVpiHandle vh_test_cb = vpi_register_cb(&cb_data); CHECK_RESULT_NZ(vh_test_cb); cb_iter = cbs_to_test.cbegin(); @@ -250,7 +251,6 @@ int main(int argc, char** argv, char** env) { double sim_time = 100; bool cbs_called; Verilated::commandArgs(argc, argv); - Verilated::debug(0); VM_PREFIX* topp = new VM_PREFIX(""); // Note null name - we're flattening it out diff --git a/test_regress/t/t_vpi_memory.cpp b/test_regress/t/t_vpi_memory.cpp index 6292fbe83..6478c4b58 100644 --- a/test_regress/t/t_vpi_memory.cpp +++ b/test_regress/t/t_vpi_memory.cpp @@ -80,26 +80,32 @@ unsigned int main_time = 0; #define CHECK_RESULT_CSTR_STRIP(got, exp) CHECK_RESULT_CSTR(got + strspn(got, " "), exp) -int _mon_check_range(TestVpiHandle& handle, int size, int left, int right) { - TestVpiHandle iter_h, left_h, right_h; +int _mon_check_range(const TestVpiHandle& handle, int size, int left, int right) { s_vpi_value value; value.format = vpiIntVal; value.value.integer = 0; // check size of object - int vpisize = vpi_get(vpiSize, handle); - CHECK_RESULT(vpisize, size); - // check left hand side of range - left_h = vpi_handle(vpiLeftRange, handle); - CHECK_RESULT_NZ(left_h); - vpi_get_value(left_h, &value); - CHECK_RESULT(value.value.integer, left); - int coherency = value.value.integer; - // check right hand side of range - right_h = vpi_handle(vpiRightRange, handle); - CHECK_RESULT_NZ(right_h); - vpi_get_value(right_h, &value); - CHECK_RESULT(value.value.integer, right); - coherency -= value.value.integer; + { + int vpisize = vpi_get(vpiSize, handle); + CHECK_RESULT(vpisize, size); + } + int coherency; + { + // check left hand side of range + TestVpiHandle left_h = vpi_handle(vpiLeftRange, handle); + CHECK_RESULT_NZ(left_h); + vpi_get_value(left_h, &value); + CHECK_RESULT(value.value.integer, left); + coherency = value.value.integer; + } + { + // check right hand side of range + TestVpiHandle right_h = vpi_handle(vpiRightRange, handle); + CHECK_RESULT_NZ(right_h); + vpi_get_value(right_h, &value); + CHECK_RESULT(value.value.integer, right); + coherency -= value.value.integer; + } // calculate size & check coherency = abs(coherency) + 1; CHECK_RESULT(coherency, size); @@ -107,40 +113,46 @@ int _mon_check_range(TestVpiHandle& handle, int size, int left, int right) { } int _mon_check_memory() { - int cnt; - TestVpiHandle mem_h, lcl_h, side_h; - vpiHandle iter_h; // Icarus does not like auto free of iterator handles s_vpi_value value; value.format = vpiIntVal; value.value.integer = 0; s_vpi_error_info e; vpi_printf((PLI_BYTE8*)"Check memory vpi ...\n"); - mem_h = vpi_handle_by_name((PLI_BYTE8*)TestSimulator::rooted("mem0"), NULL); + TestVpiHandle mem_h = vpi_handle_by_name((PLI_BYTE8*)TestSimulator::rooted("mem0"), NULL); CHECK_RESULT_NZ(mem_h); - // check type - int vpitype = vpi_get(vpiType, mem_h); - CHECK_RESULT(vpitype, vpiMemory); + { + // check type + int vpitype = vpi_get(vpiType, mem_h); + CHECK_RESULT(vpitype, vpiMemory); + } if (int status = _mon_check_range(mem_h, 16, 16, 1)) return status; // iterate and store - iter_h = vpi_iterate(vpiMemoryWord, mem_h); - cnt = 0; - while ((lcl_h = vpi_scan(iter_h))) { - value.value.integer = ++cnt; - vpi_put_value(lcl_h, &value, NULL, vpiNoDelay); - // check size and range - if (int status = _mon_check_range(lcl_h, 32, 31, 0)) return status; + { + TestVpiHandle iter_h = vpi_iterate(vpiMemoryWord, mem_h); + int cnt = 0; + while (TestVpiHandle lcl_h = vpi_scan(iter_h)) { + value.value.integer = ++cnt; + vpi_put_value(lcl_h, &value, NULL, vpiNoDelay); + // check size and range + if (int status = _mon_check_range(lcl_h, 32, 31, 0)) return status; + } + iter_h.freed(); // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + CHECK_RESULT(cnt, 16); // should be 16 addresses } - CHECK_RESULT(cnt, 16); // should be 16 addresses - // iterate and accumulate - iter_h = vpi_iterate(vpiMemoryWord, mem_h); - cnt = 0; - while ((lcl_h = vpi_scan(iter_h))) { - ++cnt; - vpi_get_value(lcl_h, &value); - CHECK_RESULT(value.value.integer, cnt); + { + // iterate and accumulate + TestVpiHandle iter_h = vpi_iterate(vpiMemoryWord, mem_h); + int cnt = 0; + while (TestVpiHandle lcl_h = vpi_scan(iter_h)) { + ++cnt; + vpi_get_value(lcl_h, &value); + CHECK_RESULT(value.value.integer, cnt); + } + iter_h.freed(); // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + CHECK_RESULT(cnt, 16); // should be 16 addresses } - CHECK_RESULT(cnt, 16); // should be 16 addresses + // don't care for non verilator // (crashes on Icarus) if (TestSimulator::is_icarus()) { @@ -148,40 +160,49 @@ int _mon_check_memory() { TestSimulator::get_info().product); return 0; // Ok } - // make sure trying to get properties that don't exist - // doesn't crash - int should_be_0 = vpi_get(vpiSize, iter_h); - CHECK_RESULT(should_be_0, 0); - should_be_0 = vpi_get(vpiIndex, iter_h); - CHECK_RESULT(should_be_0, 0); - vpiHandle should_be_NULL = vpi_handle(vpiLeftRange, iter_h); - CHECK_RESULT(should_be_NULL, 0); - should_be_NULL = vpi_handle(vpiRightRange, iter_h); - CHECK_RESULT(should_be_NULL, 0); - should_be_NULL = vpi_handle(vpiScope, iter_h); - CHECK_RESULT(should_be_NULL, 0); - - // check vpiRange - iter_h = vpi_iterate(vpiRange, mem_h); - CHECK_RESULT_NZ(iter_h); - lcl_h = vpi_scan(iter_h); - CHECK_RESULT_NZ(lcl_h); - side_h = vpi_handle(vpiLeftRange, lcl_h); - CHECK_RESULT_NZ(side_h); - vpi_get_value(side_h, &value); - CHECK_RESULT(value.value.integer, 16); - side_h = vpi_handle(vpiRightRange, lcl_h); - CHECK_RESULT_NZ(side_h); - vpi_get_value(side_h, &value); - CHECK_RESULT(value.value.integer, 1); - // iterator should exhaust after 1 dimension - lcl_h = vpi_scan(iter_h); - CHECK_RESULT(lcl_h, 0); - - // check writing to vpiConstant - vpi_put_value(side_h, &value, NULL, vpiNoDelay); - CHECK_RESULT_NZ(vpi_chk_error(&e)); - + { + // make sure trying to get properties that don't exist + // doesn't crash + TestVpiHandle iter_h = vpi_iterate(vpiMemoryWord, mem_h); + int should_be_0 = vpi_get(vpiSize, iter_h); + CHECK_RESULT(should_be_0, 0); + should_be_0 = vpi_get(vpiIndex, iter_h); + CHECK_RESULT(should_be_0, 0); + vpiHandle should_be_NULL = vpi_handle(vpiLeftRange, iter_h); + CHECK_RESULT(should_be_NULL, 0); + should_be_NULL = vpi_handle(vpiRightRange, iter_h); + CHECK_RESULT(should_be_NULL, 0); + should_be_NULL = vpi_handle(vpiScope, iter_h); + CHECK_RESULT(should_be_NULL, 0); + } + { + // check vpiRange + TestVpiHandle iter_h = vpi_iterate(vpiRange, mem_h); + CHECK_RESULT_NZ(iter_h); + TestVpiHandle lcl_h = vpi_scan(iter_h); + CHECK_RESULT_NZ(lcl_h); + { + TestVpiHandle side_h = vpi_handle(vpiLeftRange, lcl_h); + CHECK_RESULT_NZ(side_h); + vpi_get_value(side_h, &value); + CHECK_RESULT(value.value.integer, 16); + } + { + TestVpiHandle side_h = vpi_handle(vpiRightRange, lcl_h); + CHECK_RESULT_NZ(side_h); + vpi_get_value(side_h, &value); + CHECK_RESULT(value.value.integer, 1); + // check writing to vpiConstant + vpi_put_value(side_h, &value, NULL, vpiNoDelay); + CHECK_RESULT_NZ(vpi_chk_error(&e)); + } + { + // iterator should exhaust after 1 dimension + TestVpiHandle zero_h = vpi_scan(iter_h); + iter_h.freed(); // IEEE 37.2.2 vpi_scan at end does a vpi_release_handle + CHECK_RESULT(zero_h, 0); + } + } return 0; // Ok } diff --git a/test_regress/t/t_vpi_time_cb.cpp b/test_regress/t/t_vpi_time_cb.cpp index 210abb609..e5e16e0d4 100644 --- a/test_regress/t/t_vpi_time_cb.cpp +++ b/test_regress/t/t_vpi_time_cb.cpp @@ -106,7 +106,7 @@ static int _time_cb1(p_cb_data cb_data) { t.low = 1; cb_data_n.time = &t; cb_data_n.cb_rtn = _time_cb1; - vpi_register_cb(&cb_data_n); + TestVpiHandle cb_data_n1_h = vpi_register_cb(&cb_data_n); return 0; } @@ -127,7 +127,7 @@ static int _time_cb2(p_cb_data cb_data) { t.low = 1; cb_data_n.time = &t; cb_data_n.cb_rtn = _time_cb2; - vpi_register_cb(&cb_data_n); + TestVpiHandle cb_data_n2_h = vpi_register_cb(&cb_data_n); return 0; } @@ -143,7 +143,7 @@ static int _start_of_sim_cb(p_cb_data cb_data) { t1.low = 3; cb_data_n1.time = &t1; cb_data_n1.cb_rtn = _time_cb1; - vpi_register_cb(&cb_data_n1); + TestVpiHandle cb_data_n1_h = vpi_register_cb(&cb_data_n1); cb_data_n2.reason = cbAfterDelay; t2.type = vpiSimTime; @@ -151,7 +151,7 @@ static int _start_of_sim_cb(p_cb_data cb_data) { t2.low = 4; cb_data_n2.time = &t2; cb_data_n2.cb_rtn = _time_cb2; - vpi_register_cb(&cb_data_n2); + TestVpiHandle cb_data_n2_h = vpi_register_cb(&cb_data_n2); callback_count_start_of_sim++; return 0; } @@ -177,12 +177,12 @@ void vpi_compat_bootstrap(void) { cb_data.reason = cbStartOfSimulation; cb_data.time = 0; cb_data.cb_rtn = _start_of_sim_cb; - vpi_register_cb(&cb_data); + TestVpiHandle _start_of_sim_cb_h = vpi_register_cb(&cb_data); cb_data.reason = cbEndOfSimulation; cb_data.time = 0; cb_data.cb_rtn = _end_of_sim_cb; - vpi_register_cb(&cb_data); + TestVpiHandle _end_of_sim_cb_h = vpi_register_cb(&cb_data); } // icarus entry diff --git a/test_regress/t/t_vpi_var.cpp b/test_regress/t/t_vpi_var.cpp index a048ad11a..b09f9b3dd 100644 --- a/test_regress/t/t_vpi_var.cpp +++ b/test_regress/t/t_vpi_var.cpp @@ -49,6 +49,12 @@ unsigned int callback_count_strs_max = 500; //====================================================================== +#ifdef TEST_VERBOSE +bool verbose = true; +#else +bool verbose = false; +#endif + #define CHECK_RESULT_VH(got, exp) \ if ((got) != (exp)) { \ printf("%%Error: %s:%d: GOT = %p EXP = %p\n", FILENM, __LINE__, (got), (exp)); \ @@ -138,17 +144,18 @@ int _mon_check_callbacks() { cb_data.value = NULL; cb_data.time = NULL; - vpiHandle vh = vpi_register_cb(&cb_data); + TestVpiHandle vh = vpi_register_cb(&cb_data); CHECK_RESULT_NZ(vh); PLI_INT32 status = vpi_remove_cb(vh); + vh.freed(); CHECK_RESULT_NZ(status); return 0; } int _value_callback(p_cb_data cb_data) { - + if (verbose) vpi_printf(const_cast(" _value_callback:\n")); if (TestSimulator::is_verilator()) { // this check only makes sense in Verilator CHECK_RESULT(cb_data->value->value.integer + 10, main_time); @@ -178,50 +185,59 @@ int _value_callback_quad(p_cb_data cb_data) { } int _mon_check_value_callbacks() { - vpiHandle vh1 = VPI_HANDLE("count"); - CHECK_RESULT_NZ(vh1); - s_vpi_value v; v.format = vpiIntVal; - vpi_get_value(vh1, &v); t_cb_data cb_data; cb_data.reason = cbValueChange; - cb_data.cb_rtn = _value_callback; - cb_data.obj = vh1; - cb_data.value = &v; cb_data.time = NULL; - vpiHandle vh = vpi_register_cb(&cb_data); - CHECK_RESULT_NZ(vh); + { + TestVpiHandle vh1 = VPI_HANDLE("count"); + CHECK_RESULT_NZ(vh1); - vh1 = VPI_HANDLE("half_count"); - CHECK_RESULT_NZ(vh1); + vpi_get_value(vh1, &v); + cb_data.value = &v; + cb_data.obj = vh1; + cb_data.cb_rtn = _value_callback; - cb_data.obj = vh1; - cb_data.cb_rtn = _value_callback_half; + if (verbose) vpi_printf(const_cast(" vpi_register_cb(_value_callback):\n")); + TestVpiHandle callback_h = vpi_register_cb(&cb_data); + CHECK_RESULT_NZ(callback_h); + } + { + TestVpiHandle vh1 = VPI_HANDLE("half_count"); + CHECK_RESULT_NZ(vh1); - vh = vpi_register_cb(&cb_data); - CHECK_RESULT_NZ(vh); + cb_data.obj = vh1; + cb_data.cb_rtn = _value_callback_half; - vh1 = VPI_HANDLE("quads"); - CHECK_RESULT_NZ(vh1); + TestVpiHandle callback_h = vpi_register_cb(&cb_data); + CHECK_RESULT_NZ(callback_h); + } + { + TestVpiHandle vh1 = VPI_HANDLE("quads"); + CHECK_RESULT_NZ(vh1); - v.format = vpiVectorVal; - cb_data.obj = vh1; - cb_data.cb_rtn = _value_callback_quad; + v.format = vpiVectorVal; + cb_data.obj = vh1; + cb_data.cb_rtn = _value_callback_quad; - vh = vpi_register_cb(&cb_data); - CHECK_RESULT_NZ(vh); + TestVpiHandle callback_h = vpi_register_cb(&cb_data); + CHECK_RESULT_NZ(callback_h); + } + { + TestVpiHandle vh1 = VPI_HANDLE("quads"); + CHECK_RESULT_NZ(vh1); + TestVpiHandle vh2 = vpi_handle_by_index(vh1, 2); + CHECK_RESULT_NZ(vh2); - vh1 = vpi_handle_by_index(vh1, 2); - CHECK_RESULT_NZ(vh1); + cb_data.obj = vh2; + cb_data.cb_rtn = _value_callback_quad; - cb_data.obj = vh1; - cb_data.cb_rtn = _value_callback_quad; - - vh = vpi_register_cb(&cb_data); - CHECK_RESULT_NZ(vh); + TestVpiHandle callback_h = vpi_register_cb(&cb_data); + CHECK_RESULT_NZ(callback_h); + } return 0; } @@ -463,7 +479,10 @@ int _mon_check_putget_str(p_cb_data cb_data) { s_vpi_vecval vector[4]; } value; // reference } data[129]; + if (cb_data) { + if (verbose) vpi_printf(const_cast(" _mon_check_putget_str callback:\n")); + // this is the callback static unsigned int seed = 1; s_vpi_time t; @@ -542,6 +561,7 @@ int _mon_check_putget_str(p_cb_data cb_data) { } if (++callback_count_strs == callback_count_strs_max) { int success = vpi_remove_cb(cb); + cb.freed(); CHECK_RESULT_NZ(success); }; } else { @@ -560,7 +580,7 @@ int _mon_check_putget_str(p_cb_data cb_data) { static t_cb_data cb_data; static s_vpi_value v; - static TestVpiHandle count_h = VPI_HANDLE("count"); + TestVpiHandle count_h = VPI_HANDLE("count"); cb_data.reason = cbValueChange; cb_data.cb_rtn = _mon_check_putget_str; // this function @@ -570,6 +590,7 @@ int _mon_check_putget_str(p_cb_data cb_data) { v.format = vpiIntVal; cb = vpi_register_cb(&cb_data); + // It is legal to free the callback handle immediately if not otherwise needed CHECK_RESULT_NZ(cb); } return 0; diff --git a/test_regress/t/t_vpi_zero_time_cb.cpp b/test_regress/t/t_vpi_zero_time_cb.cpp index 54303e04f..00b3a9c28 100644 --- a/test_regress/t/t_vpi_zero_time_cb.cpp +++ b/test_regress/t/t_vpi_zero_time_cb.cpp @@ -100,7 +100,7 @@ static int _start_of_sim_cb(p_cb_data cb_data) { t.low = 0; cb_data_n.time = &t; cb_data_n.cb_rtn = _zero_time_cb; - vpi_register_cb(&cb_data_n); + TestVpiHandle _cb_data_n_h = vpi_register_cb(&cb_data_n); callback_count_start_of_sim++; return 0; } @@ -127,12 +127,12 @@ void vpi_compat_bootstrap(void) { cb_data.reason = cbStartOfSimulation; cb_data.time = 0; cb_data.cb_rtn = _start_of_sim_cb; - vpi_register_cb(&cb_data); + TestVpiHandle _start_of_sim_cb_h = vpi_register_cb(&cb_data); cb_data.reason = cbEndOfSimulation; cb_data.time = 0; cb_data.cb_rtn = _end_of_sim_cb; - vpi_register_cb(&cb_data); + TestVpiHandle _end_of_sim_cb_h = vpi_register_cb(&cb_data); } // icarus entry