mirror of
https://github.com/verilator/verilator.git
synced 2025-04-05 20:22:41 +00:00
Improve robustness of trace configuration
Always fail if adding a model to a trace file that has already executed a dump. We used to do this before as well, though in a less robust way. We will be relying on this property more in the future, so improve the check.
This commit is contained in:
parent
3a002b6cf2
commit
efb5caad22
@ -208,7 +208,6 @@ protected:
|
|||||||
uint32_t* m_sigs_oldvalp = nullptr; // Previous value store
|
uint32_t* m_sigs_oldvalp = nullptr; // Previous value store
|
||||||
EData* m_sigs_enabledp = nullptr; // Bit vector of enabled codes (nullptr = all on)
|
EData* m_sigs_enabledp = nullptr; // Bit vector of enabled codes (nullptr = all on)
|
||||||
private:
|
private:
|
||||||
uint64_t m_timeLastDump = 0; // Last time we did a dump
|
|
||||||
std::vector<bool> m_sigs_enabledVec; // Staging for m_sigs_enabledp
|
std::vector<bool> m_sigs_enabledVec; // Staging for m_sigs_enabledp
|
||||||
std::vector<CallbackRecord> m_initCbs; // Routines to initialize tracing
|
std::vector<CallbackRecord> m_initCbs; // Routines to initialize tracing
|
||||||
std::vector<CallbackRecord> m_fullCbs; // Routines to perform full dump
|
std::vector<CallbackRecord> m_fullCbs; // Routines to perform full dump
|
||||||
@ -226,6 +225,8 @@ private:
|
|||||||
char m_scopeEscape = '.';
|
char m_scopeEscape = '.';
|
||||||
double m_timeRes = 1e-9; // Time resolution (ns/ms etc)
|
double m_timeRes = 1e-9; // Time resolution (ns/ms etc)
|
||||||
double m_timeUnit = 1e-0; // Time units (ns/ms etc)
|
double m_timeUnit = 1e-0; // Time units (ns/ms etc)
|
||||||
|
uint64_t m_timeLastDump = 0; // Last time we did a dump
|
||||||
|
bool m_didSomeDump = false; // Did at least one dump (i.e.: m_timeLastDump is valid)
|
||||||
|
|
||||||
void addModel(VerilatedModel*) VL_MT_SAFE_EXCLUDES(m_mutex);
|
void addModel(VerilatedModel*) VL_MT_SAFE_EXCLUDES(m_mutex);
|
||||||
|
|
||||||
@ -290,7 +291,6 @@ protected:
|
|||||||
uint32_t numSignals() const { return m_numSignals; }
|
uint32_t numSignals() const { return m_numSignals; }
|
||||||
uint32_t maxBits() const { return m_maxBits; }
|
uint32_t maxBits() const { return m_maxBits; }
|
||||||
void fullDump(bool value) { m_fullDump = value; }
|
void fullDump(bool value) { m_fullDump = value; }
|
||||||
uint64_t timeLastDump() { return m_timeLastDump; }
|
|
||||||
|
|
||||||
double timeRes() const { return m_timeRes; }
|
double timeRes() const { return m_timeRes; }
|
||||||
double timeUnit() const { return m_timeUnit; }
|
double timeUnit() const { return m_timeUnit; }
|
||||||
|
@ -559,13 +559,14 @@ void VerilatedTrace<VL_SUB_T, VL_BUF_T>::dump(uint64_t timeui) VL_MT_SAFE_EXCLUD
|
|||||||
// This does get the mutex, but if multiple threads are trying to dump
|
// This does get the mutex, but if multiple threads are trying to dump
|
||||||
// chances are the data being dumped will have other problems
|
// chances are the data being dumped will have other problems
|
||||||
const VerilatedLockGuard lock{m_mutex};
|
const VerilatedLockGuard lock{m_mutex};
|
||||||
if (VL_UNCOVERABLE(m_timeLastDump && timeui <= m_timeLastDump)) { // LCOV_EXCL_START
|
if (VL_UNCOVERABLE(m_didSomeDump && timeui <= m_timeLastDump)) { // LCOV_EXCL_START
|
||||||
VL_PRINTF_MT("%%Warning: previous dump at t=%" PRIu64 ", requesting t=%" PRIu64
|
VL_PRINTF_MT("%%Warning: previous dump at t=%" PRIu64 ", requesting t=%" PRIu64
|
||||||
", dump call ignored\n",
|
", dump call ignored\n",
|
||||||
m_timeLastDump, timeui);
|
m_timeLastDump, timeui);
|
||||||
return;
|
return;
|
||||||
} // LCOV_EXCL_STOP
|
} // LCOV_EXCL_STOP
|
||||||
m_timeLastDump = timeui;
|
m_timeLastDump = timeui;
|
||||||
|
m_didSomeDump = true;
|
||||||
|
|
||||||
Verilated::quiesce();
|
Verilated::quiesce();
|
||||||
|
|
||||||
@ -646,11 +647,14 @@ void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addModel(VerilatedModel* modelp)
|
|||||||
VL_MT_SAFE_EXCLUDES(m_mutex) {
|
VL_MT_SAFE_EXCLUDES(m_mutex) {
|
||||||
const VerilatedLockGuard lock{m_mutex};
|
const VerilatedLockGuard lock{m_mutex};
|
||||||
VerilatedContext* const contextp = modelp->contextp();
|
VerilatedContext* const contextp = modelp->contextp();
|
||||||
if (m_contextp && contextp != m_contextp) {
|
if (VL_UNCOVERABLE(m_contextp && contextp != m_contextp)) { // LCOV_EXCL_START
|
||||||
VL_FATAL_MT(
|
VL_FATAL_MT(__FILE__, __LINE__, "",
|
||||||
__FILE__, __LINE__, "",
|
"A trace file instance can only handle models from the same context");
|
||||||
"A trace file instance can only handle models from the same simulation context");
|
|
||||||
}
|
}
|
||||||
|
if (VL_UNCOVERABLE(m_didSomeDump)) {
|
||||||
|
VL_FATAL_MT(__FILE__, __LINE__, "",
|
||||||
|
"Cannot add models to a trace file if 'dump' has already been called");
|
||||||
|
} // LCOV_EXCL_STOP
|
||||||
m_contextp = contextp;
|
m_contextp = contextp;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -659,11 +663,6 @@ void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addCallbackRecord(std::vector<CallbackR
|
|||||||
CallbackRecord&& cbRec)
|
CallbackRecord&& cbRec)
|
||||||
VL_MT_SAFE_EXCLUDES(m_mutex) {
|
VL_MT_SAFE_EXCLUDES(m_mutex) {
|
||||||
const VerilatedLockGuard lock{m_mutex};
|
const VerilatedLockGuard lock{m_mutex};
|
||||||
if (VL_UNCOVERABLE(timeLastDump() != 0)) { // LCOV_EXCL_START
|
|
||||||
const std::string msg = (std::string{"Internal: "} + __FILE__ + "::" + __FUNCTION__
|
|
||||||
+ " called with already open file");
|
|
||||||
VL_FATAL_MT(__FILE__, __LINE__, "", msg.c_str());
|
|
||||||
} // LCOV_EXCL_STOP
|
|
||||||
cbVec.push_back(cbRec);
|
cbVec.push_back(cbRec);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -676,29 +675,29 @@ void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addInitCb(initCb_t cb, void* userp,
|
|||||||
template <>
|
template <>
|
||||||
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addFullCb(dumpCb_t cb, void* userp,
|
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addFullCb(dumpCb_t cb, void* userp,
|
||||||
VerilatedModel* modelp) VL_MT_SAFE {
|
VerilatedModel* modelp) VL_MT_SAFE {
|
||||||
assert(!offload());
|
|
||||||
addModel(modelp);
|
addModel(modelp);
|
||||||
|
assert(!offload());
|
||||||
addCallbackRecord(m_fullCbs, CallbackRecord{cb, userp});
|
addCallbackRecord(m_fullCbs, CallbackRecord{cb, userp});
|
||||||
}
|
}
|
||||||
template <>
|
template <>
|
||||||
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addFullCb(dumpOffloadCb_t cb, void* userp,
|
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addFullCb(dumpOffloadCb_t cb, void* userp,
|
||||||
VerilatedModel* modelp) VL_MT_SAFE {
|
VerilatedModel* modelp) VL_MT_SAFE {
|
||||||
assert(offload());
|
|
||||||
addModel(modelp);
|
addModel(modelp);
|
||||||
|
assert(offload());
|
||||||
addCallbackRecord(m_fullOffloadCbs, CallbackRecord{cb, userp});
|
addCallbackRecord(m_fullOffloadCbs, CallbackRecord{cb, userp});
|
||||||
}
|
}
|
||||||
template <>
|
template <>
|
||||||
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addChgCb(dumpCb_t cb, void* userp,
|
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addChgCb(dumpCb_t cb, void* userp,
|
||||||
VerilatedModel* modelp) VL_MT_SAFE {
|
VerilatedModel* modelp) VL_MT_SAFE {
|
||||||
assert(!offload());
|
|
||||||
addModel(modelp);
|
addModel(modelp);
|
||||||
|
assert(!offload());
|
||||||
addCallbackRecord(m_chgCbs, CallbackRecord{cb, userp});
|
addCallbackRecord(m_chgCbs, CallbackRecord{cb, userp});
|
||||||
}
|
}
|
||||||
template <>
|
template <>
|
||||||
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addChgCb(dumpOffloadCb_t cb, void* userp,
|
void VerilatedTrace<VL_SUB_T, VL_BUF_T>::addChgCb(dumpOffloadCb_t cb, void* userp,
|
||||||
VerilatedModel* modelp) VL_MT_SAFE {
|
VerilatedModel* modelp) VL_MT_SAFE {
|
||||||
assert(offload());
|
|
||||||
addModel(modelp);
|
addModel(modelp);
|
||||||
|
assert(offload());
|
||||||
addCallbackRecord(m_chgOffloadCbs, CallbackRecord{cb, userp});
|
addCallbackRecord(m_chgOffloadCbs, CallbackRecord{cb, userp});
|
||||||
}
|
}
|
||||||
template <>
|
template <>
|
||||||
|
Loading…
Reference in New Issue
Block a user