blob: e1afd6d924f23a90ce443060b6f5a97a01c8d883 [file] [log] [blame]
Chris Lattner6953a072008-06-26 18:38:35 +00001//===--- SemaDeclAttr.cpp - Declaration Attribute Handling ----------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file implements decl-related attribute processing.
11//
12//===----------------------------------------------------------------------===//
13
14#include "Sema.h"
15#include "clang/AST/ASTContext.h"
Daniel Dunbare0ad2152008-08-11 06:23:49 +000016#include "clang/AST/DeclObjC.h"
17#include "clang/AST/Expr.h"
Daniel Dunbarcc7b1602008-08-11 03:45:03 +000018#include "clang/Basic/Diagnostic.h"
Chris Lattnerdc789562008-06-27 22:18:37 +000019#include "clang/Basic/TargetInfo.h"
Daniel Dunbar8d03cbe2008-08-11 03:27:53 +000020#include "clang/Parse/DeclSpec.h"
Ted Kremenek26151d12008-07-22 16:56:21 +000021#include <llvm/ADT/StringExtras.h>
Chris Lattner6953a072008-06-26 18:38:35 +000022using namespace clang;
23
Chris Lattner2024f0a2008-06-29 00:16:31 +000024//===----------------------------------------------------------------------===//
25// Helper functions
26//===----------------------------------------------------------------------===//
27
Daniel Dunbar0ea20472008-10-19 02:04:16 +000028static const FunctionType *getFunctionType(Decl *d) {
Chris Lattner6953a072008-06-26 18:38:35 +000029 QualType Ty;
Chris Lattner6953a072008-06-26 18:38:35 +000030 if (ValueDecl *decl = dyn_cast<ValueDecl>(d))
31 Ty = decl->getType();
32 else if (FieldDecl *decl = dyn_cast<FieldDecl>(d))
33 Ty = decl->getType();
34 else if (TypedefDecl* decl = dyn_cast<TypedefDecl>(d))
35 Ty = decl->getUnderlyingType();
36 else
37 return 0;
38
39 if (Ty->isFunctionPointerType())
40 Ty = Ty->getAsPointerType()->getPointeeType();
Daniel Dunbar0ea20472008-10-19 02:04:16 +000041
42 return Ty->getAsFunctionType();
Chris Lattner6953a072008-06-26 18:38:35 +000043}
44
Daniel Dunbard1d847c2008-09-26 04:12:28 +000045// FIXME: We should provide an abstraction around a method or function
46// to provide the following bits of information.
47
Daniel Dunbar0ea20472008-10-19 02:04:16 +000048/// isFunctionOrMethod - Return true if the given decl has function
49/// type (function or function-typed variable) or an Objective-C
50/// method.
Daniel Dunbard1d847c2008-09-26 04:12:28 +000051static bool isFunctionOrMethod(Decl *d) {
Daniel Dunbar0ea20472008-10-19 02:04:16 +000052 return getFunctionType(d) || isa<ObjCMethodDecl>(d);
Daniel Dunbard1d847c2008-09-26 04:12:28 +000053}
54
Daniel Dunbar0ea20472008-10-19 02:04:16 +000055/// hasFunctionProto - Return true if the given decl has a argument
56/// information. This decl should have already passed
57/// isFunctionOrMethod.
58static bool hasFunctionProto(Decl *d) {
59 if (const FunctionType *FnTy = getFunctionType(d)) {
60 return isa<FunctionTypeProto>(FnTy);
61 } else {
62 assert(isa<ObjCMethodDecl>(d));
63 return true;
64 }
65}
66
67/// getFunctionOrMethodNumArgs - Return number of function or method
68/// arguments. It is an error to call this on a K&R function (use
69/// hasFunctionProto first).
Daniel Dunbard1d847c2008-09-26 04:12:28 +000070static unsigned getFunctionOrMethodNumArgs(Decl *d) {
Daniel Dunbar0ea20472008-10-19 02:04:16 +000071 if (const FunctionType *FnTy = getFunctionType(d)) {
72 const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
Daniel Dunbard1d847c2008-09-26 04:12:28 +000073 return proto->getNumArgs();
74 } else {
75 return cast<ObjCMethodDecl>(d)->getNumParams();
76 }
77}
78
79static QualType getFunctionOrMethodArgType(Decl *d, unsigned Idx) {
Daniel Dunbar0ea20472008-10-19 02:04:16 +000080 if (const FunctionType *FnTy = getFunctionType(d)) {
81 const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
Daniel Dunbard1d847c2008-09-26 04:12:28 +000082 return proto->getArgType(Idx);
83 } else {
84 return cast<ObjCMethodDecl>(d)->getParamDecl(Idx)->getType();
85 }
86}
87
88static bool isFunctionOrMethodVariadic(Decl *d) {
Daniel Dunbar0ea20472008-10-19 02:04:16 +000089 if (const FunctionType *FnTy = getFunctionType(d)) {
90 const FunctionTypeProto *proto = cast<FunctionTypeProto>(FnTy);
Daniel Dunbard1d847c2008-09-26 04:12:28 +000091 return proto->isVariadic();
92 } else {
93 return cast<ObjCMethodDecl>(d)->isVariadic();
94 }
95}
96
Chris Lattner6953a072008-06-26 18:38:35 +000097static inline bool isNSStringType(QualType T, ASTContext &Ctx) {
Chris Lattnerd5a56aa2008-07-26 22:17:49 +000098 const PointerType *PT = T->getAsPointerType();
99 if (!PT)
Chris Lattner6953a072008-06-26 18:38:35 +0000100 return false;
101
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000102 const ObjCInterfaceType *ClsT =PT->getPointeeType()->getAsObjCInterfaceType();
Chris Lattner6953a072008-06-26 18:38:35 +0000103 if (!ClsT)
104 return false;
105
106 IdentifierInfo* ClsName = ClsT->getDecl()->getIdentifier();
107
108 // FIXME: Should we walk the chain of classes?
109 return ClsName == &Ctx.Idents.get("NSString") ||
110 ClsName == &Ctx.Idents.get("NSMutableString");
111}
112
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000113static inline bool isCFStringType(QualType T, ASTContext &Ctx) {
114 const PointerType *PT = T->getAsPointerType();
115 if (!PT)
116 return false;
117
118 const RecordType *RT = PT->getPointeeType()->getAsRecordType();
119 if (!RT)
120 return false;
121
122 const RecordDecl *RD = RT->getDecl();
123 if (RD->getTagKind() != TagDecl::TK_struct)
124 return false;
125
126 return RD->getIdentifier() == &Ctx.Idents.get("__CFString");
127}
128
Chris Lattner2024f0a2008-06-29 00:16:31 +0000129//===----------------------------------------------------------------------===//
Chris Lattner2024f0a2008-06-29 00:16:31 +0000130// Attribute Implementations
131//===----------------------------------------------------------------------===//
132
Daniel Dunbar8716a012008-07-31 22:40:48 +0000133// FIXME: All this manual attribute parsing code is gross. At the
134// least add some helper functions to check most argument patterns (#
135// and types of args).
136
Chris Lattner703c52d2008-06-29 00:43:07 +0000137static void HandleExtVectorTypeAttr(Decl *d, const AttributeList &Attr,
138 Sema &S) {
Chris Lattner1c151132008-06-28 23:36:30 +0000139 TypedefDecl *tDecl = dyn_cast<TypedefDecl>(d);
140 if (tDecl == 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000141 S.Diag(Attr.getLoc(), diag::err_typecheck_ext_vector_not_typedef);
Chris Lattner1c151132008-06-28 23:36:30 +0000142 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000143 }
144
Chris Lattner6953a072008-06-26 18:38:35 +0000145 QualType curType = tDecl->getUnderlyingType();
146 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000147 if (Attr.getNumArgs() != 1) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000148 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
149 std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000150 return;
151 }
Chris Lattner1c151132008-06-28 23:36:30 +0000152 Expr *sizeExpr = static_cast<Expr *>(Attr.getArg(0));
Chris Lattner6953a072008-06-26 18:38:35 +0000153 llvm::APSInt vecSize(32);
Chris Lattner703c52d2008-06-29 00:43:07 +0000154 if (!sizeExpr->isIntegerConstantExpr(vecSize, S.Context)) {
155 S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int,
156 "ext_vector_type", sizeExpr->getSourceRange());
Chris Lattner6953a072008-06-26 18:38:35 +0000157 return;
158 }
159 // unlike gcc's vector_size attribute, we do not allow vectors to be defined
160 // in conjunction with complex types (pointers, arrays, functions, etc.).
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000161 if (!curType->isIntegerType() && !curType->isRealFloatingType()) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000162 S.Diag(Attr.getLoc(), diag::err_attribute_invalid_vector_type,
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000163 curType.getAsString());
Chris Lattner6953a072008-06-26 18:38:35 +0000164 return;
165 }
166 // unlike gcc's vector_size attribute, the size is specified as the
167 // number of elements, not the number of bytes.
168 unsigned vectorSize = static_cast<unsigned>(vecSize.getZExtValue());
169
170 if (vectorSize == 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000171 S.Diag(Attr.getLoc(), diag::err_attribute_zero_size,
172 sizeExpr->getSourceRange());
Chris Lattner6953a072008-06-26 18:38:35 +0000173 return;
174 }
175 // Instantiate/Install the vector type, the number of elements is > 0.
Chris Lattner703c52d2008-06-29 00:43:07 +0000176 tDecl->setUnderlyingType(S.Context.getExtVectorType(curType, vectorSize));
Chris Lattner6953a072008-06-26 18:38:35 +0000177 // Remember this typedef decl, we will need it later for diagnostics.
Chris Lattner703c52d2008-06-29 00:43:07 +0000178 S.ExtVectorDecls.push_back(tDecl);
Chris Lattner6953a072008-06-26 18:38:35 +0000179}
180
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000181
182/// HandleVectorSizeAttribute - this attribute is only applicable to
183/// integral and float scalars, although arrays, pointers, and function
184/// return values are allowed in conjunction with this construct. Aggregates
185/// with this attribute are invalid, even if they are of the same size as a
186/// corresponding scalar.
187/// The raw attribute should contain precisely 1 argument, the vector size
188/// for the variable, measured in bytes. If curType and rawAttr are well
189/// formed, this routine will return a new vector type.
Chris Lattner703c52d2008-06-29 00:43:07 +0000190static void HandleVectorSizeAttr(Decl *D, const AttributeList &Attr, Sema &S) {
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000191 QualType CurType;
192 if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
193 CurType = VD->getType();
194 else if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
195 CurType = TD->getUnderlyingType();
196 else {
Chris Lattner703c52d2008-06-29 00:43:07 +0000197 S.Diag(D->getLocation(), diag::err_attr_wrong_decl,
198 std::string("vector_size"),
199 SourceRange(Attr.getLoc(), Attr.getLoc()));
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000200 return;
201 }
202
203 // Check the attribute arugments.
Chris Lattner1c151132008-06-28 23:36:30 +0000204 if (Attr.getNumArgs() != 1) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000205 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
206 std::string("1"));
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000207 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000208 }
Chris Lattner1c151132008-06-28 23:36:30 +0000209 Expr *sizeExpr = static_cast<Expr *>(Attr.getArg(0));
Chris Lattner6953a072008-06-26 18:38:35 +0000210 llvm::APSInt vecSize(32);
Chris Lattner703c52d2008-06-29 00:43:07 +0000211 if (!sizeExpr->isIntegerConstantExpr(vecSize, S.Context)) {
212 S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int,
213 "vector_size", sizeExpr->getSourceRange());
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000214 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000215 }
216 // navigate to the base type - we need to provide for vector pointers,
217 // vector arrays, and functions returning vectors.
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000218 if (CurType->isPointerType() || CurType->isArrayType() ||
219 CurType->isFunctionType()) {
Chris Lattner6953a072008-06-26 18:38:35 +0000220 assert(0 && "HandleVector(): Complex type construction unimplemented");
221 /* FIXME: rebuild the type from the inside out, vectorizing the inner type.
222 do {
223 if (PointerType *PT = dyn_cast<PointerType>(canonType))
224 canonType = PT->getPointeeType().getTypePtr();
225 else if (ArrayType *AT = dyn_cast<ArrayType>(canonType))
226 canonType = AT->getElementType().getTypePtr();
227 else if (FunctionType *FT = dyn_cast<FunctionType>(canonType))
228 canonType = FT->getResultType().getTypePtr();
229 } while (canonType->isPointerType() || canonType->isArrayType() ||
230 canonType->isFunctionType());
231 */
232 }
233 // the base type must be integer or float.
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000234 if (!CurType->isIntegerType() && !CurType->isRealFloatingType()) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000235 S.Diag(Attr.getLoc(), diag::err_attribute_invalid_vector_type,
Chris Lattnerd5a56aa2008-07-26 22:17:49 +0000236 CurType.getAsString());
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000237 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000238 }
Chris Lattner703c52d2008-06-29 00:43:07 +0000239 unsigned typeSize = static_cast<unsigned>(S.Context.getTypeSize(CurType));
Chris Lattner6953a072008-06-26 18:38:35 +0000240 // vecSize is specified in bytes - convert to bits.
241 unsigned vectorSize = static_cast<unsigned>(vecSize.getZExtValue() * 8);
242
243 // the vector size needs to be an integral multiple of the type size.
244 if (vectorSize % typeSize) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000245 S.Diag(Attr.getLoc(), diag::err_attribute_invalid_size,
246 sizeExpr->getSourceRange());
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000247 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000248 }
249 if (vectorSize == 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000250 S.Diag(Attr.getLoc(), diag::err_attribute_zero_size,
251 sizeExpr->getSourceRange());
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000252 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000253 }
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000254
255 // Success! Instantiate the vector type, the number of elements is > 0, and
256 // not required to be a power of 2, unlike GCC.
Chris Lattner703c52d2008-06-29 00:43:07 +0000257 CurType = S.Context.getVectorType(CurType, vectorSize/typeSize);
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000258
259 if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
260 VD->setType(CurType);
261 else
262 cast<TypedefDecl>(D)->setUnderlyingType(CurType);
Chris Lattner6953a072008-06-26 18:38:35 +0000263}
264
Chris Lattner703c52d2008-06-29 00:43:07 +0000265static void HandlePackedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000266 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000267 if (Attr.getNumArgs() > 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000268 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
269 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000270 return;
271 }
272
273 if (TagDecl *TD = dyn_cast<TagDecl>(d))
Daniel Dunbar2cb762f2008-10-16 02:34:03 +0000274 TD->addAttr(new PackedAttr(1));
Chris Lattner6953a072008-06-26 18:38:35 +0000275 else if (FieldDecl *FD = dyn_cast<FieldDecl>(d)) {
276 // If the alignment is less than or equal to 8 bits, the packed attribute
277 // has no effect.
278 if (!FD->getType()->isIncompleteType() &&
Chris Lattner703c52d2008-06-29 00:43:07 +0000279 S.Context.getTypeAlign(FD->getType()) <= 8)
280 S.Diag(Attr.getLoc(),
281 diag::warn_attribute_ignored_for_field_of_type,
282 Attr.getName()->getName(), FD->getType().getAsString());
Chris Lattner6953a072008-06-26 18:38:35 +0000283 else
Daniel Dunbar2cb762f2008-10-16 02:34:03 +0000284 FD->addAttr(new PackedAttr(1));
Chris Lattner6953a072008-06-26 18:38:35 +0000285 } else
Chris Lattner703c52d2008-06-29 00:43:07 +0000286 S.Diag(Attr.getLoc(), diag::warn_attribute_ignored,
287 Attr.getName()->getName());
Chris Lattner6953a072008-06-26 18:38:35 +0000288}
289
Ted Kremenek4e5bb122008-07-15 22:26:48 +0000290static void HandleIBOutletAttr(Decl *d, const AttributeList &Attr, Sema &S) {
291 // check the attribute arguments.
292 if (Attr.getNumArgs() > 0) {
293 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
294 std::string("0"));
295 return;
296 }
297
298 // The IBOutlet attribute only applies to instance variables of Objective-C
299 // classes.
300 if (ObjCIvarDecl *ID = dyn_cast<ObjCIvarDecl>(d))
301 ID->addAttr(new IBOutletAttr());
302 else
303 S.Diag(Attr.getLoc(), diag::err_attribute_iboutlet_non_ivar);
304}
305
Ted Kremenekc0af2542008-07-21 21:53:04 +0000306static void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Ted Kremenekc0af2542008-07-21 21:53:04 +0000307 // GCC ignores the nonnull attribute on K&R style function
308 // prototypes, so we ignore it as well
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000309 if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
Ted Kremenekc0af2542008-07-21 21:53:04 +0000310 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
311 "nonnull", "function");
312 return;
313 }
314
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000315 unsigned NumArgs = getFunctionOrMethodNumArgs(d);
Ted Kremenekc0af2542008-07-21 21:53:04 +0000316
317 // The nonnull attribute only applies to pointers.
318 llvm::SmallVector<unsigned, 10> NonNullArgs;
319
320 for (AttributeList::arg_iterator I=Attr.arg_begin(),
321 E=Attr.arg_end(); I!=E; ++I) {
322
323
324 // The argument must be an integer constant expression.
325 Expr *Ex = static_cast<Expr *>(Attr.getArg(0));
326 llvm::APSInt ArgNum(32);
327 if (!Ex->isIntegerConstantExpr(ArgNum, S.Context)) {
328 S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int,
329 "nonnull", Ex->getSourceRange());
330 return;
331 }
332
333 unsigned x = (unsigned) ArgNum.getZExtValue();
334
335 if (x < 1 || x > NumArgs) {
336 S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds,
Ted Kremenek26151d12008-07-22 16:56:21 +0000337 "nonnull", llvm::utostr_32(I.getArgNum()), Ex->getSourceRange());
Ted Kremenekc0af2542008-07-21 21:53:04 +0000338 return;
339 }
Ted Kremenek178cee22008-07-21 22:09:15 +0000340
341 --x;
Ted Kremenekc0af2542008-07-21 21:53:04 +0000342
343 // Is the function argument a pointer type?
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000344 if (!getFunctionOrMethodArgType(d, x)->isPointerType()) {
Ted Kremenekc0af2542008-07-21 21:53:04 +0000345 // FIXME: Should also highlight argument in decl.
346 S.Diag(Attr.getLoc(), diag::err_nonnull_pointers_only,
347 "nonnull", Ex->getSourceRange());
Ted Kremenek7af441b2008-09-01 19:57:52 +0000348 continue;
Ted Kremenekc0af2542008-07-21 21:53:04 +0000349 }
350
351 NonNullArgs.push_back(x);
352 }
353
Ted Kremenek7af441b2008-09-01 19:57:52 +0000354 // If no arguments were specified to __attribute__((nonnull)) then all
355 // pointer arguments have a nonnull attribute.
356 if (NonNullArgs.empty()) {
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000357 for (unsigned I = 0, E = getFunctionOrMethodNumArgs(d); I != E; ++I)
358 if (getFunctionOrMethodArgType(d, I)->isPointerType())
359 NonNullArgs.push_back(I);
Ted Kremenek7af441b2008-09-01 19:57:52 +0000360
361 if (NonNullArgs.empty()) {
362 S.Diag(Attr.getLoc(), diag::warn_attribute_nonnull_no_pointers);
363 return;
364 }
Ted Kremenekc0af2542008-07-21 21:53:04 +0000365 }
Ted Kremenek7af441b2008-09-01 19:57:52 +0000366
367 unsigned* start = &NonNullArgs[0];
368 unsigned size = NonNullArgs.size();
369 std::sort(start, start + size);
370 d->addAttr(new NonNullAttr(start, size));
Ted Kremenekc0af2542008-07-21 21:53:04 +0000371}
372
Chris Lattner703c52d2008-06-29 00:43:07 +0000373static void HandleAliasAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000374 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000375 if (Attr.getNumArgs() != 1) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000376 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
377 std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000378 return;
379 }
380
Chris Lattner1c151132008-06-28 23:36:30 +0000381 Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
Chris Lattner6953a072008-06-26 18:38:35 +0000382 Arg = Arg->IgnoreParenCasts();
383 StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
384
385 if (Str == 0 || Str->isWide()) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000386 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string,
387 "alias", std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000388 return;
389 }
390
391 const char *Alias = Str->getStrData();
392 unsigned AliasLen = Str->getByteLength();
393
394 // FIXME: check if target symbol exists in current file
395
396 d->addAttr(new AliasAttr(std::string(Alias, AliasLen)));
397}
398
Daniel Dunbar0a2da712008-10-28 00:17:57 +0000399static void HandleAlwaysInlineAttr(Decl *d, const AttributeList &Attr,
400 Sema &S) {
401 // check the attribute arguments.
402 if (Attr.getNumArgs() != 0) {
403 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
404 std::string("0"));
405 return;
406 }
407
408 d->addAttr(new AlwaysInlineAttr());
409}
410
Chris Lattner703c52d2008-06-29 00:43:07 +0000411static void HandleNoReturnAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000412 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000413 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000414 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
415 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000416 return;
417 }
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000418
419 if (!isFunctionOrMethod(d)) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000420 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
421 "noreturn", "function");
Chris Lattner6953a072008-06-26 18:38:35 +0000422 return;
423 }
424
425 d->addAttr(new NoReturnAttr());
426}
427
Ted Kremenekce13e1e2008-07-25 04:39:19 +0000428static void HandleUnusedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
429 // check the attribute arguments.
430 if (Attr.getNumArgs() != 0) {
431 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
432 std::string("0"));
433 return;
434 }
435
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000436 if (!isa<VarDecl>(d) && !isFunctionOrMethod(d)) {
Ted Kremenekce13e1e2008-07-25 04:39:19 +0000437 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
Ted Kremenek4159e8a2008-08-07 01:02:05 +0000438 "unused", "variable and function");
Ted Kremenekce13e1e2008-07-25 04:39:19 +0000439 return;
440 }
441
442 d->addAttr(new UnusedAttr());
443}
444
Daniel Dunbar8716a012008-07-31 22:40:48 +0000445static void HandleConstructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
446 // check the attribute arguments.
447 if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
448 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments, "0 or 1");
449 return;
450 }
451
452 int priority = 65535; // FIXME: Do not hardcode such constants.
453 if (Attr.getNumArgs() > 0) {
454 Expr *E = static_cast<Expr *>(Attr.getArg(0));
455 llvm::APSInt Idx(32);
456 if (!E->isIntegerConstantExpr(Idx, S.Context)) {
457 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
458 "constructor", "1", E->getSourceRange());
459 return;
460 }
461 priority = Idx.getZExtValue();
462 }
463
464 FunctionDecl *Fn = dyn_cast<FunctionDecl>(d);
465 if (!Fn) {
466 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
467 "constructor", "function");
468 return;
469 }
470
471 d->addAttr(new ConstructorAttr(priority));
472}
473
474static void HandleDestructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
475 // check the attribute arguments.
476 if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
477 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments, "0 or 1");
478 return;
479 }
480
481 int priority = 65535; // FIXME: Do not hardcode such constants.
482 if (Attr.getNumArgs() > 0) {
483 Expr *E = static_cast<Expr *>(Attr.getArg(0));
484 llvm::APSInt Idx(32);
485 if (!E->isIntegerConstantExpr(Idx, S.Context)) {
486 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
487 "destructor", "1", E->getSourceRange());
488 return;
489 }
490 priority = Idx.getZExtValue();
491 }
492
Anders Carlsson03e49652008-08-22 22:10:48 +0000493 if (!isa<FunctionDecl>(d)) {
Daniel Dunbar8716a012008-07-31 22:40:48 +0000494 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
495 "destructor", "function");
496 return;
497 }
498
499 d->addAttr(new DestructorAttr(priority));
500}
501
Chris Lattner703c52d2008-06-29 00:43:07 +0000502static void HandleDeprecatedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000503 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000504 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000505 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
506 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000507 return;
508 }
509
510 d->addAttr(new DeprecatedAttr());
511}
512
Chris Lattner703c52d2008-06-29 00:43:07 +0000513static void HandleVisibilityAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000514 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000515 if (Attr.getNumArgs() != 1) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000516 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
517 std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000518 return;
519 }
520
Chris Lattner1c151132008-06-28 23:36:30 +0000521 Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
Chris Lattner6953a072008-06-26 18:38:35 +0000522 Arg = Arg->IgnoreParenCasts();
523 StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
524
525 if (Str == 0 || Str->isWide()) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000526 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string,
527 "visibility", std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000528 return;
529 }
530
531 const char *TypeStr = Str->getStrData();
532 unsigned TypeLen = Str->getByteLength();
533 VisibilityAttr::VisibilityTypes type;
534
535 if (TypeLen == 7 && !memcmp(TypeStr, "default", 7))
536 type = VisibilityAttr::DefaultVisibility;
537 else if (TypeLen == 6 && !memcmp(TypeStr, "hidden", 6))
538 type = VisibilityAttr::HiddenVisibility;
539 else if (TypeLen == 8 && !memcmp(TypeStr, "internal", 8))
540 type = VisibilityAttr::HiddenVisibility; // FIXME
541 else if (TypeLen == 9 && !memcmp(TypeStr, "protected", 9))
542 type = VisibilityAttr::ProtectedVisibility;
543 else {
Chris Lattner703c52d2008-06-29 00:43:07 +0000544 S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported,
545 "visibility", TypeStr);
Chris Lattner6953a072008-06-26 18:38:35 +0000546 return;
547 }
548
549 d->addAttr(new VisibilityAttr(type));
550}
551
Anders Carlssond83141a52008-08-23 23:22:21 +0000552static void HandleObjCGCAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Anders Carlssone0f00b32008-08-24 16:33:25 +0000553 if (!Attr.getParameterName()) {
Anders Carlssond83141a52008-08-23 23:22:21 +0000554 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string,
555 "objc_gc", std::string("1"));
556 return;
557 }
558
559 if (Attr.getNumArgs() != 0) {
560 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
561 std::string("1"));
562 return;
563 }
564
565 const char *TypeStr = Attr.getParameterName()->getName();
566 unsigned TypeLen = Attr.getParameterName()->getLength();
567
568 ObjCGCAttr::GCAttrTypes type;
569
570 if (TypeLen == 4 && !memcmp(TypeStr, "weak", 4))
571 type = ObjCGCAttr::Weak;
Anders Carlssone0f00b32008-08-24 16:33:25 +0000572 else if (TypeLen == 6 && !memcmp(TypeStr, "strong", 6))
Anders Carlssond83141a52008-08-23 23:22:21 +0000573 type = ObjCGCAttr::Strong;
574 else {
575 S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported,
576 "objc_gc", TypeStr);
577 return;
578 }
579
580 d->addAttr(new ObjCGCAttr(type));
581}
582
Steve Naroffe1cecba2008-09-18 16:44:58 +0000583static void HandleBlocksAttr(Decl *d, const AttributeList &Attr, Sema &S) {
584 if (!Attr.getParameterName()) {
585 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string,
586 "blocks", std::string("1"));
587 return;
588 }
589
590 if (Attr.getNumArgs() != 0) {
591 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
592 std::string("1"));
593 return;
594 }
595 const char *TypeStr = Attr.getParameterName()->getName();
596 unsigned TypeLen = Attr.getParameterName()->getLength();
597
598 BlocksAttr::BlocksAttrTypes type;
599
600 if (TypeLen == 5 && !memcmp(TypeStr, "byref", 5))
601 type = BlocksAttr::ByRef;
602 else {
603 S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported,
604 "blocks", TypeStr);
605 return;
606 }
607
608 d->addAttr(new BlocksAttr(type));
609}
610
Anders Carlsson244d99b2008-10-05 18:05:59 +0000611static void HandleSentinelAttr(Decl *d, const AttributeList &Attr, Sema &S) {
612 // check the attribute arguments.
613 if (Attr.getNumArgs() > 2) {
614 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments, "0, 1 or 2");
615 return;
616 }
617
618 int sentinel = 0;
619 if (Attr.getNumArgs() > 0) {
620 Expr *E = static_cast<Expr *>(Attr.getArg(0));
621 llvm::APSInt Idx(32);
622 if (!E->isIntegerConstantExpr(Idx, S.Context)) {
623 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
624 "sentinel", "1", E->getSourceRange());
625 return;
626 }
627 sentinel = Idx.getZExtValue();
628
629 if (sentinel < 0) {
630 S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_less_than_zero,
631 E->getSourceRange());
632 return;
633 }
634 }
635
636 int nullPos = 0;
637 if (Attr.getNumArgs() > 1) {
638 Expr *E = static_cast<Expr *>(Attr.getArg(1));
639 llvm::APSInt Idx(32);
640 if (!E->isIntegerConstantExpr(Idx, S.Context)) {
641 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
642 "sentinel", "2", E->getSourceRange());
643 return;
644 }
645 nullPos = Idx.getZExtValue();
646
647 if (nullPos > 1 || nullPos < 0) {
648 // FIXME: This error message could be improved, it would be nice
649 // to say what the bounds actually are.
650 S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_not_zero_or_one,
651 E->getSourceRange());
652 return;
653 }
654 }
655
656 if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d)) {
657 QualType FT = FD->getType();
658 if (!FT->getAsFunctionTypeProto()->isVariadic()) {
659 S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic);
660 return;
661 }
662 } else if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d)) {
663 if (!MD->isVariadic()) {
664 S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic);
665 return;
666 }
667 } else {
668 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
669 "sentinel", "function or method");
670 return;
671 }
672
673 // FIXME: Actually create the attribute.
674}
675
Chris Lattner703c52d2008-06-29 00:43:07 +0000676static void HandleWeakAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000677 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000678 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000679 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
680 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000681 return;
682 }
683
684 d->addAttr(new WeakAttr());
685}
686
Chris Lattner703c52d2008-06-29 00:43:07 +0000687static void HandleDLLImportAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000688 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000689 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000690 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
691 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000692 return;
693 }
694
695 d->addAttr(new DLLImportAttr());
696}
697
Chris Lattner703c52d2008-06-29 00:43:07 +0000698static void HandleDLLExportAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000699 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000700 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000701 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
702 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000703 return;
704 }
705
706 d->addAttr(new DLLExportAttr());
707}
708
Chris Lattner703c52d2008-06-29 00:43:07 +0000709static void HandleStdCallAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000710 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000711 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000712 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
713 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000714 return;
715 }
716
717 d->addAttr(new StdCallAttr());
718}
719
Chris Lattner703c52d2008-06-29 00:43:07 +0000720static void HandleFastCallAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000721 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000722 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000723 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
724 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000725 return;
726 }
727
728 d->addAttr(new FastCallAttr());
729}
730
Chris Lattner703c52d2008-06-29 00:43:07 +0000731static void HandleNothrowAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000732 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000733 if (Attr.getNumArgs() != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000734 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
735 std::string("0"));
Chris Lattner6953a072008-06-26 18:38:35 +0000736 return;
737 }
738
739 d->addAttr(new NoThrowAttr());
740}
741
Anders Carlssondd6791c2008-10-05 23:32:53 +0000742static void HandleConstAttr(Decl *d, const AttributeList &Attr, Sema &S) {
743 // check the attribute arguments.
744 if (Attr.getNumArgs() != 0) {
745 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
746 std::string("0"));
747 return;
748 }
749
750 d->addAttr(new ConstAttr());
751}
752
753static void HandlePureAttr(Decl *d, const AttributeList &Attr, Sema &S) {
754 // check the attribute arguments.
755 if (Attr.getNumArgs() != 0) {
756 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
757 std::string("0"));
758 return;
759 }
760
761 d->addAttr(new PureAttr());
762}
763
Chris Lattner6953a072008-06-26 18:38:35 +0000764/// Handle __attribute__((format(type,idx,firstarg))) attributes
765/// based on http://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html
Chris Lattner703c52d2008-06-29 00:43:07 +0000766static void HandleFormatAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000767
Chris Lattner1c151132008-06-28 23:36:30 +0000768 if (!Attr.getParameterName()) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000769 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string,
Chris Lattner6953a072008-06-26 18:38:35 +0000770 "format", std::string("1"));
771 return;
772 }
773
Chris Lattner1c151132008-06-28 23:36:30 +0000774 if (Attr.getNumArgs() != 2) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000775 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
776 std::string("3"));
Chris Lattner6953a072008-06-26 18:38:35 +0000777 return;
778 }
779
Daniel Dunbar0ea20472008-10-19 02:04:16 +0000780 if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000781 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
782 "format", "function");
Chris Lattner6953a072008-06-26 18:38:35 +0000783 return;
784 }
785
786 // FIXME: in C++ the implicit 'this' function parameter also counts.
787 // this is needed in order to be compatible with GCC
788 // the index must start in 1 and the limit is numargs+1
Daniel Dunbard1d847c2008-09-26 04:12:28 +0000789 unsigned NumArgs = getFunctionOrMethodNumArgs(d);
Chris Lattner6953a072008-06-26 18:38:35 +0000790 unsigned FirstIdx = 1;
791
Chris Lattner1c151132008-06-28 23:36:30 +0000792 const char *Format = Attr.getParameterName()->getName();
793 unsigned FormatLen = Attr.getParameterName()->getLength();
Chris Lattner6953a072008-06-26 18:38:35 +0000794
795 // Normalize the argument, __foo__ becomes foo.
796 if (FormatLen > 4 && Format[0] == '_' && Format[1] == '_' &&
797 Format[FormatLen - 2] == '_' && Format[FormatLen - 1] == '_') {
798 Format += 2;
799 FormatLen -= 4;
800 }
801
802 bool Supported = false;
803 bool is_NSString = false;
804 bool is_strftime = false;
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000805 bool is_CFString = false;
Chris Lattner6953a072008-06-26 18:38:35 +0000806
807 switch (FormatLen) {
808 default: break;
Chris Lattner703c52d2008-06-29 00:43:07 +0000809 case 5: Supported = !memcmp(Format, "scanf", 5); break;
810 case 6: Supported = !memcmp(Format, "printf", 6); break;
811 case 7: Supported = !memcmp(Format, "strfmon", 7); break;
Chris Lattner6953a072008-06-26 18:38:35 +0000812 case 8:
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000813 Supported = (is_strftime = !memcmp(Format, "strftime", 8)) ||
814 (is_NSString = !memcmp(Format, "NSString", 8)) ||
815 (is_CFString = !memcmp(Format, "CFString", 8));
Chris Lattner6953a072008-06-26 18:38:35 +0000816 break;
817 }
818
819 if (!Supported) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000820 S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported,
Chris Lattner1c151132008-06-28 23:36:30 +0000821 "format", Attr.getParameterName()->getName());
Chris Lattner6953a072008-06-26 18:38:35 +0000822 return;
823 }
824
825 // checks for the 2nd argument
Chris Lattner1c151132008-06-28 23:36:30 +0000826 Expr *IdxExpr = static_cast<Expr *>(Attr.getArg(0));
Chris Lattner703c52d2008-06-29 00:43:07 +0000827 llvm::APSInt Idx(32);
828 if (!IdxExpr->isIntegerConstantExpr(Idx, S.Context)) {
829 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
Chris Lattner6953a072008-06-26 18:38:35 +0000830 "format", std::string("2"), IdxExpr->getSourceRange());
831 return;
832 }
833
834 if (Idx.getZExtValue() < FirstIdx || Idx.getZExtValue() > NumArgs) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000835 S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds,
Chris Lattner6953a072008-06-26 18:38:35 +0000836 "format", std::string("2"), IdxExpr->getSourceRange());
837 return;
838 }
839
840 // FIXME: Do we need to bounds check?
841 unsigned ArgIdx = Idx.getZExtValue() - 1;
842
843 // make sure the format string is really a string
Daniel Dunbard1d847c2008-09-26 04:12:28 +0000844 QualType Ty = getFunctionOrMethodArgType(d, ArgIdx);
Chris Lattner6953a072008-06-26 18:38:35 +0000845
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000846 if (is_CFString) {
847 if (!isCFStringType(Ty, S.Context)) {
848 S.Diag(Attr.getLoc(), diag::err_format_attribute_not,
849 "a CFString", IdxExpr->getSourceRange());
850 return;
851 }
852 } else if (is_NSString) {
Chris Lattner6953a072008-06-26 18:38:35 +0000853 // FIXME: do we need to check if the type is NSString*? What are
854 // the semantics?
Chris Lattner703c52d2008-06-29 00:43:07 +0000855 if (!isNSStringType(Ty, S.Context)) {
Chris Lattner6953a072008-06-26 18:38:35 +0000856 // FIXME: Should highlight the actual expression that has the
857 // wrong type.
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000858 S.Diag(Attr.getLoc(), diag::err_format_attribute_not,
859 "an NSString", IdxExpr->getSourceRange());
Chris Lattner6953a072008-06-26 18:38:35 +0000860 return;
861 }
862 } else if (!Ty->isPointerType() ||
863 !Ty->getAsPointerType()->getPointeeType()->isCharType()) {
864 // FIXME: Should highlight the actual expression that has the
865 // wrong type.
Daniel Dunbarbddb14a2008-09-26 03:32:58 +0000866 S.Diag(Attr.getLoc(), diag::err_format_attribute_not,
867 "a string type", IdxExpr->getSourceRange());
Chris Lattner6953a072008-06-26 18:38:35 +0000868 return;
869 }
870
871 // check the 3rd argument
Chris Lattner1c151132008-06-28 23:36:30 +0000872 Expr *FirstArgExpr = static_cast<Expr *>(Attr.getArg(1));
Chris Lattner703c52d2008-06-29 00:43:07 +0000873 llvm::APSInt FirstArg(32);
874 if (!FirstArgExpr->isIntegerConstantExpr(FirstArg, S.Context)) {
875 S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int,
Chris Lattner6953a072008-06-26 18:38:35 +0000876 "format", std::string("3"), FirstArgExpr->getSourceRange());
877 return;
878 }
879
880 // check if the function is variadic if the 3rd argument non-zero
881 if (FirstArg != 0) {
Daniel Dunbard1d847c2008-09-26 04:12:28 +0000882 if (isFunctionOrMethodVariadic(d)) {
Chris Lattner6953a072008-06-26 18:38:35 +0000883 ++NumArgs; // +1 for ...
884 } else {
Chris Lattner703c52d2008-06-29 00:43:07 +0000885 S.Diag(d->getLocation(), diag::err_format_attribute_requires_variadic);
Chris Lattner6953a072008-06-26 18:38:35 +0000886 return;
887 }
888 }
889
890 // strftime requires FirstArg to be 0 because it doesn't read from any variable
891 // the input is just the current time + the format string
892 if (is_strftime) {
893 if (FirstArg != 0) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000894 S.Diag(Attr.getLoc(), diag::err_format_strftime_third_parameter,
Chris Lattner6953a072008-06-26 18:38:35 +0000895 FirstArgExpr->getSourceRange());
896 return;
897 }
898 // if 0 it disables parameter checking (to use with e.g. va_list)
899 } else if (FirstArg != 0 && FirstArg != NumArgs) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000900 S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds,
Chris Lattner6953a072008-06-26 18:38:35 +0000901 "format", std::string("3"), FirstArgExpr->getSourceRange());
902 return;
903 }
904
905 d->addAttr(new FormatAttr(std::string(Format, FormatLen),
906 Idx.getZExtValue(), FirstArg.getZExtValue()));
907}
908
Chris Lattnerf6690152008-06-29 00:28:59 +0000909static void HandleTransparentUnionAttr(Decl *d, const AttributeList &Attr,
910 Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000911 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000912 if (Attr.getNumArgs() != 0) {
Chris Lattnerf6690152008-06-29 00:28:59 +0000913 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
Chris Lattner6953a072008-06-26 18:38:35 +0000914 std::string("0"));
915 return;
916 }
917
Eli Friedman283b3622008-09-02 05:19:23 +0000918 // FIXME: This shouldn't be restricted to typedefs
919 TypedefDecl *TD = dyn_cast<TypedefDecl>(d);
920 if (!TD || !TD->getUnderlyingType()->isUnionType()) {
Chris Lattnerf6690152008-06-29 00:28:59 +0000921 S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type,
Chris Lattner6953a072008-06-26 18:38:35 +0000922 "transparent_union", "union");
923 return;
924 }
925
Eli Friedman283b3622008-09-02 05:19:23 +0000926 RecordDecl* RD = TD->getUnderlyingType()->getAsUnionType()->getDecl();
Chris Lattner6953a072008-06-26 18:38:35 +0000927
Eli Friedman283b3622008-09-02 05:19:23 +0000928 // FIXME: Should we do a check for RD->isDefinition()?
929
930 // FIXME: This isn't supposed to be restricted to pointers, but otherwise
931 // we might silently generate incorrect code; see following code
932 for (int i = 0; i < RD->getNumMembers(); i++) {
933 if (!RD->getMember(i)->getType()->isPointerType()) {
934 S.Diag(Attr.getLoc(), diag::warn_transparent_union_nonpointer);
935 return;
936 }
937 }
938
939 // FIXME: This is a complete hack; we should be properly propagating
940 // transparent_union through Sema. That said, this is close enough to
941 // correctly compile all the common cases of transparent_union without
942 // errors or warnings
943 QualType NewTy = S.Context.VoidPtrTy;
944 NewTy.addConst();
945 TD->setUnderlyingType(NewTy);
Chris Lattner6953a072008-06-26 18:38:35 +0000946}
947
Chris Lattnerf6690152008-06-29 00:28:59 +0000948static void HandleAnnotateAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000949 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000950 if (Attr.getNumArgs() != 1) {
Chris Lattnerf6690152008-06-29 00:28:59 +0000951 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
952 std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000953 return;
954 }
Chris Lattner1c151132008-06-28 23:36:30 +0000955 Expr *argExpr = static_cast<Expr *>(Attr.getArg(0));
Chris Lattner6953a072008-06-26 18:38:35 +0000956 StringLiteral *SE = dyn_cast<StringLiteral>(argExpr);
957
958 // Make sure that there is a string literal as the annotation's single
959 // argument.
960 if (!SE) {
Chris Lattnerf6690152008-06-29 00:28:59 +0000961 S.Diag(Attr.getLoc(), diag::err_attribute_annotate_no_string);
Chris Lattner6953a072008-06-26 18:38:35 +0000962 return;
963 }
964 d->addAttr(new AnnotateAttr(std::string(SE->getStrData(),
965 SE->getByteLength())));
966}
967
Chris Lattner703c52d2008-06-29 00:43:07 +0000968static void HandleAlignedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
Chris Lattner6953a072008-06-26 18:38:35 +0000969 // check the attribute arguments.
Chris Lattner1c151132008-06-28 23:36:30 +0000970 if (Attr.getNumArgs() > 1) {
Chris Lattner703c52d2008-06-29 00:43:07 +0000971 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
972 std::string("1"));
Chris Lattner6953a072008-06-26 18:38:35 +0000973 return;
974 }
975
976 unsigned Align = 0;
Chris Lattner1c151132008-06-28 23:36:30 +0000977 if (Attr.getNumArgs() == 0) {
Chris Lattner6953a072008-06-26 18:38:35 +0000978 // FIXME: This should be the target specific maximum alignment.
979 // (For now we just use 128 bits which is the maximum on X86.
980 Align = 128;
981 return;
Chris Lattner6953a072008-06-26 18:38:35 +0000982 }
Chris Lattnerb0011ca2008-06-28 23:50:44 +0000983
984 Expr *alignmentExpr = static_cast<Expr *>(Attr.getArg(0));
985 llvm::APSInt Alignment(32);
Chris Lattner703c52d2008-06-29 00:43:07 +0000986 if (!alignmentExpr->isIntegerConstantExpr(Alignment, S.Context)) {
987 S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int,
988 "aligned", alignmentExpr->getSourceRange());
Chris Lattnerb0011ca2008-06-28 23:50:44 +0000989 return;
990 }
991 d->addAttr(new AlignedAttr(Alignment.getZExtValue() * 8));
Chris Lattner6953a072008-06-26 18:38:35 +0000992}
Chris Lattnerdc789562008-06-27 22:18:37 +0000993
Chris Lattnerf6690152008-06-29 00:28:59 +0000994/// HandleModeAttr - This attribute modifies the width of a decl with
Chris Lattner8ed14aa2008-06-28 23:48:25 +0000995/// primitive type.
Chris Lattnerdc789562008-06-27 22:18:37 +0000996///
997/// Despite what would be logical, the mode attribute is a decl attribute,
998/// not a type attribute: 'int ** __attribute((mode(HI))) *G;' tries to make
999/// 'G' be HImode, not an intermediate pointer.
1000///
Chris Lattnerf6690152008-06-29 00:28:59 +00001001static void HandleModeAttr(Decl *D, const AttributeList &Attr, Sema &S) {
Chris Lattnerdc789562008-06-27 22:18:37 +00001002 // This attribute isn't documented, but glibc uses it. It changes
1003 // the width of an int or unsigned int to the specified size.
1004
1005 // Check that there aren't any arguments
1006 if (Attr.getNumArgs() != 0) {
Chris Lattnerf6690152008-06-29 00:28:59 +00001007 S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments,
1008 std::string("0"));
Chris Lattnerdc789562008-06-27 22:18:37 +00001009 return;
1010 }
1011
1012 IdentifierInfo *Name = Attr.getParameterName();
1013 if (!Name) {
Chris Lattnerf6690152008-06-29 00:28:59 +00001014 S.Diag(Attr.getLoc(), diag::err_attribute_missing_parameter_name);
Chris Lattnerdc789562008-06-27 22:18:37 +00001015 return;
1016 }
1017 const char *Str = Name->getName();
1018 unsigned Len = Name->getLength();
1019
1020 // Normalize the attribute name, __foo__ becomes foo.
1021 if (Len > 4 && Str[0] == '_' && Str[1] == '_' &&
1022 Str[Len - 2] == '_' && Str[Len - 1] == '_') {
1023 Str += 2;
1024 Len -= 4;
1025 }
1026
1027 unsigned DestWidth = 0;
1028 bool IntegerMode = true;
1029 switch (Len) {
1030 case 2:
1031 if (!memcmp(Str, "QI", 2)) { DestWidth = 8; break; }
1032 if (!memcmp(Str, "HI", 2)) { DestWidth = 16; break; }
1033 if (!memcmp(Str, "SI", 2)) { DestWidth = 32; break; }
1034 if (!memcmp(Str, "DI", 2)) { DestWidth = 64; break; }
1035 if (!memcmp(Str, "TI", 2)) { DestWidth = 128; break; }
1036 if (!memcmp(Str, "SF", 2)) { DestWidth = 32; IntegerMode = false; break; }
1037 if (!memcmp(Str, "DF", 2)) { DestWidth = 64; IntegerMode = false; break; }
1038 if (!memcmp(Str, "XF", 2)) { DestWidth = 96; IntegerMode = false; break; }
1039 if (!memcmp(Str, "TF", 2)) { DestWidth = 128; IntegerMode = false; break; }
1040 break;
1041 case 4:
1042 // FIXME: glibc uses 'word' to define register_t; this is narrower than a
1043 // pointer on PIC16 and other embedded platforms.
1044 if (!memcmp(Str, "word", 4))
Chris Lattnerf6690152008-06-29 00:28:59 +00001045 DestWidth = S.Context.Target.getPointerWidth(0);
Chris Lattnerdc789562008-06-27 22:18:37 +00001046 if (!memcmp(Str, "byte", 4))
Chris Lattnerf6690152008-06-29 00:28:59 +00001047 DestWidth = S.Context.Target.getCharWidth();
Chris Lattnerdc789562008-06-27 22:18:37 +00001048 break;
1049 case 7:
1050 if (!memcmp(Str, "pointer", 7))
Chris Lattnerf6690152008-06-29 00:28:59 +00001051 DestWidth = S.Context.Target.getPointerWidth(0);
Chris Lattnerdc789562008-06-27 22:18:37 +00001052 break;
1053 }
1054
1055 QualType OldTy;
1056 if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
1057 OldTy = TD->getUnderlyingType();
1058 else if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
1059 OldTy = VD->getType();
1060 else {
Chris Lattnerf6690152008-06-29 00:28:59 +00001061 S.Diag(D->getLocation(), diag::err_attr_wrong_decl, "mode",
1062 SourceRange(Attr.getLoc(), Attr.getLoc()));
Chris Lattnerdc789562008-06-27 22:18:37 +00001063 return;
1064 }
1065
1066 // FIXME: Need proper fixed-width types
1067 QualType NewTy;
1068 switch (DestWidth) {
1069 case 0:
Chris Lattnerf6690152008-06-29 00:28:59 +00001070 S.Diag(Attr.getLoc(), diag::err_unknown_machine_mode, Name->getName());
Chris Lattnerdc789562008-06-27 22:18:37 +00001071 return;
1072 default:
Chris Lattnerf6690152008-06-29 00:28:59 +00001073 S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode, Name->getName());
Chris Lattnerdc789562008-06-27 22:18:37 +00001074 return;
1075 case 8:
1076 assert(IntegerMode);
1077 if (OldTy->isSignedIntegerType())
Chris Lattnerf6690152008-06-29 00:28:59 +00001078 NewTy = S.Context.SignedCharTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001079 else
Chris Lattnerf6690152008-06-29 00:28:59 +00001080 NewTy = S.Context.UnsignedCharTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001081 break;
1082 case 16:
1083 assert(IntegerMode);
1084 if (OldTy->isSignedIntegerType())
Chris Lattnerf6690152008-06-29 00:28:59 +00001085 NewTy = S.Context.ShortTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001086 else
Chris Lattnerf6690152008-06-29 00:28:59 +00001087 NewTy = S.Context.UnsignedShortTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001088 break;
1089 case 32:
1090 if (!IntegerMode)
Chris Lattnerf6690152008-06-29 00:28:59 +00001091 NewTy = S.Context.FloatTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001092 else if (OldTy->isSignedIntegerType())
Chris Lattnerf6690152008-06-29 00:28:59 +00001093 NewTy = S.Context.IntTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001094 else
Chris Lattnerf6690152008-06-29 00:28:59 +00001095 NewTy = S.Context.UnsignedIntTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001096 break;
1097 case 64:
1098 if (!IntegerMode)
Chris Lattnerf6690152008-06-29 00:28:59 +00001099 NewTy = S.Context.DoubleTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001100 else if (OldTy->isSignedIntegerType())
Chris Lattnerf6690152008-06-29 00:28:59 +00001101 NewTy = S.Context.LongLongTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001102 else
Chris Lattnerf6690152008-06-29 00:28:59 +00001103 NewTy = S.Context.UnsignedLongLongTy;
Chris Lattnerdc789562008-06-27 22:18:37 +00001104 break;
1105 }
1106
1107 if (!OldTy->getAsBuiltinType())
Chris Lattnerf6690152008-06-29 00:28:59 +00001108 S.Diag(Attr.getLoc(), diag::err_mode_not_primitive);
Chris Lattnerdc789562008-06-27 22:18:37 +00001109 else if (!(IntegerMode && OldTy->isIntegerType()) &&
1110 !(!IntegerMode && OldTy->isFloatingType())) {
Chris Lattnerf6690152008-06-29 00:28:59 +00001111 S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
Chris Lattnerdc789562008-06-27 22:18:37 +00001112 }
1113
1114 // Install the new type.
1115 if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
1116 TD->setUnderlyingType(NewTy);
1117 else
1118 cast<ValueDecl>(D)->setType(NewTy);
1119}
Chris Lattnera72440d2008-06-29 00:23:49 +00001120
1121//===----------------------------------------------------------------------===//
1122// Top Level Sema Entry Points
1123//===----------------------------------------------------------------------===//
1124
Chris Lattner703c52d2008-06-29 00:43:07 +00001125/// HandleDeclAttribute - Apply the specific attribute to the specified decl if
1126/// the attribute applies to decls. If the attribute is a type attribute, just
1127/// silently ignore it.
1128static void ProcessDeclAttribute(Decl *D, const AttributeList &Attr, Sema &S) {
1129 switch (Attr.getKind()) {
Daniel Dunbar8716a012008-07-31 22:40:48 +00001130 case AttributeList::AT_IBOutlet: HandleIBOutletAttr (D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001131 case AttributeList::AT_address_space:
1132 // Ignore this, this is a type attribute, handled by ProcessTypeAttributes.
1133 break;
Daniel Dunbar8716a012008-07-31 22:40:48 +00001134 case AttributeList::AT_alias: HandleAliasAttr (D, Attr, S); break;
1135 case AttributeList::AT_aligned: HandleAlignedAttr (D, Attr, S); break;
Daniel Dunbar0a2da712008-10-28 00:17:57 +00001136 case AttributeList::AT_always_inline:
1137 HandleAlwaysInlineAttr (D, Attr, S); break;
Daniel Dunbar8716a012008-07-31 22:40:48 +00001138 case AttributeList::AT_annotate: HandleAnnotateAttr (D, Attr, S); break;
1139 case AttributeList::AT_constructor: HandleConstructorAttr(D, Attr, S); break;
1140 case AttributeList::AT_deprecated: HandleDeprecatedAttr(D, Attr, S); break;
1141 case AttributeList::AT_destructor: HandleDestructorAttr(D, Attr, S); break;
1142 case AttributeList::AT_dllexport: HandleDLLExportAttr (D, Attr, S); break;
1143 case AttributeList::AT_dllimport: HandleDLLImportAttr (D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001144 case AttributeList::AT_ext_vector_type:
1145 HandleExtVectorTypeAttr(D, Attr, S);
1146 break;
Daniel Dunbar8716a012008-07-31 22:40:48 +00001147 case AttributeList::AT_fastcall: HandleFastCallAttr (D, Attr, S); break;
1148 case AttributeList::AT_format: HandleFormatAttr (D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001149 case AttributeList::AT_mode: HandleModeAttr (D, Attr, S); break;
Daniel Dunbar8716a012008-07-31 22:40:48 +00001150 case AttributeList::AT_nonnull: HandleNonNullAttr (D, Attr, S); break;
1151 case AttributeList::AT_noreturn: HandleNoReturnAttr (D, Attr, S); break;
1152 case AttributeList::AT_nothrow: HandleNothrowAttr (D, Attr, S); break;
1153 case AttributeList::AT_packed: HandlePackedAttr (D, Attr, S); break;
1154 case AttributeList::AT_stdcall: HandleStdCallAttr (D, Attr, S); break;
1155 case AttributeList::AT_unused: HandleUnusedAttr (D, Attr, S); break;
1156 case AttributeList::AT_vector_size: HandleVectorSizeAttr(D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001157 case AttributeList::AT_visibility: HandleVisibilityAttr(D, Attr, S); break;
1158 case AttributeList::AT_weak: HandleWeakAttr (D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001159 case AttributeList::AT_transparent_union:
1160 HandleTransparentUnionAttr(D, Attr, S);
1161 break;
Anders Carlssond83141a52008-08-23 23:22:21 +00001162 case AttributeList::AT_objc_gc: HandleObjCGCAttr (D, Attr, S); break;
Steve Naroffe1cecba2008-09-18 16:44:58 +00001163 case AttributeList::AT_blocks: HandleBlocksAttr (D, Attr, S); break;
Anders Carlsson244d99b2008-10-05 18:05:59 +00001164 case AttributeList::AT_sentinel: HandleSentinelAttr (D, Attr, S); break;
Anders Carlssondd6791c2008-10-05 23:32:53 +00001165 case AttributeList::AT_const: HandleConstAttr (D, Attr, S); break;
1166 case AttributeList::AT_pure: HandlePureAttr (D, Attr, S); break;
Chris Lattner703c52d2008-06-29 00:43:07 +00001167 default:
1168#if 0
1169 // TODO: when we have the full set of attributes, warn about unknown ones.
1170 S.Diag(Attr->getLoc(), diag::warn_attribute_ignored,
1171 Attr->getName()->getName());
1172#endif
1173 break;
1174 }
1175}
1176
1177/// ProcessDeclAttributeList - Apply all the decl attributes in the specified
1178/// attribute list to the specified decl, ignoring any type attributes.
1179void Sema::ProcessDeclAttributeList(Decl *D, const AttributeList *AttrList) {
1180 while (AttrList) {
1181 ProcessDeclAttribute(D, *AttrList, *this);
1182 AttrList = AttrList->getNext();
1183 }
1184}
1185
1186
Chris Lattnera72440d2008-06-29 00:23:49 +00001187/// ProcessDeclAttributes - Given a declarator (PD) with attributes indicated in
1188/// it, apply them to D. This is a bit tricky because PD can have attributes
1189/// specified in many different places, and we need to find and apply them all.
1190void Sema::ProcessDeclAttributes(Decl *D, const Declarator &PD) {
1191 // Apply decl attributes from the DeclSpec if present.
1192 if (const AttributeList *Attrs = PD.getDeclSpec().getAttributes())
1193 ProcessDeclAttributeList(D, Attrs);
Chris Lattner703c52d2008-06-29 00:43:07 +00001194
Chris Lattnera72440d2008-06-29 00:23:49 +00001195 // Walk the declarator structure, applying decl attributes that were in a type
1196 // position to the decl itself. This handles cases like:
1197 // int *__attr__(x)** D;
1198 // when X is a decl attribute.
1199 for (unsigned i = 0, e = PD.getNumTypeObjects(); i != e; ++i)
1200 if (const AttributeList *Attrs = PD.getTypeObject(i).getAttrs())
1201 ProcessDeclAttributeList(D, Attrs);
1202
1203 // Finally, apply any attributes on the decl itself.
1204 if (const AttributeList *Attrs = PD.getAttributes())
1205 ProcessDeclAttributeList(D, Attrs);
1206}
1207