mirror of
https://github.com/verilator/verilator.git
synced 2025-01-03 21:27:35 +00:00
519792d02b
Event-triggered coroutines live in two stages: 'uncommitted' and 'ready'. First they land in 'uncommitted', meaning they can't be resumed yet. Only after coroutines from the 'ready' queue are resumed, the 'uncommitted' ones are moved to the 'ready' queue, and can be resumed. This is to avoid self-triggering in situations like waiting for an event immediately after triggering it. However, there is an issue with `wait` statements. If you have a `wait(b)`, it's being translated into a loop that awaits a change in `b` as long as `b` is false. If `b` is false at first, the coroutine is put into the `uncommitted` queue. If `b` is set to true before it's committed, the coroutine won't get resumed. This patch fixes that by immediately committing event controls created from `wait` statements. That means the coroutine from the example above will get resumed from now on.
426 lines
17 KiB
C++
426 lines
17 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 header
|
|
///
|
|
/// This file is included automatically by Verilator in some of the C++ files
|
|
/// it generates if timing features are used.
|
|
///
|
|
/// This file is not part of the Verilated public-facing API.
|
|
/// It is only for internal use.
|
|
///
|
|
/// See the internals documentation docs/internals.rst for details.
|
|
///
|
|
//*************************************************************************
|
|
#ifndef VERILATOR_VERILATED_TIMING_H_
|
|
#define VERILATOR_VERILATED_TIMING_H_
|
|
|
|
#include "verilated.h"
|
|
|
|
// clang-format off
|
|
// Some preprocessor magic to support both Clang and GCC coroutines with both libc++ and libstdc++
|
|
#if defined _LIBCPP_VERSION // libc++
|
|
# if __clang_major__ > 13 // Clang > 13 warns that coroutine types in std::experimental are deprecated
|
|
# pragma clang diagnostic push
|
|
# pragma clang diagnostic ignored "-Wdeprecated-experimental-coroutine"
|
|
# endif
|
|
# include <experimental/coroutine>
|
|
namespace std {
|
|
using namespace experimental; // Bring std::experimental into the std namespace
|
|
}
|
|
#else
|
|
# if defined __clang__ && defined __GLIBCXX__
|
|
# define __cpp_impl_coroutine 1 // Clang doesn't define this, but it's needed for libstdc++
|
|
# endif
|
|
# include <coroutine>
|
|
# if __clang_major__ < 14
|
|
namespace std { // Bring coroutine library into std::experimental, as Clang < 14 expects it to be there
|
|
namespace experimental {
|
|
using namespace std;
|
|
}
|
|
}
|
|
# endif
|
|
#endif
|
|
// clang-format on
|
|
|
|
// Placeholder for compiling with --protect-ids
|
|
#define VL_UNKNOWN "<unknown>"
|
|
|
|
//=============================================================================
|
|
// VlFileLineDebug stores a SystemVerilog source code location. Used in VlCoroutineHandle for
|
|
// debugging purposes.
|
|
|
|
class VlFileLineDebug final {
|
|
// MEMBERS
|
|
#ifdef VL_DEBUG
|
|
const char* m_filename = nullptr;
|
|
int m_lineno = 0;
|
|
#endif
|
|
|
|
public:
|
|
// CONSTRUCTORS
|
|
// Construct
|
|
VlFileLineDebug() = default;
|
|
VlFileLineDebug(const char* filename, int lineno)
|
|
#ifdef VL_DEBUG
|
|
: m_filename{filename}
|
|
, m_lineno{lineno}
|
|
#endif
|
|
{
|
|
}
|
|
|
|
// METHODS
|
|
#ifdef VL_DEBUG
|
|
const char* filename() const { return m_filename; }
|
|
int lineno() const { return m_lineno; }
|
|
#endif
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlCoroutineHandle is a non-copyable (but movable) coroutine handle. On resume, the handle is
|
|
// cleared, as we assume that either the coroutine has finished and deleted itself, or, if it got
|
|
// suspended, another VlCoroutineHandle was created to manage it.
|
|
|
|
class VlCoroutineHandle final {
|
|
VL_UNCOPYABLE(VlCoroutineHandle);
|
|
|
|
// MEMBERS
|
|
std::coroutine_handle<> m_coro; // The wrapped coroutine handle
|
|
VlFileLineDebug m_fileline;
|
|
|
|
public:
|
|
// CONSTRUCTORS
|
|
// Construct
|
|
VlCoroutineHandle()
|
|
: m_coro{nullptr} {}
|
|
VlCoroutineHandle(std::coroutine_handle<> coro, VlFileLineDebug fileline)
|
|
: m_coro{coro}
|
|
, m_fileline{fileline} {}
|
|
// Move the handle, leaving a nullptr
|
|
VlCoroutineHandle(VlCoroutineHandle&& moved)
|
|
: m_coro{std::exchange(moved.m_coro, nullptr)}
|
|
, m_fileline{moved.m_fileline} {}
|
|
// Destroy if the handle isn't null
|
|
~VlCoroutineHandle() {
|
|
// Usually these coroutines should get resumed; we only need to clean up if we destroy a
|
|
// model with some coroutines suspended
|
|
if (VL_UNLIKELY(m_coro)) m_coro.destroy();
|
|
}
|
|
// METHODS
|
|
// Move the handle, leaving a null handle
|
|
auto& operator=(VlCoroutineHandle&& moved) {
|
|
m_coro = std::exchange(moved.m_coro, nullptr);
|
|
return *this;
|
|
}
|
|
// Resume the coroutine if the handle isn't null
|
|
void resume();
|
|
#ifdef VL_DEBUG
|
|
void dump() const;
|
|
#endif
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlDelayScheduler stores coroutines to be resumed at a certain simulation time. If the current
|
|
// time is equal to a coroutine's resume time, the coroutine gets resumed.
|
|
|
|
class VlDelayScheduler final {
|
|
// TYPES
|
|
struct VlDelayedCoroutine {
|
|
uint64_t m_timestep; // Simulation time when the coroutine should be resumed
|
|
VlCoroutineHandle m_handle; // The suspended coroutine to be resumed
|
|
|
|
// Comparison operator for std::push_heap(), std::pop_heap()
|
|
bool operator<(const VlDelayedCoroutine& other) const {
|
|
return m_timestep > other.m_timestep;
|
|
}
|
|
#ifdef VL_DEBUG
|
|
void dump() const;
|
|
#endif
|
|
};
|
|
using VlDelayedCoroutineQueue = std::vector<VlDelayedCoroutine>;
|
|
|
|
// MEMBERS
|
|
VerilatedContext& m_context;
|
|
VlDelayedCoroutineQueue m_queue; // Coroutines to be restored at a certain simulation time
|
|
|
|
public:
|
|
// CONSTRUCTORS
|
|
explicit VlDelayScheduler(VerilatedContext& context)
|
|
: m_context{context} {}
|
|
// METHODS
|
|
// Resume coroutines waiting for the current simulation time
|
|
void resume();
|
|
// Returns the simulation time of the next time slot (aborts if there are no delayed
|
|
// coroutines)
|
|
uint64_t nextTimeSlot() const;
|
|
// Are there no delayed coroutines awaiting?
|
|
bool empty() const { return m_queue.empty(); }
|
|
// Are there coroutines to resume at the current simulation time?
|
|
bool awaitingCurrentTime() const {
|
|
return !empty() && m_queue.front().m_timestep <= m_context.time();
|
|
}
|
|
#ifdef VL_DEBUG
|
|
void dump() const;
|
|
#endif
|
|
// Used by coroutines for co_awaiting a certain simulation time
|
|
auto delay(uint64_t delay, const char* filename = VL_UNKNOWN, int lineno = 0) {
|
|
struct Awaitable {
|
|
VlDelayedCoroutineQueue& queue;
|
|
uint64_t delay;
|
|
VlFileLineDebug fileline;
|
|
|
|
bool await_ready() const { return false; } // Always suspend
|
|
void await_suspend(std::coroutine_handle<> coro) {
|
|
queue.push_back({delay, VlCoroutineHandle{coro, fileline}});
|
|
// Move last element to the proper place in the max-heap
|
|
std::push_heap(queue.begin(), queue.end());
|
|
}
|
|
void await_resume() const {}
|
|
};
|
|
return Awaitable{m_queue, m_context.time() + delay, VlFileLineDebug{filename, lineno}};
|
|
}
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlTriggerScheduler stores coroutines to be resumed by a trigger. It does not keep track of its
|
|
// trigger, relying on calling code to resume when appropriate. Coroutines are kept in two stages
|
|
// - 'uncommitted' and 'ready'. Whenever a coroutine is suspended, it lands in the 'uncommitted'
|
|
// stage. Only when commit() is called, these coroutines get moved to the 'ready' stage. That's
|
|
// when they can be resumed. This is done to avoid resuming processes before they start waiting.
|
|
|
|
class VlTriggerScheduler final {
|
|
// TYPES
|
|
using VlCoroutineVec = std::vector<VlCoroutineHandle>;
|
|
|
|
// MEMBERS
|
|
VlCoroutineVec m_uncommitted; // Coroutines suspended before commit() was called
|
|
// (not resumable)
|
|
VlCoroutineVec m_ready; // Coroutines that can be resumed (all coros from m_uncommitted are
|
|
// moved here in commit())
|
|
VlCoroutineVec m_resumeQueue; // Coroutines being resumed by resume(); kept as a field to
|
|
// avoid reallocation. Resumed coroutines are moved to
|
|
// m_resumeQueue to allow adding coroutines to m_ready
|
|
// during resume(). Outside of resume() should always be empty.
|
|
|
|
public:
|
|
// METHODS
|
|
// Resumes all coroutines from the 'ready' stage
|
|
void resume(const char* eventDescription = VL_UNKNOWN);
|
|
// Moves all coroutines from m_uncommitted to m_ready
|
|
void commit(const char* eventDescription = VL_UNKNOWN);
|
|
// Are there no coroutines awaiting?
|
|
bool empty() const { return m_ready.empty() && m_uncommitted.empty(); }
|
|
#ifdef VL_DEBUG
|
|
void dump(const char* eventDescription) const;
|
|
#endif
|
|
// Used by coroutines for co_awaiting a certain trigger
|
|
auto trigger(bool commit, const char* eventDescription = VL_UNKNOWN,
|
|
const char* filename = VL_UNKNOWN, int lineno = 0) {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Suspending process waiting for %s at %s:%d\n",
|
|
eventDescription, filename, lineno););
|
|
struct Awaitable {
|
|
VlCoroutineVec& suspended; // Coros waiting on trigger
|
|
VlFileLineDebug fileline;
|
|
|
|
bool await_ready() const { return false; } // Always suspend
|
|
void await_suspend(std::coroutine_handle<> coro) {
|
|
suspended.emplace_back(coro, fileline);
|
|
}
|
|
void await_resume() const {}
|
|
};
|
|
return Awaitable{commit ? m_ready : m_uncommitted, VlFileLineDebug{filename, lineno}};
|
|
}
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlDynamicTriggerScheduler is used for cases where triggers cannot be statically referenced and
|
|
// evaluated. Coroutines that make use of this scheduler must adhere to a certain procedure:
|
|
// __Vtrigger = 0;
|
|
// <locals and inits required for trigger eval>
|
|
// while (!__Vtrigger) {
|
|
// co_await __VdynSched.evaluation();
|
|
// <pre updates>;
|
|
// __Vtrigger = <trigger eval>;
|
|
// [optionally] co_await __VdynSched.postUpdate();
|
|
// <post updates>;
|
|
// }
|
|
// co_await __VdynSched.resumption();
|
|
// The coroutines get resumed at trigger evaluation time, evaluate their local triggers, optionally
|
|
// await the post update step, and if the trigger is set, await proper resumption in the 'act' eval
|
|
// step.
|
|
|
|
class VlDynamicTriggerScheduler final {
|
|
// TYPES
|
|
using VlCoroutineVec = std::vector<VlCoroutineHandle>;
|
|
|
|
// MEMBERS
|
|
VlCoroutineVec m_suspended; // Suspended coroutines awaiting trigger evaluation
|
|
VlCoroutineVec m_evaluated; // Coroutines currently being evaluated (for evaluate())
|
|
VlCoroutineVec m_triggered; // Coroutines whose triggers were set, and are awaiting resumption
|
|
VlCoroutineVec m_post; // Coroutines awaiting the post update step (only relevant for triggers
|
|
// with destructive post updates, e.g. named events)
|
|
|
|
// METHODS
|
|
auto awaitable(VlCoroutineVec& queue, const char* filename, int lineno) {
|
|
struct Awaitable {
|
|
VlCoroutineVec& suspended; // Coros waiting on trigger
|
|
VlFileLineDebug fileline;
|
|
|
|
bool await_ready() const { return false; } // Always suspend
|
|
void await_suspend(std::coroutine_handle<> coro) {
|
|
suspended.emplace_back(coro, fileline);
|
|
}
|
|
void await_resume() const {}
|
|
};
|
|
return Awaitable{queue, VlFileLineDebug{filename, lineno}};
|
|
}
|
|
|
|
public:
|
|
// Evaluates all dynamic triggers (resumed coroutines that co_await evaluation())
|
|
bool evaluate();
|
|
// Runs post updates for all dynamic triggers (resumes coroutines that co_await postUpdate())
|
|
void doPostUpdates();
|
|
// Resumes all coroutines whose triggers are set (those that co_await resumption())
|
|
void resume();
|
|
#ifdef VL_DEBUG
|
|
void dump() const;
|
|
#endif
|
|
// Used by coroutines for co_awaiting trigger evaluation
|
|
auto evaluation(const char* eventDescription, const char* filename, int lineno) {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Suspending process waiting for %s at %s:%d\n",
|
|
eventDescription, filename, lineno););
|
|
return awaitable(m_suspended, filename, lineno);
|
|
}
|
|
// Used by coroutines for co_awaiting the trigger post update step
|
|
auto postUpdate(const char* eventDescription, const char* filename, int lineno) {
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" Process waiting for %s at %s:%d awaiting the post update step\n",
|
|
eventDescription, filename, lineno););
|
|
return awaitable(m_post, filename, lineno);
|
|
}
|
|
// Used by coroutines for co_awaiting the resumption step (in 'act' eval)
|
|
auto resumption(const char* eventDescription, const char* filename, int lineno) {
|
|
VL_DEBUG_IF(VL_DBG_MSGF(" Process waiting for %s at %s:%d awaiting resumption\n",
|
|
eventDescription, filename, lineno););
|
|
return awaitable(m_triggered, filename, lineno);
|
|
}
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlForever is a helper awaitable type for suspending coroutines forever. Used for constant
|
|
// wait statements.
|
|
|
|
struct VlForever {
|
|
bool await_ready() const { return false; } // Always suspend
|
|
void await_suspend(std::coroutine_handle<> coro) const { coro.destroy(); }
|
|
void await_resume() const {}
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlForkSync is used to manage fork..join and fork..join_any constructs.
|
|
|
|
class VlForkSync final {
|
|
// VlJoin stores the handle of a suspended coroutine that did a fork..join or fork..join_any.
|
|
// If the counter reaches 0, the suspended coroutine shall be resumed.
|
|
struct VlJoin {
|
|
size_t m_counter = 0; // When reaches 0, resume suspended coroutine
|
|
VlCoroutineHandle m_susp; // Coroutine to resume
|
|
};
|
|
|
|
// The join info is shared among all forked processes
|
|
std::shared_ptr<VlJoin> m_join;
|
|
|
|
public:
|
|
// Create the join object and set the counter to the specified number
|
|
void init(size_t count) { m_join.reset(new VlJoin{count, {}}); }
|
|
// Called whenever any of the forked processes finishes. If the join counter reaches 0, the
|
|
// main process gets resumed
|
|
void done(const char* filename = VL_UNKNOWN, int lineno = 0);
|
|
// Used by coroutines for co_awaiting a join
|
|
auto join(const char* filename = VL_UNKNOWN, int lineno = 0) {
|
|
assert(m_join);
|
|
VL_DEBUG_IF(
|
|
VL_DBG_MSGF(" Awaiting join of fork at: %s:%d\n", filename, lineno););
|
|
struct Awaitable {
|
|
const std::shared_ptr<VlJoin> join; // Join to await on
|
|
VlFileLineDebug fileline;
|
|
|
|
bool await_ready() { return join->m_counter == 0; } // Suspend if join still exists
|
|
void await_suspend(std::coroutine_handle<> coro) { join->m_susp = {coro, fileline}; }
|
|
void await_resume() const {}
|
|
};
|
|
return Awaitable{m_join, VlFileLineDebug{filename, lineno}};
|
|
}
|
|
};
|
|
|
|
//=============================================================================
|
|
// VlCoroutine
|
|
// Return value of a coroutine. Used for chaining coroutine suspension/resumption.
|
|
|
|
class VlCoroutine final {
|
|
private:
|
|
// TYPES
|
|
struct VlPromise {
|
|
std::coroutine_handle<> m_continuation; // Coroutine to resume after this one finishes
|
|
VlCoroutine* m_corop = nullptr; // Pointer to the coroutine return object
|
|
|
|
~VlPromise();
|
|
|
|
VlCoroutine get_return_object() { return {this}; }
|
|
|
|
// Never suspend at the start of the coroutine
|
|
std::suspend_never initial_suspend() const { return {}; }
|
|
|
|
// Never suspend at the end of the coroutine (thanks to this, the coroutine will clean up
|
|
// after itself)
|
|
std::suspend_never final_suspend() noexcept;
|
|
|
|
void unhandled_exception() const { std::abort(); }
|
|
void return_void() const {}
|
|
};
|
|
|
|
// MEMBERS
|
|
VlPromise* m_promisep; // The promise created for this coroutine
|
|
|
|
public:
|
|
// TYPES
|
|
using promise_type = VlPromise; // promise_type has to be public
|
|
|
|
// CONSTRUCTORS
|
|
// Construct
|
|
// cppcheck-suppress noExplicitConstructor
|
|
VlCoroutine(VlPromise* promisep)
|
|
: m_promisep{promisep} {
|
|
m_promisep->m_corop = this;
|
|
}
|
|
// Move. Update the pointers each time the return object is moved
|
|
// cppcheck-suppress noExplicitConstructor
|
|
VlCoroutine(VlCoroutine&& other)
|
|
: m_promisep{std::exchange(other.m_promisep, nullptr)} {
|
|
if (m_promisep) m_promisep->m_corop = this;
|
|
}
|
|
~VlCoroutine() {
|
|
// Indicate to the promise that the return object is gone
|
|
if (m_promisep) m_promisep->m_corop = nullptr;
|
|
}
|
|
|
|
// METHODS
|
|
// Suspend the awaiter if the coroutine is suspended (the promise exists)
|
|
bool await_ready() const noexcept { return !m_promisep; }
|
|
// Set the awaiting coroutine as the continuation of the current coroutine
|
|
void await_suspend(std::coroutine_handle<> coro) { m_promisep->m_continuation = coro; }
|
|
void await_resume() const noexcept {}
|
|
};
|
|
|
|
#endif // Guard
|