2014-11-22 16:48:39 +00:00
|
|
|
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
// DESCRIPTION: Verilator: Error handling
|
|
|
|
|
//
|
|
|
|
|
// Code available from: http://www.veripool.org/verilator
|
|
|
|
|
//
|
|
|
|
|
//*************************************************************************
|
|
|
|
|
//
|
2019-01-04 00:17:22 +00:00
|
|
|
|
// Copyright 2003-2019 by Wilson Snyder. This program is free software; you can
|
2014-11-22 16:48:39 +00:00
|
|
|
|
// 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.
|
|
|
|
|
//
|
|
|
|
|
// 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 _V3FileLine_H_
|
|
|
|
|
#define _V3FileLine_H_ 1
|
2018-10-14 17:43:24 +00:00
|
|
|
|
|
2014-11-22 16:48:39 +00:00
|
|
|
|
#include "config_build.h"
|
|
|
|
|
#include "verilatedos.h"
|
2018-10-14 17:43:24 +00:00
|
|
|
|
|
|
|
|
|
#include "V3Error.h"
|
|
|
|
|
#include "V3LangCode.h"
|
|
|
|
|
|
2014-11-22 16:48:39 +00:00
|
|
|
|
#include <sstream>
|
|
|
|
|
#include <bitset>
|
|
|
|
|
#include <map>
|
|
|
|
|
#include <set>
|
|
|
|
|
#include <deque>
|
|
|
|
|
|
|
|
|
|
//######################################################################
|
|
|
|
|
|
|
|
|
|
class FileLine;
|
|
|
|
|
|
|
|
|
|
//! Singleton class with tables of per-file data.
|
|
|
|
|
|
|
|
|
|
//! This singleton class contains tables of data that are unchanging in each
|
|
|
|
|
//! source file (each with its own unique filename number).
|
|
|
|
|
class FileLineSingleton {
|
|
|
|
|
// TYPES
|
2018-02-02 02:24:41 +00:00
|
|
|
|
typedef std::map<string,int> FileNameNumMap;
|
|
|
|
|
typedef std::map<string,V3LangCode> FileLangNumMap;
|
2014-11-22 16:48:39 +00:00
|
|
|
|
// MEMBERS
|
2019-05-19 20:13:13 +00:00
|
|
|
|
FileNameNumMap m_namemap; // filenameno for each filename
|
2018-02-02 02:24:41 +00:00
|
|
|
|
std::deque<string> m_names; // filename text for each filenameno
|
2019-05-19 20:13:13 +00:00
|
|
|
|
std::deque<V3LangCode> m_languages; // language for each filenameno
|
2019-07-11 02:04:58 +00:00
|
|
|
|
// CONSTRUCTORS
|
2014-11-22 16:48:39 +00:00
|
|
|
|
FileLineSingleton() { }
|
|
|
|
|
~FileLineSingleton() { }
|
|
|
|
|
protected:
|
|
|
|
|
friend class FileLine;
|
|
|
|
|
int nameToNumber(const string& filename);
|
|
|
|
|
const string numberToName(int filenameno) const { return m_names[filenameno]; }
|
|
|
|
|
const V3LangCode numberToLang(int filenameno) const { return m_languages[filenameno]; }
|
|
|
|
|
void numberToLang(int filenameno, const V3LangCode& l) { m_languages[filenameno] = l; }
|
|
|
|
|
void clear() { m_namemap.clear(); m_names.clear(); m_languages.clear(); }
|
2018-02-02 02:24:41 +00:00
|
|
|
|
void fileNameNumMapDumpXml(std::ostream& os);
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static const string filenameLetters(int fileno);
|
|
|
|
|
};
|
|
|
|
|
|
2019-07-15 01:42:03 +00:00
|
|
|
|
//! All source lines from a file/stream, to enable errors to show sources
|
|
|
|
|
class VFileContent {
|
|
|
|
|
// MEMBERS
|
|
|
|
|
int m_id; // Content ID number
|
|
|
|
|
std::deque<string> m_lines; // Source text lines
|
|
|
|
|
public:
|
|
|
|
|
VFileContent() { static int s_id = 0; m_id = ++s_id; }
|
|
|
|
|
~VFileContent() { }
|
|
|
|
|
// METHODS
|
|
|
|
|
void pushText(const string& text); // Add arbitrary text (need not be line-by-line)
|
|
|
|
|
string getLine(int lineno) const;
|
|
|
|
|
string ascii() const { return "ct"+cvtToStr(m_id); }
|
|
|
|
|
static int debug();
|
|
|
|
|
};
|
|
|
|
|
std::ostream& operator<<(std::ostream& os, VFileContent* contentp);
|
|
|
|
|
|
2014-11-22 16:48:39 +00:00
|
|
|
|
//! File and line number of an object, mostly for error reporting
|
|
|
|
|
|
|
|
|
|
//! This class is instantiated for every source code line (potentially
|
|
|
|
|
//! millions). To save space, per-file information (e.g. filename, source
|
|
|
|
|
//! language is held in tables in the FileLineSingleton class.
|
|
|
|
|
class FileLine {
|
2019-07-15 01:42:03 +00:00
|
|
|
|
// CONSTANTS
|
|
|
|
|
enum { SHOW_SOURCE_MAX_LENGTH = 400 }; // Don't show source lines > this long
|
|
|
|
|
|
2019-07-11 02:04:58 +00:00
|
|
|
|
// MEMBERS
|
2019-07-15 01:42:03 +00:00
|
|
|
|
// Columns here means number of chars from beginning (i.e. tabs count as one)
|
|
|
|
|
int m_firstLineno; // `line corrected token's first line number
|
|
|
|
|
int m_firstColumn; // `line corrected token's first column number
|
|
|
|
|
int m_lastLineno; // `line corrected token's last line number
|
|
|
|
|
int m_lastColumn; // `line corrected token's last column number
|
2019-06-29 01:43:37 +00:00
|
|
|
|
int m_filenameno; // `line corrected filename number
|
2019-07-15 01:42:03 +00:00
|
|
|
|
int m_contentLineno; // Line number within source stream
|
|
|
|
|
VFileContent* m_contentp; // Source text contents line is within
|
2019-06-22 21:01:39 +00:00
|
|
|
|
FileLine* m_parent; // Parent line that included this line
|
2018-02-02 02:24:41 +00:00
|
|
|
|
std::bitset<V3ErrorCode::_ENUM_MAX> m_warnOn;
|
2014-11-22 16:48:39 +00:00
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
// User routines should never need to change line numbers
|
|
|
|
|
// We are storing pointers, so we CAN'T change them after initial reading.
|
|
|
|
|
friend class FileLineSingleton;
|
|
|
|
|
friend class V3ParseImp;
|
|
|
|
|
friend class V3PreLex;
|
|
|
|
|
friend class V3PreProcImp;
|
2019-03-10 18:09:22 +00:00
|
|
|
|
friend class V3PreShellImp;
|
2019-07-11 02:04:58 +00:00
|
|
|
|
private:
|
|
|
|
|
// CONSTRUCTORS
|
|
|
|
|
static FileLineSingleton& singleton() {
|
|
|
|
|
static FileLineSingleton s;
|
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
static FileLine& defaultFileLine() {
|
|
|
|
|
static FileLine* defFilelinep = new FileLine(FileLine::EmptySecret());
|
|
|
|
|
return *defFilelinep;
|
|
|
|
|
}
|
2014-11-22 16:48:39 +00:00
|
|
|
|
public:
|
2019-07-11 02:04:58 +00:00
|
|
|
|
FileLine(const string& filename) {
|
2019-07-15 01:42:03 +00:00
|
|
|
|
m_lastLineno = m_firstLineno = 0;
|
|
|
|
|
m_lastColumn = m_firstColumn = 0;
|
2019-05-19 20:13:13 +00:00
|
|
|
|
m_filenameno = singleton().nameToNumber(filename);
|
2019-07-15 01:42:03 +00:00
|
|
|
|
m_contentLineno = 0;
|
|
|
|
|
m_contentp = NULL;
|
2019-06-22 21:01:39 +00:00
|
|
|
|
m_parent = NULL;
|
2019-05-19 20:13:13 +00:00
|
|
|
|
m_warnOn = defaultFileLine().m_warnOn;
|
|
|
|
|
}
|
2018-08-25 13:52:45 +00:00
|
|
|
|
explicit FileLine(FileLine* fromp) {
|
2019-07-15 01:42:03 +00:00
|
|
|
|
m_firstLineno = fromp->m_firstLineno;
|
|
|
|
|
m_firstColumn = fromp->m_firstColumn;
|
|
|
|
|
m_lastLineno = fromp->m_lastLineno;
|
|
|
|
|
m_lastColumn = fromp->m_lastColumn;
|
2019-05-19 20:13:13 +00:00
|
|
|
|
m_filenameno = fromp->m_filenameno;
|
2019-07-15 01:42:03 +00:00
|
|
|
|
m_contentLineno = fromp->m_contentLineno;
|
|
|
|
|
m_contentp = fromp->m_contentp;
|
2019-06-22 21:01:39 +00:00
|
|
|
|
m_parent = fromp->m_parent;
|
2019-05-19 20:13:13 +00:00
|
|
|
|
m_warnOn = fromp->m_warnOn;
|
|
|
|
|
}
|
2019-07-11 02:04:58 +00:00
|
|
|
|
struct EmptySecret {}; // Constructor selection
|
2018-08-25 13:52:45 +00:00
|
|
|
|
explicit FileLine(EmptySecret);
|
2019-07-11 02:04:58 +00:00
|
|
|
|
FileLine* copyOrSameFileLine();
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static void deleteAllRemaining();
|
2019-07-11 02:04:58 +00:00
|
|
|
|
~FileLine() { }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
#ifdef VL_LEAK_CHECKS
|
|
|
|
|
static void* operator new(size_t size);
|
|
|
|
|
static void operator delete(void* obj, size_t size);
|
|
|
|
|
#endif
|
2019-07-15 01:42:03 +00:00
|
|
|
|
void newContent() { m_contentp = new VFileContent; m_contentLineno = 1; }
|
2019-07-11 02:04:58 +00:00
|
|
|
|
// METHODS
|
2019-07-15 01:42:03 +00:00
|
|
|
|
void lineno(int num) { m_firstLineno = num; m_lastLineno = num;
|
|
|
|
|
m_firstColumn = m_lastColumn = 1; }
|
2019-07-11 02:04:58 +00:00
|
|
|
|
void language(V3LangCode lang) { singleton().numberToLang(m_filenameno, lang); }
|
|
|
|
|
void filename(const string& name) { m_filenameno = singleton().nameToNumber(name); }
|
|
|
|
|
void parent(FileLine* fileline) { m_parent = fileline; }
|
|
|
|
|
void lineDirective(const char* textp, int& enterExitRef);
|
2019-07-15 01:42:03 +00:00
|
|
|
|
void linenoInc() { m_lastLineno++; m_lastColumn = 1; m_contentLineno++; }
|
|
|
|
|
void startToken() { m_firstLineno = m_lastLineno;
|
|
|
|
|
m_firstColumn = m_lastColumn; }
|
|
|
|
|
// Advance last line/column based on given text
|
|
|
|
|
void forwardToken(const char* textp, size_t size, bool trackLines=true);
|
|
|
|
|
int firstLineno() const { return m_firstLineno; }
|
|
|
|
|
int firstColumn() const { return m_firstColumn; }
|
|
|
|
|
int lastLineno() const { return m_lastLineno; }
|
|
|
|
|
int lastColumn() const { return m_lastColumn; }
|
|
|
|
|
VFileContent* contentp() const { return m_contentp; }
|
|
|
|
|
// If not otherwise more specific, use last lineno for errors etc,
|
|
|
|
|
// as the parser errors etc generally make more sense pointing at the last parse point
|
|
|
|
|
int lineno() const { return m_lastLineno; }
|
|
|
|
|
string source() const;
|
|
|
|
|
string prettySource() const; // Source, w/stripped unprintables and newlines
|
2019-06-22 21:01:39 +00:00
|
|
|
|
FileLine* parent() const { return m_parent; }
|
2018-08-25 13:52:45 +00:00
|
|
|
|
V3LangCode language() const { return singleton().numberToLang(m_filenameno); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
string ascii() const;
|
2019-07-15 01:42:03 +00:00
|
|
|
|
string asciiLineCol() const;
|
2018-08-25 13:52:45 +00:00
|
|
|
|
const string filename() const { return singleton().numberToName(m_filenameno); }
|
2019-06-12 11:00:56 +00:00
|
|
|
|
bool filenameIsGlobal() const { return (filename() == commandLineFilename()
|
2019-06-29 11:39:17 +00:00
|
|
|
|
|| filename() == builtInFilename()); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
const string filenameLetters() const { return singleton().filenameLetters(m_filenameno); }
|
2018-08-25 13:52:45 +00:00
|
|
|
|
const string filebasename() const;
|
|
|
|
|
const string filebasenameNoExt() const;
|
2014-11-22 16:48:39 +00:00
|
|
|
|
const string profileFuncname() const;
|
2019-07-15 01:42:03 +00:00
|
|
|
|
const string xml() const { return "fl=\""+filenameLetters()+cvtToStr(lastLineno())+"\""; }
|
2018-10-14 22:39:33 +00:00
|
|
|
|
string lineDirectiveStrg(int enterExit) const;
|
2019-06-22 21:01:39 +00:00
|
|
|
|
|
2019-05-19 20:13:13 +00:00
|
|
|
|
// Turn on/off warning messages on this line.
|
|
|
|
|
void warnOn(V3ErrorCode code, bool flag) { m_warnOn.set(code, flag); }
|
|
|
|
|
void warnOff(V3ErrorCode code, bool flag) { warnOn(code, !flag); }
|
2018-10-14 22:39:33 +00:00
|
|
|
|
bool warnOff(const string& msg, bool flag); // Returns 1 if ok
|
2014-11-22 16:48:39 +00:00
|
|
|
|
bool warnIsOff(V3ErrorCode code) const;
|
|
|
|
|
void warnLintOff(bool flag);
|
|
|
|
|
void warnStyleOff(bool flag);
|
2019-05-19 20:13:13 +00:00
|
|
|
|
void warnStateFrom(const FileLine& from) { m_warnOn = from.m_warnOn; }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
void warnResetDefault() { warnStateFrom(defaultFileLine()); }
|
|
|
|
|
|
|
|
|
|
// Specific flag ACCESSORS/METHODS
|
|
|
|
|
bool coverageOn() const { return m_warnOn.test(V3ErrorCode::I_COVERAGE); }
|
2019-05-19 20:13:13 +00:00
|
|
|
|
void coverageOn(bool flag) { warnOn(V3ErrorCode::I_COVERAGE, flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
bool tracingOn() const { return m_warnOn.test(V3ErrorCode::I_TRACING); }
|
2019-05-19 20:13:13 +00:00
|
|
|
|
void tracingOn(bool flag) { warnOn(V3ErrorCode::I_TRACING, flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
|
|
|
|
|
// METHODS - Global
|
2019-06-29 11:39:17 +00:00
|
|
|
|
// <command-line> and <built-in> match what GCC outputs
|
|
|
|
|
static string commandLineFilename() { return "<command-line>"; }
|
|
|
|
|
static string builtInFilename() { return "<built-in>"; }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static void globalWarnLintOff(bool flag) {
|
2019-05-19 20:13:13 +00:00
|
|
|
|
defaultFileLine().warnLintOff(flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static void globalWarnStyleOff(bool flag) {
|
2019-05-19 20:13:13 +00:00
|
|
|
|
defaultFileLine().warnStyleOff(flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static void globalWarnOff(V3ErrorCode code, bool flag) {
|
2019-05-19 20:13:13 +00:00
|
|
|
|
defaultFileLine().warnOff(code, flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
static bool globalWarnOff(const string& code, bool flag) {
|
2019-05-19 20:13:13 +00:00
|
|
|
|
return defaultFileLine().warnOff(code, flag); }
|
2018-02-02 02:24:41 +00:00
|
|
|
|
static void fileNameNumMapDumpXml(std::ostream& os) {
|
2019-05-19 20:13:13 +00:00
|
|
|
|
singleton().fileNameNumMapDumpXml(os); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
|
|
|
|
|
// METHODS - Called from netlist
|
|
|
|
|
// Merge warning disables from another fileline
|
|
|
|
|
void modifyStateInherit(const FileLine* fromp);
|
|
|
|
|
// Change the current fileline due to actions discovered after parsing
|
|
|
|
|
// and may have side effects on other nodes sharing this FileLine.
|
|
|
|
|
// Use only when this is intended
|
2019-05-19 20:13:13 +00:00
|
|
|
|
void modifyWarnOff(V3ErrorCode code, bool flag) { warnOff(code, flag); }
|
2014-11-22 16:48:39 +00:00
|
|
|
|
|
|
|
|
|
// OPERATORS
|
2019-07-26 16:52:38 +00:00
|
|
|
|
void v3errorEnd(std::ostringstream& str, const string& locationStr = "");
|
2018-02-02 02:24:41 +00:00
|
|
|
|
void v3errorEndFatal(std::ostringstream& str);
|
2019-06-22 21:01:39 +00:00
|
|
|
|
/// When building an error, prefix for printing continuation lines
|
2019-07-11 23:15:40 +00:00
|
|
|
|
/// e.g. information referring to the same FileLine as before
|
2014-11-22 16:48:39 +00:00
|
|
|
|
string warnMore() const;
|
2019-07-11 23:15:40 +00:00
|
|
|
|
/// When building an error, prefix for printing secondary information
|
|
|
|
|
/// from a different FileLine than the original error
|
|
|
|
|
string warnOther() const;
|
2019-06-22 21:01:39 +00:00
|
|
|
|
/// When building an error, current location in include etc
|
|
|
|
|
/// If not used in a given error, automatically pasted at end of error
|
|
|
|
|
string warnContextPrimary() const { return warnContext(false); }
|
|
|
|
|
/// When building an error, additional location for additional references
|
|
|
|
|
/// Simplified information vs warnContextPrimary() to make dump clearer
|
|
|
|
|
string warnContextSecondary() const { return warnContext(true); }
|
2019-07-11 02:04:58 +00:00
|
|
|
|
bool operator==(FileLine rhs) const {
|
2019-07-15 01:42:03 +00:00
|
|
|
|
return (m_firstLineno == rhs.m_firstLineno
|
|
|
|
|
&& m_firstColumn == rhs.m_firstColumn
|
|
|
|
|
&& m_lastLineno == rhs.m_lastLineno
|
|
|
|
|
&& m_lastColumn == rhs.m_lastColumn
|
2019-07-11 02:04:58 +00:00
|
|
|
|
&& m_filenameno == rhs.m_filenameno
|
|
|
|
|
&& m_warnOn == rhs.m_warnOn);
|
2014-11-22 16:48:39 +00:00
|
|
|
|
}
|
2017-09-13 23:27:59 +00:00
|
|
|
|
private:
|
2018-02-02 02:24:41 +00:00
|
|
|
|
void v3errorEndFatalGuts(std::ostringstream& str);
|
2019-06-22 21:01:39 +00:00
|
|
|
|
string warnContext(bool secondary) const;
|
2014-11-22 16:48:39 +00:00
|
|
|
|
};
|
2018-02-02 02:24:41 +00:00
|
|
|
|
std::ostream& operator<<(std::ostream& os, FileLine* fileline);
|
2014-11-22 16:48:39 +00:00
|
|
|
|
|
2019-05-15 00:51:28 +00:00
|
|
|
|
inline void FileLine::v3errorEndFatal(std::ostringstream& str) {
|
|
|
|
|
v3errorEnd(str); assert(0); VL_UNREACHABLE
|
|
|
|
|
}
|
2017-07-07 00:25:59 +00:00
|
|
|
|
|
2019-05-15 00:51:28 +00:00
|
|
|
|
#endif // Guard
|