2012-08-27 01:13:47 +00:00
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
|
|
|
//=============================================================================
|
|
|
|
//
|
|
|
|
// THIS MODULE IS PUBLICLY LICENSED
|
|
|
|
//
|
2017-01-15 17:09:59 +00:00
|
|
|
// Copyright 2012-2017 by Wilson Snyder. This program is free software;
|
2012-08-27 01:13:47 +00:00
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// This is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
//=============================================================================
|
|
|
|
///
|
|
|
|
/// \file
|
|
|
|
/// \brief Save-restore serialization of verilated modules
|
|
|
|
///
|
|
|
|
//=============================================================================
|
|
|
|
|
|
|
|
#ifndef _VERILATED_SAVE_C_H_
|
|
|
|
#define _VERILATED_SAVE_C_H_ 1
|
|
|
|
|
|
|
|
#include "verilatedos.h"
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
//=============================================================================
|
2017-10-11 22:55:31 +00:00
|
|
|
// VerilatedSerialize - convert structures to a stream representation
|
2017-10-27 00:05:42 +00:00
|
|
|
// This class is not thread safe, it must be called by a single thread
|
2012-08-27 01:13:47 +00:00
|
|
|
|
2017-10-11 22:55:31 +00:00
|
|
|
class VerilatedSerialize {
|
2012-08-27 01:13:47 +00:00
|
|
|
protected:
|
|
|
|
// MEMBERS
|
|
|
|
// For speed, keep m_cp as the first member of this structure
|
|
|
|
vluint8_t* m_cp; ///< Current pointer into m_bufp buffer
|
|
|
|
vluint8_t* m_bufp; ///< Output buffer
|
|
|
|
bool m_isOpen; ///< True indicates open file/stream
|
2017-10-11 22:55:31 +00:00
|
|
|
std::string m_filename; ///< Filename, for error messages
|
2012-08-27 01:13:47 +00:00
|
|
|
|
|
|
|
inline static size_t bufferSize() { return 256*1024; } // See below for slack calculation
|
|
|
|
inline static size_t bufferInsertSize() { return 16*1024; }
|
|
|
|
|
2017-10-11 22:55:31 +00:00
|
|
|
void header();
|
|
|
|
void trailer();
|
|
|
|
public:
|
2012-08-27 01:13:47 +00:00
|
|
|
// CREATORS
|
2017-10-11 22:55:31 +00:00
|
|
|
VerilatedSerialize() {
|
2012-08-27 01:13:47 +00:00
|
|
|
m_isOpen = false;
|
|
|
|
m_bufp = new vluint8_t [bufferSize()];
|
|
|
|
m_cp = m_bufp;
|
|
|
|
}
|
2017-10-11 22:55:31 +00:00
|
|
|
virtual ~VerilatedSerialize() {
|
2012-08-27 01:13:47 +00:00
|
|
|
close();
|
|
|
|
if (m_bufp) { delete m_bufp; m_bufp=NULL; }
|
|
|
|
}
|
|
|
|
// METHODS
|
|
|
|
bool isOpen() const { return m_isOpen; }
|
2017-09-23 11:32:37 +00:00
|
|
|
std::string filename() const { return m_filename; }
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual void close() { flush(); }
|
|
|
|
virtual void flush() {}
|
|
|
|
inline VerilatedSerialize& write (const void* __restrict datap, size_t size) {
|
|
|
|
const vluint8_t* __restrict dp = (const vluint8_t* __restrict)datap;
|
|
|
|
while (size) {
|
|
|
|
bufferCheck();
|
|
|
|
size_t blk = size; if (blk>bufferInsertSize()) blk = bufferInsertSize();
|
|
|
|
const vluint8_t* __restrict maxp = dp + blk;
|
|
|
|
while (dp < maxp) *m_cp++ = *dp++;
|
|
|
|
size -= blk;
|
|
|
|
}
|
|
|
|
return *this; // For function chaining
|
|
|
|
}
|
2017-10-11 22:55:31 +00:00
|
|
|
private:
|
2017-10-26 00:00:38 +00:00
|
|
|
VerilatedSerialize(const VerilatedSerialize&) VL_EQ_DELETE; ///< N/A, no copy constructor
|
2017-10-11 22:55:31 +00:00
|
|
|
VerilatedSerialize& bufferCheck() {
|
|
|
|
// Flush the write buffer if there's not enough space left for new information
|
|
|
|
// We only call this once per vector, so we need enough slop for a very wide "b###" line
|
|
|
|
if (VL_UNLIKELY(m_cp > (m_bufp+(bufferSize()-bufferInsertSize())))) {
|
|
|
|
flush();
|
|
|
|
}
|
|
|
|
return *this; // For function chaining
|
|
|
|
}
|
2012-08-27 01:13:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// VerilatedDeserial - load structures from a stream representation
|
2017-10-27 00:05:42 +00:00
|
|
|
// This class is not thread safe, it must be called by a single thread
|
2012-08-27 01:13:47 +00:00
|
|
|
|
2017-10-11 22:55:31 +00:00
|
|
|
class VerilatedDeserialize {
|
2012-08-27 01:13:47 +00:00
|
|
|
protected:
|
2017-10-11 22:55:31 +00:00
|
|
|
// MEMBERS
|
|
|
|
// For speed, keep m_cp as the first member of this structure
|
|
|
|
vluint8_t* m_cp; ///< Current pointer into m_bufp buffer
|
|
|
|
vluint8_t* m_bufp; ///< Output buffer
|
2012-08-27 01:13:47 +00:00
|
|
|
vluint8_t* m_endp; ///< Last valid byte in m_bufp buffer
|
2017-10-11 22:55:31 +00:00
|
|
|
bool m_isOpen; ///< True indicates open file/stream
|
|
|
|
std::string m_filename; ///< Filename, for error messages
|
|
|
|
|
|
|
|
inline static size_t bufferSize() { return 256*1024; } // See below for slack calculation
|
|
|
|
inline static size_t bufferInsertSize() { return 16*1024; }
|
|
|
|
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual void fill() = 0;
|
|
|
|
void header();
|
|
|
|
void trailer();
|
|
|
|
public:
|
|
|
|
// CREATORS
|
2017-10-11 22:55:31 +00:00
|
|
|
VerilatedDeserialize() {
|
|
|
|
m_isOpen = false;
|
|
|
|
m_bufp = new vluint8_t [bufferSize()];
|
|
|
|
m_cp = m_bufp;
|
|
|
|
m_endp = NULL;
|
|
|
|
}
|
|
|
|
virtual ~VerilatedDeserialize() {
|
|
|
|
close();
|
|
|
|
if (m_bufp) { delete m_bufp; m_bufp=NULL; }
|
|
|
|
}
|
2012-08-27 01:13:47 +00:00
|
|
|
// METHODS
|
2017-10-11 22:55:31 +00:00
|
|
|
bool isOpen() const { return m_isOpen; }
|
|
|
|
std::string filename() const { return m_filename; }
|
|
|
|
virtual void close() { flush(); }
|
|
|
|
virtual void flush() {}
|
2012-08-27 01:13:47 +00:00
|
|
|
inline VerilatedDeserialize& read (void* __restrict datap, size_t size) {
|
|
|
|
vluint8_t* __restrict dp = (vluint8_t* __restrict)datap;
|
|
|
|
while (size) {
|
|
|
|
bufferCheck();
|
|
|
|
size_t blk = size; if (blk>bufferInsertSize()) blk = bufferInsertSize();
|
|
|
|
const vluint8_t* __restrict maxp = dp + blk;
|
|
|
|
while (dp < maxp) *dp++ = *m_cp++;
|
|
|
|
size -= blk;
|
|
|
|
}
|
|
|
|
return *this; // For function chaining
|
|
|
|
}
|
|
|
|
// Read a datum and compare with expected value
|
|
|
|
bool readDiffers (const void* __restrict datap, size_t size);
|
|
|
|
VerilatedDeserialize& readAssert (const void* __restrict datap, size_t size);
|
|
|
|
VerilatedDeserialize& readAssert (vluint64_t data) { return readAssert(&data, sizeof(data)); }
|
2017-10-11 22:55:31 +00:00
|
|
|
private:
|
2017-10-26 00:00:38 +00:00
|
|
|
VerilatedDeserialize(const VerilatedDeserialize&) VL_EQ_DELETE; ///< N/A, no copy constructor
|
2012-08-27 01:13:47 +00:00
|
|
|
VerilatedDeserialize& bufferCheck() {
|
|
|
|
// Flush the write buffer if there's not enough space left for new information
|
|
|
|
// We only call this once per vector, so we need enough slop for a very wide "b###" line
|
|
|
|
if (VL_UNLIKELY((m_cp+bufferInsertSize()) > m_endp)) {
|
|
|
|
fill();
|
|
|
|
}
|
|
|
|
return *this; // For function chaining
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// VerilatedSave - serialize to a file
|
|
|
|
|
|
|
|
class VerilatedSave : public VerilatedSerialize {
|
|
|
|
private:
|
|
|
|
int m_fd; ///< File descriptor we're writing to
|
|
|
|
|
|
|
|
public:
|
|
|
|
// CREATORS
|
2013-02-03 18:27:37 +00:00
|
|
|
VerilatedSave() { m_fd=-1; }
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual ~VerilatedSave() { close(); }
|
|
|
|
// METHODS
|
|
|
|
void open(const char* filenamep); ///< Open the file; call isOpen() to see if errors
|
2017-09-23 11:32:37 +00:00
|
|
|
void open(const std::string& filename) { open(filename.c_str()); }
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual void close();
|
|
|
|
virtual void flush();
|
|
|
|
};
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
// VerilatedRestore - deserialize from a file
|
|
|
|
|
|
|
|
class VerilatedRestore : public VerilatedDeserialize {
|
|
|
|
private:
|
|
|
|
int m_fd; ///< File descriptor we're writing to
|
|
|
|
|
|
|
|
public:
|
|
|
|
// CREATORS
|
2013-02-03 18:27:37 +00:00
|
|
|
VerilatedRestore() { m_fd=-1; }
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual ~VerilatedRestore() { close(); }
|
|
|
|
|
|
|
|
// METHODS
|
|
|
|
void open(const char* filenamep); ///< Open the file; call isOpen() to see if errors
|
2017-09-23 11:32:37 +00:00
|
|
|
void open(const std::string& filename) { open(filename.c_str()); }
|
2012-08-27 01:13:47 +00:00
|
|
|
virtual void close();
|
|
|
|
virtual void flush() {}
|
|
|
|
virtual void fill();
|
|
|
|
};
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, vluint64_t& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, vluint64_t& rhs){
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, vluint32_t& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, vluint32_t& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, vluint16_t& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, vluint16_t& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, vluint8_t& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, vluint8_t& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, bool& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, bool& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, double& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, double& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, float& rhs) {
|
|
|
|
return os.write(&rhs, sizeof(rhs));
|
|
|
|
}
|
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, float& rhs) {
|
|
|
|
return os.read(&rhs, sizeof(rhs));
|
|
|
|
}
|
2017-09-23 11:32:37 +00:00
|
|
|
inline VerilatedSerialize& operator<<(VerilatedSerialize& os, std::string& rhs) {
|
2012-08-27 01:13:47 +00:00
|
|
|
vluint32_t len=rhs.length();
|
|
|
|
os<<len;
|
|
|
|
return os.write(rhs.data(), len);
|
|
|
|
}
|
2017-09-23 11:32:37 +00:00
|
|
|
inline VerilatedDeserialize& operator>>(VerilatedDeserialize& os, std::string& rhs) {
|
2015-10-04 17:11:32 +00:00
|
|
|
vluint32_t len=0;
|
2012-08-27 01:13:47 +00:00
|
|
|
os>>len;
|
|
|
|
rhs.resize(len);
|
|
|
|
return os.read((void*)rhs.data(), len);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // guard
|