2012-04-13 01:08:20 +00:00
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
2010-03-17 12:22:49 +00:00
|
|
|
//*************************************************************************
|
|
|
|
//
|
2011-01-01 23:21:19 +00:00
|
|
|
// Copyright 2010-2011 by Wilson Snyder. This program is free software; you can
|
2010-03-17 12:22:49 +00:00
|
|
|
// redistribute it and/or modify it under the terms of either the GNU
|
2020-03-21 15:24:24 +00:00
|
|
|
// Lesser General Public License Version 3 or the Perl Artistic License
|
2010-03-17 12:22:49 +00:00
|
|
|
// Version 2.0.
|
2020-03-21 15:24:24 +00:00
|
|
|
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
2010-03-17 12:22:49 +00:00
|
|
|
//
|
|
|
|
//*************************************************************************
|
|
|
|
|
2020-01-12 09:03:17 +00:00
|
|
|
#include VM_PREFIX_INCLUDE
|
2010-03-17 12:22:49 +00:00
|
|
|
#include "verilated.h"
|
|
|
|
#include "svdpi.h"
|
|
|
|
|
|
|
|
#include "verilated_syms.h"
|
|
|
|
|
|
|
|
//======================================================================
|
|
|
|
|
|
|
|
struct MyMon {
|
|
|
|
vluint32_t* sigsp[2];
|
2019-11-10 01:35:12 +00:00
|
|
|
MyMon() {
|
2020-08-24 22:49:36 +00:00
|
|
|
sigsp[0] = NULL;
|
|
|
|
sigsp[1] = NULL;
|
2019-11-10 01:35:12 +00:00
|
|
|
}
|
2010-03-17 12:22:49 +00:00
|
|
|
};
|
|
|
|
MyMon mons[2];
|
|
|
|
|
|
|
|
void mon_register_a(const char* namep, void* sigp, bool isOut) {
|
|
|
|
// Callback from initial block in monitor
|
|
|
|
#ifdef TEST_VERBOSE
|
|
|
|
VL_PRINTF("- mon_register_a(\"%s\", %p, %d);\n", namep, sigp, isOut);
|
|
|
|
#endif
|
|
|
|
mons[0].sigsp[isOut] = (vluint32_t*)sigp;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mon_do(MyMon* monp) {
|
2019-11-10 01:35:12 +00:00
|
|
|
if (!monp->sigsp[0]) vl_fatal(__FILE__, __LINE__, "", "never registered");
|
|
|
|
if (!monp->sigsp[1]) vl_fatal(__FILE__, __LINE__, "", "never registered");
|
|
|
|
*monp->sigsp[1] = (*(monp->sigsp[0])) + 1;
|
2010-03-17 12:22:49 +00:00
|
|
|
|
|
|
|
#ifdef TEST_VERBOSE
|
2019-11-10 01:35:12 +00:00
|
|
|
VL_PRINTF("- mon_do(%08x(&%p) -> %08x(&%p));\n", *(monp->sigsp[0]), monp->sigsp[0],
|
|
|
|
*(monp->sigsp[1]), monp->sigsp[1]);
|
2010-03-17 12:22:49 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void mon_class_name(const char* namep) {
|
|
|
|
#ifdef TEST_VERBOSE
|
|
|
|
VL_PRINTF("- mon_class_name(\"%s\");\n", namep);
|
|
|
|
#endif
|
|
|
|
// Check the C's calling name of "" doesn't lead to extra dots in the name()
|
2019-11-10 01:35:12 +00:00
|
|
|
if (namep && namep[0] == '.')
|
|
|
|
vl_fatal(__FILE__, __LINE__, "", (std::string("Unexp class name ") + namep).c_str());
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void mon_scope_name(const char* namep);
|
|
|
|
void mon_scope_name(const char* namep) {
|
|
|
|
const char* modp = svGetNameFromScope(svGetScope());
|
|
|
|
#ifdef TEST_VERBOSE
|
|
|
|
VL_PRINTF("- mon_scope_name('%s', \"%s\");\n", modp, namep);
|
|
|
|
#endif
|
2019-11-10 01:35:12 +00:00
|
|
|
if (strcmp(namep, "t.sub"))
|
|
|
|
vl_fatal(__FILE__, __LINE__, "", (std::string("Unexp scope name ") + namep).c_str());
|
|
|
|
if (strcmp(modp, "t.sub"))
|
|
|
|
vl_fatal(__FILE__, __LINE__, "", (std::string("Unexp dpiscope name ") + modp).c_str());
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void mon_register_b(const char* namep, int isOut);
|
|
|
|
void mon_register_b(const char* namep, int isOut) {
|
|
|
|
const char* modp = svGetNameFromScope(svGetScope());
|
|
|
|
#ifdef TEST_VERBOSE
|
|
|
|
VL_PRINTF("- mon_register_b('%s', \"%s\", %d);\n", modp, namep, isOut);
|
|
|
|
#endif
|
|
|
|
// Use scope to get pointer and size of signal
|
|
|
|
const VerilatedScope* scopep = Verilated::dpiScope();
|
|
|
|
const VerilatedVar* varp = scopep->varFind(namep);
|
|
|
|
if (!varp) {
|
2019-05-08 02:34:09 +00:00
|
|
|
VL_PRINTF("%%Warning: mon_register_b signal not found: \"%s\"\n", namep);
|
2010-03-17 12:22:49 +00:00
|
|
|
} else if (varp->vltype() != VLVT_UINT32) {
|
2019-05-08 02:34:09 +00:00
|
|
|
VL_PRINTF("%%Warning: wrong type for signal: \"%s\"\n", namep);
|
2010-03-17 12:22:49 +00:00
|
|
|
} else {
|
2019-05-08 02:34:09 +00:00
|
|
|
vluint32_t* datap = (vluint32_t*)(varp->datap());
|
|
|
|
VL_PRINTF("- mon_register_b('%s', \"%s\", %p, %d);\n", modp, namep, datap, isOut);
|
|
|
|
mons[1].sigsp[isOut] = (vluint32_t*)(varp->datap());
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void mon_register_done();
|
|
|
|
void mon_register_done() {
|
|
|
|
#ifdef TEST_VERBOSE
|
2019-10-25 01:48:45 +00:00
|
|
|
const char* modp = svGetNameFromScope(svGetScope());
|
2010-03-17 12:22:49 +00:00
|
|
|
VL_PRINTF("- mon_register_done('%s');\n", modp);
|
|
|
|
#endif
|
|
|
|
// Print list of all signals - if we didn't register2 anything we'd pick them off here
|
|
|
|
const VerilatedScope* scopep = Verilated::dpiScope();
|
|
|
|
if (VerilatedVarNameMap* varsp = scopep->varsp()) {
|
2019-11-10 01:35:12 +00:00
|
|
|
for (VerilatedVarNameMap::const_iterator it = varsp->begin(); it != varsp->end(); ++it) {
|
2019-05-08 02:34:09 +00:00
|
|
|
VL_PRINTF("- mon2: %s\n", it->first);
|
|
|
|
}
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-06 00:01:17 +00:00
|
|
|
extern "C" void mon_eval();
|
|
|
|
void mon_eval() {
|
2010-03-17 12:22:49 +00:00
|
|
|
// Callback from always@ negedge
|
|
|
|
mon_do(&mons[0]);
|
|
|
|
mon_do(&mons[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
//======================================================================
|
|
|
|
|
2020-02-29 13:56:49 +00:00
|
|
|
unsigned int main_time = 0;
|
2010-03-17 12:22:49 +00:00
|
|
|
|
2019-11-10 01:35:12 +00:00
|
|
|
double sc_time_stamp() { return main_time; }
|
|
|
|
int main(int argc, char** argv, char** env) {
|
2021-02-17 01:14:13 +00:00
|
|
|
vluint64_t sim_time = 1100;
|
2010-03-17 12:22:49 +00:00
|
|
|
Verilated::commandArgs(argc, argv);
|
|
|
|
Verilated::debug(0);
|
|
|
|
|
2018-08-25 13:52:45 +00:00
|
|
|
VM_PREFIX* topp = new VM_PREFIX(""); // Note null name - we're flattening it out
|
2010-03-17 12:22:49 +00:00
|
|
|
|
2020-10-28 22:42:36 +00:00
|
|
|
// clang-format off
|
2010-03-17 12:22:49 +00:00
|
|
|
#ifdef VERILATOR
|
|
|
|
# ifdef TEST_VERBOSE
|
|
|
|
Verilated::scopesDump();
|
|
|
|
# endif
|
|
|
|
#endif
|
2020-10-28 22:42:36 +00:00
|
|
|
// clang-format on
|
2010-03-17 12:22:49 +00:00
|
|
|
|
|
|
|
topp->eval();
|
|
|
|
topp->clk = 0;
|
|
|
|
main_time += 10;
|
|
|
|
|
2021-02-17 01:14:13 +00:00
|
|
|
while (vl_time_stamp64() < sim_time && !Verilated::gotFinish()) {
|
2019-05-08 02:34:09 +00:00
|
|
|
main_time += 1;
|
|
|
|
topp->eval();
|
|
|
|
topp->clk = !topp->clk;
|
2019-11-10 01:35:12 +00:00
|
|
|
// mon_do();
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
if (!Verilated::gotFinish()) {
|
2019-11-10 01:35:12 +00:00
|
|
|
vl_fatal(__FILE__, __LINE__, "main", "%Error: Timeout; never got a $finish");
|
2010-03-17 12:22:49 +00:00
|
|
|
}
|
|
|
|
topp->final();
|
|
|
|
|
2020-01-17 01:17:11 +00:00
|
|
|
VL_DO_DANGLING(delete topp, topp);
|
2010-03-17 12:22:49 +00:00
|
|
|
exit(0L);
|
|
|
|
}
|