summaryrefslogtreecommitdiff
path: root/utils/TableGen/ClangDiagnosticsEmitter.cpp
diff options
context:
space:
mode:
authorTed Kremenek <kremenek@apple.com>2009-03-13 22:21:17 +0000
committerTed Kremenek <kremenek@apple.com>2009-03-13 22:21:17 +0000
commit04a847e70616cba0208c654729736a9d8f4047ac (patch)
tree287a1db1b51984ef5252724a0b947cd81a046470 /utils/TableGen/ClangDiagnosticsEmitter.cpp
parent02475f1fd502fa1729d1fc447417087acf9e96c8 (diff)
downloadllvm-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.cpp84
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