| #include "SourcePos.h" |
| |
| #include <stdarg.h> |
| #include <cstdio> |
| #include <set> |
| |
| using namespace std; |
| |
| const SourcePos GENERATED_POS("<generated>", -1); |
| |
| // ErrorPos |
| // ============================================================================= |
| struct ErrorPos |
| { |
| string file; |
| int line; |
| string error; |
| |
| ErrorPos(); |
| ErrorPos(const ErrorPos& that); |
| ErrorPos(const string& file, int line, const string& error); |
| ~ErrorPos(); |
| bool operator<(const ErrorPos& rhs) const; |
| bool operator==(const ErrorPos& rhs) const; |
| ErrorPos& operator=(const ErrorPos& rhs); |
| |
| void Print(FILE* to) const; |
| }; |
| |
| static set<ErrorPos> g_errors; |
| |
| ErrorPos::ErrorPos() |
| { |
| } |
| |
| ErrorPos::ErrorPos(const ErrorPos& that) |
| :file(that.file), |
| line(that.line), |
| error(that.error) |
| { |
| } |
| |
| ErrorPos::ErrorPos(const string& f, int l, const string& e) |
| :file(f), |
| line(l), |
| error(e) |
| { |
| } |
| |
| ErrorPos::~ErrorPos() |
| { |
| } |
| |
| bool |
| ErrorPos::operator<(const ErrorPos& rhs) const |
| { |
| if (this->file < rhs.file) return true; |
| if (this->file == rhs.file) { |
| if (this->line < rhs.line) return true; |
| if (this->line == rhs.line) { |
| if (this->error < rhs.error) return true; |
| } |
| } |
| return false; |
| } |
| |
| bool |
| ErrorPos::operator==(const ErrorPos& rhs) const |
| { |
| return this->file == rhs.file |
| && this->line == rhs.line |
| && this->error == rhs.error; |
| } |
| |
| ErrorPos& |
| ErrorPos::operator=(const ErrorPos& rhs) |
| { |
| this->file = rhs.file; |
| this->line = rhs.line; |
| this->error = rhs.error; |
| return *this; |
| } |
| |
| void |
| ErrorPos::Print(FILE* to) const |
| { |
| if (this->line >= 0) { |
| fprintf(to, "%s:%d: %s\n", this->file.c_str(), this->line, this->error.c_str()); |
| } else { |
| fprintf(to, "%s: %s\n", this->file.c_str(), this->error.c_str()); |
| } |
| } |
| |
| // SourcePos |
| // ============================================================================= |
| SourcePos::SourcePos(const string& f, int l) |
| : file(f), line(l) |
| { |
| } |
| |
| SourcePos::SourcePos(const SourcePos& that) |
| : file(that.file), line(that.line) |
| { |
| } |
| |
| SourcePos::SourcePos() |
| : file("???", 0) |
| { |
| } |
| |
| SourcePos::~SourcePos() |
| { |
| } |
| |
| string |
| SourcePos::ToString() const |
| { |
| char buf[1024]; |
| if (this->line >= 0) { |
| snprintf(buf, sizeof(buf)-1, "%s:%d", this->file.c_str(), this->line); |
| } else { |
| snprintf(buf, sizeof(buf)-1, "%s:", this->file.c_str()); |
| } |
| buf[sizeof(buf)-1] = '\0'; |
| return string(buf); |
| } |
| |
| int |
| SourcePos::Error(const char* fmt, ...) const |
| { |
| int retval=0; |
| char buf[1024]; |
| va_list ap; |
| va_start(ap, fmt); |
| retval = vsnprintf(buf, sizeof(buf), fmt, ap); |
| va_end(ap); |
| char* p = buf + retval - 1; |
| while (p > buf && *p == '\n') { |
| *p = '\0'; |
| p--; |
| } |
| ErrorPos err(this->file, this->line, string(buf)); |
| if (g_errors.find(err) == g_errors.end()) { |
| err.Print(stderr); |
| g_errors.insert(err); |
| } |
| return retval; |
| } |
| |
| bool |
| SourcePos::HasErrors() |
| { |
| return g_errors.size() > 0; |
| } |
| |
| void |
| SourcePos::PrintErrors(FILE* to) |
| { |
| set<ErrorPos>::const_iterator it; |
| for (it=g_errors.begin(); it!=g_errors.end(); it++) { |
| it->Print(to); |
| } |
| } |
| |
| |
| |
| |