2018-10-02 22:42:53 +00:00
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
|
|
|
//=============================================================================
|
|
|
|
//
|
|
|
|
// THIS MODULE IS PUBLICLY LICENSED
|
|
|
|
//
|
2020-03-21 15:24:24 +00:00
|
|
|
// Copyright 2001-2020 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
|
2018-10-02 22:42:53 +00:00
|
|
|
//
|
|
|
|
//=============================================================================
|
|
|
|
///
|
|
|
|
/// \file
|
|
|
|
/// \brief C++ Tracing in FST Format
|
|
|
|
///
|
|
|
|
//=============================================================================
|
|
|
|
// SPDIFF_OFF
|
|
|
|
|
2020-04-13 23:13:10 +00:00
|
|
|
// clang-format off
|
|
|
|
|
2020-02-08 12:09:41 +00:00
|
|
|
#define __STDC_LIMIT_MACROS // UINT64_MAX
|
2018-10-02 22:42:53 +00:00
|
|
|
#include "verilated.h"
|
|
|
|
#include "verilated_fst_c.h"
|
2018-10-14 17:43:24 +00:00
|
|
|
|
2019-05-03 00:33:05 +00:00
|
|
|
// GTKWave configuration
|
2020-04-21 22:49:07 +00:00
|
|
|
#ifdef VL_TRACE_FST_WRITER_THREAD
|
2019-05-03 00:33:05 +00:00
|
|
|
# define HAVE_LIBPTHREAD
|
|
|
|
# define FST_WRITER_PARALLEL
|
|
|
|
#endif
|
|
|
|
|
2018-10-02 22:42:53 +00:00
|
|
|
// Include the GTKWave implementation directly
|
2019-11-16 08:54:05 +00:00
|
|
|
#define FST_CONFIG_INCLUDE "fst_config.h"
|
2018-10-02 22:42:53 +00:00
|
|
|
#include "gtkwave/fastlz.c"
|
2018-10-14 17:43:24 +00:00
|
|
|
#include "gtkwave/fstapi.c"
|
2018-10-02 22:42:53 +00:00
|
|
|
#include "gtkwave/lz4.c"
|
|
|
|
|
2018-10-14 11:04:18 +00:00
|
|
|
#include <algorithm>
|
2018-10-02 22:42:53 +00:00
|
|
|
#include <cerrno>
|
|
|
|
#include <ctime>
|
2018-10-14 11:04:18 +00:00
|
|
|
#include <fcntl.h>
|
2018-10-02 22:42:53 +00:00
|
|
|
#include <iterator>
|
2018-10-14 11:04:18 +00:00
|
|
|
#include <sstream>
|
|
|
|
#include <sys/stat.h>
|
2018-10-02 22:42:53 +00:00
|
|
|
|
|
|
|
#if defined(_WIN32) && !defined(__MINGW32__) && !defined(__CYGWIN__)
|
|
|
|
# include <io.h>
|
|
|
|
#else
|
2020-02-08 12:09:41 +00:00
|
|
|
# include <stdint.h>
|
2018-10-02 22:42:53 +00:00
|
|
|
# include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2020-04-13 23:13:10 +00:00
|
|
|
// clang-format on
|
|
|
|
|
2018-10-02 22:42:53 +00:00
|
|
|
//=============================================================================
|
2020-04-19 22:57:36 +00:00
|
|
|
// Specialization of the generics for this trace format
|
2018-10-02 22:42:53 +00:00
|
|
|
|
2020-04-19 22:57:36 +00:00
|
|
|
#define VL_DERIVED_T VerilatedFst
|
|
|
|
#include "verilated_trace_imp.cpp"
|
|
|
|
#undef VL_DERIVED_T
|
2018-10-02 22:42:53 +00:00
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// VerilatedFst
|
|
|
|
|
|
|
|
VerilatedFst::VerilatedFst(void* fst)
|
2020-03-07 23:39:58 +00:00
|
|
|
: m_fst(fst)
|
2020-04-19 22:57:36 +00:00
|
|
|
, m_symbolp(NULL) {}
|
2018-10-02 22:42:53 +00:00
|
|
|
|
2020-04-08 21:54:35 +00:00
|
|
|
VerilatedFst::~VerilatedFst() {
|
|
|
|
if (m_fst) fstWriterClose(m_fst);
|
|
|
|
if (m_symbolp) VL_DO_CLEAR(delete[] m_symbolp, m_symbolp = NULL);
|
|
|
|
}
|
|
|
|
|
2018-10-02 22:42:53 +00:00
|
|
|
void VerilatedFst::open(const char* filename) VL_MT_UNSAFE {
|
|
|
|
m_assertOne.check();
|
2018-10-08 00:39:51 +00:00
|
|
|
m_fst = fstWriterCreate(filename, 1);
|
2019-05-02 23:55:16 +00:00
|
|
|
fstWriterSetPackType(m_fst, FST_WR_PT_LZ4);
|
2020-04-19 22:57:36 +00:00
|
|
|
fstWriterSetTimescaleFromString(m_fst, timeResStr().c_str()); // lintok-begin-on-ref
|
2020-04-21 22:49:07 +00:00
|
|
|
#ifdef VL_TRACE_FST_WRITER_THREAD
|
2019-05-03 00:33:05 +00:00
|
|
|
fstWriterSetParallelMode(m_fst, 1);
|
|
|
|
#endif
|
2020-04-21 22:49:07 +00:00
|
|
|
|
2018-10-02 22:42:53 +00:00
|
|
|
m_curScope.clear();
|
|
|
|
|
2020-04-19 22:57:36 +00:00
|
|
|
VerilatedTrace<VerilatedFst>::traceInit();
|
2018-10-02 22:42:53 +00:00
|
|
|
|
|
|
|
// Clear the scope stack
|
|
|
|
std::list<std::string>::iterator it = m_curScope.begin();
|
|
|
|
while (it != m_curScope.end()) {
|
|
|
|
fstWriterSetUpscope(m_fst);
|
|
|
|
it = m_curScope.erase(it);
|
|
|
|
}
|
2020-04-08 21:54:35 +00:00
|
|
|
|
|
|
|
// convert m_code2symbol into an array for fast lookup
|
|
|
|
if (!m_symbolp) {
|
2020-04-19 22:57:36 +00:00
|
|
|
m_symbolp = new fstHandle[nextCode()];
|
2020-04-08 21:54:35 +00:00
|
|
|
for (Code2SymbolType::iterator it = m_code2symbol.begin(); it != m_code2symbol.end();
|
|
|
|
++it) {
|
|
|
|
m_symbolp[it->first] = it->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_code2symbol.clear();
|
2018-10-02 22:42:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-21 22:49:07 +00:00
|
|
|
void VerilatedFst::close() {
|
|
|
|
m_assertOne.check();
|
|
|
|
VerilatedTrace<VerilatedFst>::close();
|
|
|
|
fstWriterClose(m_fst);
|
|
|
|
m_fst = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VerilatedFst::flush() {
|
|
|
|
VerilatedTrace<VerilatedFst>::flush();
|
|
|
|
fstWriterFlushContext(m_fst);
|
|
|
|
}
|
|
|
|
|
2020-04-19 22:57:36 +00:00
|
|
|
void VerilatedFst::emitTimeChange(vluint64_t timeui) { fstWriterEmitTimeChange(m_fst, timeui); }
|
2018-10-02 22:42:53 +00:00
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// Decl
|
|
|
|
|
2018-10-08 11:21:22 +00:00
|
|
|
void VerilatedFst::declDTypeEnum(int dtypenum, const char* name, vluint32_t elements,
|
2020-04-04 17:45:24 +00:00
|
|
|
unsigned int minValbits, const char** itemNamesp,
|
|
|
|
const char** itemValuesp) {
|
|
|
|
fstEnumHandle enumNum
|
|
|
|
= fstWriterCreateEnumTable(m_fst, name, elements, minValbits, itemNamesp, itemValuesp);
|
2018-10-08 11:21:22 +00:00
|
|
|
m_local2fstdtype[dtypenum] = enumNum;
|
|
|
|
}
|
|
|
|
|
2020-03-07 23:39:58 +00:00
|
|
|
void VerilatedFst::declSymbol(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum, vluint32_t len,
|
|
|
|
vluint32_t bits) {
|
2020-04-19 22:57:36 +00:00
|
|
|
VerilatedTrace<VerilatedFst>::declCode(code, bits, false);
|
2020-03-07 23:39:58 +00:00
|
|
|
|
2018-10-02 22:42:53 +00:00
|
|
|
std::pair<Code2SymbolType::iterator, bool> p
|
2018-10-14 22:39:33 +00:00
|
|
|
= m_code2symbol.insert(std::make_pair(code, static_cast<fstHandle>(NULL)));
|
2018-10-02 22:42:53 +00:00
|
|
|
std::istringstream nameiss(name);
|
|
|
|
std::istream_iterator<std::string> beg(nameiss), end;
|
|
|
|
std::list<std::string> tokens(beg, end); // Split name
|
|
|
|
std::string symbol_name(tokens.back());
|
|
|
|
tokens.pop_back(); // Remove symbol name from hierarchy
|
2020-04-19 22:57:36 +00:00
|
|
|
tokens.insert(tokens.begin(), moduleName()); // Add current module to the hierarchy
|
2018-10-02 22:42:53 +00:00
|
|
|
|
|
|
|
// Find point where current and new scope diverge
|
|
|
|
std::list<std::string>::iterator cur_it = m_curScope.begin();
|
|
|
|
std::list<std::string>::iterator new_it = tokens.begin();
|
|
|
|
while (cur_it != m_curScope.end() && new_it != tokens.end()) {
|
2020-04-04 17:45:24 +00:00
|
|
|
if (*cur_it != *new_it) break;
|
2018-10-02 22:42:53 +00:00
|
|
|
++cur_it;
|
|
|
|
++new_it;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go back to the common point
|
|
|
|
while (cur_it != m_curScope.end()) {
|
|
|
|
fstWriterSetUpscope(m_fst);
|
|
|
|
cur_it = m_curScope.erase(cur_it);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Follow the hierarchy of the new variable from the common scope point
|
|
|
|
while (new_it != tokens.end()) {
|
|
|
|
fstWriterSetScope(m_fst, FST_ST_VCD_SCOPE, new_it->c_str(), NULL);
|
|
|
|
m_curScope.push_back(*new_it);
|
|
|
|
new_it = tokens.erase(new_it);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::stringstream name_ss;
|
|
|
|
name_ss << symbol_name;
|
2020-01-08 12:32:31 +00:00
|
|
|
if (array) name_ss << "(" << arraynum << ")";
|
2018-10-02 22:42:53 +00:00
|
|
|
std::string name_str = name_ss.str();
|
|
|
|
|
2018-10-08 11:21:22 +00:00
|
|
|
if (dtypenum > 0) {
|
|
|
|
fstEnumHandle enumNum = m_local2fstdtype[dtypenum];
|
|
|
|
fstWriterEmitEnumTableRef(m_fst, enumNum);
|
|
|
|
}
|
2018-10-02 22:42:53 +00:00
|
|
|
if (p.second) { // New
|
2018-10-03 23:51:05 +00:00
|
|
|
p.first->second = fstWriterCreateVar(m_fst, vartype, vardir, len, name_str.c_str(), 0);
|
2018-10-02 22:42:53 +00:00
|
|
|
assert(p.first->second);
|
|
|
|
} else { // Alias
|
2018-10-03 23:51:05 +00:00
|
|
|
fstWriterCreateVar(m_fst, vartype, vardir, len, name_str.c_str(), p.first->second);
|
2018-10-02 22:42:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 22:57:36 +00:00
|
|
|
void VerilatedFst::declBit(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, 1, 1);
|
2018-10-02 22:42:53 +00:00
|
|
|
}
|
2020-04-19 22:57:36 +00:00
|
|
|
void VerilatedFst::declBus(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum, int msb, int lsb) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, msb - lsb + 1,
|
|
|
|
msb - lsb + 1);
|
|
|
|
}
|
|
|
|
void VerilatedFst::declQuad(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum, int msb, int lsb) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, msb - lsb + 1,
|
|
|
|
msb - lsb + 1);
|
|
|
|
}
|
|
|
|
void VerilatedFst::declArray(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum, int msb, int lsb) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, msb - lsb + 1,
|
|
|
|
msb - lsb + 1);
|
|
|
|
}
|
|
|
|
void VerilatedFst::declFloat(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, 1, 32);
|
|
|
|
}
|
|
|
|
void VerilatedFst::declDouble(vluint32_t code, const char* name, int dtypenum, fstVarDir vardir,
|
|
|
|
fstVarType vartype, bool array, int arraynum) {
|
|
|
|
declSymbol(code, name, dtypenum, vardir, vartype, array, arraynum, 2, 64);
|
2018-10-02 22:42:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-19 22:57:36 +00:00
|
|
|
void VerilatedFst::emitBit(vluint32_t code, vluint32_t newval) {
|
|
|
|
fstWriterEmitValueChange(m_fst, m_symbolp[code], newval ? "1" : "0");
|
|
|
|
}
|
|
|
|
template <int T_Bits> void VerilatedFst::emitBus(vluint32_t code, vluint32_t newval) {
|
|
|
|
fstWriterEmitValueChange32(m_fst, m_symbolp[code], T_Bits, newval);
|
|
|
|
}
|
|
|
|
void VerilatedFst::emitQuad(vluint32_t code, vluint64_t newval, int bits) {
|
|
|
|
fstWriterEmitValueChange64(m_fst, m_symbolp[code], bits, newval);
|
|
|
|
}
|
|
|
|
void VerilatedFst::emitArray(vluint32_t code, const vluint32_t* newvalp, int bits) {
|
|
|
|
fstWriterEmitValueChangeVec32(m_fst, m_symbolp[code], bits, newvalp);
|
|
|
|
}
|
|
|
|
void VerilatedFst::emitFloat(vluint32_t code, float newval) {
|
|
|
|
fstWriterEmitValueChange(m_fst, m_symbolp[code], &newval);
|
|
|
|
}
|
|
|
|
void VerilatedFst::emitDouble(vluint32_t code, double newval) {
|
|
|
|
fstWriterEmitValueChange(m_fst, m_symbolp[code], &newval);
|
|
|
|
}
|