Implement support for GCC's -dD mode, which dumps -E output *and*
macro definitions.


git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@68884 91177308-0d34-0410-b5e6-96231b3b80d8
diff --git a/include/clang/Driver/Options.def b/include/clang/Driver/Options.def
index 0a141a1..0f59230 100644
--- a/include/clang/Driver/Options.def
+++ b/include/clang/Driver/Options.def
@@ -400,6 +400,7 @@
 OPTION("-c", c, Flag, INVALID, INVALID, "d", 0, 
        "Only run preprocess, compile, and assemble steps", 0)
 OPTION("-dA", dA, Flag, d_Group, INVALID, "", 0, 0, 0)
+OPTION("-dD", dD, Flag, d_Group, INVALID, "", 0, 0, 0)
 OPTION("-dM", dM, Flag, d_Group, INVALID, "", 0, 0, 0)
 OPTION("-dead_strip", dead__strip, Flag, INVALID, INVALID, "", 0, 0, 0)
 OPTION("-dependency-file", dependency_file, Separate, INVALID, INVALID, "", 0, 0, 0)
diff --git a/lib/Driver/Tools.cpp b/lib/Driver/Tools.cpp
index 4d1aca4..7c73780 100644
--- a/lib/Driver/Tools.cpp
+++ b/lib/Driver/Tools.cpp
@@ -474,6 +474,7 @@
     CmdArgs.push_back("-fno-common");
 
   Args.AddLastArg(CmdArgs, options::OPT_dM);
+  Args.AddLastArg(CmdArgs, options::OPT_dD);
 
   Args.AddAllArgValues(CmdArgs, options::OPT_Xclang);
 
diff --git a/test/Preprocessor/dump-options.c b/test/Preprocessor/dump-options.c
new file mode 100644
index 0000000..6910e1a
--- /dev/null
+++ b/test/Preprocessor/dump-options.c
@@ -0,0 +1,3 @@
+// RUN: clang %s -E -dD | grep __INTMAX_MAX__ &&
+// RUN: clang %s -E -dM | grep __INTMAX_MAX__
+
diff --git a/tools/clang-cc/PrintPreprocessedOutput.cpp b/tools/clang-cc/PrintPreprocessedOutput.cpp
index 6ac31a5..a2047fa 100644
--- a/tools/clang-cc/PrintPreprocessedOutput.cpp
+++ b/tools/clang-cc/PrintPreprocessedOutput.cpp
@@ -30,6 +30,53 @@
 #include <cstdio>
 using namespace clang;
 
+/// PrintMacroDefinition - Print a macro definition in a form that will be
+/// properly accepted back as a definition.
+static void PrintMacroDefinition(const IdentifierInfo &II, const MacroInfo &MI,
+                                 Preprocessor &PP, llvm::raw_ostream &OS) {
+  OS << "#define " << II.getName();
+  
+  if (MI.isFunctionLike()) {
+    OS << '(';
+    if (MI.arg_empty())
+      ;
+    else if (MI.getNumArgs() == 1) 
+      OS << (*MI.arg_begin())->getName();
+    else {
+      MacroInfo::arg_iterator AI = MI.arg_begin(), E = MI.arg_end();
+      OS << (*AI++)->getName();
+      while (AI != E)
+        OS << ',' << (*AI++)->getName();
+    }
+    
+    if (MI.isVariadic()) {
+      if (!MI.arg_empty())
+        OS << ',';
+      OS << "...";
+    }
+    OS << ')';
+  }
+  
+  // GCC always emits a space, even if the macro body is empty.  However, do not
+  // want to emit two spaces if the first token has a leading space.
+  if (MI.tokens_empty() || !MI.tokens_begin()->hasLeadingSpace())
+    OS << ' ';
+  
+  llvm::SmallVector<char, 128> SpellingBuffer;
+  for (MacroInfo::tokens_iterator I = MI.tokens_begin(), E = MI.tokens_end();
+       I != E; ++I) {
+    if (I->hasLeadingSpace())
+      OS << ' ';
+    
+    // Make sure we have enough space in the spelling buffer.
+    if (I->getLength() < SpellingBuffer.size())
+      SpellingBuffer.resize(I->getLength());
+    const char *Buffer = &SpellingBuffer[0];
+    unsigned SpellingLen = PP.getSpelling(*I, Buffer);
+    OS.write(Buffer, SpellingLen);
+  }
+}
+
 //===----------------------------------------------------------------------===//
 // Preprocessed token printer
 //===----------------------------------------------------------------------===//
@@ -46,6 +93,9 @@
 DumpMacros("dM", llvm::cl::desc("Print macro definitions in -E mode instead of"
                                 " normal output"));
 
+static llvm::cl::opt<bool>
+DumpDefines("dD", llvm::cl::desc("Print macro definitions in -E mode in "
+                                "addition to normal output"));
 
 namespace {
 class PrintPPOutputPPCallbacks : public PPCallbacks {
@@ -85,6 +135,10 @@
     return ConcatInfo.AvoidConcat(PrevTok, Tok);
   }
   void WriteLineInfo(unsigned LineNo, const char *Extra=0, unsigned ExtraLen=0);
+  
+  /// MacroDefined - This hook is called whenever a macro definition is seen.
+  void MacroDefined(const IdentifierInfo *II, const MacroInfo *MI);
+  
 };
 }  // end anonymous namespace
 
@@ -211,6 +265,19 @@
   EmittedTokensOnThisLine = true;
 }
 
+/// MacroDefined - This hook is called whenever a macro definition is seen.
+void PrintPPOutputPPCallbacks::MacroDefined(const IdentifierInfo *II,
+                                            const MacroInfo *MI) {
+  // Only print out macro definitions in -dD mode.
+  if (!DumpDefines ||
+      // Ignore __FILE__ etc.
+      MI->isBuiltinMacro()) return;
+  
+  MoveToLine(MI->getDefinitionLoc());
+  PrintMacroDefinition(*II, *MI, PP, OS);
+}
+
+
 void PrintPPOutputPPCallbacks::PragmaComment(SourceLocation Loc,
                                              const IdentifierInfo *Kind, 
                                              const std::string &Str) {
@@ -338,56 +405,6 @@
   }
 }
 
-/// PrintMacroDefinition - Print a macro definition in a form that will be
-/// properly accepted back as a definition.
-static void PrintMacroDefinition(IdentifierInfo &II, const MacroInfo &MI,
-                                 Preprocessor &PP, llvm::raw_ostream &OS) {
-  // Ignore computed macros like __LINE__ and friends. 
-  if (MI.isBuiltinMacro()) return;
-  OS << "#define " << II.getName();
-
-  if (MI.isFunctionLike()) {
-    OS << '(';
-    if (MI.arg_empty())
-      ;
-    else if (MI.getNumArgs() == 1) 
-      OS << (*MI.arg_begin())->getName();
-    else {
-      MacroInfo::arg_iterator AI = MI.arg_begin(), E = MI.arg_end();
-      OS << (*AI++)->getName();
-      while (AI != E)
-        OS << ',' << (*AI++)->getName();
-    }
-    
-    if (MI.isVariadic()) {
-      if (!MI.arg_empty())
-        OS << ',';
-      OS << "...";
-    }
-    OS << ')';
-  }
-  
-  // GCC always emits a space, even if the macro body is empty.  However, do not
-  // want to emit two spaces if the first token has a leading space.
-  if (MI.tokens_empty() || !MI.tokens_begin()->hasLeadingSpace())
-    OS << ' ';
-  
-  llvm::SmallVector<char, 128> SpellingBuffer;
-  for (MacroInfo::tokens_iterator I = MI.tokens_begin(), E = MI.tokens_end();
-       I != E; ++I) {
-    if (I->hasLeadingSpace())
-      OS << ' ';
-    
-    // Make sure we have enough space in the spelling buffer.
-    if (I->getLength() < SpellingBuffer.size())
-      SpellingBuffer.resize(I->getLength());
-    const char *Buffer = &SpellingBuffer[0];
-    unsigned SpellingLen = PP.getSpelling(*I, Buffer);
-    OS.write(Buffer, SpellingLen);
-  }
-  OS << "\n";
-}
-
 namespace {
   struct SortMacrosByID {
     typedef std::pair<IdentifierInfo*, MacroInfo*> id_macro_pair;
@@ -430,12 +447,17 @@
       MacrosByID.push_back(*I);
     std::sort(MacrosByID.begin(), MacrosByID.end(), SortMacrosByID());
     
-    for (unsigned i = 0, e = MacrosByID.size(); i != e; ++i)
-      PrintMacroDefinition(*MacrosByID[i].first, *MacrosByID[i].second, PP, OS);
+    for (unsigned i = 0, e = MacrosByID.size(); i != e; ++i) {
+      MacroInfo &MI = *MacrosByID[i].second;
+      // Ignore computed macros like __LINE__ and friends. 
+      if (MI.isBuiltinMacro()) continue;
+
+      PrintMacroDefinition(*MacrosByID[i].first, MI, PP, OS);
+      OS << "\n";
+    }
     
   } else {
-    PrintPPOutputPPCallbacks *Callbacks
-      = new PrintPPOutputPPCallbacks(PP, OS);
+    PrintPPOutputPPCallbacks *Callbacks = new PrintPPOutputPPCallbacks(PP, OS);
     PP.AddPragmaHandler(0, new UnknownPragmaHandler("#pragma", Callbacks));
     PP.AddPragmaHandler("GCC", new UnknownPragmaHandler("#pragma GCC",
                                                         Callbacks));