| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 1 | //===--- SemaDeclSpec.cpp - Declaration Specifier Semantic Analysis -------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
| Chris Lattner | 0bc735f | 2007-12-29 19:59:25 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | //  This file implements semantic analysis for declaration specifiers. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
|  | 14 | #include "clang/Parse/DeclSpec.h" | 
| Chris Lattner | 500d329 | 2009-01-29 05:15:15 +0000 | [diff] [blame] | 15 | #include "clang/Parse/ParseDiagnostic.h" | 
| Douglas Gregor | 9b3064b | 2009-04-01 22:41:11 +0000 | [diff] [blame] | 16 | #include "clang/Lex/Preprocessor.h" | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 17 | #include "clang/Basic/LangOptions.h" | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 18 | #include "llvm/ADT/STLExtras.h" | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 19 | #include "llvm/Support/ErrorHandling.h" | 
| Douglas Gregor | e4e5b05 | 2009-03-19 00:18:19 +0000 | [diff] [blame] | 20 | #include <cstring> | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 21 | using namespace clang; | 
|  | 22 |  | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 23 |  | 
|  | 24 | static DiagnosticBuilder Diag(Diagnostic &D, SourceLocation Loc, | 
|  | 25 | SourceManager &SrcMgr, unsigned DiagID) { | 
|  | 26 | return D.Report(FullSourceLoc(Loc, SrcMgr), DiagID); | 
|  | 27 | } | 
|  | 28 |  | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 29 | /// DeclaratorChunk::getFunction - Return a DeclaratorChunk for a function. | 
|  | 30 | /// "TheDeclarator" is the declarator that this will be added to. | 
|  | 31 | DeclaratorChunk DeclaratorChunk::getFunction(bool hasProto, bool isVariadic, | 
| Douglas Gregor | 965acbb | 2009-02-18 07:07:28 +0000 | [diff] [blame] | 32 | SourceLocation EllipsisLoc, | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 33 | ParamInfo *ArgInfo, | 
|  | 34 | unsigned NumArgs, | 
|  | 35 | unsigned TypeQuals, | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 36 | bool hasExceptionSpec, | 
| Sebastian Redl | 3cc9726 | 2009-05-31 11:47:27 +0000 | [diff] [blame] | 37 | SourceLocation ThrowLoc, | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 38 | bool hasAnyExceptionSpec, | 
|  | 39 | ActionBase::TypeTy **Exceptions, | 
| Sebastian Redl | ef65f06 | 2009-05-29 18:02:33 +0000 | [diff] [blame] | 40 | SourceRange *ExceptionRanges, | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 41 | unsigned NumExceptions, | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 42 | SourceLocation Loc, | 
|  | 43 | Declarator &TheDeclarator) { | 
|  | 44 | DeclaratorChunk I; | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 45 | I.Kind                 = Function; | 
|  | 46 | I.Loc                  = Loc; | 
|  | 47 | I.Fun.hasPrototype     = hasProto; | 
|  | 48 | I.Fun.isVariadic       = isVariadic; | 
|  | 49 | I.Fun.EllipsisLoc      = EllipsisLoc.getRawEncoding(); | 
|  | 50 | I.Fun.DeleteArgInfo    = false; | 
|  | 51 | I.Fun.TypeQuals        = TypeQuals; | 
|  | 52 | I.Fun.NumArgs          = NumArgs; | 
|  | 53 | I.Fun.ArgInfo          = 0; | 
|  | 54 | I.Fun.hasExceptionSpec = hasExceptionSpec; | 
| Sebastian Redl | 3cc9726 | 2009-05-31 11:47:27 +0000 | [diff] [blame] | 55 | I.Fun.ThrowLoc         = ThrowLoc.getRawEncoding(); | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 56 | I.Fun.hasAnyExceptionSpec = hasAnyExceptionSpec; | 
|  | 57 | I.Fun.NumExceptions    = NumExceptions; | 
|  | 58 | I.Fun.Exceptions       = 0; | 
|  | 59 |  | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 60 | // new[] an argument array if needed. | 
|  | 61 | if (NumArgs) { | 
|  | 62 | // If the 'InlineParams' in Declarator is unused and big enough, put our | 
|  | 63 | // parameter list there (in an effort to avoid new/delete traffic).  If it | 
|  | 64 | // is already used (consider a function returning a function pointer) or too | 
|  | 65 | // small (function taking too many arguments), go to the heap. | 
|  | 66 | if (!TheDeclarator.InlineParamsUsed && | 
|  | 67 | NumArgs <= llvm::array_lengthof(TheDeclarator.InlineParams)) { | 
|  | 68 | I.Fun.ArgInfo = TheDeclarator.InlineParams; | 
|  | 69 | I.Fun.DeleteArgInfo = false; | 
|  | 70 | TheDeclarator.InlineParamsUsed = true; | 
|  | 71 | } else { | 
|  | 72 | I.Fun.ArgInfo = new DeclaratorChunk::ParamInfo[NumArgs]; | 
|  | 73 | I.Fun.DeleteArgInfo = true; | 
|  | 74 | } | 
|  | 75 | memcpy(I.Fun.ArgInfo, ArgInfo, sizeof(ArgInfo[0])*NumArgs); | 
|  | 76 | } | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 77 | // new[] an exception array if needed | 
|  | 78 | if (NumExceptions) { | 
| Sebastian Redl | ef65f06 | 2009-05-29 18:02:33 +0000 | [diff] [blame] | 79 | I.Fun.Exceptions = new DeclaratorChunk::TypeAndRange[NumExceptions]; | 
|  | 80 | for (unsigned i = 0; i != NumExceptions; ++i) { | 
|  | 81 | I.Fun.Exceptions[i].Ty = Exceptions[i]; | 
|  | 82 | I.Fun.Exceptions[i].Range = ExceptionRanges[i]; | 
|  | 83 | } | 
| Sebastian Redl | 7dc8134 | 2009-04-29 17:30:04 +0000 | [diff] [blame] | 84 | } | 
| Chris Lattner | 5af2f35 | 2009-01-20 19:11:22 +0000 | [diff] [blame] | 85 | return I; | 
|  | 86 | } | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 87 |  | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 88 | /// getParsedSpecifiers - Return a bitmask of which flavors of specifiers this | 
| Chris Lattner | 2a327d1 | 2009-02-27 18:35:46 +0000 | [diff] [blame] | 89 | /// declaration specifier includes. | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 90 | /// | 
|  | 91 | unsigned DeclSpec::getParsedSpecifiers() const { | 
|  | 92 | unsigned Res = 0; | 
|  | 93 | if (StorageClassSpec != SCS_unspecified || | 
|  | 94 | SCS_thread_specified) | 
|  | 95 | Res |= PQ_StorageClassSpecifier; | 
| Mike Stump | d420433 | 2008-06-19 19:52:46 +0000 | [diff] [blame] | 96 |  | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 97 | if (TypeQualifiers != TQ_unspecified) | 
|  | 98 | Res |= PQ_TypeQualifier; | 
|  | 99 |  | 
|  | 100 | if (hasTypeSpecifier()) | 
|  | 101 | Res |= PQ_TypeSpecifier; | 
|  | 102 |  | 
| Douglas Gregor | b48fe38 | 2008-10-31 09:07:45 +0000 | [diff] [blame] | 103 | if (FS_inline_specified || FS_virtual_specified || FS_explicit_specified) | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 104 | Res |= PQ_FunctionSpecifier; | 
|  | 105 | return Res; | 
|  | 106 | } | 
|  | 107 |  | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 108 | template <class T> static bool BadSpecifier(T TPrev, const char *&PrevSpec) { | 
|  | 109 | PrevSpec = DeclSpec::getSpecifierName(TPrev); | 
|  | 110 | return true; | 
|  | 111 | } | 
|  | 112 |  | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 113 | const char *DeclSpec::getSpecifierName(DeclSpec::SCS S) { | 
|  | 114 | switch (S) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 115 | case DeclSpec::SCS_unspecified: return "unspecified"; | 
|  | 116 | case DeclSpec::SCS_typedef:     return "typedef"; | 
|  | 117 | case DeclSpec::SCS_extern:      return "extern"; | 
|  | 118 | case DeclSpec::SCS_static:      return "static"; | 
|  | 119 | case DeclSpec::SCS_auto:        return "auto"; | 
|  | 120 | case DeclSpec::SCS_register:    return "register"; | 
| Eli Friedman | 63054b3 | 2009-04-19 20:27:55 +0000 | [diff] [blame] | 121 | case DeclSpec::SCS_private_extern: return "__private_extern__"; | 
| Sebastian Redl | 669d5d7 | 2008-11-14 23:42:31 +0000 | [diff] [blame] | 122 | case DeclSpec::SCS_mutable:     return "mutable"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 123 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 124 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 125 | } | 
|  | 126 |  | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 127 | const char *DeclSpec::getSpecifierName(TSW W) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 128 | switch (W) { | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 129 | case TSW_unspecified: return "unspecified"; | 
|  | 130 | case TSW_short:       return "short"; | 
|  | 131 | case TSW_long:        return "long"; | 
|  | 132 | case TSW_longlong:    return "long long"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 133 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 134 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 135 | } | 
|  | 136 |  | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 137 | const char *DeclSpec::getSpecifierName(TSC C) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 138 | switch (C) { | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 139 | case TSC_unspecified: return "unspecified"; | 
|  | 140 | case TSC_imaginary:   return "imaginary"; | 
|  | 141 | case TSC_complex:     return "complex"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 142 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 143 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 144 | } | 
|  | 145 |  | 
|  | 146 |  | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 147 | const char *DeclSpec::getSpecifierName(TSS S) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 148 | switch (S) { | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 149 | case TSS_unspecified: return "unspecified"; | 
|  | 150 | case TSS_signed:      return "signed"; | 
|  | 151 | case TSS_unsigned:    return "unsigned"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 152 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 153 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 154 | } | 
|  | 155 |  | 
|  | 156 | const char *DeclSpec::getSpecifierName(DeclSpec::TST T) { | 
|  | 157 | switch (T) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 158 | case DeclSpec::TST_unspecified: return "unspecified"; | 
|  | 159 | case DeclSpec::TST_void:        return "void"; | 
|  | 160 | case DeclSpec::TST_char:        return "char"; | 
| Argyrios Kyrtzidis | 64c438a | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 161 | case DeclSpec::TST_wchar:       return "wchar_t"; | 
| Alisdair Meredith | f5c209d | 2009-07-14 06:30:34 +0000 | [diff] [blame] | 162 | case DeclSpec::TST_char16:      return "char16_t"; | 
|  | 163 | case DeclSpec::TST_char32:      return "char32_t"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 164 | case DeclSpec::TST_int:         return "int"; | 
|  | 165 | case DeclSpec::TST_float:       return "float"; | 
|  | 166 | case DeclSpec::TST_double:      return "double"; | 
|  | 167 | case DeclSpec::TST_bool:        return "_Bool"; | 
|  | 168 | case DeclSpec::TST_decimal32:   return "_Decimal32"; | 
|  | 169 | case DeclSpec::TST_decimal64:   return "_Decimal64"; | 
|  | 170 | case DeclSpec::TST_decimal128:  return "_Decimal128"; | 
|  | 171 | case DeclSpec::TST_enum:        return "enum"; | 
| Chris Lattner | 99dc914 | 2008-04-13 18:59:07 +0000 | [diff] [blame] | 172 | case DeclSpec::TST_class:       return "class"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 173 | case DeclSpec::TST_union:       return "union"; | 
|  | 174 | case DeclSpec::TST_struct:      return "struct"; | 
| Douglas Gregor | 1a51b4a | 2009-02-09 15:09:02 +0000 | [diff] [blame] | 175 | case DeclSpec::TST_typename:    return "type-name"; | 
| Steve Naroff | d1861fd | 2007-07-31 12:34:36 +0000 | [diff] [blame] | 176 | case DeclSpec::TST_typeofType: | 
|  | 177 | case DeclSpec::TST_typeofExpr:  return "typeof"; | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 178 | case DeclSpec::TST_auto:        return "auto"; | 
|  | 179 | case DeclSpec::TST_decltype:    return "(decltype)"; | 
|  | 180 | case DeclSpec::TST_error:       return "(error)"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 181 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 182 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 183 | } | 
|  | 184 |  | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 185 | const char *DeclSpec::getSpecifierName(TQ T) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 186 | switch (T) { | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 187 | case DeclSpec::TQ_unspecified: return "unspecified"; | 
|  | 188 | case DeclSpec::TQ_const:       return "const"; | 
|  | 189 | case DeclSpec::TQ_restrict:    return "restrict"; | 
|  | 190 | case DeclSpec::TQ_volatile:    return "volatile"; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 191 | } | 
| John McCall | 32d335e | 2009-08-03 18:47:27 +0000 | [diff] [blame^] | 192 | llvm::llvm_unreachable("Unknown typespec!"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 193 | } | 
|  | 194 |  | 
|  | 195 | bool DeclSpec::SetStorageClassSpec(SCS S, SourceLocation Loc, | 
|  | 196 | const char *&PrevSpec) { | 
|  | 197 | if (StorageClassSpec != SCS_unspecified) | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 198 | return BadSpecifier((SCS)StorageClassSpec, PrevSpec); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 199 | StorageClassSpec = S; | 
|  | 200 | StorageClassSpecLoc = Loc; | 
| Sebastian Redl | 669d5d7 | 2008-11-14 23:42:31 +0000 | [diff] [blame] | 201 | assert((unsigned)S == StorageClassSpec && "SCS constants overflow bitfield"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 202 | return false; | 
|  | 203 | } | 
|  | 204 |  | 
|  | 205 | bool DeclSpec::SetStorageClassSpecThread(SourceLocation Loc, | 
|  | 206 | const char *&PrevSpec) { | 
|  | 207 | if (SCS_thread_specified) { | 
|  | 208 | PrevSpec = "__thread"; | 
|  | 209 | return true; | 
|  | 210 | } | 
|  | 211 | SCS_thread_specified = true; | 
|  | 212 | SCS_threadLoc = Loc; | 
|  | 213 | return false; | 
|  | 214 | } | 
|  | 215 |  | 
|  | 216 |  | 
|  | 217 | /// These methods set the specified attribute of the DeclSpec, but return true | 
|  | 218 | /// and ignore the request if invalid (e.g. "extern" then "auto" is | 
|  | 219 | /// specified). | 
|  | 220 | bool DeclSpec::SetTypeSpecWidth(TSW W, SourceLocation Loc, | 
|  | 221 | const char *&PrevSpec) { | 
|  | 222 | if (TypeSpecWidth != TSW_unspecified && | 
|  | 223 | // Allow turning long -> long long. | 
|  | 224 | (W != TSW_longlong || TypeSpecWidth != TSW_long)) | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 225 | return BadSpecifier((TSW)TypeSpecWidth, PrevSpec); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 226 | TypeSpecWidth = W; | 
|  | 227 | TSWLoc = Loc; | 
|  | 228 | return false; | 
|  | 229 | } | 
|  | 230 |  | 
|  | 231 | bool DeclSpec::SetTypeSpecComplex(TSC C, SourceLocation Loc, | 
|  | 232 | const char *&PrevSpec) { | 
|  | 233 | if (TypeSpecComplex != TSC_unspecified) | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 234 | return BadSpecifier((TSC)TypeSpecComplex, PrevSpec); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 235 | TypeSpecComplex = C; | 
|  | 236 | TSCLoc = Loc; | 
|  | 237 | return false; | 
|  | 238 | } | 
|  | 239 |  | 
|  | 240 | bool DeclSpec::SetTypeSpecSign(TSS S, SourceLocation Loc, | 
|  | 241 | const char *&PrevSpec) { | 
|  | 242 | if (TypeSpecSign != TSS_unspecified) | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 243 | return BadSpecifier((TSS)TypeSpecSign, PrevSpec); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 244 | TypeSpecSign = S; | 
|  | 245 | TSSLoc = Loc; | 
|  | 246 | return false; | 
|  | 247 | } | 
|  | 248 |  | 
|  | 249 | bool DeclSpec::SetTypeSpecType(TST T, SourceLocation Loc, | 
| Douglas Gregor | 402abb5 | 2009-05-28 23:31:59 +0000 | [diff] [blame] | 250 | const char *&PrevSpec, void *Rep, | 
|  | 251 | bool Owned) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 252 | if (TypeSpecType != TST_unspecified) | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 253 | return BadSpecifier((TST)TypeSpecType, PrevSpec); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 254 | TypeSpecType = T; | 
|  | 255 | TypeRep = Rep; | 
|  | 256 | TSTLoc = Loc; | 
| Douglas Gregor | 402abb5 | 2009-05-28 23:31:59 +0000 | [diff] [blame] | 257 | TypeSpecOwned = Owned; | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 258 | return false; | 
|  | 259 | } | 
|  | 260 |  | 
| Douglas Gregor | ddc29e1 | 2009-02-06 22:42:48 +0000 | [diff] [blame] | 261 | bool DeclSpec::SetTypeSpecError() { | 
|  | 262 | TypeSpecType = TST_error; | 
|  | 263 | TypeRep = 0; | 
|  | 264 | TSTLoc = SourceLocation(); | 
|  | 265 | return false; | 
|  | 266 | } | 
|  | 267 |  | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 268 | bool DeclSpec::SetTypeQual(TQ T, SourceLocation Loc, const char *&PrevSpec, | 
|  | 269 | const LangOptions &Lang) { | 
|  | 270 | // Duplicates turn into warnings pre-C99. | 
|  | 271 | if ((TypeQualifiers & T) && !Lang.C99) | 
|  | 272 | return BadSpecifier(T, PrevSpec); | 
|  | 273 | TypeQualifiers |= T; | 
|  | 274 |  | 
|  | 275 | switch (T) { | 
|  | 276 | default: assert(0 && "Unknown type qualifier!"); | 
|  | 277 | case TQ_const:    TQ_constLoc = Loc; break; | 
|  | 278 | case TQ_restrict: TQ_restrictLoc = Loc; break; | 
|  | 279 | case TQ_volatile: TQ_volatileLoc = Loc; break; | 
|  | 280 | } | 
|  | 281 | return false; | 
|  | 282 | } | 
|  | 283 |  | 
|  | 284 | bool DeclSpec::SetFunctionSpecInline(SourceLocation Loc, const char *&PrevSpec){ | 
|  | 285 | // 'inline inline' is ok. | 
|  | 286 | FS_inline_specified = true; | 
|  | 287 | FS_inlineLoc = Loc; | 
|  | 288 | return false; | 
|  | 289 | } | 
|  | 290 |  | 
| Douglas Gregor | b48fe38 | 2008-10-31 09:07:45 +0000 | [diff] [blame] | 291 | bool DeclSpec::SetFunctionSpecVirtual(SourceLocation Loc, const char *&PrevSpec){ | 
|  | 292 | // 'virtual virtual' is ok. | 
|  | 293 | FS_virtual_specified = true; | 
|  | 294 | FS_virtualLoc = Loc; | 
|  | 295 | return false; | 
|  | 296 | } | 
|  | 297 |  | 
|  | 298 | bool DeclSpec::SetFunctionSpecExplicit(SourceLocation Loc, const char *&PrevSpec){ | 
|  | 299 | // 'explicit explicit' is ok. | 
|  | 300 | FS_explicit_specified = true; | 
|  | 301 | FS_explicitLoc = Loc; | 
|  | 302 | return false; | 
|  | 303 | } | 
|  | 304 |  | 
| Anders Carlsson | f47f7a1 | 2009-05-06 04:46:28 +0000 | [diff] [blame] | 305 | bool DeclSpec::SetFriendSpec(SourceLocation Loc, const char *&PrevSpec) { | 
|  | 306 | if (Friend_specified) { | 
|  | 307 | PrevSpec = "friend"; | 
|  | 308 | return true; | 
|  | 309 | } | 
|  | 310 |  | 
|  | 311 | Friend_specified = true; | 
|  | 312 | FriendLoc = Loc; | 
|  | 313 | return false; | 
|  | 314 | } | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 315 |  | 
|  | 316 | /// Finish - This does final analysis of the declspec, rejecting things like | 
|  | 317 | /// "_Imaginary" (lacking an FP type).  This returns a diagnostic to issue or | 
|  | 318 | /// diag::NUM_DIAGNOSTICS if there is no error.  After calling this method, | 
|  | 319 | /// DeclSpec is guaranteed self-consistent, even if an error occurred. | 
| Douglas Gregor | 9b3064b | 2009-04-01 22:41:11 +0000 | [diff] [blame] | 320 | void DeclSpec::Finish(Diagnostic &D, Preprocessor &PP) { | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 321 | // Check the type specifier components first. | 
| Douglas Gregor | 9b3064b | 2009-04-01 22:41:11 +0000 | [diff] [blame] | 322 | SourceManager &SrcMgr = PP.getSourceManager(); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 323 |  | 
| Argyrios Kyrtzidis | 64c438a | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 324 | // signed/unsigned are only valid with int/char/wchar_t. | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 325 | if (TypeSpecSign != TSS_unspecified) { | 
|  | 326 | if (TypeSpecType == TST_unspecified) | 
|  | 327 | TypeSpecType = TST_int; // unsigned -> unsigned int, signed -> signed int. | 
| Argyrios Kyrtzidis | 64c438a | 2008-08-09 16:51:54 +0000 | [diff] [blame] | 328 | else if (TypeSpecType != TST_int  && | 
|  | 329 | TypeSpecType != TST_char && TypeSpecType != TST_wchar) { | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 330 | Diag(D, TSSLoc, SrcMgr, diag::err_invalid_sign_spec) | 
|  | 331 | << getSpecifierName((TST)TypeSpecType); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 332 | // signed double -> double. | 
|  | 333 | TypeSpecSign = TSS_unspecified; | 
|  | 334 | } | 
|  | 335 | } | 
|  | 336 |  | 
|  | 337 | // Validate the width of the type. | 
|  | 338 | switch (TypeSpecWidth) { | 
|  | 339 | case TSW_unspecified: break; | 
|  | 340 | case TSW_short:    // short int | 
|  | 341 | case TSW_longlong: // long long int | 
|  | 342 | if (TypeSpecType == TST_unspecified) | 
|  | 343 | TypeSpecType = TST_int; // short -> short int, long long -> long long int. | 
|  | 344 | else if (TypeSpecType != TST_int) { | 
| Ted Kremenek | 7a9d49f | 2007-12-11 21:27:55 +0000 | [diff] [blame] | 345 | Diag(D, TSWLoc, SrcMgr, | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 346 | TypeSpecWidth == TSW_short ? diag::err_invalid_short_spec | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 347 | : diag::err_invalid_longlong_spec) | 
|  | 348 | <<  getSpecifierName((TST)TypeSpecType); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 349 | TypeSpecType = TST_int; | 
|  | 350 | } | 
|  | 351 | break; | 
|  | 352 | case TSW_long:  // long double, long int | 
|  | 353 | if (TypeSpecType == TST_unspecified) | 
|  | 354 | TypeSpecType = TST_int;  // long -> long int. | 
|  | 355 | else if (TypeSpecType != TST_int && TypeSpecType != TST_double) { | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 356 | Diag(D, TSWLoc, SrcMgr, diag::err_invalid_long_spec) | 
|  | 357 | << getSpecifierName((TST)TypeSpecType); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 358 | TypeSpecType = TST_int; | 
|  | 359 | } | 
|  | 360 | break; | 
|  | 361 | } | 
|  | 362 |  | 
|  | 363 | // TODO: if the implementation does not implement _Complex or _Imaginary, | 
|  | 364 | // disallow their use.  Need information about the backend. | 
|  | 365 | if (TypeSpecComplex != TSC_unspecified) { | 
|  | 366 | if (TypeSpecType == TST_unspecified) { | 
| Douglas Gregor | 9b3064b | 2009-04-01 22:41:11 +0000 | [diff] [blame] | 367 | Diag(D, TSCLoc, SrcMgr, diag::ext_plain_complex) | 
|  | 368 | << CodeModificationHint::CreateInsertion( | 
|  | 369 | PP.getLocForEndOfToken(getTypeSpecComplexLoc()), | 
|  | 370 | " double"); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 371 | TypeSpecType = TST_double;   // _Complex -> _Complex double. | 
|  | 372 | } else if (TypeSpecType == TST_int || TypeSpecType == TST_char) { | 
|  | 373 | // Note that this intentionally doesn't include _Complex _Bool. | 
| Ted Kremenek | 7a9d49f | 2007-12-11 21:27:55 +0000 | [diff] [blame] | 374 | Diag(D, TSTLoc, SrcMgr, diag::ext_integer_complex); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 375 | } else if (TypeSpecType != TST_float && TypeSpecType != TST_double) { | 
| Chris Lattner | 254be6a | 2008-11-22 08:32:36 +0000 | [diff] [blame] | 376 | Diag(D, TSCLoc, SrcMgr, diag::err_invalid_complex_spec) | 
|  | 377 | << getSpecifierName((TST)TypeSpecType); | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 378 | TypeSpecComplex = TSC_unspecified; | 
|  | 379 | } | 
|  | 380 | } | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 381 |  | 
|  | 382 | // Okay, now we can infer the real type. | 
| Reid Spencer | 5f016e2 | 2007-07-11 17:01:13 +0000 | [diff] [blame] | 383 |  | 
|  | 384 | // TODO: return "auto function" and other bad things based on the real type. | 
|  | 385 |  | 
|  | 386 | // 'data definition has no type or storage class'? | 
|  | 387 | } | 
| Daniel Dunbar | e4858a6 | 2008-08-11 03:45:03 +0000 | [diff] [blame] | 388 |  | 
| Sebastian Redl | a4ed0d8 | 2008-12-28 15:28:59 +0000 | [diff] [blame] | 389 | bool DeclSpec::isMissingDeclaratorOk() { | 
|  | 390 | TST tst = getTypeSpecType(); | 
|  | 391 | return (tst == TST_union | 
|  | 392 | || tst == TST_struct | 
|  | 393 | || tst == TST_class | 
|  | 394 | || tst == TST_enum | 
| Douglas Gregor | 4920f1f | 2009-01-12 22:49:06 +0000 | [diff] [blame] | 395 | ) && getTypeRep() != 0 && StorageClassSpec != DeclSpec::SCS_typedef; | 
| Sebastian Redl | a4ed0d8 | 2008-12-28 15:28:59 +0000 | [diff] [blame] | 396 | } |