forked from github/verilator
125 lines
3.1 KiB
C++
125 lines
3.1 KiB
C++
|
//
|
||
|
// DESCRIPTION: Verilator: Verilog Multiple Model Test Module
|
||
|
//
|
||
|
// This file ONLY is placed under the Creative Commons Public Domain, for
|
||
|
// any use, without warranty, 2020-2021 by Andreas Kuster.
|
||
|
// SPDX-License-Identifier: CC0-1.0
|
||
|
//
|
||
|
|
||
|
#include <iostream>
|
||
|
#include <thread>
|
||
|
|
||
|
#include <verilated.h>
|
||
|
#include <verilated_cov.h>
|
||
|
|
||
|
#include VM_PREFIX_INCLUDE
|
||
|
|
||
|
double sc_time_stamp() { return 0; }
|
||
|
|
||
|
VerilatedMutex outputMutex;
|
||
|
|
||
|
#ifdef T_WRAPPER_CONTEXT
|
||
|
#elif defined(T_WRAPPER_CONTEXT_SEQ)
|
||
|
VerilatedMutex sequentialMutex;
|
||
|
#elif defined(T_WRAPPER_CONTEXT_FST)
|
||
|
#else
|
||
|
#error "Unexpected test name"
|
||
|
#endif
|
||
|
|
||
|
void sim(VM_PREFIX* topp) {
|
||
|
#ifdef T_WRAPPER_CONTEXT_SEQ
|
||
|
// Run each sim sequentially
|
||
|
const VerilatedLockGuard seqLock(sequentialMutex);
|
||
|
#endif
|
||
|
|
||
|
VerilatedContext* contextp = topp->contextp();
|
||
|
// This test created a thread, so need to associate VerilatedContext with it
|
||
|
Verilated::threadContextp(contextp);
|
||
|
|
||
|
// reset
|
||
|
topp->clk = 0;
|
||
|
topp->rst = 1;
|
||
|
topp->stop = (topp->trace_number == 0);
|
||
|
topp->eval();
|
||
|
|
||
|
contextp->timeInc(1);
|
||
|
topp->clk = 1;
|
||
|
topp->eval();
|
||
|
|
||
|
contextp->timeInc(1);
|
||
|
topp->rst = 0;
|
||
|
topp->clk = 0;
|
||
|
topp->eval();
|
||
|
|
||
|
// simulate until done
|
||
|
while (!contextp->gotFinish()) {
|
||
|
// increment time
|
||
|
contextp->timeInc(1);
|
||
|
|
||
|
{
|
||
|
const VerilatedLockGuard lock(outputMutex);
|
||
|
#ifdef TEST_VERBOSE
|
||
|
// std::endl needed to flush output before mutex release
|
||
|
std::cout << "{top" << topp->trace_number
|
||
|
<< ", ctx=" << reinterpret_cast<void*>(contextp) << "} [" << contextp->time()
|
||
|
<< "]" << std::endl;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// toggle clk
|
||
|
topp->clk = !topp->clk;
|
||
|
|
||
|
// evaluate model
|
||
|
topp->eval();
|
||
|
}
|
||
|
|
||
|
std::string filename
|
||
|
= std::string(VL_STRINGIFY(TEST_OBJ_DIR) "/coverage_") + topp->name() + ".dat";
|
||
|
contextp->coveragep()->write(filename.c_str());
|
||
|
}
|
||
|
|
||
|
int main(int argc, char** argv, char** env) {
|
||
|
// Create contexts
|
||
|
std::unique_ptr<VerilatedContext> context0p{new VerilatedContext};
|
||
|
std::unique_ptr<VerilatedContext> context1p{new VerilatedContext};
|
||
|
|
||
|
// configuration
|
||
|
context0p->fatalOnError(false);
|
||
|
context1p->fatalOnError(false);
|
||
|
context0p->traceEverOn(true);
|
||
|
context1p->traceEverOn(true);
|
||
|
|
||
|
// instantiate verilated design
|
||
|
std::unique_ptr<VM_PREFIX> top0p{new VM_PREFIX{context0p.get(), "top0"}};
|
||
|
std::unique_ptr<VM_PREFIX> top1p{new VM_PREFIX{context1p.get(), "top1"}};
|
||
|
|
||
|
top0p->trace_number = 0;
|
||
|
top0p->trace_number = 1;
|
||
|
|
||
|
std::cout << "Below '%Error: ... Verilog $stop' is expected as part of the test\n";
|
||
|
|
||
|
// create threads
|
||
|
std::thread t0(sim, top0p.get());
|
||
|
std::thread t1(sim, top1p.get());
|
||
|
|
||
|
// wait to finish
|
||
|
t0.join();
|
||
|
t1.join();
|
||
|
|
||
|
// check if both finished
|
||
|
bool pass = true;
|
||
|
if (top0p->done_o && top1p->done_o) {
|
||
|
std::cout << "*-* All Finished *-*" << std::endl;
|
||
|
} else {
|
||
|
std::cout << "Error: Early termination!" << std::endl;
|
||
|
pass = false;
|
||
|
}
|
||
|
|
||
|
// final model cleanup
|
||
|
top0p->final();
|
||
|
top1p->final();
|
||
|
|
||
|
// exit successful
|
||
|
return pass ? 0 : 10;
|
||
|
}
|