SemaDeclAttr.cpp revision fd6ad3cf9c8fc6904bd5f33212207aa69743fd45
16b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//===--- SemaDeclAttr.cpp - Declaration Attribute Handling ----------------===//
26b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//
36b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//                     The LLVM Compiler Infrastructure
46b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//
56b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner// This file is distributed under the University of Illinois Open Source
66b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner// License. See LICENSE.TXT for details.
76b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//
86b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//===----------------------------------------------------------------------===//
96b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//
106b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//  This file implements decl-related attribute processing.
116b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//
126b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner//===----------------------------------------------------------------------===//
136b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
146b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner#include "Sema.h"
156b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner#include "clang/AST/ASTContext.h"
16acc5f3e42334525bf28c86471551f83dfce222d5Daniel Dunbar#include "clang/AST/DeclObjC.h"
17acc5f3e42334525bf28c86471551f83dfce222d5Daniel Dunbar#include "clang/AST/Expr.h"
18fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner#include "clang/Basic/TargetInfo.h"
1912bc692a78582f1cc32791325981aadcffb04c5eDaniel Dunbar#include "clang/Parse/DeclSpec.h"
206e1eb87c04a3acd50888375dad59fac06b7ceb1fTed Kremenek#include <llvm/ADT/StringExtras.h>
216b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattnerusing namespace clang;
226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
23e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner//===----------------------------------------------------------------------===//
24e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner//  Helper functions
25e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner//===----------------------------------------------------------------------===//
26e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner
27755f9d2c65f75d539a2440e5de82d881e4417397Fariborz Jahanianstatic const FunctionType *getFunctionType(Decl *d, bool blocksToo = true) {
286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  QualType Ty;
296b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (ValueDecl *decl = dyn_cast<ValueDecl>(d))
306b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    Ty = decl->getType();
316b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (FieldDecl *decl = dyn_cast<FieldDecl>(d))
326b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    Ty = decl->getType();
336b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (TypedefDecl* decl = dyn_cast<TypedefDecl>(d))
346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    Ty = decl->getUnderlyingType();
356b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else
366b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return 0;
37bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
386b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (Ty->isFunctionPointerType())
396217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek    Ty = Ty->getAs<PointerType>()->getPointeeType();
40755f9d2c65f75d539a2440e5de82d881e4417397Fariborz Jahanian  else if (blocksToo && Ty->isBlockPointerType())
416217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek    Ty = Ty->getAs<BlockPointerType>()->getPointeeType();
42d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar
43d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  return Ty->getAsFunctionType();
446b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
456b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
463568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar// FIXME: We should provide an abstraction around a method or function
473568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar// to provide the following bits of information.
483568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
49d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// isFunctionOrMethod - Return true if the given decl has function
50d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// type (function or function-typed variable) or an Objective-C
51d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// method.
523568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbarstatic bool isFunctionOrMethod(Decl *d) {
53755f9d2c65f75d539a2440e5de82d881e4417397Fariborz Jahanian  return getFunctionType(d, false) || isa<ObjCMethodDecl>(d);
54d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar}
553568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
56620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian/// isFunctionOrMethodOrBlock - Return true if the given decl has function
57620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian/// type (function or function-typed variable) or an Objective-C
58620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian/// method or a block.
59620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanianstatic bool isFunctionOrMethodOrBlock(Decl *d) {
60620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  if (isFunctionOrMethod(d))
61620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian    return true;
62620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  // check for block is more involved.
63620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  if (const VarDecl *V = dyn_cast<VarDecl>(d)) {
64620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian    QualType Ty = V->getType();
65620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian    return Ty->isBlockPointerType();
66620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  }
67d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian  return isa<BlockDecl>(d);
68620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian}
69620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian
70d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// hasFunctionProto - Return true if the given decl has a argument
71d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// information. This decl should have already passed
72620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian/// isFunctionOrMethod or isFunctionOrMethodOrBlock.
73d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbarstatic bool hasFunctionProto(Decl *d) {
74620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  if (const FunctionType *FnTy = getFunctionType(d))
7572564e73277e29f6db3305d1f27ba408abb7ed88Douglas Gregor    return isa<FunctionProtoType>(FnTy);
76620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  else {
77d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian    assert(isa<ObjCMethodDecl>(d) || isa<BlockDecl>(d));
78d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar    return true;
79d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  }
803568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar}
813568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
82d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// getFunctionOrMethodNumArgs - Return number of function or method
83d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// arguments. It is an error to call this on a K&R function (use
84d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar/// hasFunctionProto first).
853568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbarstatic unsigned getFunctionOrMethodNumArgs(Decl *d) {
8689951a86b594513c2a013532ed45d197413b1087Chris Lattner  if (const FunctionType *FnTy = getFunctionType(d))
8772564e73277e29f6db3305d1f27ba408abb7ed88Douglas Gregor    return cast<FunctionProtoType>(FnTy)->getNumArgs();
88d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian  if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
89d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian    return BD->getNumParams();
9089951a86b594513c2a013532ed45d197413b1087Chris Lattner  return cast<ObjCMethodDecl>(d)->param_size();
913568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar}
923568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
933568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbarstatic QualType getFunctionOrMethodArgType(Decl *d, unsigned Idx) {
9489951a86b594513c2a013532ed45d197413b1087Chris Lattner  if (const FunctionType *FnTy = getFunctionType(d))
9572564e73277e29f6db3305d1f27ba408abb7ed88Douglas Gregor    return cast<FunctionProtoType>(FnTy)->getArgType(Idx);
96d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian  if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
97d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian    return BD->getParamDecl(Idx)->getType();
98bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
9989951a86b594513c2a013532ed45d197413b1087Chris Lattner  return cast<ObjCMethodDecl>(d)->param_begin()[Idx]->getType();
1003568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar}
1013568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
1025b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanianstatic QualType getFunctionOrMethodResultType(Decl *d) {
1035b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (const FunctionType *FnTy = getFunctionType(d))
1045b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return cast<FunctionProtoType>(FnTy)->getResultType();
1055b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  return cast<ObjCMethodDecl>(d)->getResultType();
1065b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian}
1075b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian
1083568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbarstatic bool isFunctionOrMethodVariadic(Decl *d) {
109d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  if (const FunctionType *FnTy = getFunctionType(d)) {
11072564e73277e29f6db3305d1f27ba408abb7ed88Douglas Gregor    const FunctionProtoType *proto = cast<FunctionProtoType>(FnTy);
1113568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar    return proto->isVariadic();
112d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian  } else if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
113d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian    return BD->IsVariadic();
114d66f22d9f8423579322a6dd16587ed52b0a58834Fariborz Jahanian  else {
1153568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar    return cast<ObjCMethodDecl>(d)->isVariadic();
1163568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar  }
1173568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar}
1183568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar
1196b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattnerstatic inline bool isNSStringType(QualType T, ASTContext &Ctx) {
12014108da7f7fc059772711e4ffee1322a27b152a7Steve Naroff  const ObjCObjectPointerType *PT = T->getAsObjCObjectPointerType();
121b77792eabf5882cf9af8cc810599b20432fda6c2Chris Lattner  if (!PT)
1226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return false;
123bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
124b77792eabf5882cf9af8cc810599b20432fda6c2Chris Lattner  const ObjCInterfaceType *ClsT =PT->getPointeeType()->getAsObjCInterfaceType();
1256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (!ClsT)
1266b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return false;
127bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  IdentifierInfo* ClsName = ClsT->getDecl()->getIdentifier();
129bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1306b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // FIXME: Should we walk the chain of classes?
1316b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  return ClsName == &Ctx.Idents.get("NSString") ||
1326b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner         ClsName == &Ctx.Idents.get("NSMutableString");
1336b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
1346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
135085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbarstatic inline bool isCFStringType(QualType T, ASTContext &Ctx) {
1366217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek  const PointerType *PT = T->getAs<PointerType>();
137085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  if (!PT)
138085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    return false;
139085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar
1406217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek  const RecordType *RT = PT->getPointeeType()->getAs<RecordType>();
141085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  if (!RT)
142085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    return false;
143bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
144085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  const RecordDecl *RD = RT->getDecl();
145085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  if (RD->getTagKind() != TagDecl::TK_struct)
146085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    return false;
147085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar
148085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  return RD->getIdentifier() == &Ctx.Idents.get("__CFString");
149085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar}
150085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar
151e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner//===----------------------------------------------------------------------===//
152e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner// Attribute Implementations
153e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner//===----------------------------------------------------------------------===//
154e5c5ee1cff9ec084f176fa252774262677857ad2Chris Lattner
1553068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar// FIXME: All this manual attribute parsing code is gross. At the
1563068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar// least add some helper functions to check most argument patterns (#
1573068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar// and types of args).
1583068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
159bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stumpstatic void HandleExtVectorTypeAttr(Scope *scope, Decl *d,
1609cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor                                    const AttributeList &Attr, Sema &S) {
161545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  TypedefDecl *tDecl = dyn_cast<TypedefDecl>(d);
162545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (tDecl == 0) {
163803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    S.Diag(Attr.getLoc(), diag::err_typecheck_ext_vector_not_typedef);
164545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner    return;
1656b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
166bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1676b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  QualType curType = tDecl->getUnderlyingType();
1689cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor
1699cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  Expr *sizeExpr;
1709cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor
1719cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  // Special case where the argument is a template id.
1729cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  if (Attr.getParameterName()) {
1739cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    sizeExpr = S.ActOnDeclarationNameExpr(scope, Attr.getLoc(),
1749cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor                               Attr.getParameterName(),
1759cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor                               false, 0, false).takeAs<Expr>();
1769cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  } else {
1779cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    // check the attribute arguments.
1789cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    if (Attr.getNumArgs() != 1) {
1799cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor      S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
1809cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor      return;
1819cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    }
1829cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    sizeExpr = static_cast<Expr *>(Attr.getArg(0));
1836b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1849cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor
1859cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  // Instantiate/Install the vector type, and let Sema build the type for us.
1869cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  // This will run the reguired checks.
1879cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  QualType T = S.BuildExtVectorType(curType, S.Owned(sizeExpr), Attr.getLoc());
1889cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor  if (!T.isNull()) {
1899cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    tDecl->setUnderlyingType(T);
190bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1919cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    // Remember this typedef decl, we will need it later for diagnostics.
1929cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    S.ExtVectorDecls.push_back(tDecl);
1936b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1946b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
1956b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
196065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner
197bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// HandleVectorSizeAttribute - this attribute is only applicable to integral
198bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// and float scalars, although arrays, pointers, and function return values are
199bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// allowed in conjunction with this construct. Aggregates with this attribute
200bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// are invalid, even if they are of the same size as a corresponding scalar.
201bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// The raw attribute should contain precisely 1 argument, the vector size for
202bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// the variable, measured in bytes. If curType and rawAttr are well formed,
203bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// this routine will return a new vector type.
204803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleVectorSizeAttr(Decl *D, const AttributeList &Attr, Sema &S) {
205065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  QualType CurType;
206065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
207065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    CurType = VD->getType();
208065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  else if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
209065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    CurType = TD->getUnderlyingType();
210065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  else {
211fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(D->getLocation(), diag::err_attr_wrong_decl)
212fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "vector_size" << SourceRange(Attr.getLoc(), Attr.getLoc());
213065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
214065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  }
215bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
216065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  // Check the attribute arugments.
217545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 1) {
2183c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
219065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
2206b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
221545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *sizeExpr = static_cast<Expr *>(Attr.getArg(0));
2226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  llvm::APSInt vecSize(32);
223803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  if (!sizeExpr->isIntegerConstantExpr(vecSize, S.Context)) {
224fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
225fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "vector_size" << sizeExpr->getSourceRange();
226065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
2276b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
228bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // navigate to the base type - we need to provide for vector pointers, vector
229bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // arrays, and functions returning vectors.
230b77792eabf5882cf9af8cc810599b20432fda6c2Chris Lattner  if (CurType->isPointerType() || CurType->isArrayType() ||
231b77792eabf5882cf9af8cc810599b20432fda6c2Chris Lattner      CurType->isFunctionType()) {
2322db15bdd945163eacfa4623fd2e32a536ed2dd3bChris Lattner    S.Diag(Attr.getLoc(), diag::err_unsupported_vector_size) << CurType;
2332db15bdd945163eacfa4623fd2e32a536ed2dd3bChris Lattner    return;
2346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    /* FIXME: rebuild the type from the inside out, vectorizing the inner type.
2356b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     do {
2366b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     if (PointerType *PT = dyn_cast<PointerType>(canonType))
2376b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     canonType = PT->getPointeeType().getTypePtr();
2386b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     else if (ArrayType *AT = dyn_cast<ArrayType>(canonType))
2396b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     canonType = AT->getElementType().getTypePtr();
2406b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     else if (FunctionType *FT = dyn_cast<FunctionType>(canonType))
2416b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     canonType = FT->getResultType().getTypePtr();
2426b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     } while (canonType->isPointerType() || canonType->isArrayType() ||
2436b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     canonType->isFunctionType());
2446b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner     */
2456b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
24682afa2d97d39cc0d5a4897716ec0a80aeab9e14bChris Lattner  // the base type must be integer or float, and can't already be a vector.
24782afa2d97d39cc0d5a4897716ec0a80aeab9e14bChris Lattner  if (CurType->isVectorType() ||
24882afa2d97d39cc0d5a4897716ec0a80aeab9e14bChris Lattner      (!CurType->isIntegerType() && !CurType->isRealFloatingType())) {
249d162584991885ab004a02573a73ce06422b921fcChris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_invalid_vector_type) << CurType;
250065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
2516b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
252803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  unsigned typeSize = static_cast<unsigned>(S.Context.getTypeSize(CurType));
2536b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // vecSize is specified in bytes - convert to bits.
254bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  unsigned vectorSize = static_cast<unsigned>(vecSize.getZExtValue() * 8);
255bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
2566b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // the vector size needs to be an integral multiple of the type size.
2576b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (vectorSize % typeSize) {
258fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_invalid_size)
259fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << sizeExpr->getSourceRange();
260065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
2616b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
2626b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (vectorSize == 0) {
263fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_zero_size)
264fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << sizeExpr->getSourceRange();
265065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    return;
2666b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
267bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
268065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  // Success! Instantiate the vector type, the number of elements is > 0, and
269065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  // not required to be a power of 2, unlike GCC.
270803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  CurType = S.Context.getVectorType(CurType, vectorSize/typeSize);
271bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
272065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner  if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
273065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    VD->setType(CurType);
274bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  else
275065c5a801f46dda8ff0cd6ad954c4e5f94a3be75Chris Lattner    cast<TypedefDecl>(D)->setUnderlyingType(CurType);
2766b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
2776b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
278803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandlePackedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
2796b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
280545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() > 0) {
2813c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
2826b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
2836b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
284bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
2856b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (TagDecl *TD = dyn_cast<TagDecl>(d))
286a860e755f1f9f071b6a6a2f96128a6a258f5c331Anders Carlsson    TD->addAttr(::new (S.Context) PackedAttr);
2876b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (FieldDecl *FD = dyn_cast<FieldDecl>(d)) {
2886b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    // If the alignment is less than or equal to 8 bits, the packed attribute
2896b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    // has no effect.
2906b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    if (!FD->getType()->isIncompleteType() &&
291803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner        S.Context.getTypeAlign(FD->getType()) <= 8)
292fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::warn_attribute_ignored_for_field_of_type)
29308631c5fa053867146b5ee8be658c229f6bf127cChris Lattner        << Attr.getName() << FD->getType();
2946b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    else
295a860e755f1f9f071b6a6a2f96128a6a258f5c331Anders Carlsson      FD->addAttr(::new (S.Context) PackedAttr);
2966b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  } else
2973c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << Attr.getName();
2986b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
2996b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
30096329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenekstatic void HandleIBOutletAttr(Decl *d, const AttributeList &Attr, Sema &S) {
30196329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  // check the attribute arguments.
30296329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  if (Attr.getNumArgs() > 0) {
3033c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
30496329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek    return;
30596329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  }
306bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
30796329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  // The IBOutlet attribute only applies to instance variables of Objective-C
30896329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  // classes.
309327426076e1acc8217307cb236269ccf08c18fe6Ted Kremenek  if (isa<ObjCIvarDecl>(d) || isa<ObjCPropertyDecl>(d))
31040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    d->addAttr(::new (S.Context) IBOutletAttr());
31196329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek  else
312327426076e1acc8217307cb236269ccf08c18fe6Ted Kremenek    S.Diag(Attr.getLoc(), diag::err_attribute_iboutlet);
31396329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek}
31496329d4e07a9bcddb5a927892b70408c8fd8c474Ted Kremenek
315eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenekstatic void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
316bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // GCC ignores the nonnull attribute on K&R style function prototypes, so we
317bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // ignore it as well
318d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
319fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
3205dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
321eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    return;
322eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  }
323bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
324d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  unsigned NumArgs = getFunctionOrMethodNumArgs(d);
325eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek
326eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  // The nonnull attribute only applies to pointers.
327eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  llvm::SmallVector<unsigned, 10> NonNullArgs;
328bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
329eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  for (AttributeList::arg_iterator I=Attr.arg_begin(),
330eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek                                   E=Attr.arg_end(); I!=E; ++I) {
331bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
332bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
333eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    // The argument must be an integer constant expression.
334f5e883474796afd26e52a010cd9bf90374fa1915Ted Kremenek    Expr *Ex = static_cast<Expr *>(*I);
335eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    llvm::APSInt ArgNum(32);
336eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    if (!Ex->isIntegerConstantExpr(ArgNum, S.Context)) {
337fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
338fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << "nonnull" << Ex->getSourceRange();
339eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek      return;
340eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    }
341bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
342eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    unsigned x = (unsigned) ArgNum.getZExtValue();
343bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
344eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    if (x < 1 || x > NumArgs) {
345fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
34630bc96544346bea42921cf6837e66cef80d664b4Chris Lattner       << "nonnull" << I.getArgNum() << Ex->getSourceRange();
347eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek      return;
348eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    }
349bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
350465172f304248a9aab6f2c398a836ce4e25efbbfTed Kremenek    --x;
351eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek
352eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    // Is the function argument a pointer type?
353bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    QualType T = getFunctionOrMethodArgType(d, x);
354dbfe99ef39163fd3574332673ee175c2bb6ef3caTed Kremenek    if (!T->isAnyPointerType() && !T->isBlockPointerType()) {
355eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek      // FIXME: Should also highlight argument in decl.
356fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_nonnull_pointers_only)
357fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << "nonnull" << Ex->getSourceRange();
3587fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek      continue;
359eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    }
360bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
361eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek    NonNullArgs.push_back(x);
362eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  }
363bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
364bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // If no arguments were specified to __attribute__((nonnull)) then all pointer
365bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // arguments have a nonnull attribute.
3667fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek  if (NonNullArgs.empty()) {
36746bbacac37141ed9d01d5b6473e8211554b02710Ted Kremenek    for (unsigned I = 0, E = getFunctionOrMethodNumArgs(d); I != E; ++I) {
36846bbacac37141ed9d01d5b6473e8211554b02710Ted Kremenek      QualType T = getFunctionOrMethodArgType(d, I);
369dbfe99ef39163fd3574332673ee175c2bb6ef3caTed Kremenek      if (T->isAnyPointerType() || T->isBlockPointerType())
370d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar        NonNullArgs.push_back(I);
37146bbacac37141ed9d01d5b6473e8211554b02710Ted Kremenek    }
372bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
3737fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek    if (NonNullArgs.empty()) {
3747fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek      S.Diag(Attr.getLoc(), diag::warn_attribute_nonnull_no_pointers);
3757fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek      return;
3767fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek    }
377eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  }
3787fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek
3797fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek  unsigned* start = &NonNullArgs[0];
3807fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek  unsigned size = NonNullArgs.size();
3817fb43c17eb2b4102f40a80a355629aacd70589adTed Kremenek  std::sort(start, start + size);
38240b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) NonNullAttr(start, size));
383eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek}
384eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek
385803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleAliasAttr(Decl *d, const AttributeList &Attr, Sema &S) {
3866b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
387545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 1) {
3883c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
3896b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
3906b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
391bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
392545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
3936b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  Arg = Arg->IgnoreParenCasts();
3946b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
395bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
3966b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (Str == 0 || Str->isWide()) {
397fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
3983c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "alias" << 1;
3996b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
4006b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
401bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
4026b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  const char *Alias = Str->getStrData();
4036b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  unsigned AliasLen = Str->getByteLength();
404bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
4056b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // FIXME: check if target symbol exists in current file
406bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
40740b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) AliasAttr(std::string(Alias, AliasLen)));
4086b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
4096b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
410bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stumpstatic void HandleAlwaysInlineAttr(Decl *d, const AttributeList &Attr,
411af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar                                   Sema &S) {
412af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar  // check the attribute arguments.
413af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar  if (Attr.getNumArgs() != 0) {
4143c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
415af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar    return;
416af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar  }
4175bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson
418c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  if (!isa<FunctionDecl>(d)) {
4195bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
4205dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    << Attr.getName() << 0 /*function*/;
4215bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    return;
4225bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  }
423bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
42440b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) AlwaysInlineAttr());
425af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar}
426af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar
42776168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynnstatic void HandleMallocAttr(Decl *d, const AttributeList &Attr, Sema &S) {
42876168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  // check the attribute arguments.
42976168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  if (Attr.getNumArgs() != 0) {
43076168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
43176168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn    return;
43276168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  }
43376168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn
43476168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  if (!isFunctionOrMethod(d)) {
43576168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
43676168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn      << Attr.getName() << 0 /*function*/;
43776168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn    return;
43876168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  }
43976168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn
440fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d)) {
441fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn    if (!FD->getResultType()->isPointerType()) {
442fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn      S.Diag(Attr.getLoc(), diag::warn_attribute_malloc_pointer_only);
443fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn      return;
444fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn    }
445fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn  }
446fd6ad3cf9c8fc6904bd5f33212207aa69743fd45Ryan Flynn
44776168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  d->addAttr(::new (S.Context) MallocAttr());
44876168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn}
44976168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn
450b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenekstatic bool HandleCommonNoReturnAttr(Decl *d, const AttributeList &Attr,
4515dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek                                     Sema &S) {
4526b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
453545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
4543c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
455b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek    return false;
4566b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
457d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar
45819c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump  if (!isFunctionOrMethod(d) && !isa<BlockDecl>(d)) {
45919c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump    ValueDecl *VD = dyn_cast<ValueDecl>(d);
46019c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump    if (VD == 0 || !VD->getType()->isBlockPointerType()) {
46119c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump      S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
4625dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek        << Attr.getName() << 0 /*function*/;
46319c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump      return false;
46419c30c00e5e01e4608a43c7deb504f343f09e46dMike Stump    }
4656b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
466bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
467b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek  return true;
468b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek}
469b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek
470b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenekstatic void HandleNoReturnAttr(Decl *d, const AttributeList &Attr, Sema &S) {
471bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  if (HandleCommonNoReturnAttr(d, Attr, S))
47240b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    d->addAttr(::new (S.Context) NoReturnAttr());
473b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek}
474b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek
475b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenekstatic void HandleAnalyzerNoReturnAttr(Decl *d, const AttributeList &Attr,
476b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek                                       Sema &S) {
477bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  if (HandleCommonNoReturnAttr(d, Attr, S))
47840b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    d->addAttr(::new (S.Context) AnalyzerNoReturnAttr());
4796b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
4806b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
48173798892751e378cbcdef43579c1d41685091fd0Ted Kremenekstatic void HandleUnusedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
48273798892751e378cbcdef43579c1d41685091fd0Ted Kremenek  // check the attribute arguments.
48373798892751e378cbcdef43579c1d41685091fd0Ted Kremenek  if (Attr.getNumArgs() != 0) {
4843c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
48573798892751e378cbcdef43579c1d41685091fd0Ted Kremenek    return;
48673798892751e378cbcdef43579c1d41685091fd0Ted Kremenek  }
487bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
488d3f2c10f881311831a84114179342ff4db55e0c3Daniel Dunbar  if (!isa<VarDecl>(d) && !isFunctionOrMethod(d)) {
489fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
4905dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 2 /*variable and function*/;
49173798892751e378cbcdef43579c1d41685091fd0Ted Kremenek    return;
49273798892751e378cbcdef43579c1d41685091fd0Ted Kremenek  }
493bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
49440b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) UnusedAttr());
49573798892751e378cbcdef43579c1d41685091fd0Ted Kremenek}
49673798892751e378cbcdef43579c1d41685091fd0Ted Kremenek
497b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbarstatic void HandleUsedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
498b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  // check the attribute arguments.
499b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  if (Attr.getNumArgs() != 0) {
500b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
501b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar    return;
502b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  }
503bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
504b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  if (const VarDecl *VD = dyn_cast<VarDecl>(d)) {
505186204bfcf9c53d48143ec300d4c3d036fed4140Daniel Dunbar    if (VD->hasLocalStorage() || VD->hasExternalStorage()) {
506b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar      S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "used";
507b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar      return;
508b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar    }
509b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  } else if (!isFunctionOrMethod(d)) {
510b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
5115dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 2 /*variable and function*/;
512b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar    return;
513b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  }
514bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
51540b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) UsedAttr());
516b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar}
517b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar
5183068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbarstatic void HandleConstructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
5193068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  // check the attribute arguments.
5203068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
521fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
522fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "0 or 1";
5233068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    return;
524bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
5253068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
5263068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  int priority = 65535; // FIXME: Do not hardcode such constants.
5273068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  if (Attr.getNumArgs() > 0) {
5283068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    Expr *E = static_cast<Expr *>(Attr.getArg(0));
5293068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    llvm::APSInt Idx(32);
5303068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    if (!E->isIntegerConstantExpr(Idx, S.Context)) {
531fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
5323c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner        << "constructor" << 1 << E->getSourceRange();
5333068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar      return;
5343068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    }
5353068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    priority = Idx.getZExtValue();
5363068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  }
537bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
538c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  if (!isa<FunctionDecl>(d)) {
539fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
5405dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
5413068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    return;
5423068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  }
5433068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
54440b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) ConstructorAttr(priority));
5453068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar}
5463068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
5473068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbarstatic void HandleDestructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
5483068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  // check the attribute arguments.
5493068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
550fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
551fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner       << "0 or 1";
5523068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    return;
553bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
5543068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
5553068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  int priority = 65535; // FIXME: Do not hardcode such constants.
5563068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  if (Attr.getNumArgs() > 0) {
5573068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    Expr *E = static_cast<Expr *>(Attr.getArg(0));
5583068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    llvm::APSInt Idx(32);
5593068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    if (!E->isIntegerConstantExpr(Idx, S.Context)) {
560fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
5613c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner        << "destructor" << 1 << E->getSourceRange();
5623068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar      return;
5633068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    }
5643068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    priority = Idx.getZExtValue();
5653068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  }
566bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
5676782fc6925a85c3772253e272745589a0c799c15Anders Carlsson  if (!isa<FunctionDecl>(d)) {
568fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
5695dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
5703068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    return;
5713068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  }
5723068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
57340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) DestructorAttr(priority));
5743068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar}
5753068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar
576803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleDeprecatedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
5776b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
578545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
5793c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
5806b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
5816b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
582bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
58340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) DeprecatedAttr());
5846b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
5856b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
586bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanianstatic void HandleUnavailableAttr(Decl *d, const AttributeList &Attr, Sema &S) {
587bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian  // check the attribute arguments.
588bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian  if (Attr.getNumArgs() != 0) {
589bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
590bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian    return;
591bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian  }
592bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
59340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) UnavailableAttr());
594bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian}
595bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian
596803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleVisibilityAttr(Decl *d, const AttributeList &Attr, Sema &S) {
5976b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
598545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 1) {
5993c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
6006b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
6016b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
602bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
603545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
6046b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  Arg = Arg->IgnoreParenCasts();
6056b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
606bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6076b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (Str == 0 || Str->isWide()) {
608fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
6093c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "visibility" << 1;
6106b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
6116b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
612bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6136b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  const char *TypeStr = Str->getStrData();
6146b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  unsigned TypeLen = Str->getByteLength();
6156b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  VisibilityAttr::VisibilityTypes type;
616bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6176b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (TypeLen == 7 && !memcmp(TypeStr, "default", 7))
6186b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    type = VisibilityAttr::DefaultVisibility;
6196b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (TypeLen == 6 && !memcmp(TypeStr, "hidden", 6))
6206b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    type = VisibilityAttr::HiddenVisibility;
6216b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (TypeLen == 8 && !memcmp(TypeStr, "internal", 8))
6226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    type = VisibilityAttr::HiddenVisibility; // FIXME
6236b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else if (TypeLen == 9 && !memcmp(TypeStr, "protected", 9))
6246b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    type = VisibilityAttr::ProtectedVisibility;
6256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  else {
62608631c5fa053867146b5ee8be658c229f6bf127cChris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_unknown_visibility) << TypeStr;
6276b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
6286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
629bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
63040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) VisibilityAttr(type));
6316b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
6326b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
6330db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattnerstatic void HandleObjCExceptionAttr(Decl *D, const AttributeList &Attr,
6340db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner                                    Sema &S) {
6350db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  if (Attr.getNumArgs() != 0) {
6360db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
6370db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    return;
6380db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  }
639bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6400db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  ObjCInterfaceDecl *OCI = dyn_cast<ObjCInterfaceDecl>(D);
6410db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  if (OCI == 0) {
6420db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_requires_objc_interface);
6430db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    return;
6440db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  }
645bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
64640b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) ObjCExceptionAttr());
6470db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner}
6480db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner
6490db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattnerstatic void HandleObjCNSObject(Decl *D, const AttributeList &Attr, Sema &S) {
650fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian  if (Attr.getNumArgs() != 0) {
651fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
652fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian    return;
653fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian  }
6540db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
655fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian    QualType T = TD->getUnderlyingType();
656fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian    if (!T->isPointerType() ||
6576217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek        !T->getAs<PointerType>()->getPointeeType()->isRecordType()) {
658fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian      S.Diag(TD->getLocation(), diag::err_nsobject_attribute);
659fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian      return;
660fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian    }
661fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian  }
66240b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) ObjCNSObjectAttr());
663fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian}
664fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian
665bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stumpstatic void
666f9201e0ff1779567150b70856753d9f2c6a91467Douglas GregorHandleOverloadableAttr(Decl *D, const AttributeList &Attr, Sema &S) {
667f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor  if (Attr.getNumArgs() != 0) {
668f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
669f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor    return;
670f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor  }
671f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor
672f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor  if (!isa<FunctionDecl>(D)) {
673f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor    S.Diag(Attr.getLoc(), diag::err_attribute_overloadable_not_function);
674f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor    return;
675f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor  }
676f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor
67740b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) OverloadableAttr());
678f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor}
679f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor
6809eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroffstatic void HandleBlocksAttr(Decl *d, const AttributeList &Attr, Sema &S) {
681bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  if (!Attr.getParameterName()) {
682fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
6833c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "blocks" << 1;
6849eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff    return;
6859eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  }
686bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6879eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  if (Attr.getNumArgs() != 0) {
6883c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
6899eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff    return;
6909eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  }
691bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
6929eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  BlocksAttr::BlocksAttrTypes type;
69392e62b02226410bcad8584541b8f1ff4d35ebab9Chris Lattner  if (Attr.getParameterName()->isStr("byref"))
6949eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff    type = BlocksAttr::ByRef;
6959eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  else {
696fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported)
6973c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "blocks" << Attr.getParameterName();
6989eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff    return;
6999eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  }
700bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
70140b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) BlocksAttr(type));
7029eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff}
7039eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff
704770918281c5bdc7b5b3942285c407e3d62270053Anders Carlssonstatic void HandleSentinelAttr(Decl *d, const AttributeList &Attr, Sema &S) {
705770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  // check the attribute arguments.
706770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  if (Attr.getNumArgs() > 2) {
707fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
708fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "0, 1 or 2";
709770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    return;
710bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
711bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
712770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  int sentinel = 0;
713770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  if (Attr.getNumArgs() > 0) {
714770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    Expr *E = static_cast<Expr *>(Attr.getArg(0));
715770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    llvm::APSInt Idx(32);
716770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    if (!E->isIntegerConstantExpr(Idx, S.Context)) {
717fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
7183c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner       << "sentinel" << 1 << E->getSourceRange();
719770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
720770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    }
721770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    sentinel = Idx.getZExtValue();
722bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
723770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    if (sentinel < 0) {
724fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_less_than_zero)
725fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << E->getSourceRange();
726770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
727770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    }
728770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  }
729770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson
730770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  int nullPos = 0;
731770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  if (Attr.getNumArgs() > 1) {
732770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    Expr *E = static_cast<Expr *>(Attr.getArg(1));
733770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    llvm::APSInt Idx(32);
734770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    if (!E->isIntegerConstantExpr(Idx, S.Context)) {
735fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
7363c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner        << "sentinel" << 2 << E->getSourceRange();
737770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
738770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    }
739770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    nullPos = Idx.getZExtValue();
740bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
741770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    if (nullPos > 1 || nullPos < 0) {
742770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      // FIXME: This error message could be improved, it would be nice
743770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      // to say what the bounds actually are.
744fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_not_zero_or_one)
745fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << E->getSourceRange();
746770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
747770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    }
748770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  }
749770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson
750770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d)) {
751897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner    const FunctionType *FT = FD->getType()->getAsFunctionType();
752897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner    assert(FT && "FunctionDecl has non-function type?");
753bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
754897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner    if (isa<FunctionNoProtoType>(FT)) {
755897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner      S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_named_arguments);
756897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner      return;
757897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner    }
758bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
759897cd90fef4cd5139999585f3af31d85c2d07720Chris Lattner    if (!cast<FunctionProtoType>(FT)->isVariadic()) {
7603bba33d6f58844d4924ab1e221dc2ff44c521624Fariborz Jahanian      S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << 0;
761770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
762bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    }
763770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  } else if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d)) {
764770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    if (!MD->isVariadic()) {
7653bba33d6f58844d4924ab1e221dc2ff44c521624Fariborz Jahanian      S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << 0;
766770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson      return;
7672f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian    }
7682f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian  } else if (isa<BlockDecl>(d)) {
769bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    // Note! BlockDecl is typeless. Variadic diagnostics will be issued by the
770bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    // caller.
7712f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian    ;
7722f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian  } else if (const VarDecl *V = dyn_cast<VarDecl>(d)) {
7732f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian    QualType Ty = V->getType();
774daf0415583e33d5d279197c65e9227c1ed92474bFariborz Jahanian    if (Ty->isBlockPointerType() || Ty->isFunctionPointerType()) {
775bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump      const FunctionType *FT = Ty->isFunctionPointerType() ? getFunctionType(d)
7766217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek        : Ty->getAs<BlockPointerType>()->getPointeeType()->getAsFunctionType();
7772f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian      if (!cast<FunctionProtoType>(FT)->isVariadic()) {
7783bba33d6f58844d4924ab1e221dc2ff44c521624Fariborz Jahanian        int m = Ty->isFunctionPointerType() ? 0 : 1;
7793bba33d6f58844d4924ab1e221dc2ff44c521624Fariborz Jahanian        S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << m;
7802f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian        return;
7812f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian      }
782ac5fc7c6bcb494b60fee7ce615ac931c5db6135eMike Stump    } else {
7832f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian      S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
784ffb0081d0c0509eb4884143381cb3e5a5f6947b4Fariborz Jahanian      << Attr.getName() << 6 /*function, method or block */;
7852f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian      return;
7862f7c39246a968b921a6d95c7f8037fb3429e9501Fariborz Jahanian    }
787770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  } else {
788fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
789ffb0081d0c0509eb4884143381cb3e5a5f6947b4Fariborz Jahanian      << Attr.getName() << 6 /*function, method or block */;
790770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson    return;
791770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  }
79240b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) SentinelAttr(sentinel, nullPos));
793770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson}
794770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson
795026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattnerstatic void HandleWarnUnusedResult(Decl *D, const AttributeList &Attr, Sema &S) {
796026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  // check the attribute arguments.
797026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  if (Attr.getNumArgs() != 0) {
798026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
799026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner    return;
800026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  }
801026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner
802026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  // TODO: could also be applied to methods?
803026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  FunctionDecl *Fn = dyn_cast<FunctionDecl>(D);
804026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  if (!Fn) {
805026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
8065dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
807026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner    return;
808026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  }
809bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
81040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  Fn->addAttr(::new (S.Context) WarnUnusedResultAttr());
811026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner}
812026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner
813026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattnerstatic void HandleWeakAttr(Decl *D, const AttributeList &Attr, Sema &S) {
8146b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
815545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
8163c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
8176b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
8186b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
8196e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar
820f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  /* weak only applies to non-static declarations */
821f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  bool isStatic = false;
822f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
823f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian    isStatic = VD->getStorageClass() == VarDecl::Static;
824f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  } else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
825f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian    isStatic = FD->getStorageClass() == FunctionDecl::Static;
826f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  }
827f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  if (isStatic) {
828f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_weak_static) <<
829f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian      dyn_cast<NamedDecl>(D)->getNameAsString();
830f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian    return;
831f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian  }
832f23ecd91bf0205b776dfab2c5231e895019a7400Fariborz Jahanian
8336e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  // TODO: could also be applied to methods?
8346e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  if (!isa<FunctionDecl>(D) && !isa<VarDecl>(D)) {
8356e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
8365dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 2 /*variable and function*/;
8376e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    return;
8386e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  }
839bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
84040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) WeakAttr());
8416b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
8426b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
8436e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbarstatic void HandleWeakImportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
8446e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  // check the attribute arguments.
8456e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  if (Attr.getNumArgs() != 0) {
8466e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
8476e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    return;
848bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
8496e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar
8506e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  // weak_import only applies to variable & function declarations.
8516e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  bool isDef = false;
8526e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
8536e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    isDef = (!VD->hasExternalStorage() || VD->getInit());
8546e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  } else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
8556fb0aee4f9dc261bbec72e1283ad8dc0557a6d96Argyrios Kyrtzidis    isDef = FD->getBody();
856d4edddde6d3966ad4a4f60d9af0f9dd36995495cFariborz Jahanian  } else if (isa<ObjCPropertyDecl>(D) || isa<ObjCMethodDecl>(D)) {
857d4edddde6d3966ad4a4f60d9af0f9dd36995495cFariborz Jahanian    // We ignore weak import on properties and methods
8581c90f4dc686ab872013544664c797604a309c563Mike Stump    return;
8596e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  } else {
8606e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
8615dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    << Attr.getName() << 2 /*variable and function*/;
8626e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    return;
8636e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  }
8646e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar
8656e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  // Merge should handle any subsequent violations.
8666e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  if (isDef) {
867bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    S.Diag(Attr.getLoc(),
8686e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar           diag::warn_attribute_weak_import_invalid_on_definition)
8696e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar      << "weak_import" << 2 /*variable and function*/;
8706e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar    return;
8716e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  }
8726e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar
87340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) WeakImportAttr());
8746e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar}
8756e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar
876026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattnerstatic void HandleDLLImportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
8776b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
878545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
8793c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
8806b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
8816b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
8827b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
8832f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // Attribute can be applied only to functions or variables.
884026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  if (isa<VarDecl>(D)) {
88540b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    D->addAttr(::new (S.Context) DLLImportAttr());
8862f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
8872f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
8882f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
889026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  FunctionDecl *FD = dyn_cast<FunctionDecl>(D);
8902f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  if (!FD) {
8912f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
8925dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 2 /*variable and function*/;
8932f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
8942f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
8952f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
8962f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // Currently, the dllimport attribute is ignored for inlined functions.
8972f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // Warning is emitted.
8982f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  if (FD->isInline()) {
8992f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
9002f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
9012f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9022f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
9032f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // The attribute is also overridden by a subsequent declaration as dllexport.
9042f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // Warning is emitted.
9052f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  for (AttributeList *nextAttr = Attr.getNext(); nextAttr;
9062f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov       nextAttr = nextAttr->getNext()) {
9072f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    if (nextAttr->getKind() == AttributeList::AT_dllexport) {
9082f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov      S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
9092f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov      return;
9102f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    }
9112f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9122f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
91340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  if (D->getAttr<DLLExportAttr>()) {
9142f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllimport";
9152f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
9162f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9172f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
91840b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) DLLImportAttr());
9196b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
9206b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
921026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattnerstatic void HandleDLLExportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
9226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
923545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
9243c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
9256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
9266b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
9277b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
9282f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  // Attribute can be applied only to functions or variables.
929026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  if (isa<VarDecl>(D)) {
93040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    D->addAttr(::new (S.Context) DLLExportAttr());
9312f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
9322f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9332f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
934026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  FunctionDecl *FD = dyn_cast<FunctionDecl>(D);
9352f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  if (!FD) {
9362f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
9375dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 2 /*variable and function*/;
9382f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
9392f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9402f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
941bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // Currently, the dllexport attribute is ignored for inlined functions, unless
942bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // the -fkeep-inline-functions flag has been used. Warning is emitted;
9432f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  if (FD->isInline()) {
9442f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    // FIXME: ... unless the -fkeep-inline-functions flag has been used.
9452f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "dllexport";
9462f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov    return;
9472f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov  }
9482f402708e62f89fb875442802e3d3f20fc909d33Anton Korobeynikov
94940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) DLLExportAttr());
9506b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
9516b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
9526f3d838867538638b9bbf412028e8537ae12f3e5Nate Begemanstatic void HandleReqdWorkGroupSize(Decl *D, const AttributeList &Attr,
9536f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman                                    Sema &S) {
9546f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  // Attribute has 3 arguments.
9556f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  if (Attr.getNumArgs() != 3) {
9566f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
9576f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    return;
9586f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  }
9596f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman
9606f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  unsigned WGSize[3];
9616f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  for (unsigned i = 0; i < 3; ++i) {
9626f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    Expr *E = static_cast<Expr *>(Attr.getArg(i));
9636f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    llvm::APSInt ArgNum(32);
9646f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    if (!E->isIntegerConstantExpr(ArgNum, S.Context)) {
9656f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman      S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
9666f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman        << "reqd_work_group_size" << E->getSourceRange();
9676f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman      return;
9686f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    }
9696f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    WGSize[i] = (unsigned) ArgNum.getZExtValue();
9706f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  }
97140b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) ReqdWorkGroupSizeAttr(WGSize[0], WGSize[1],
9726f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman                                                     WGSize[2]));
9736f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman}
9746f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman
975026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattnerstatic void HandleSectionAttr(Decl *D, const AttributeList &Attr, Sema &S) {
97617f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  // Attribute has no arguments.
97717f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  if (Attr.getNumArgs() != 1) {
97817f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
97917f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    return;
98017f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  }
98117f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar
98217f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  // Make sure that there is a string literal as the sections's single
98317f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  // argument.
984bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  StringLiteral *SE =
98517f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    dyn_cast<StringLiteral>(static_cast<Expr *>(Attr.getArg(0)));
98617f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  if (!SE) {
98717f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    // FIXME
98817f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    S.Diag(Attr.getLoc(), diag::err_attribute_annotate_no_string);
98917f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar    return;
99017f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  }
99140b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  D->addAttr(::new (S.Context) SectionAttr(std::string(SE->getStrData(),
9920b2b6e1cb1573bb295c0a65813dc4df8d57f305bChris Lattner                                                     SE->getByteLength())));
99317f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar}
99417f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar
995803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleStdCallAttr(Decl *d, const AttributeList &Attr, Sema &S) {
9967b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  // Attribute has no arguments.
997545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
9983c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
9996b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
10006b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
10017b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
10027b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  // Attribute can be applied only to functions.
10037b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  if (!isa<FunctionDecl>(d)) {
10047b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
10055dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
10067b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    return;
10077b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  }
10087b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
10097b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  // stdcall and fastcall attributes are mutually incompatible.
101040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  if (d->getAttr<FastCallAttr>()) {
10117b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    S.Diag(Attr.getLoc(), diag::err_attributes_are_not_compatible)
10127b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov      << "stdcall" << "fastcall";
10137b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    return;
10147b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  }
10157b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
101640b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) StdCallAttr());
10176b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
10186b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1019803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleFastCallAttr(Decl *d, const AttributeList &Attr, Sema &S) {
10207b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  // Attribute has no arguments.
1021545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
10223c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
10236b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
10246b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
10257b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
10267b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  if (!isa<FunctionDecl>(d)) {
10277b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
10285dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
10297b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    return;
10307b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  }
10317b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
10327b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  // stdcall and fastcall attributes are mutually incompatible.
103340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  if (d->getAttr<StdCallAttr>()) {
10347b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    S.Diag(Attr.getLoc(), diag::err_attributes_are_not_compatible)
10357b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov      << "fastcall" << "stdcall";
10367b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov    return;
10377b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov  }
10387b0a52f982e3514425fc8a3c8fc728f17c27c08eAnton Korobeynikov
103940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) FastCallAttr());
10406b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
10416b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1042803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleNothrowAttr(Decl *d, const AttributeList &Attr, Sema &S) {
10436b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
1044545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
10453c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
10466b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
10476b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1048bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
104940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) NoThrowAttr());
10506b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
10516b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1052232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlssonstatic void HandleConstAttr(Decl *d, const AttributeList &Attr, Sema &S) {
1053232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  // check the attribute arguments.
1054232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  if (Attr.getNumArgs() != 0) {
10553c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
1056232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson    return;
1057232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  }
1058bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
105940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) ConstAttr());
1060232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson}
1061232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson
1062232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlssonstatic void HandlePureAttr(Decl *d, const AttributeList &Attr, Sema &S) {
1063232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  // check the attribute arguments.
1064232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  if (Attr.getNumArgs() != 0) {
10653c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
1066232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson    return;
1067232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  }
1068bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
106940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) PureAttr());
1070232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson}
1071232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson
1072f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlssonstatic void HandleCleanupAttr(Decl *d, const AttributeList &Attr, Sema &S) {
107389941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  // Match gcc which ignores cleanup attrs when compiling C++.
107489941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  if (S.getLangOptions().CPlusPlus)
107589941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson    return;
1076bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1077bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  if (!Attr.getParameterName()) {
1078f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
1079f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1080f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1081bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1082f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  if (Attr.getNumArgs() != 0) {
1083f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
1084f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1085f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1086bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1087f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  VarDecl *VD = dyn_cast<VarDecl>(d);
1088bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1089f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  if (!VD || !VD->hasLocalStorage()) {
1090f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "cleanup";
1091f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1092f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1093bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1094f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  // Look up the function
109547b9a1ca55e61e37f5a368740e29de190345acc6Douglas Gregor  NamedDecl *CleanupDecl = S.LookupName(S.TUScope, Attr.getParameterName(),
109647b9a1ca55e61e37f5a368740e29de190345acc6Douglas Gregor                                        Sema::LookupOrdinaryName);
1097f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  if (!CleanupDecl) {
109889941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_arg_not_found) <<
1099f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson      Attr.getParameterName();
1100f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1101f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1102bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1103f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  FunctionDecl *FD = dyn_cast<FunctionDecl>(CleanupDecl);
1104f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  if (!FD) {
110589941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_arg_not_function) <<
1106f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson      Attr.getParameterName();
1107f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1108f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1109f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson
1110f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  if (FD->getNumParams() != 1) {
111189941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_func_must_take_one_arg) <<
1112f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson      Attr.getParameterName();
1113f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson    return;
1114f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  }
1115bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
111689941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  // We're currently more strict than GCC about what function types we accept.
111789941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  // If this ever proves to be a problem it should be easy to fix.
111889941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  QualType Ty = S.Context.getPointerType(VD->getType());
111989941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  QualType ParamTy = FD->getParamDecl(0)->getType();
1120d5e3e8ec50d6ea481b3bc841dcbe853175d05122Eli Friedman  if (S.CheckAssignmentConstraints(ParamTy, Ty) != Sema::Compatible) {
1121bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    S.Diag(Attr.getLoc(),
112289941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson           diag::err_attribute_cleanup_func_arg_incompatible_type) <<
112389941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson      Attr.getParameterName() << ParamTy << Ty;
112489941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson    return;
112589941c1c68d8e4eec3c8ea8ee68e34d9e3c7b083Anders Carlsson  }
1126bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
112740b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) CleanupAttr(FD));
1128f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson}
1129f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson
1130bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// Handle __attribute__((format_arg((idx)))) attribute based on
1131bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// http://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html
1132bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stumpstatic void HandleFormatArgAttr(Decl *d, const AttributeList &Attr, Sema &S) {
11335b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (Attr.getNumArgs() != 1) {
11345b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
11355b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
11365b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  }
11375b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
11385b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
11395b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    << Attr.getName() << 0 /*function*/;
11405b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
11415b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  }
1142bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // FIXME: in C++ the implicit 'this' function parameter also counts.  this is
1143bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // needed in order to be compatible with GCC the index must start with 1.
11445b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  unsigned NumArgs  = getFunctionOrMethodNumArgs(d);
11455b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  unsigned FirstIdx = 1;
11465b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  // checks for the 2nd argument
11475b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  Expr *IdxExpr = static_cast<Expr *>(Attr.getArg(0));
11485b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  llvm::APSInt Idx(32);
11495b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (!IdxExpr->isIntegerConstantExpr(Idx, S.Context)) {
11505b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
11515b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    << "format" << 2 << IdxExpr->getSourceRange();
11525b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
11535b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  }
1154bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
11555b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (Idx.getZExtValue() < FirstIdx || Idx.getZExtValue() > NumArgs) {
11565b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
11575b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    << "format" << 2 << IdxExpr->getSourceRange();
11585b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
11595b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  }
1160bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
11615b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  unsigned ArgIdx = Idx.getZExtValue() - 1;
1162bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
11635b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  // make sure the format string is really a string
11645b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  QualType Ty = getFunctionOrMethodArgType(d, ArgIdx);
1165bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
11665b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  bool not_nsstring_type = !isNSStringType(Ty, S.Context);
11675b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (not_nsstring_type &&
11685b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian      !isCFStringType(Ty, S.Context) &&
11695b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian      (!Ty->isPointerType() ||
11706217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek       !Ty->getAs<PointerType>()->getPointeeType()->isCharType())) {
11715b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    // FIXME: Should highlight the actual expression that has the wrong type.
11725b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
1173bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    << (not_nsstring_type ? "a string type" : "an NSString")
11745b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian       << IdxExpr->getSourceRange();
11755b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
1176bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
11775b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  Ty = getFunctionOrMethodResultType(d);
11785b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  if (!isNSStringType(Ty, S.Context) &&
11795b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian      !isCFStringType(Ty, S.Context) &&
11805b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian      (!Ty->isPointerType() ||
11816217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek       !Ty->getAs<PointerType>()->getPointeeType()->isCharType())) {
11825b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    // FIXME: Should highlight the actual expression that has the wrong type.
11835b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    S.Diag(Attr.getLoc(), diag::err_format_attribute_result_not)
1184bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    << (not_nsstring_type ? "string type" : "NSString")
11855b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian       << IdxExpr->getSourceRange();
11865b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian    return;
1187bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  }
1188bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
118940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) FormatArgAttr(Idx.getZExtValue()));
11905b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian}
11915b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian
1192bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// Handle __attribute__((format(type,idx,firstarg))) attributes based on
1193bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// http://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html
1194803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleFormatAttr(Decl *d, const AttributeList &Attr, Sema &S) {
11956b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1196545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (!Attr.getParameterName()) {
1197fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
11983c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "format" << 1;
11996b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12006b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12016b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1202545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 2) {
12033c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 3;
12046b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12056b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12066b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1207620d89ca4eb5dcb6be13a42aafa4849eaa9b834bFariborz Jahanian  if (!isFunctionOrMethodOrBlock(d) || !hasFunctionProto(d)) {
1208fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
12095dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
12106b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12116b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12126b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1213bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // FIXME: in C++ the implicit 'this' function parameter also counts.  this is
1214bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // needed in order to be compatible with GCC the index must start in 1 and the
1215bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  // limit is numargs+1
12163568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar  unsigned NumArgs  = getFunctionOrMethodNumArgs(d);
12176b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  unsigned FirstIdx = 1;
12186b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1219545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  const char *Format = Attr.getParameterName()->getName();
1220545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  unsigned FormatLen = Attr.getParameterName()->getLength();
12216b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // Normalize the argument, __foo__ becomes foo.
12236b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (FormatLen > 4 && Format[0] == '_' && Format[1] == '_' &&
12246b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner      Format[FormatLen - 2] == '_' && Format[FormatLen - 1] == '_') {
12256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    Format += 2;
12266b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    FormatLen -= 4;
12276b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12296b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  bool Supported = false;
12306b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  bool is_NSString = false;
12316b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  bool is_strftime = false;
1232085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  bool is_CFString = false;
1233bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
12346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  switch (FormatLen) {
12356b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  default: break;
1236803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case 5: Supported = !memcmp(Format, "scanf", 5); break;
1237803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case 6: Supported = !memcmp(Format, "printf", 6); break;
12384403a5e1f956fa86d515492dbe7c7a2817d8780dRyan Flynn  case 7: Supported = !memcmp(Format, "printf0", 7) ||
12394403a5e1f956fa86d515492dbe7c7a2817d8780dRyan Flynn                      !memcmp(Format, "strfmon", 7); break;
12406b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  case 8:
1241085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    Supported = (is_strftime = !memcmp(Format, "strftime", 8)) ||
1242085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar                (is_NSString = !memcmp(Format, "NSString", 8)) ||
1243085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar                (is_CFString = !memcmp(Format, "CFString", 8));
12446b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    break;
12456b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1246bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
12476b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (!Supported) {
1248fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported)
1249fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "format" << Attr.getParameterName()->getName();
12506b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12516b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12526b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12536b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // checks for the 2nd argument
1254545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *IdxExpr = static_cast<Expr *>(Attr.getArg(0));
1255803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  llvm::APSInt Idx(32);
1256803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  if (!IdxExpr->isIntegerConstantExpr(Idx, S.Context)) {
1257fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
12583c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "format" << 2 << IdxExpr->getSourceRange();
12596b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12606b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12616b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12626b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (Idx.getZExtValue() < FirstIdx || Idx.getZExtValue() > NumArgs) {
1263fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
12643c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "format" << 2 << IdxExpr->getSourceRange();
12656b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12666b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12676b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12686b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // FIXME: Do we need to bounds check?
12696b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  unsigned ArgIdx = Idx.getZExtValue() - 1;
1270bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
12716b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // make sure the format string is really a string
12723568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar  QualType Ty = getFunctionOrMethodArgType(d, ArgIdx);
12736b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1274085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  if (is_CFString) {
1275085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    if (!isCFStringType(Ty, S.Context)) {
1276fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
1277fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << "a CFString" << IdxExpr->getSourceRange();
1278085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar      return;
1279085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar    }
1280085e8f7da37a227ceee7f98b724e0a42e04d01caDaniel Dunbar  } else if (is_NSString) {
1281390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump    // FIXME: do we need to check if the type is NSString*?  What are the
1282390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump    // semantics?
1283803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    if (!isNSStringType(Ty, S.Context)) {
1284390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump      // FIXME: Should highlight the actual expression that has the wrong type.
1285fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
1286fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << "an NSString" << IdxExpr->getSourceRange();
12876b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner      return;
1288bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    }
12896b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  } else if (!Ty->isPointerType() ||
12906217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek             !Ty->getAs<PointerType>()->getPointeeType()->isCharType()) {
1291390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump    // FIXME: Should highlight the actual expression that has the wrong type.
1292fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
1293fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "a string type" << IdxExpr->getSourceRange();
12946b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
12956b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
12966b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
12976b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the 3rd argument
1298545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *FirstArgExpr = static_cast<Expr *>(Attr.getArg(1));
1299803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  llvm::APSInt FirstArg(32);
1300803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  if (!FirstArgExpr->isIntegerConstantExpr(FirstArg, S.Context)) {
1301fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
13023c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "format" << 3 << FirstArgExpr->getSourceRange();
13036b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
13046b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
13056b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
13066b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check if the function is variadic if the 3rd argument non-zero
13076b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (FirstArg != 0) {
13083568249c2d72d58b835a22d9186f5a6b4fc4bcd6Daniel Dunbar    if (isFunctionOrMethodVariadic(d)) {
13096b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner      ++NumArgs; // +1 for ...
13106b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    } else {
1311803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner      S.Diag(d->getLocation(), diag::err_format_attribute_requires_variadic);
13126b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner      return;
13136b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    }
13146b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
13156b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
13163c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner  // strftime requires FirstArg to be 0 because it doesn't read from any
13173c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner  // variable the input is just the current time + the format string.
13186b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (is_strftime) {
13196b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    if (FirstArg != 0) {
1320fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      S.Diag(Attr.getLoc(), diag::err_format_strftime_third_parameter)
1321fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner        << FirstArgExpr->getSourceRange();
13226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner      return;
13236b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    }
13246b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // if 0 it disables parameter checking (to use with e.g. va_list)
13256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  } else if (FirstArg != 0 && FirstArg != NumArgs) {
1326fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
13273c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner      << "format" << 3 << FirstArgExpr->getSourceRange();
13286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
13296b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
13306b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
133140b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) FormatAttr(std::string(Format, FormatLen),
13326b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner                            Idx.getZExtValue(), FirstArg.getZExtValue()));
13336b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
13346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
13350b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattnerstatic void HandleTransparentUnionAttr(Decl *d, const AttributeList &Attr,
13360b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner                                       Sema &S) {
13376b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
1338545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 0) {
13393c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
13406b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
13416b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
13426b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
13430c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  // Try to find the underlying union declaration.
13440c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  RecordDecl *RD = 0;
1345bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman  TypedefDecl *TD = dyn_cast<TypedefDecl>(d);
13460c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  if (TD && TD->getUnderlyingType()->isUnionType())
13470c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    RD = TD->getUnderlyingType()->getAsUnionType()->getDecl();
13480c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  else
13490c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    RD = dyn_cast<RecordDecl>(d);
13500c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor
13510c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  if (!RD || !RD->isUnion()) {
1352fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
13535dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 1 /*union*/;
13546b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
13556b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
13566b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
13570c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  if (!RD->isDefinition()) {
1358bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    S.Diag(Attr.getLoc(),
13590c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor        diag::warn_transparent_union_attribute_not_definition);
13600c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    return;
13610c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  }
13620c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor
136317945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis  RecordDecl::field_iterator Field = RD->field_begin(),
136417945a0f64fe03ff6ec0c2146005a87636e3ac12Argyrios Kyrtzidis                          FieldEnd = RD->field_end();
13650c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  if (Field == FieldEnd) {
13660c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    S.Diag(Attr.getLoc(), diag::warn_transparent_union_attribute_zero_fields);
13670c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    return;
13680c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  }
1369bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman
13700c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  FieldDecl *FirstField = *Field;
13710c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  QualType FirstType = FirstField->getType();
13720c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  if (FirstType->isFloatingType() || FirstType->isVectorType()) {
1373bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    S.Diag(FirstField->getLocation(),
13740c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor           diag::warn_transparent_union_attribute_floating);
13750c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    return;
13760c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  }
1377bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman
13780c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  uint64_t FirstSize = S.Context.getTypeSize(FirstType);
13790c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  uint64_t FirstAlign = S.Context.getTypeAlign(FirstType);
13800c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor  for (; Field != FieldEnd; ++Field) {
13810c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    QualType FieldType = Field->getType();
13820c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor    if (S.Context.getTypeSize(FieldType) != FirstSize ||
13830c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor        S.Context.getTypeAlign(FieldType) != FirstAlign) {
13840c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor      // Warn if we drop the attribute.
13850c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor      bool isSize = S.Context.getTypeSize(FieldType) != FirstSize;
1386bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump      unsigned FieldBits = isSize? S.Context.getTypeSize(FieldType)
13870c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor                                 : S.Context.getTypeAlign(FieldType);
1388bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump      S.Diag(Field->getLocation(),
13890c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor          diag::warn_transparent_union_attribute_field_size_align)
13900c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor        << isSize << Field->getDeclName() << FieldBits;
13910c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor      unsigned FirstBits = isSize? FirstSize : FirstAlign;
1392bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump      S.Diag(FirstField->getLocation(),
13930c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor             diag::note_transparent_union_first_field_size_align)
13940c74e8a4e8865ec9ebb8efc0af247a3c077236c4Douglas Gregor        << isSize << FirstBits;
1395bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman      return;
1396bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman    }
1397bc88745b43f440341e60ed93b0d27bac7c418029Eli Friedman  }
13986b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
139940b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  RD->addAttr(::new (S.Context) TransparentUnionAttr());
14006b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
14016b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
14020b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattnerstatic void HandleAnnotateAttr(Decl *d, const AttributeList &Attr, Sema &S) {
14036b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
1404545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() != 1) {
14053c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
14066b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
14076b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1408545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  Expr *argExpr = static_cast<Expr *>(Attr.getArg(0));
14096b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  StringLiteral *SE = dyn_cast<StringLiteral>(argExpr);
1410bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
14116b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // Make sure that there is a string literal as the annotation's single
14126b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // argument.
14136b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  if (!SE) {
14140b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_annotate_no_string);
14156b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
14166b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
141740b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) AnnotateAttr(std::string(SE->getStrData(),
14180b2b6e1cb1573bb295c0a65813dc4df8d57f305bChris Lattner                                                        SE->getByteLength())));
14196b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
14206b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
1421803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattnerstatic void HandleAlignedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
14226b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  // check the attribute arguments.
1423545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() > 1) {
14243c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
14256b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
14266b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
14276b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner
14286b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  unsigned Align = 0;
1429545dd3401e7f31c256d69cb948a45d5ca781064cChris Lattner  if (Attr.getNumArgs() == 0) {
14306b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    // FIXME: This should be the target specific maximum alignment.
14317549c5589ac0d2087e55f2bdd4854adef23f29fdDaniel Dunbar    // (For now we just use 128 bits which is the maximum on X86).
14326b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    Align = 128;
143340b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis    d->addAttr(::new (S.Context) AlignedAttr(Align));
14346b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner    return;
14356b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner  }
1436bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
143749e2d34f74f98bef23e37c415ce90cd783cdea24Chris Lattner  Expr *alignmentExpr = static_cast<Expr *>(Attr.getArg(0));
143849e2d34f74f98bef23e37c415ce90cd783cdea24Chris Lattner  llvm::APSInt Alignment(32);
1439803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  if (!alignmentExpr->isIntegerConstantExpr(Alignment, S.Context)) {
1440fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
1441fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "aligned" << alignmentExpr->getSourceRange();
144249e2d34f74f98bef23e37c415ce90cd783cdea24Chris Lattner    return;
144349e2d34f74f98bef23e37c415ce90cd783cdea24Chris Lattner  }
1444396b2a22788b0134018760d6a476de1e20f81334Daniel Dunbar  if (!llvm::isPowerOf2_64(Alignment.getZExtValue())) {
1445bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    S.Diag(Attr.getLoc(), diag::err_attribute_aligned_not_power_of_two)
1446396b2a22788b0134018760d6a476de1e20f81334Daniel Dunbar      << alignmentExpr->getSourceRange();
1447396b2a22788b0134018760d6a476de1e20f81334Daniel Dunbar    return;
1448396b2a22788b0134018760d6a476de1e20f81334Daniel Dunbar  }
1449396b2a22788b0134018760d6a476de1e20f81334Daniel Dunbar
145040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) AlignedAttr(Alignment.getZExtValue() * 8));
14516b6b5372f4b60b1c5ee101709e71a04642c835f4Chris Lattner}
1452fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1453bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// HandleModeAttr - This attribute modifies the width of a decl with primitive
1454bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// type.
1455fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner///
1456bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// Despite what would be logical, the mode attribute is a decl attribute, not a
1457bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// type attribute: 'int ** __attribute((mode(HI))) *G;' tries to make 'G' be
1458bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump/// HImode, not an intermediate pointer.
14590b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattnerstatic void HandleModeAttr(Decl *D, const AttributeList &Attr, Sema &S) {
1460fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  // This attribute isn't documented, but glibc uses it.  It changes
1461fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  // the width of an int or unsigned int to the specified size.
1462fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1463fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  // Check that there aren't any arguments
1464fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  if (Attr.getNumArgs() != 0) {
14653c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
1466fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    return;
1467fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1468fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1469fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  IdentifierInfo *Name = Attr.getParameterName();
1470fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  if (!Name) {
14710b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_missing_parameter_name);
1472fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    return;
1473fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1474fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  const char *Str = Name->getName();
1475fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  unsigned Len = Name->getLength();
1476fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1477fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  // Normalize the attribute name, __foo__ becomes foo.
1478fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  if (Len > 4 && Str[0] == '_' && Str[1] == '_' &&
1479fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner      Str[Len - 2] == '_' && Str[Len - 1] == '_') {
1480fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    Str += 2;
1481fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    Len -= 4;
1482fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1483fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1484fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  unsigned DestWidth = 0;
1485fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  bool IntegerMode = true;
148673397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  bool ComplexMode = false;
1487fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  switch (Len) {
1488fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 2:
148973397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    switch (Str[0]) {
149073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'Q': DestWidth = 8; break;
149173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'H': DestWidth = 16; break;
149273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'S': DestWidth = 32; break;
149373397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'D': DestWidth = 64; break;
149473397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'X': DestWidth = 96; break;
149573397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    case 'T': DestWidth = 128; break;
149673397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    }
149773397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (Str[1] == 'F') {
149873397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      IntegerMode = false;
149973397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    } else if (Str[1] == 'C') {
150073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      IntegerMode = false;
150173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      ComplexMode = true;
150273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    } else if (Str[1] != 'I') {
150373397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      DestWidth = 0;
150473397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    }
1505fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1506fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 4:
1507fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    // FIXME: glibc uses 'word' to define register_t; this is narrower than a
1508fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    // pointer on PIC16 and other embedded platforms.
1509fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (!memcmp(Str, "word", 4))
15100b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      DestWidth = S.Context.Target.getPointerWidth(0);
1511fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (!memcmp(Str, "byte", 4))
15120b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      DestWidth = S.Context.Target.getCharWidth();
1513fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1514fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 7:
1515fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (!memcmp(Str, "pointer", 7))
15160b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      DestWidth = S.Context.Target.getPointerWidth(0);
1517fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1518fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1519fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1520fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  QualType OldTy;
1521fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
1522fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    OldTy = TD->getUnderlyingType();
1523fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  else if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
1524fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    OldTy = VD->getType();
1525fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  else {
1526fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner    S.Diag(D->getLocation(), diag::err_attr_wrong_decl)
1527fa25bbb351f4fdd977f51254119cdfc2b525ce90Chris Lattner      << "mode" << SourceRange(Attr.getLoc(), Attr.getLoc());
1528fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    return;
1529fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
153073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman
153173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  if (!OldTy->getAsBuiltinType() && !OldTy->isComplexType())
153273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    S.Diag(Attr.getLoc(), diag::err_mode_not_primitive);
153373397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  else if (IntegerMode) {
153473397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (!OldTy->isIntegralType())
153573397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
153673397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  } else if (ComplexMode) {
153773397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (!OldTy->isComplexType())
153873397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
153973397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  } else {
154073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (!OldTy->isFloatingType())
154173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
154273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  }
154373397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman
1544390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump  // FIXME: Sync this with InitializePredefinedMacros; we need to match int8_t
1545390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump  // and friends, at least with glibc.
1546390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump  // FIXME: Make sure 32/64-bit integers don't get defined to types of the wrong
1547390b4cc8b45a05612349269ef08faab3e4688f06Mike Stump  // width on unusual platforms.
1548f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman  // FIXME: Make sure floating-point mappings are accurate
1549f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman  // FIXME: Support XF and TF types
1550fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  QualType NewTy;
1551fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  switch (DestWidth) {
1552fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 0:
15533c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_unknown_machine_mode) << Name;
1554fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    return;
1555fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  default:
15563c73c41cefcfe76f36b7bed72c9f1ec195490951Chris Lattner    S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
1557fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    return;
1558fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 8:
155973397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (!IntegerMode) {
156073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
156173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      return;
156273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    }
1563fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (OldTy->isSignedIntegerType())
15640b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.SignedCharTy;
1565fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else
15660b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.UnsignedCharTy;
1567fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1568fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 16:
156973397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    if (!IntegerMode) {
157073397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
157173397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman      return;
157273397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    }
1573fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (OldTy->isSignedIntegerType())
15740b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.ShortTy;
1575fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else
15760b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.UnsignedShortTy;
1577fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1578fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 32:
1579fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (!IntegerMode)
15800b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.FloatTy;
1581fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else if (OldTy->isSignedIntegerType())
15820b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.IntTy;
1583fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else
15840b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.UnsignedIntTy;
1585fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
1586fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  case 64:
1587fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    if (!IntegerMode)
15880b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.DoubleTy;
1589fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else if (OldTy->isSignedIntegerType())
15900b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.LongLongTy;
1591fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    else
15920b2f4da7651feb6edab5e4a657fce058f0dd514aChris Lattner      NewTy = S.Context.UnsignedLongLongTy;
1593fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    break;
159473397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  case 96:
159573397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    NewTy = S.Context.LongDoubleTy;
159673397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    break;
1597f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman  case 128:
1598f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman    if (!IntegerMode) {
1599f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman      S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
1600f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman      return;
1601f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman    }
1602f98aba35e6c3da5aae61843fc01334939e4e12ecEli Friedman    NewTy = S.Context.getFixedWidthIntType(128, OldTy->isSignedIntegerType());
160373397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    break;
1604fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1605fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
160673397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman  if (ComplexMode) {
160773397496fec250f565f49e27f8ba79f94f4e7427Eli Friedman    NewTy = S.Context.getComplexType(NewTy);
1608fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  }
1609fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner
1610fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  // Install the new type.
1611fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
1612fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    TD->setUnderlyingType(NewTy);
1613fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner  else
1614fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner    cast<ValueDecl>(D)->setType(NewTy);
1615fbf1347e1e225cbc206563bba3f0a75f9ceaa571Chris Lattner}
16160744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner
1617d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlssonstatic void HandleNodebugAttr(Decl *d, const AttributeList &Attr, Sema &S) {
1618d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson  // check the attribute arguments.
1619d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson  if (Attr.getNumArgs() > 0) {
1620d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
1621d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson    return;
1622d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson  }
1623e896d98548b02223c7740d807a0aa6e20fba7079Anders Carlsson
16245bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  if (!isFunctionOrMethod(d)) {
1625d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
16265dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
1627d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson    return;
1628d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson  }
1629bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
163040b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) NodebugAttr());
1631d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson}
1632d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson
16335bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlssonstatic void HandleNoinlineAttr(Decl *d, const AttributeList &Attr, Sema &S) {
16345bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  // check the attribute arguments.
16355bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  if (Attr.getNumArgs() != 0) {
16365bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
16375bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    return;
16385bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  }
1639bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1640c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  if (!isa<FunctionDecl>(d)) {
16415bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
16425dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    << Attr.getName() << 0 /*function*/;
16435bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson    return;
16445bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  }
1645bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
164640b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) NoinlineAttr());
16475bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson}
16485bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson
1649cf2a7211b4785068c7efa836baab90b198a4d2a6Chris Lattnerstatic void HandleGNUInlineAttr(Decl *d, const AttributeList &Attr, Sema &S) {
165026e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner  // check the attribute arguments.
165126e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner  if (Attr.getNumArgs() != 0) {
165226e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
165326e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner    return;
165426e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner  }
1655bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1656c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  FunctionDecl *Fn = dyn_cast<FunctionDecl>(d);
1657c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  if (Fn == 0) {
165826e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
16595dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName() << 0 /*function*/;
166026e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner    return;
166126e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner  }
1662bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1663c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  if (!Fn->isInline()) {
1664cf2a7211b4785068c7efa836baab90b198a4d2a6Chris Lattner    S.Diag(Attr.getLoc(), diag::warn_gnu_inline_attribute_requires_inline);
1665c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner    return;
1666c51974328b3a378c3c40b1fa527ecb928ed2bfdaChris Lattner  }
1667bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
166840b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) GNUInlineAttr());
166926e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner}
167026e25545b26ec06f5d674dbae00fb168e6688d90Chris Lattner
1671ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanianstatic void HandleRegparmAttr(Decl *d, const AttributeList &Attr, Sema &S) {
1672ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian  // check the attribute arguments.
1673ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian  if (Attr.getNumArgs() != 1) {
167455d3aaf9a537888734762170823daf750ea9036dEli Friedman    S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
1675ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian    return;
1676ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian  }
167755d3aaf9a537888734762170823daf750ea9036dEli Friedman
1678ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian  if (!isFunctionOrMethod(d)) {
1679ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
16805dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    << Attr.getName() << 0 /*function*/;
1681ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian    return;
1682ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian  }
168355d3aaf9a537888734762170823daf750ea9036dEli Friedman
168455d3aaf9a537888734762170823daf750ea9036dEli Friedman  Expr *NumParamsExpr = static_cast<Expr *>(Attr.getArg(0));
168555d3aaf9a537888734762170823daf750ea9036dEli Friedman  llvm::APSInt NumParams(32);
168655d3aaf9a537888734762170823daf750ea9036dEli Friedman  if (!NumParamsExpr->isIntegerConstantExpr(NumParams, S.Context)) {
168755d3aaf9a537888734762170823daf750ea9036dEli Friedman    S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
168855d3aaf9a537888734762170823daf750ea9036dEli Friedman      << "regparm" << NumParamsExpr->getSourceRange();
168955d3aaf9a537888734762170823daf750ea9036dEli Friedman    return;
169055d3aaf9a537888734762170823daf750ea9036dEli Friedman  }
169155d3aaf9a537888734762170823daf750ea9036dEli Friedman
1692264a76cdf382c507f4d43e64c89f1503f003ac95Anton Korobeynikov  if (S.Context.Target.getRegParmMax() == 0) {
1693264a76cdf382c507f4d43e64c89f1503f003ac95Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::err_attribute_regparm_wrong_platform)
169455d3aaf9a537888734762170823daf750ea9036dEli Friedman      << NumParamsExpr->getSourceRange();
169555d3aaf9a537888734762170823daf750ea9036dEli Friedman    return;
169655d3aaf9a537888734762170823daf750ea9036dEli Friedman  }
169755d3aaf9a537888734762170823daf750ea9036dEli Friedman
1698348f28ab6a574df6501ff8b76f9fc6753c155badAnton Korobeynikov  if (NumParams.getLimitedValue(255) > S.Context.Target.getRegParmMax()) {
1699264a76cdf382c507f4d43e64c89f1503f003ac95Anton Korobeynikov    S.Diag(Attr.getLoc(), diag::err_attribute_regparm_invalid_number)
1700264a76cdf382c507f4d43e64c89f1503f003ac95Anton Korobeynikov      << S.Context.Target.getRegParmMax() << NumParamsExpr->getSourceRange();
170155d3aaf9a537888734762170823daf750ea9036dEli Friedman    return;
170255d3aaf9a537888734762170823daf750ea9036dEli Friedman  }
170355d3aaf9a537888734762170823daf750ea9036dEli Friedman
170440b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis  d->addAttr(::new (S.Context) RegparmAttr(NumParams.getZExtValue()));
1705ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian}
1706ee760330a415635369556796a97afcfd6207f4dcFariborz Jahanian
17070744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner//===----------------------------------------------------------------------===//
1708b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek// Checker-specific attribute handlers.
1709b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek//===----------------------------------------------------------------------===//
1710b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek
1711b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenekstatic void HandleNSReturnsRetainedAttr(Decl *d, const AttributeList &Attr,
1712b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek                                        Sema &S) {
1713b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek
17145dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek  QualType RetTy;
1715bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
17165dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek  if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d))
17175dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    RetTy = MD->getResultType();
17185dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek  else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d))
17195dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    RetTy = FD->getResultType();
17205dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek  else {
17215dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
17225dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek        << Attr.getName() << 3 /* function or method */;
1723b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek    return;
1724b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  }
1725bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
17266217b80b7a1379b74cced1c076338262c3c980b3Ted Kremenek  if (!(S.Context.isObjCNSObjectType(RetTy) || RetTy->getAs<PointerType>()
172714108da7f7fc059772711e4ffee1322a27b152a7Steve Naroff        || RetTy->getAsObjCObjectPointerType())) {
17285dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek    S.Diag(Attr.getLoc(), diag::warn_ns_attribute_wrong_return_type)
17295dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek      << Attr.getName();
1730bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    return;
17315dc53c9c2328b5bea5422005b04960c18afd83adTed Kremenek  }
1732bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
1733b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  switch (Attr.getKind()) {
1734b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek    default:
1735b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek      assert(0 && "invalid ownership attribute");
1736b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek      return;
1737b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek    case AttributeList::AT_cf_returns_retained:
173840b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis      d->addAttr(::new (S.Context) CFReturnsRetainedAttr());
1739b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek      return;
1740b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek    case AttributeList::AT_ns_returns_retained:
174140b598eea1310ec9ed554d56ce3e25b34c585458Argyrios Kyrtzidis      d->addAttr(::new (S.Context) NSReturnsRetainedAttr());
1742b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek      return;
1743b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  };
1744b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek}
1745b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek
1746b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek//===----------------------------------------------------------------------===//
17470744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner// Top Level Sema Entry Points
17480744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner//===----------------------------------------------------------------------===//
17490744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner
1750a89d82c1c819d17042ec2db4283326a850229b21Sebastian Redl/// ProcessDeclAttribute - Apply the specific attribute to the specified decl if
1751803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner/// the attribute applies to decls.  If the attribute is a type attribute, just
1752803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner/// silently ignore it.
1753bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stumpstatic void ProcessDeclAttribute(Scope *scope, Decl *D,
1754bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump                                 const AttributeList &Attr, Sema &S) {
1755290eeb0ec2b6b91f3621e05ef541deb257fbea73Eli Friedman  if (Attr.isDeclspecAttribute())
1756290eeb0ec2b6b91f3621e05ef541deb257fbea73Eli Friedman    // FIXME: Try to deal with __declspec attributes!
1757290eeb0ec2b6b91f3621e05ef541deb257fbea73Eli Friedman    return;
1758803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  switch (Attr.getKind()) {
17593068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_IBOutlet:    HandleIBOutletAttr  (D, Attr, S); break;
1760803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_address_space:
1761ba372b85524f712e5b97a176f6ce0197d365835dFariborz Jahanian  case AttributeList::AT_objc_gc:
1762bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    // Ignore these, these are type attributes, handled by
1763bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    // ProcessTypeAttributes.
1764803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    break;
1765803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_alias:       HandleAliasAttr     (D, Attr, S); break;
17663068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_aligned:     HandleAlignedAttr   (D, Attr, S); break;
1767bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  case AttributeList::AT_always_inline:
1768af668b0e7d3581dea3b4f29a9262686e83887e5bDaniel Dunbar    HandleAlwaysInlineAttr  (D, Attr, S); break;
1769b725232b46e92f3e36b03a32a6fc75748c312122Ted Kremenek  case AttributeList::AT_analyzer_noreturn:
1770bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump    HandleAnalyzerNoReturnAttr  (D, Attr, S); break;
17713068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_annotate:    HandleAnnotateAttr  (D, Attr, S); break;
17723068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_constructor: HandleConstructorAttr(D, Attr, S); break;
1773803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_deprecated:  HandleDeprecatedAttr(D, Attr, S); break;
17743068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_destructor:  HandleDestructorAttr(D, Attr, S); break;
1775803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_dllexport:   HandleDLLExportAttr (D, Attr, S); break;
17763068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_dllimport:   HandleDLLImportAttr (D, Attr, S); break;
17773068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_ext_vector_type:
17789cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    HandleExtVectorTypeAttr(scope, D, Attr, S);
17793068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar    break;
1780803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_fastcall:    HandleFastCallAttr  (D, Attr, S); break;
1781803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_format:      HandleFormatAttr    (D, Attr, S); break;
17825b160927672440076aa53c31d84149f70fd8d40eFariborz Jahanian  case AttributeList::AT_format_arg:  HandleFormatArgAttr (D, Attr, S); break;
1783cf2a7211b4785068c7efa836baab90b198a4d2a6Chris Lattner  case AttributeList::AT_gnu_inline:  HandleGNUInlineAttr(D, Attr, S); break;
17843068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_mode:        HandleModeAttr      (D, Attr, S); break;
178576168e289ca4b307259e3bc9b3353f03b05bb6b9Ryan Flynn  case AttributeList::AT_malloc:      HandleMallocAttr    (D, Attr, S); break;
1786eb2b2a3f1898f146c6e153a64ec58ec5e1750bd2Ted Kremenek  case AttributeList::AT_nonnull:     HandleNonNullAttr   (D, Attr, S); break;
17873068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_noreturn:    HandleNoReturnAttr  (D, Attr, S); break;
17883068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_nothrow:     HandleNothrowAttr   (D, Attr, S); break;
1789b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek
1790b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  // Checker-specific.
1791b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  case AttributeList::AT_ns_returns_retained:
1792b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek  case AttributeList::AT_cf_returns_retained:
1793b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek    HandleNSReturnsRetainedAttr(D, Attr, S); break;
1794b71368d28532908ae1c2dc23f91761781205b3d0Ted Kremenek
17956f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman  case AttributeList::AT_reqd_wg_size:
17966f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman    HandleReqdWorkGroupSize(D, Attr, S); break;
17976f3d838867538638b9bbf412028e8537ae12f3e5Nate Begeman
17983068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_packed:      HandlePackedAttr    (D, Attr, S); break;
179917f194f4393a67fd28ad822c06d32b8cb99bad3fDaniel Dunbar  case AttributeList::AT_section:     HandleSectionAttr   (D, Attr, S); break;
18003068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_stdcall:     HandleStdCallAttr   (D, Attr, S); break;
1801bc1c877fe28fb6a825f0b226a0a2da99e713ea03Fariborz Jahanian  case AttributeList::AT_unavailable: HandleUnavailableAttr(D, Attr, S); break;
180273798892751e378cbcdef43579c1d41685091fd0Ted Kremenek  case AttributeList::AT_unused:      HandleUnusedAttr    (D, Attr, S); break;
1803b805dad4aa386aeae0f72512895bd238678d37a5Daniel Dunbar  case AttributeList::AT_used:        HandleUsedAttr      (D, Attr, S); break;
18043068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_vector_size: HandleVectorSizeAttr(D, Attr, S); break;
18053068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_visibility:  HandleVisibilityAttr(D, Attr, S); break;
1806026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner  case AttributeList::AT_warn_unused_result: HandleWarnUnusedResult(D,Attr,S);
1807026dc96ac6ece60da5e1b98e2a71bd0ff0939fd8Chris Lattner    break;
18083068ae0feb5d477477f45045f7ec9d0414fe57f3Daniel Dunbar  case AttributeList::AT_weak:        HandleWeakAttr      (D, Attr, S); break;
18096e775dbafba2ab6634decc489eb3b4301b4b506bDaniel Dunbar  case AttributeList::AT_weak_import: HandleWeakImportAttr(D, Attr, S); break;
1810803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  case AttributeList::AT_transparent_union:
1811803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    HandleTransparentUnionAttr(D, Attr, S);
1812803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    break;
18130db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner  case AttributeList::AT_objc_exception:
18140db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    HandleObjCExceptionAttr(D, Attr, S);
18150db29ece81d360dcefbe912339c34abe5917f6a9Chris Lattner    break;
1816f9201e0ff1779567150b70856753d9f2c6a91467Douglas Gregor  case AttributeList::AT_overloadable:HandleOverloadableAttr(D, Attr, S); break;
1817fa23c1d9adc99c662c1c0e192817185809d95614Fariborz Jahanian  case AttributeList::AT_nsobject:    HandleObjCNSObject  (D, Attr, S); break;
18189eae5761c0691c0f11d7a823b8ee54f05786cbbeSteve Naroff  case AttributeList::AT_blocks:      HandleBlocksAttr    (D, Attr, S); break;
1819770918281c5bdc7b5b3942285c407e3d62270053Anders Carlsson  case AttributeList::AT_sentinel:    HandleSentinelAttr  (D, Attr, S); break;
1820232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  case AttributeList::AT_const:       HandleConstAttr     (D, Attr, S); break;
1821232eb7d33b96ad8f99de3b5ae840421b3a7c6cb7Anders Carlsson  case AttributeList::AT_pure:        HandlePureAttr      (D, Attr, S); break;
1822f6e35d0b9f1e9f1b4c5d3ef924415fa5e7c89849Anders Carlsson  case AttributeList::AT_cleanup:     HandleCleanupAttr   (D, Attr, S); break;
1823d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson  case AttributeList::AT_nodebug:     HandleNodebugAttr   (D, Attr, S); break;
18245bab788d40026ad6e932a3cd9b86bc13f8a27661Anders Carlsson  case AttributeList::AT_noinline:    HandleNoinlineAttr  (D, Attr, S); break;
182555d3aaf9a537888734762170823daf750ea9036dEli Friedman  case AttributeList::AT_regparm:     HandleRegparmAttr   (D, Attr, S); break;
1826bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump  case AttributeList::IgnoredAttribute:
18275e204486a7dd1e5f7e14e941a2c7e707a8ad1a3bChris Lattner  case AttributeList::AT_no_instrument_function:  // Interacts with -pg.
182805f8e471aae971c9867dbac148eba1275a570814Anders Carlsson    // Just ignore
182905f8e471aae971c9867dbac148eba1275a570814Anders Carlsson    break;
1830803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  default:
1831d87df37e0adaba0d5e33da7b1a14d7f1d94c5eefAnders Carlsson    S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << Attr.getName();
1832803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    break;
1833803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  }
1834803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner}
1835803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner
1836803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner/// ProcessDeclAttributeList - Apply all the decl attributes in the specified
1837803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner/// attribute list to the specified decl, ignoring any type attributes.
18389cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregorvoid Sema::ProcessDeclAttributeList(Scope *S, Decl *D, const AttributeList *AttrList) {
1839803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  while (AttrList) {
18409cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    ProcessDeclAttribute(S, D, *AttrList, *this);
1841803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner    AttrList = AttrList->getNext();
1842803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner  }
1843803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner}
1844803d08039c5194cf51071ed1d8fbc5b18b3ec38bChris Lattner
1845e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn/// DeclClonePragmaWeak - clone existing decl (maybe definition),
1846e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn/// #pragma weak needs a non-definition decl and source may not have one
18477b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan FlynnNamedDecl * Sema::DeclClonePragmaWeak(NamedDecl *ND, IdentifierInfo *II)
1848e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn{
18497b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn  assert(isa<FunctionDecl>(ND) || isa<VarDecl>(ND));
1850e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  NamedDecl *NewD = 0;
1851e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(ND)) {
1852e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    NewD = FunctionDecl::Create(FD->getASTContext(), FD->getDeclContext(),
1853e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn                                FD->getLocation(), DeclarationName(II),
1854e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn                                FD->getType());
1855e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  } else if (VarDecl *VD = dyn_cast<VarDecl>(ND)) {
1856e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    NewD = VarDecl::Create(VD->getASTContext(), VD->getDeclContext(),
1857e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn                           VD->getLocation(), II,
1858e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn                           VD->getType(), VD->getStorageClass());
1859e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  }
1860e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  return NewD;
1861e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn}
1862e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn
1863e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn/// DeclApplyPragmaWeak - A declaration (maybe definition) needs #pragma weak
1864e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn/// applied to it, possibly with an alias.
18657b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynnvoid Sema::DeclApplyPragmaWeak(Scope *S, NamedDecl *ND, WeakInfo &W) {
1866e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  if (!W.getUsed()) { // only do this once
1867e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    W.setUsed(true);
1868e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    if (W.getAlias()) { // clone decl, impersonate __attribute(weak,alias(...))
1869e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      IdentifierInfo *NDId = ND->getIdentifier();
1870e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      NamedDecl *NewD = DeclClonePragmaWeak(ND, W.getAlias());
1871e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      NewD->addAttr(::new (Context) AliasAttr(NDId->getName()));
1872e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      NewD->addAttr(::new (Context) WeakAttr());
18737b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      WeakTopLevelDecl.push_back(NewD);
18747b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      // FIXME: "hideous" code from Sema::LazilyCreateBuiltin
18757b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      // to insert Decl at TU scope, sorry.
18767b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      DeclContext *SavedContext = CurContext;
18777b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      CurContext = Context.getTranslationUnitDecl();
18787b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      PushOnScopeChains(NewD, S);
18797b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn      CurContext = SavedContext;
1880e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    } else { // just add weak to existing
1881e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      ND->addAttr(::new (Context) WeakAttr());
1882e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    }
1883e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  }
1884e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn}
1885e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn
18860744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner/// ProcessDeclAttributes - Given a declarator (PD) with attributes indicated in
18870744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner/// it, apply them to D.  This is a bit tricky because PD can have attributes
18880744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner/// specified in many different places, and we need to find and apply them all.
18899cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregorvoid Sema::ProcessDeclAttributes(Scope *S, Decl *D, const Declarator &PD) {
1890e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  // Handle #pragma weak
1891e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  if (NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
1892e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    if (ND->hasLinkage()) {
1893e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      WeakInfo W = WeakUndeclaredIdentifiers.lookup(ND->getIdentifier());
1894e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      if (W != WeakInfo()) {
18957b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn        // Identifier referenced by #pragma weak before it was declared
18967b1fdbda2757cc4a7f25664475be44119d7f8e59Ryan Flynn        DeclApplyPragmaWeak(S, ND, W);
1897e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn        WeakUndeclaredIdentifiers[ND->getIdentifier()] = W;
1898e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn      }
1899e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn    }
1900e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn  }
1901e25ff83fb7eee9eeda89b6f2371bc33a37bf1028Ryan Flynn
19020744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  // Apply decl attributes from the DeclSpec if present.
19030744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  if (const AttributeList *Attrs = PD.getDeclSpec().getAttributes())
19049cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    ProcessDeclAttributeList(S, D, Attrs);
1905bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
19060744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  // Walk the declarator structure, applying decl attributes that were in a type
19070744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  // position to the decl itself.  This handles cases like:
19080744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  //   int *__attr__(x)** D;
19090744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  // when X is a decl attribute.
19100744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  for (unsigned i = 0, e = PD.getNumTypeObjects(); i != e; ++i)
19110744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner    if (const AttributeList *Attrs = PD.getTypeObject(i).getAttrs())
19129cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor      ProcessDeclAttributeList(S, D, Attrs);
1913bf91650557d9afa08e3d311a9c0a10d73c62cbfcMike Stump
19140744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  // Finally, apply any attributes on the decl itself.
19150744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner  if (const AttributeList *Attrs = PD.getAttributes())
19169cdda0cf8528e3d595be9bfa002f0450074beb4dDouglas Gregor    ProcessDeclAttributeList(S, D, Attrs);
19170744e5f3325e2d2107506002e43c37ea0155a5acChris Lattner}
1918