mirror of
https://github.com/verilator/verilator.git
synced 2025-01-03 21:27:35 +00:00
da7ad35577
Signed-off-by: Krzysztof Bieganski <kbieganski@antmicro.com>
182 lines
5.9 KiB
C++
182 lines
5.9 KiB
C++
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
|
//*************************************************************************
|
|
//
|
|
// Code available from: https://verilator.org
|
|
//
|
|
// Copyright 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
|
|
//
|
|
//=========================================================================
|
|
///
|
|
/// \file
|
|
/// \brief Verilated timing implementation code
|
|
///
|
|
/// This file must be compiled and linked against all Verilated objects
|
|
/// that use timing features.
|
|
///
|
|
/// See the internals documentation docs/internals.rst for details.
|
|
///
|
|
//=========================================================================
|
|
|
|
#include "verilated_timing.h"
|
|
|
|
//======================================================================
|
|
// VlCoroutineHandle:: Methods
|
|
|
|
void VlCoroutineHandle::resume() {
|
|
// Only null if we have a fork..join_any and one of the other child processes resumed the
|
|
// main process
|
|
if (VL_LIKELY(m_coro)) {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Resuming: "); dump(););
|
|
m_coro();
|
|
m_coro = nullptr;
|
|
}
|
|
}
|
|
|
|
#ifdef VL_DEBUG
|
|
void VlCoroutineHandle::dump() {
|
|
VL_DEBUG_IF(VL_PRINTF("Process waiting at %s:%d\n", m_filename, m_linenum););
|
|
}
|
|
#endif
|
|
|
|
//======================================================================
|
|
// VlDelayScheduler:: Methods
|
|
|
|
#ifdef VL_DEBUG
|
|
void VlDelayScheduler::VlDelayedCoroutine::dump() {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Awaiting time %lu: ", m_timestep););
|
|
m_handle.dump();
|
|
}
|
|
#endif
|
|
|
|
void VlDelayScheduler::resume() {
|
|
#ifdef VL_DEBUG
|
|
dump();
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Resuming delayed processes\n"););
|
|
#endif
|
|
while (awaitingCurrentTime()) {
|
|
if (m_queue.front().m_timestep != m_context.time()) {
|
|
VL_FATAL_MT(__FILE__, __LINE__, "",
|
|
"%Error: Encountered process that should've been resumed at an "
|
|
"earlier simulation time. Missed a time slot?");
|
|
}
|
|
// Move max element in the heap to the end
|
|
std::pop_heap(m_queue.begin(), m_queue.end());
|
|
VlCoroutineHandle handle = std::move(m_queue.back().m_handle);
|
|
m_queue.pop_back();
|
|
handle.resume();
|
|
}
|
|
}
|
|
|
|
uint64_t VlDelayScheduler::nextTimeSlot() {
|
|
if (empty()) {
|
|
VL_FATAL_MT(__FILE__, __LINE__, "", "%Error: There is no next time slot scheduled");
|
|
}
|
|
return m_queue.front().m_timestep;
|
|
}
|
|
|
|
#ifdef VL_DEBUG
|
|
void VlDelayScheduler::dump() {
|
|
if (m_queue.empty()) {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" No delayed processes:\n"););
|
|
} else {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Delayed processes:\n"););
|
|
for (auto& susp : m_queue) susp.dump();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
//======================================================================
|
|
// VlTriggerScheduler:: Methods
|
|
|
|
void VlTriggerScheduler::resume(const char* eventDescription) {
|
|
#ifdef VL_DEBUG
|
|
dump(eventDescription);
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Resuming processes waiting for %s\n", eventDescription););
|
|
#endif
|
|
for (auto& susp : m_ready) susp.resume();
|
|
m_ready.clear();
|
|
commit(eventDescription);
|
|
}
|
|
|
|
void VlTriggerScheduler::commit(const char* eventDescription) {
|
|
#ifdef VL_DEBUG
|
|
if (!m_uncommitted.empty()) {
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" Committing processes waiting for %s:\n", eventDescription);
|
|
for (auto& susp
|
|
: m_uncommitted) {
|
|
VL_DBG_MSGF(" - ");
|
|
susp.dump();
|
|
});
|
|
}
|
|
#endif
|
|
m_ready.reserve(m_ready.size() + m_uncommitted.size());
|
|
m_ready.insert(m_ready.end(), std::make_move_iterator(m_uncommitted.begin()),
|
|
std::make_move_iterator(m_uncommitted.end()));
|
|
m_uncommitted.clear();
|
|
}
|
|
|
|
#ifdef VL_DEBUG
|
|
void VlTriggerScheduler::dump(const char* eventDescription) {
|
|
if (m_ready.empty()) {
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" No ready processes waiting for %s\n", eventDescription););
|
|
} else {
|
|
VL_DEBUG_IF(for (auto& susp
|
|
: m_ready) {
|
|
VL_DBG_MSGF(" Ready processes waiting for %s:\n", eventDescription);
|
|
VL_DBG_MSGF(" - ");
|
|
susp.dump();
|
|
});
|
|
}
|
|
if (!m_uncommitted.empty()) {
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" Uncommitted processes waiting for %s:\n", eventDescription);
|
|
for (auto& susp
|
|
: m_uncommitted) {
|
|
VL_DBG_MSGF(" - ");
|
|
susp.dump();
|
|
});
|
|
}
|
|
}
|
|
#endif
|
|
|
|
//======================================================================
|
|
// VlForkSync:: Methods
|
|
|
|
void VlForkSync::done(const char* filename, int linenum) {
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" Process forked at %s:%d finished\n", filename, linenum););
|
|
if (m_join->m_counter > 0) m_join->m_counter--;
|
|
if (m_join->m_counter == 0) m_join->m_susp.resume();
|
|
}
|
|
|
|
//======================================================================
|
|
// VlCoroutine:: Methods
|
|
|
|
VlCoroutine::VlPromise::~VlPromise() {
|
|
// Indicate to the return object that the coroutine has finished or been destroyed
|
|
if (m_corop) m_corop->m_promisep = nullptr;
|
|
// If there is a continuation, destroy it
|
|
if (m_continuation) m_continuation.destroy();
|
|
}
|
|
|
|
std::suspend_never VlCoroutine::VlPromise::final_suspend() noexcept {
|
|
// Indicate to the return object that the coroutine has finished
|
|
if (m_corop) {
|
|
m_corop->m_promisep = nullptr;
|
|
// Forget the return value, we won't need it and it won't be able to let us know if
|
|
// it's destroyed
|
|
m_corop = nullptr;
|
|
}
|
|
// If there is a continuation, resume it
|
|
if (m_continuation) {
|
|
m_continuation();
|
|
m_continuation = nullptr;
|
|
}
|
|
return {};
|
|
}
|