diff options
author | Ted Kremenek <kremenek@apple.com> | 2009-03-13 22:21:17 +0000 |
---|---|---|
committer | Ted Kremenek <kremenek@apple.com> | 2009-03-13 22:21:17 +0000 |
commit | 04a847e70616cba0208c654729736a9d8f4047ac (patch) | |
tree | 287a1db1b51984ef5252724a0b947cd81a046470 /utils/TableGen/ClangDiagnosticsEmitter.cpp | |
parent | 02475f1fd502fa1729d1fc447417087acf9e96c8 (diff) | |
download | llvm-04a847e70616cba0208c654729736a9d8f4047ac.tar.gz llvm-04a847e70616cba0208c654729736a9d8f4047ac.tar.bz2 llvm-04a847e70616cba0208c654729736a9d8f4047ac.tar.xz |
Add initial implementation of a TableGen backend for converting Clang-warnings
tablegen files to the original .def preprocessor include files. This is my first
TableGen backend; I don't claim that it is awesome.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@66971 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'utils/TableGen/ClangDiagnosticsEmitter.cpp')
-rw-r--r-- | utils/TableGen/ClangDiagnosticsEmitter.cpp | 84 |
1 files changed, 84 insertions, 0 deletions
diff --git a/utils/TableGen/ClangDiagnosticsEmitter.cpp b/utils/TableGen/ClangDiagnosticsEmitter.cpp new file mode 100644 index 0000000000..fa139e7729 --- /dev/null +++ b/utils/TableGen/ClangDiagnosticsEmitter.cpp @@ -0,0 +1,84 @@ +//=- ClangDiagnosticsEmitter.cpp - Generate Clang diagnostics tables -*- C++ -*- +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// These tablegen backends emit Clang diagnostics tables. +// +//===----------------------------------------------------------------------===// + +#include "ClangDiagnosticsEmitter.h" +#include "Record.h" +#include "llvm/Support/Debug.h" +#include "llvm/Support/Streams.h" +#include "llvm/ADT/VectorExtras.h" + +using namespace llvm; +typedef std::vector<Record*> RecordVector; +typedef std::vector<Record*> SuperClassVector; +typedef std::vector<RecordVal> RecordValVector; + +static const RecordVal* findRecordVal(const RecordValVector& Vals, + const std::string &key) { + for (RecordValVector::const_iterator I=Vals.begin(), E=Vals.end(); I!=E; ++I) + if ((*I).getName() == key) + return &*I; + + return 0; +} + +static const Record* getDiagKind(const Record* DiagClass, const Record &R) { + const SuperClassVector &SC = R.getSuperClasses(); + for (SuperClassVector::const_iterator I=SC.begin(), E=SC.end(); I!=E; ++I) + if ((*I)->isSubClassOf(DiagClass)) + return *I; + + return 0; +} + +static void EmitEscaped(std::ostream& OS, const std::string &s) { + for (std::string::const_iterator I=s.begin(), E=s.end(); I!=E; ++I) + switch (*I) { + default: OS << *I; break; + case '\"': OS << "\\" << *I; break; + case '\\': OS << "\\\\"; break; + } +} + +static void ProcessDiag(std::ostream& OS, const Record* DiagClass, + const Record& R) { + + const Record* DiagKind = getDiagKind(DiagClass, R); + if (!DiagKind) + return; + + OS << "DIAG(" << R.getName() << ", "; + + const std::string &s = DiagKind->getName(); + for (std::string::const_iterator I=s.begin(), E=s.end(); I!=E; ++I) + OS << char(toupper(*I)); + + const RecordVal* Text = findRecordVal(R.getValues(), "Text"); + assert(Text && "No 'Text' entry in Diagnostic."); + const StringInit* TextVal = dynamic_cast<const StringInit*>(Text->getValue()); + assert(TextVal && "Value 'Text' must be a string."); + OS << ", \""; + EmitEscaped(OS, TextVal->getValue()); + OS << "\")\n"; +} + +void ClangDiagsDefsEmitter::run(std::ostream &OS) { + const RecordVector &Diags = Records.getAllDerivedDefinitions("Diagnostic"); + + const Record* DiagClass = Records.getClass("Diagnostic"); + assert(DiagClass && "No Diagnostic class defined."); + + for (RecordVector::const_iterator I=Diags.begin(), E=Diags.end(); I!=E; ++I) { + // FIXME: Compare the component. + ProcessDiag(OS, DiagClass, **I); + } +}
\ No newline at end of file |