]> granicus.if.org Git - clang/commitdiff
implement the microsoft/gnu "__COUNTER__" macro: rdar://4329310
authorChris Lattner <sabre@nondot.org>
Mon, 13 Apr 2009 01:29:17 +0000 (01:29 +0000)
committerChris Lattner <sabre@nondot.org>
Mon, 13 Apr 2009 01:29:17 +0000 (01:29 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@68933 91177308-0d34-0410-b5e6-96231b3b80d8

include/clang/Basic/DiagnosticLexKinds.td
include/clang/Frontend/PCHBitCodes.h
include/clang/Lex/Preprocessor.h
lib/Frontend/PCHReader.cpp
lib/Frontend/PCHWriter.cpp
lib/Lex/PPMacroExpansion.cpp
lib/Lex/Preprocessor.cpp
test/Lexer/counter.c [new file with mode: 0644]
test/PCH/variables.c
test/PCH/variables.h

index adce63c48909043a721f81b7144462cbf74c0a57..fa70cbf6fb512ed7c11607e59f356ec9d9bd7cdd 100644 (file)
@@ -132,6 +132,8 @@ def ext_pp_base_file : Extension<"__BASE_FILE__ is a language extension">;
 def ext_pp_include_level : Extension<
   "__INCLUDE_LEVEL__ is a language extension">;
 def ext_pp_timestamp : Extension<"__TIMESTAMP__ is a language extension">;
+def ext_pp_counter : Extension<
+  "__COUNTER__ is a language extension">;
 
 def err_pp_invalid_directive : Error<"invalid preprocessing directive">;
 def err_pp_hash_error : Error<"#error%0">;
index dc8b6cc7c0c7d193f497f523083d0a1350f9c3eb..7782f36f43dabbee32e37e7a84948ecacc8e3192 100644 (file)
@@ -165,8 +165,12 @@ namespace clang {
       PP_MACRO_FUNCTION_LIKE = 2,
       
       /// \brief Describes one token.
-      /// [PPTOKEN, SLoc, Length, IdentInfoID, Kind, Flags]
-      PP_TOKEN = 3
+      /// [PP_TOKEN, SLoc, Length, IdentInfoID, Kind, Flags]
+      PP_TOKEN = 3,
+      
+      /// \brief The value of the next __COUNTER__ to dispense.
+      /// [PP_COUNTER_VALUE, Val]
+      PP_COUNTER_VALUE = 4
     };
     
 
index 47f40b68f6cd45c1d5ff8522d192dac7efb83cbb..9e4278515b7f2b4286baf084e0a3c995dc6ec791 100644 (file)
@@ -67,9 +67,11 @@ class Preprocessor {
   IdentifierInfo *Ident__INCLUDE_LEVEL__;          // __INCLUDE_LEVEL__
   IdentifierInfo *Ident__BASE_FILE__;              // __BASE_FILE__
   IdentifierInfo *Ident__TIMESTAMP__;              // __TIMESTAMP__
+  IdentifierInfo *Ident__COUNTER__;                // __COUNTER__
   IdentifierInfo *Ident_Pragma, *Ident__VA_ARGS__; // _Pragma, __VA_ARGS__
   
   SourceLocation DATELoc, TIMELoc;
+  unsigned CounterValue;  // Next __COUNTER__ value.
 
   enum {
     /// MaxIncludeStackDepth - Maximum depth of #includes.
@@ -590,6 +592,8 @@ public:
   bool SawDateOrTime() const {
     return DATELoc != SourceLocation() || TIMELoc != SourceLocation();
   }
+  unsigned getCounterValue() const { return CounterValue; }
+  void setCounterValue(unsigned V) { CounterValue = V; }
   
   /// AllocateMacroInfo - Allocate a new MacroInfo object with the provide
   ///  SourceLocation.
index b916deaa6905e229163663e846d22b4f06fe2fd0..fa6ad6f0a0a9bc06ab262fb1cca0d1a7a8717f11 100644 (file)
@@ -318,7 +318,11 @@ bool PCHReader::ReadPreprocessorBlock() {
     switch (RecType) {
     default:  // Default behavior: ignore unknown records.
       break;
-        
+    case pch::PP_COUNTER_VALUE:
+      if (!Record.empty())
+        PP.setCounterValue(Record[0]);
+      break;
+
     case pch::PP_MACRO_OBJECT_LIKE:
     case pch::PP_MACRO_FUNCTION_LIKE: {
       IdentifierInfo *II = DecodeIdentifierInfo(Record[0]);
index 6faf37fb295feafd31a0d0eee9b1740b53a70f89..2cf932ce9b6034e4c11d4062cfdd0de4a5f79e87 100644 (file)
@@ -567,6 +567,13 @@ void PCHWriter::WritePreprocessor(const Preprocessor &PP) {
   
   RecordData Record;
 
+  // If the preprocessor __COUNTER__ value has been bumped, remember it.
+  if (PP.getCounterValue() != 0) {
+    Record.push_back(PP.getCounterValue());
+    S.EmitRecord(pch::PP_COUNTER_VALUE, Record);
+    Record.clear();
+  }  
+  
   // Loop over all the macro definitions that are live at the end of the file,
   // emitting each to the PP section.
   // FIXME: Eventually we want to emit an index so that we can lazily load
@@ -627,9 +634,6 @@ void PCHWriter::WritePreprocessor(const Preprocessor &PP) {
     
   }
   
-  // TODO: someday when PP supports __COUNTER__, emit a record for its value if
-  // non-zero.
-  
   S.ExitBlock();
 }
 
index b23311af19274ce658e301ffab60232d122ed291..797a5bf6f2c4304208cf464df1b97bc86857e02a 100644 (file)
@@ -55,6 +55,7 @@ void Preprocessor::RegisterBuiltinMacros() {
   Ident__FILE__ = RegisterBuiltinMacro("__FILE__");
   Ident__DATE__ = RegisterBuiltinMacro("__DATE__");
   Ident__TIME__ = RegisterBuiltinMacro("__TIME__");
+  Ident__COUNTER__ = RegisterBuiltinMacro("__COUNTER__");
   Ident_Pragma  = RegisterBuiltinMacro("_Pragma");
   
   // GCC Extensions.
@@ -557,6 +558,13 @@ void Preprocessor::ExpandBuiltinMacro(Token &Tok) {
     TmpBuffer[Len] = '"';  // Replace the newline with a quote.
     Tok.setKind(tok::string_literal);
     CreateString(TmpBuffer, Len+1, Tok, Tok.getLocation());
+  } else if (II == Ident__COUNTER__) {
+    Diag(Tok, diag::ext_pp_counter);
+    
+    // __COUNTER__ expands to a simple numeric value.
+    sprintf(TmpBuffer, "%u", CounterValue++);
+    Tok.setKind(tok::numeric_constant);
+    CreateString(TmpBuffer, strlen(TmpBuffer), Tok, Tok.getLocation());
   } else {
     assert(0 && "Unknown identifier!");
   }
index 097b4542b6e46f47a8a7750f31cee806ac5301d5..2c9b0b937a28ef5d5410c14fcea5abcd4a29da7c 100644 (file)
@@ -53,7 +53,8 @@ Preprocessor::Preprocessor(Diagnostic &diags, const LangOptions &opts,
     SourceMgr(SM), HeaderInfo(Headers), Identifiers(opts, IILookup),
     CurPPLexer(0), CurDirLookup(0), Callbacks(0) {
   ScratchBuf = new ScratchBuffer(SourceMgr);
-
+  CounterValue = 0; // __COUNTER__ starts at 0.
+      
   // Clear stats.
   NumDirectives = NumDefined = NumUndefined = NumPragma = 0;
   NumIf = NumElse = NumEndif = 0;
diff --git a/test/Lexer/counter.c b/test/Lexer/counter.c
new file mode 100644 (file)
index 0000000..9b2c2c7
--- /dev/null
@@ -0,0 +1,16 @@
+// __COUNTER__ support: rdar://4329310
+// RUN: clang -E %s > %t &&
+
+#define PASTE2(x,y) x##y
+#define PASTE1(x,y) PASTE2(x,y)
+#define UNIQUE(x) PASTE1(x,__COUNTER__)
+
+// RUN: grep "A: 0" %t &&
+A: __COUNTER__
+
+// RUN: grep "B: foo1" %t &&
+B: UNIQUE(foo);
+// RUN: grep "C: foo2" %t &&
+C: UNIQUE(foo);
+// RUN: grep "D: 3" %t
+D: __COUNTER__
index 749b7b6741b5127f80e419e8608bd2a061c0a45e..f725e7d8f05c37d43c60cb8ea745d5b6de26840b 100644 (file)
@@ -15,4 +15,8 @@ double z; // expected-error{{redefinition}}
 
 int Q = A_MACRO_IN_THE_PCH;
 
-int R = FUNCLIKE_MACRO(A_MACRO_, IN_THE_PCH);
\ No newline at end of file
+int R = FUNCLIKE_MACRO(A_MACRO_, IN_THE_PCH);
+
+
+int UNIQUE(a);  // a2
+int *Arr[] = { &a0, &a1, &a2 };
index 7b27671328ebabf4e2d859ffeea240c09c6ed715..82e87aaf4bffed45dbf657eb886f9e78c58735f2 100644 (file)
@@ -16,3 +16,11 @@ int MAKE_HAPPY(Very);
 
 #define A_MACRO_IN_THE_PCH 492
 #define FUNCLIKE_MACRO(X, Y) X ## Y
+
+#define PASTE2(x,y) x##y
+#define PASTE1(x,y) PASTE2(x,y)
+#define UNIQUE(x) PASTE1(x,__COUNTER__)
+
+int UNIQUE(a);  // a0
+int UNIQUE(a);  // a1
+