#include "SourcePos.h" #include <stdarg.h> #include <vector> using namespace std; // ErrorPos // ============================================================================= struct ErrorPos { String8 file; int line; String8 error; bool fatal; ErrorPos(); ErrorPos(const ErrorPos& that); ErrorPos(const String8& file, int line, const String8& error, bool fatal); ~ErrorPos(); bool operator<(const ErrorPos& rhs) const; bool operator==(const ErrorPos& rhs) const; ErrorPos& operator=(const ErrorPos& rhs); void print(FILE* to) const; }; static vector<ErrorPos> g_errors; ErrorPos::ErrorPos() :line(-1), fatal(false) { } ErrorPos::ErrorPos(const ErrorPos& that) :file(that.file), line(that.line), error(that.error), fatal(that.fatal) { } ErrorPos::ErrorPos(const String8& f, int l, const String8& e, bool fat) :file(f), line(l), error(e), fatal(fat) { } 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 { const char* type = fatal ? "error:" : "warning:"; if (this->line >= 0) { fprintf(to, "%s:%d: %s %s\n", this->file.string(), this->line, type, this->error.string()); } else { fprintf(to, "%s: %s %s\n", this->file.string(), type, this->error.string()); } } // SourcePos // ============================================================================= SourcePos::SourcePos(const String8& f, int l) : file(f), line(l) { } SourcePos::SourcePos(const SourcePos& that) : file(that.file), line(that.line) { } SourcePos::SourcePos() : file("???", 0), line(-1) { } SourcePos::~SourcePos() { } 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--; } g_errors.push_back(ErrorPos(this->file, this->line, String8(buf), true)); return retval; } int SourcePos::warning(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(this->file, this->line, String8(buf), false).print(stderr); return retval; } bool SourcePos::hasErrors() { return g_errors.size() > 0; } void SourcePos::printErrors(FILE* to) { vector<ErrorPos>::const_iterator it; for (it=g_errors.begin(); it!=g_errors.end(); it++) { it->print(to); } }