2008-06-10 01:25:10 +00:00
|
|
|
|
// -*- C++ -*-
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//*************************************************************************
|
|
|
|
|
// DESCRIPTION: Verilator: File stream wrapper that understands indentation
|
|
|
|
|
//
|
2008-04-25 12:14:27 +00:00
|
|
|
|
// Code available from: http://www.veripool.org/verilator
|
2006-08-26 11:35:28 +00:00
|
|
|
|
//
|
|
|
|
|
// AUTHORS: Wilson Snyder with Paul Wasson, Duane Gabli
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
//
|
2008-01-15 14:29:08 +00:00
|
|
|
|
// Copyright 2003-2008 by Wilson Snyder. This program is free software; you can
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// redistribute it and/or modify it under the terms of either the GNU
|
|
|
|
|
// General Public License or the Perl Artistic License.
|
|
|
|
|
//
|
|
|
|
|
// Verilator 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.
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
|
|
|
|
|
#ifndef _V3FILE_H_
|
|
|
|
|
#define _V3FILE_H_ 1
|
2006-12-18 19:20:45 +00:00
|
|
|
|
#include "config_build.h"
|
|
|
|
|
#include "verilatedos.h"
|
2006-08-26 11:35:28 +00:00
|
|
|
|
#include "V3Error.h"
|
2008-06-30 17:11:25 +00:00
|
|
|
|
#include <cstdio>
|
2006-08-26 11:35:28 +00:00
|
|
|
|
#include <stack>
|
|
|
|
|
#include <set>
|
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
|
// V3File: Create streams, recording dependency information
|
|
|
|
|
|
|
|
|
|
class V3File {
|
|
|
|
|
public:
|
|
|
|
|
static ifstream* new_ifstream(const string& filename) {
|
|
|
|
|
addSrcDepend(filename);
|
|
|
|
|
return new_ifstream_nodepend (filename);
|
|
|
|
|
}
|
|
|
|
|
static ifstream* new_ifstream_nodepend(const string& filename) {
|
|
|
|
|
return new ifstream(filename.c_str());
|
|
|
|
|
}
|
|
|
|
|
static ofstream* new_ofstream(const string& filename, bool append=false) {
|
|
|
|
|
addTgtDepend(filename);
|
2006-10-12 14:01:06 +00:00
|
|
|
|
return new_ofstream_nodepend (filename, append);
|
|
|
|
|
}
|
|
|
|
|
static ofstream* new_ofstream_nodepend(const string& filename, bool append=false) {
|
2008-01-31 14:49:27 +00:00
|
|
|
|
if (filename != "/dev/null") createMakeDir();
|
2006-08-26 11:35:28 +00:00
|
|
|
|
if (append) {
|
|
|
|
|
return new ofstream(filename.c_str(), ios::app);
|
|
|
|
|
} else {
|
|
|
|
|
return new ofstream(filename.c_str());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
static FILE* new_fopen_w(const string& filename) {
|
2008-01-31 14:49:27 +00:00
|
|
|
|
if (filename != "/dev/null") createMakeDir();
|
2006-08-26 11:35:28 +00:00
|
|
|
|
addTgtDepend(filename);
|
|
|
|
|
return fopen(filename.c_str(),"w");
|
|
|
|
|
}
|
2008-01-31 13:50:06 +00:00
|
|
|
|
|
2006-08-26 11:35:28 +00:00
|
|
|
|
// Dependencies
|
|
|
|
|
static void addSrcDepend(const string& filename);
|
|
|
|
|
static void addTgtDepend(const string& filename);
|
|
|
|
|
static void writeDepend(const string& filename);
|
|
|
|
|
static void writeTimes(const string& filename, const string& cmdline);
|
|
|
|
|
static bool checkTimes(const string& filename, const string& cmdline);
|
2008-01-31 13:50:06 +00:00
|
|
|
|
|
|
|
|
|
// Directory utilities
|
|
|
|
|
static void createMakeDir();
|
2006-08-26 11:35:28 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
|
// V3OutFile: A class for printing to a file, with automatic indentation of C++ code.
|
|
|
|
|
|
|
|
|
|
class V3OutFile {
|
2006-08-29 00:58:48 +00:00
|
|
|
|
// TYPES
|
|
|
|
|
enum MiscConsts {
|
|
|
|
|
INDBLK = 4, // Indentation per block level
|
|
|
|
|
WIDTH = 50, // Width after which to break at ,'s
|
|
|
|
|
MAXSPACE = 80}; // After this indent, stop indenting more
|
|
|
|
|
public:
|
|
|
|
|
enum AlignClass {
|
|
|
|
|
AL_AUTO = 0,
|
|
|
|
|
AL_STATIC = 1};
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
// MEMBERS
|
2006-08-26 11:35:28 +00:00
|
|
|
|
FILE* m_fp;
|
|
|
|
|
int m_lineno;
|
|
|
|
|
int m_column;
|
|
|
|
|
int m_nobreak; // Basic operator or begin paren, don't break next
|
|
|
|
|
bool m_prependIndent;
|
|
|
|
|
int m_indentLevel; // Current {} indentation
|
2006-08-29 00:58:48 +00:00
|
|
|
|
int m_declSAlign; // Byte alignment of next declaration, statics
|
|
|
|
|
int m_declNSAlign; // Byte alignment of next declaration, nonstatics
|
2006-08-26 11:35:28 +00:00
|
|
|
|
int m_declPadNum; // Pad variable number
|
|
|
|
|
stack<int> m_parenVec; // Stack of columns where last ( was
|
|
|
|
|
|
|
|
|
|
int endLevels(const char* strg);
|
|
|
|
|
static const char* indentStr(int levels);
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
V3OutFile(const string& filename);
|
|
|
|
|
~V3OutFile();
|
|
|
|
|
// ACCESSORS
|
|
|
|
|
int column() const { return m_column; }
|
|
|
|
|
// METHODS
|
|
|
|
|
void printf(const char* fmt...) VL_ATTR_PRINTF(2);
|
|
|
|
|
void puts(const char* strg);
|
|
|
|
|
void puts(const string& strg) { puts(strg.c_str()); }
|
|
|
|
|
void putsNoTracking(const char* strg);
|
|
|
|
|
void putsNoTracking(const string& strg) { putsNoTracking(strg.c_str()); }
|
|
|
|
|
void putBreak(); // Print linebreak if line is too wide
|
|
|
|
|
void putBreakExpr(); // Print linebreak in expression if line is too wide
|
2006-08-29 00:58:48 +00:00
|
|
|
|
void putAlign(bool isstatic/*AlignClass*/, int align, int size=0/*=align*/, const char* prefix=""); // Declare a variable, with natural alignment
|
2006-08-26 11:35:28 +00:00
|
|
|
|
void putbs(const char* strg) { putBreakExpr(); puts(strg); }
|
|
|
|
|
void putbs(const string& strg) { putBreakExpr(); puts(strg); }
|
|
|
|
|
bool exceededWidth() const { return m_column > WIDTH; }
|
|
|
|
|
bool tokenStart(const char* cp, const char* cmp);
|
|
|
|
|
bool tokenEnd(const char* cp);
|
|
|
|
|
void indentInc() { m_indentLevel += INDBLK; };
|
|
|
|
|
void indentDec() { m_indentLevel -= INDBLK;
|
|
|
|
|
UASSERT(m_indentLevel>=0,"Underflow of indentation\n");
|
|
|
|
|
}
|
|
|
|
|
void blockInc() { m_parenVec.push(m_indentLevel + INDBLK); }
|
|
|
|
|
void blockDec() { if (!m_parenVec.empty()) m_parenVec.pop(); }
|
|
|
|
|
// STATIC METHODS
|
|
|
|
|
static const string indentSpaces(int levels);
|
|
|
|
|
private:
|
|
|
|
|
void putcNoTracking(char chr);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#endif // Guard
|